Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
PojavLauncherTeam
GitHub Repository: PojavLauncherTeam/openjdk-multiarch-jdk8u
Path: blob/aarch64-shenandoah-jdk8u272-b10/hotspot/agent/src/os/bsd/MacosxDebuggerLocal.m
38833 views
1
/*
2
* Copyright (c) 2002, 2015, Oracle and/or its affiliates. All rights reserved.
3
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4
*
5
* This code is free software; you can redistribute it and/or modify it
6
* under the terms of the GNU General Public License version 2 only, as
7
* published by the Free Software Foundation.
8
*
9
* This code is distributed in the hope that it will be useful, but WITHOUT
10
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12
* version 2 for more details (a copy is included in the LICENSE file that
13
* accompanied this code).
14
*
15
* You should have received a copy of the GNU General Public License version
16
* 2 along with this work; if not, write to the Free Software Foundation,
17
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18
*
19
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20
* or visit www.oracle.com if you need additional information or have any
21
* questions.
22
*
23
*/
24
25
#include <objc/objc-runtime.h>
26
#import <Foundation/Foundation.h>
27
#import <JavaNativeFoundation/JavaNativeFoundation.h>
28
29
#include <jni.h>
30
31
#import <mach/mach.h>
32
#import <mach/mach_types.h>
33
#import <sys/sysctl.h>
34
#import <stdio.h>
35
#import <stdarg.h>
36
#import <stdlib.h>
37
#import <strings.h>
38
#import <dlfcn.h>
39
#import <limits.h>
40
#import <errno.h>
41
#import <sys/types.h>
42
#import <sys/ptrace.h>
43
#include "libproc_impl.h"
44
45
#define UNSUPPORTED_ARCH "Unsupported architecture!"
46
47
#if defined(x86_64) && !defined(amd64)
48
#define amd64 1
49
#endif
50
51
#if amd64
52
#include "sun_jvm_hotspot_debugger_amd64_AMD64ThreadContext.h"
53
#elif defined(aarch64)
54
#include "sun_jvm_hotspot_debugger_aarch64_AARCH64ThreadContext.h"
55
#else
56
#error UNSUPPORTED_ARCH
57
#endif
58
59
static jfieldID symbolicatorID = 0; // set in _init0
60
static jfieldID taskID = 0; // set in _init0
61
62
static jfieldID p_ps_prochandle_ID = 0;
63
static jfieldID loadObjectList_ID = 0;
64
static jmethodID listAdd_ID = 0;
65
66
static jmethodID createClosestSymbol_ID = 0;
67
static jmethodID createLoadObject_ID = 0;
68
static jmethodID getJavaThreadsInfo_ID = 0;
69
70
// indicator if thread id (lwpid_t) was set
71
static bool _threads_filled = false;
72
73
static void putSymbolicator(JNIEnv *env, jobject this_obj, id symbolicator) {
74
(*env)->SetLongField(env, this_obj, symbolicatorID, (jlong)(intptr_t)symbolicator);
75
}
76
77
static id getSymbolicator(JNIEnv *env, jobject this_obj) {
78
jlong ptr = (*env)->GetLongField(env, this_obj, symbolicatorID);
79
return (id)(intptr_t)ptr;
80
}
81
82
static void putTask(JNIEnv *env, jobject this_obj, task_t task) {
83
(*env)->SetLongField(env, this_obj, taskID, (jlong)task);
84
}
85
86
static task_t getTask(JNIEnv *env, jobject this_obj) {
87
jlong ptr = (*env)->GetLongField(env, this_obj, taskID);
88
return (task_t)ptr;
89
}
90
91
#define CHECK_EXCEPTION_(value) if ((*env)->ExceptionOccurred(env)) { return value; }
92
#define CHECK_EXCEPTION if ((*env)->ExceptionOccurred(env)) { return;}
93
#define THROW_NEW_DEBUGGER_EXCEPTION_(str, value) { throw_new_debugger_exception(env, str); return value; }
94
#define THROW_NEW_DEBUGGER_EXCEPTION(str) { throw_new_debugger_exception(env, str); return;}
95
#define CHECK_EXCEPTION_CLEAR if ((*env)->ExceptionOccurred(env)) { (*env)->ExceptionClear(env); }
96
#define CHECK_EXCEPTION_CLEAR_VOID if ((*env)->ExceptionOccurred(env)) { (*env)->ExceptionClear(env); return; }
97
#define CHECK_EXCEPTION_CLEAR_(value) if ((*env)->ExceptionOccurred(env)) { (*env)->ExceptionClear(env); return value; }
98
99
static void throw_new_debugger_exception(JNIEnv* env, const char* errMsg) {
100
jclass exceptionClass = (*env)->FindClass(env, "sun/jvm/hotspot/debugger/DebuggerException");
101
CHECK_EXCEPTION;
102
(*env)->ThrowNew(env, exceptionClass, errMsg);
103
}
104
105
static struct ps_prochandle* get_proc_handle(JNIEnv* env, jobject this_obj) {
106
jlong ptr = (*env)->GetLongField(env, this_obj, p_ps_prochandle_ID);
107
return (struct ps_prochandle*)(intptr_t)ptr;
108
}
109
110
#if defined(__i386__)
111
#define hsdb_thread_state_t x86_thread_state32_t
112
#define hsdb_float_state_t x86_float_state32_t
113
#define HSDB_THREAD_STATE x86_THREAD_STATE32
114
#define HSDB_FLOAT_STATE x86_FLOAT_STATE32
115
#define HSDB_THREAD_STATE_COUNT x86_THREAD_STATE32_COUNT
116
#define HSDB_FLOAT_STATE_COUNT x86_FLOAT_STATE32_COUNT
117
#elif defined(__x86_64__)
118
#define hsdb_thread_state_t x86_thread_state64_t
119
#define hsdb_float_state_t x86_float_state64_t
120
#define HSDB_THREAD_STATE x86_THREAD_STATE64
121
#define HSDB_FLOAT_STATE x86_FLOAT_STATE64
122
#define HSDB_THREAD_STATE_COUNT x86_THREAD_STATE64_COUNT
123
#define HSDB_FLOAT_STATE_COUNT x86_FLOAT_STATE64_COUNT
124
#elif defined(aarch64)
125
#define hsdb_thread_state_t arm_thread_state64_t
126
#define hsdb_float_state_t arm_neon_state64_t
127
#define HSDB_THREAD_STATE ARM_THREAD_STATE64
128
#define HSDB_FLOAT_STATE ARM_NEON_STATE64
129
#define HSDB_THREAD_STATE_COUNT ARM_THREAD_STATE64_COUNT
130
#define HSDB_FLOAT_STATE_COUNT ARM_NEON_STATE64_COUNT
131
#else
132
#error UNSUPPORTED_ARCH
133
#endif
134
135
/*
136
* Class: sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal
137
* Method: init0
138
* Signature: ()V
139
*/
140
JNIEXPORT void JNICALL
141
Java_sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal_init0(JNIEnv *env, jclass cls) {
142
symbolicatorID = (*env)->GetFieldID(env, cls, "symbolicator", "J");
143
CHECK_EXCEPTION;
144
taskID = (*env)->GetFieldID(env, cls, "task", "J");
145
CHECK_EXCEPTION;
146
147
// for core file
148
p_ps_prochandle_ID = (*env)->GetFieldID(env, cls, "p_ps_prochandle", "J");
149
CHECK_EXCEPTION;
150
loadObjectList_ID = (*env)->GetFieldID(env, cls, "loadObjectList", "Ljava/util/List;");
151
CHECK_EXCEPTION;
152
153
// methods we use
154
createClosestSymbol_ID = (*env)->GetMethodID(env, cls, "createClosestSymbol",
155
"(Ljava/lang/String;J)Lsun/jvm/hotspot/debugger/cdbg/ClosestSymbol;");
156
CHECK_EXCEPTION;
157
createLoadObject_ID = (*env)->GetMethodID(env, cls, "createLoadObject",
158
"(Ljava/lang/String;JJ)Lsun/jvm/hotspot/debugger/cdbg/LoadObject;");
159
CHECK_EXCEPTION;
160
161
// java.util.List method we call
162
jclass listClass = (*env)->FindClass(env, "java/util/List");
163
CHECK_EXCEPTION;
164
listAdd_ID = (*env)->GetMethodID(env, listClass, "add", "(Ljava/lang/Object;)Z");
165
CHECK_EXCEPTION;
166
getJavaThreadsInfo_ID = (*env)->GetMethodID(env, cls, "getJavaThreadsInfo",
167
"()[J");
168
CHECK_EXCEPTION;
169
170
init_libproc(getenv("LIBSAPROC_DEBUG") != NULL);
171
}
172
173
JNIEXPORT jint JNICALL Java_sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal_getAddressSize
174
(JNIEnv *env, jclass cls)
175
{
176
#ifdef _LP64
177
return 8;
178
#else
179
#error UNSUPPORTED_ARCH
180
#endif
181
}
182
183
/** called by Java_sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal_lookupByName0 */
184
jlong lookupByNameIncore(
185
JNIEnv *env, struct ps_prochandle *ph, jobject this_obj, jstring objectName, jstring symbolName)
186
{
187
const char *objectName_cstr, *symbolName_cstr;
188
jlong addr;
189
jboolean isCopy;
190
objectName_cstr = NULL;
191
if (objectName != NULL) {
192
objectName_cstr = (*env)->GetStringUTFChars(env, objectName, &isCopy);
193
CHECK_EXCEPTION_(0);
194
}
195
symbolName_cstr = (*env)->GetStringUTFChars(env, symbolName, &isCopy);
196
CHECK_EXCEPTION_(0);
197
198
print_debug("look for %s \n", symbolName_cstr);
199
addr = (jlong) lookup_symbol(ph, objectName_cstr, symbolName_cstr);
200
201
if (objectName_cstr != NULL) {
202
(*env)->ReleaseStringUTFChars(env, objectName, objectName_cstr);
203
}
204
(*env)->ReleaseStringUTFChars(env, symbolName, symbolName_cstr);
205
return addr;
206
}
207
208
/*
209
* Class: sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal
210
* Method: lookupByName0
211
* Signature: (Ljava/lang/String;Ljava/lang/String;)J
212
*/
213
JNIEXPORT jlong JNICALL
214
Java_sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal_lookupByName0(
215
JNIEnv *env, jobject this_obj,
216
jstring objectName, jstring symbolName)
217
{
218
struct ps_prochandle* ph = get_proc_handle(env, this_obj);
219
if (ph != NULL && ph->core != NULL) {
220
return lookupByNameIncore(env, ph, this_obj, objectName, symbolName);
221
}
222
223
jlong address = 0;
224
225
JNF_COCOA_ENTER(env);
226
NSString *symbolNameString = JNFJavaToNSString(env, symbolName);
227
228
print_debug("lookupInProcess called for %s\n", [symbolNameString UTF8String]);
229
230
id symbolicator = getSymbolicator(env, this_obj);
231
if (symbolicator != nil) {
232
uint64_t (*dynamicCall)(id, SEL, NSString *) = (uint64_t (*)(id, SEL, NSString *))&objc_msgSend;
233
address = (jlong) dynamicCall(symbolicator, @selector(addressForSymbol:), symbolNameString);
234
}
235
236
print_debug("address of symbol %s = %llx\n", [symbolNameString UTF8String], address);
237
JNF_COCOA_EXIT(env);
238
239
return address;
240
}
241
242
/*
243
* Class: sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal
244
* Method: lookupByAddress0
245
* Signature: (J)Lsun/jvm/hotspot/debugger/cdbg/ClosestSymbol;
246
*/
247
JNIEXPORT jobject JNICALL Java_sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal_lookupByAddress0
248
(JNIEnv *env, jobject this_obj, jlong addr) {
249
uintptr_t offset;
250
const char* sym = NULL;
251
jstring sym_string;
252
253
struct ps_prochandle* ph = get_proc_handle(env, this_obj);
254
if (ph != NULL && ph->core != NULL) {
255
sym = symbol_for_pc(ph, (uintptr_t) addr, &offset);
256
if (sym == NULL) return 0;
257
sym_string = (*env)->NewStringUTF(env, sym);
258
CHECK_EXCEPTION_(0);
259
return (*env)->CallObjectMethod(env, this_obj, createClosestSymbol_ID,
260
sym_string, (jlong)offset);
261
}
262
return 0;
263
}
264
265
/** called from Java_sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal_readBytesFromProcess0 */
266
jbyteArray readBytesFromCore(
267
JNIEnv *env, struct ps_prochandle *ph, jobject this_obj, jlong addr, jlong numBytes)
268
{
269
jboolean isCopy;
270
jbyteArray array;
271
jbyte *bufPtr;
272
ps_err_e err;
273
274
array = (*env)->NewByteArray(env, numBytes);
275
CHECK_EXCEPTION_(0);
276
bufPtr = (*env)->GetByteArrayElements(env, array, &isCopy);
277
CHECK_EXCEPTION_(0);
278
279
err = ps_pread(ph, (psaddr_t) (uintptr_t)addr, bufPtr, numBytes);
280
(*env)->ReleaseByteArrayElements(env, array, bufPtr, 0);
281
return (err == PS_OK)? array : 0;
282
}
283
284
/*
285
* Class: sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal
286
* Method: readBytesFromProcess0
287
* Signature: (JJ)Lsun/jvm/hotspot/debugger/ReadResult;
288
*/
289
JNIEXPORT jbyteArray JNICALL
290
Java_sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal_readBytesFromProcess0(
291
JNIEnv *env, jobject this_obj,
292
jlong addr, jlong numBytes)
293
{
294
print_debug("readBytesFromProcess called. addr = %llx numBytes = %lld\n", addr, numBytes);
295
296
// must allocate storage instead of using former parameter buf
297
jbyteArray array;
298
299
struct ps_prochandle* ph = get_proc_handle(env, this_obj);
300
if (ph != NULL && ph->core != NULL) {
301
return readBytesFromCore(env, ph, this_obj, addr, numBytes);
302
}
303
304
array = (*env)->NewByteArray(env, numBytes);
305
CHECK_EXCEPTION_(0);
306
307
unsigned long alignedAddress;
308
unsigned long alignedLength = 0;
309
kern_return_t result;
310
vm_offset_t *pages;
311
int *mapped;
312
long pageCount;
313
uint byteCount;
314
int i;
315
unsigned long remaining;
316
317
alignedAddress = trunc_page(addr);
318
if (addr != alignedAddress) {
319
alignedLength += addr - alignedAddress;
320
}
321
alignedLength = round_page(numBytes);
322
pageCount = alignedLength/vm_page_size;
323
324
// Allocate storage for pages and flags.
325
pages = malloc(pageCount * sizeof(vm_offset_t));
326
mapped = calloc(pageCount, sizeof(int));
327
328
task_t gTask = getTask(env, this_obj);
329
// Try to read each of the pages.
330
for (i = 0; i < pageCount; i++) {
331
result = vm_read(gTask, alignedAddress + i*vm_page_size, vm_page_size,
332
&pages[i], &byteCount);
333
mapped[i] = (result == KERN_SUCCESS);
334
// assume all failures are unmapped pages
335
}
336
337
print_debug("%ld pages\n", pageCount);
338
339
remaining = numBytes;
340
341
for (i = 0; i < pageCount; i++) {
342
unsigned long len = vm_page_size;
343
unsigned long start = 0;
344
345
if (i == 0) {
346
start = addr - alignedAddress;
347
len = vm_page_size - start;
348
}
349
350
if (i == (pageCount - 1)) {
351
len = remaining;
352
}
353
354
if (mapped[i]) {
355
print_debug("page %d mapped (len %ld start %ld)\n", i, len, start);
356
(*env)->SetByteArrayRegion(env, array, 0, len, ((jbyte *) pages[i] + start));
357
vm_deallocate(mach_task_self(), pages[i], vm_page_size);
358
}
359
360
remaining -= len;
361
}
362
363
free (pages);
364
free (mapped);
365
return array;
366
}
367
368
/** Only used for core file reading, set thread_id for threads which is got after core file parsed.
369
* Thread context is available in Mach-O core file but thread id is not. We can get thread id
370
* from Threads which store all java threads information when they are created. Here we can identify
371
* them as java threads by checking if a thread's rsp or rbp within a java thread's stack.
372
* Note Macosx uses unique_thread_id which is different from other platforms though printed ids
373
* are still pthread id. Function BsdDebuggerLocal.getJavaThreadsInfo returns an array of long
374
* integers to host all java threads' id, stack_start, stack_end as:
375
* [uid0, stack_start0, stack_end0, uid1, stack_start1, stack_end1, ...]
376
*
377
* The work cannot be done at init0 since Threads is not available yet(VM not initialized yet).
378
* This function should be called only once if succeeded
379
*/
380
bool fill_java_threads(JNIEnv* env, jobject this_obj, struct ps_prochandle* ph) {
381
int n = 0, i = 0, j;
382
struct reg regs;
383
384
jlongArray thrinfos = (*env)->CallObjectMethod(env, this_obj, getJavaThreadsInfo_ID);
385
CHECK_EXCEPTION_(false);
386
int len = (int)(*env)->GetArrayLength(env, thrinfos);
387
uint64_t* cinfos = (uint64_t *)(*env)->GetLongArrayElements(env, thrinfos, NULL);
388
CHECK_EXCEPTION_(false);
389
n = get_num_threads(ph);
390
print_debug("fill_java_threads called, num_of_thread = %d\n", n);
391
for (i = 0; i < n; i++) {
392
if (!get_nth_lwp_regs(ph, i, &regs)) {
393
print_debug("Could not get regs of thread %d, already set!\n", i);
394
return false;
395
}
396
for (j = 0; j < len; j += 3) {
397
lwpid_t uid = cinfos[j];
398
uint64_t beg = cinfos[j + 1];
399
uint64_t end = cinfos[j + 2];
400
#if defined(amd64)
401
if ((regs.r_rsp < end && regs.r_rsp >= beg) ||
402
(regs.r_rbp < end && regs.r_rbp >= beg)) {
403
set_lwp_id(ph, i, uid);
404
break;
405
}
406
#elif defined(aarch64)
407
if ((regs.r_sp < end && regs.r_sp >= beg) ||
408
(regs.r_fp < end && regs.r_fp >= beg)) {
409
set_lwp_id(ph, i, uid);
410
break;
411
}
412
#else
413
#error UNSUPPORTED_ARCH
414
#endif
415
}
416
}
417
(*env)->ReleaseLongArrayElements(env, thrinfos, (jlong*)cinfos, 0);
418
CHECK_EXCEPTION_(false);
419
return true;
420
}
421
422
/* For core file only, called from
423
* Java_sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal_getThreadIntegerRegisterSet0
424
*/
425
jlongArray getThreadIntegerRegisterSetFromCore(JNIEnv *env, jobject this_obj, long lwp_id, struct ps_prochandle* ph) {
426
if (!_threads_filled) {
427
if (!fill_java_threads(env, this_obj, ph)) {
428
throw_new_debugger_exception(env, "Failed to fill in threads");
429
return 0;
430
} else {
431
_threads_filled = true;
432
}
433
}
434
435
struct reg gregs;
436
jboolean isCopy;
437
jlongArray array;
438
jlong *regs;
439
440
if (get_lwp_regs(ph, lwp_id, &gregs) != true) {
441
THROW_NEW_DEBUGGER_EXCEPTION_("get_thread_regs failed for a lwp", 0);
442
}
443
444
#undef NPRGREG
445
#undef REG_INDEX
446
#if amd64
447
#define NPRGREG sun_jvm_hotspot_debugger_amd64_AMD64ThreadContext_NPRGREG
448
#define REG_INDEX(reg) sun_jvm_hotspot_debugger_amd64_AMD64ThreadContext_##reg
449
450
array = (*env)->NewLongArray(env, NPRGREG);
451
CHECK_EXCEPTION_(0);
452
regs = (*env)->GetLongArrayElements(env, array, &isCopy);
453
454
regs[REG_INDEX(R15)] = gregs.r_r15;
455
regs[REG_INDEX(R14)] = gregs.r_r14;
456
regs[REG_INDEX(R13)] = gregs.r_r13;
457
regs[REG_INDEX(R12)] = gregs.r_r12;
458
regs[REG_INDEX(RBP)] = gregs.r_rbp;
459
regs[REG_INDEX(RBX)] = gregs.r_rbx;
460
regs[REG_INDEX(R11)] = gregs.r_r11;
461
regs[REG_INDEX(R10)] = gregs.r_r10;
462
regs[REG_INDEX(R9)] = gregs.r_r9;
463
regs[REG_INDEX(R8)] = gregs.r_r8;
464
regs[REG_INDEX(RAX)] = gregs.r_rax;
465
regs[REG_INDEX(RCX)] = gregs.r_rcx;
466
regs[REG_INDEX(RDX)] = gregs.r_rdx;
467
regs[REG_INDEX(RSI)] = gregs.r_rsi;
468
regs[REG_INDEX(RDI)] = gregs.r_rdi;
469
regs[REG_INDEX(RIP)] = gregs.r_rip;
470
regs[REG_INDEX(CS)] = gregs.r_cs;
471
regs[REG_INDEX(RSP)] = gregs.r_rsp;
472
regs[REG_INDEX(SS)] = gregs.r_ss;
473
regs[REG_INDEX(FSBASE)] = 0;
474
regs[REG_INDEX(GSBASE)] = 0;
475
regs[REG_INDEX(DS)] = gregs.r_ds;
476
regs[REG_INDEX(ES)] = gregs.r_es;
477
regs[REG_INDEX(FS)] = gregs.r_fs;
478
regs[REG_INDEX(GS)] = gregs.r_gs;
479
regs[REG_INDEX(TRAPNO)] = gregs.r_trapno;
480
regs[REG_INDEX(RFL)] = gregs.r_rflags;
481
482
#elif defined(aarch64)
483
#define NPRGREG sun_jvm_hotspot_debugger_aarch64_AARCH64ThreadContext_NPRGREG
484
#define REG_INDEX(reg) sun_jvm_hotspot_debugger_aarch64_AARCH64ThreadContext_##reg
485
486
array = (*env)->NewLongArray(env, NPRGREG);
487
CHECK_EXCEPTION_(0);
488
regs = (*env)->GetLongArrayElements(env, array, &isCopy);
489
490
regs[REG_INDEX(R0)] = gregs.r_r0;
491
regs[REG_INDEX(R1)] = gregs.r_r1;
492
regs[REG_INDEX(R2)] = gregs.r_r2;
493
regs[REG_INDEX(R3)] = gregs.r_r3;
494
regs[REG_INDEX(R4)] = gregs.r_r4;
495
regs[REG_INDEX(R5)] = gregs.r_r5;
496
regs[REG_INDEX(R6)] = gregs.r_r6;
497
regs[REG_INDEX(R7)] = gregs.r_r7;
498
regs[REG_INDEX(R8)] = gregs.r_r8;
499
regs[REG_INDEX(R9)] = gregs.r_r9;
500
regs[REG_INDEX(R10)] = gregs.r_r10;
501
regs[REG_INDEX(R11)] = gregs.r_r11;
502
regs[REG_INDEX(R12)] = gregs.r_r12;
503
regs[REG_INDEX(R13)] = gregs.r_r13;
504
regs[REG_INDEX(R14)] = gregs.r_r14;
505
regs[REG_INDEX(R15)] = gregs.r_r15;
506
regs[REG_INDEX(R16)] = gregs.r_r16;
507
regs[REG_INDEX(R17)] = gregs.r_r17;
508
regs[REG_INDEX(R18)] = gregs.r_r18;
509
regs[REG_INDEX(R19)] = gregs.r_r19;
510
regs[REG_INDEX(R20)] = gregs.r_r20;
511
regs[REG_INDEX(R21)] = gregs.r_r21;
512
regs[REG_INDEX(R22)] = gregs.r_r22;
513
regs[REG_INDEX(R23)] = gregs.r_r23;
514
regs[REG_INDEX(R24)] = gregs.r_r24;
515
regs[REG_INDEX(R25)] = gregs.r_r25;
516
regs[REG_INDEX(R26)] = gregs.r_r26;
517
regs[REG_INDEX(R27)] = gregs.r_r27;
518
regs[REG_INDEX(R28)] = gregs.r_r28;
519
regs[REG_INDEX(FP)] = gregs.r_fp;
520
regs[REG_INDEX(LR)] = gregs.r_lr;
521
regs[REG_INDEX(SP)] = gregs.r_sp;
522
regs[REG_INDEX(PC)] = gregs.r_pc;
523
524
#endif /* amd64 */
525
(*env)->ReleaseLongArrayElements(env, array, regs, JNI_COMMIT);
526
return array;
527
}
528
529
/*
530
* Lookup the thread_t that corresponds to the given thread_id.
531
* The thread_id should be the result from calling thread_info() with THREAD_IDENTIFIER_INFO
532
* and reading the m_ident_info.thread_id returned.
533
* The returned thread_t is the mach send right to the kernel port for the corresponding thread.
534
*
535
* We cannot simply use the OSThread._thread_id field in the JVM. This is set to ::mach_thread_self()
536
* in the VM, but that thread port is not valid for a remote debugger to access the thread.
537
*/
538
thread_t
539
lookupThreadFromThreadId(task_t task, jlong thread_id) {
540
print_debug("lookupThreadFromThreadId thread_id=0x%llx\n", thread_id);
541
542
thread_array_t thread_list = NULL;
543
mach_msg_type_number_t thread_list_count = 0;
544
thread_t result_thread = 0;
545
int i;
546
547
// get the list of all the send rights
548
kern_return_t result = task_threads(task, &thread_list, &thread_list_count);
549
if (result != KERN_SUCCESS) {
550
print_debug("task_threads returned 0x%x\n", result);
551
return 0;
552
}
553
554
for(i = 0 ; i < thread_list_count; i++) {
555
thread_identifier_info_data_t m_ident_info;
556
mach_msg_type_number_t count = THREAD_IDENTIFIER_INFO_COUNT;
557
558
// get the THREAD_IDENTIFIER_INFO for the send right
559
result = thread_info(thread_list[i], THREAD_IDENTIFIER_INFO, (thread_info_t) &m_ident_info, &count);
560
if (result != KERN_SUCCESS) {
561
print_debug("thread_info returned 0x%x\n", result);
562
break;
563
}
564
565
// if this is the one we're looking for, return the send right
566
if (thread_id == m_ident_info.thread_id)
567
{
568
result_thread = thread_list[i];
569
break;
570
}
571
}
572
573
vm_size_t thread_list_size = (vm_size_t) (thread_list_count * sizeof (thread_t));
574
vm_deallocate(mach_task_self(), (vm_address_t) thread_list, thread_list_count);
575
576
return result_thread;
577
}
578
579
580
/*
581
* Class: sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal
582
* Method: getThreadIntegerRegisterSet0
583
* Signature: (J)[J
584
*/
585
JNIEXPORT jlongArray JNICALL
586
Java_sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal_getThreadIntegerRegisterSet0(
587
JNIEnv *env, jobject this_obj,
588
jlong thread_id)
589
{
590
print_debug("getThreadRegisterSet0 called\n");
591
592
struct ps_prochandle* ph = get_proc_handle(env, this_obj);
593
if (ph != NULL && ph->core != NULL) {
594
return getThreadIntegerRegisterSetFromCore(env, this_obj, thread_id, ph);
595
}
596
597
kern_return_t result;
598
thread_t tid;
599
mach_msg_type_number_t count = HSDB_THREAD_STATE_COUNT;
600
hsdb_thread_state_t state;
601
jlongArray registerArray;
602
jlong *primitiveArray;
603
task_t gTask = getTask(env, this_obj);
604
605
tid = lookupThreadFromThreadId(gTask, thread_id);
606
607
result = thread_get_state(tid, HSDB_THREAD_STATE, (thread_state_t)&state, &count);
608
609
if (result != KERN_SUCCESS) {
610
print_error("getregs: thread_get_state(%d) failed (%d)\n", tid, result);
611
return NULL;
612
}
613
614
// 64 bit
615
print_debug("Getting threads for a 64-bit process\n");
616
registerArray = (*env)->NewLongArray(env, NPRGREG);
617
CHECK_EXCEPTION_(0);
618
primitiveArray = (*env)->GetLongArrayElements(env, registerArray, NULL);
619
620
#if amd64
621
#define NPRGREG sun_jvm_hotspot_debugger_amd64_AMD64ThreadContext_NPRGREG
622
#undef REG_INDEX
623
#define REG_INDEX(reg) sun_jvm_hotspot_debugger_amd64_AMD64ThreadContext_##reg
624
625
primitiveArray[REG_INDEX(R15)] = state.__r15;
626
primitiveArray[REG_INDEX(R14)] = state.__r14;
627
primitiveArray[REG_INDEX(R13)] = state.__r13;
628
primitiveArray[REG_INDEX(R12)] = state.__r12;
629
primitiveArray[REG_INDEX(R11)] = state.__r11;
630
primitiveArray[REG_INDEX(R10)] = state.__r10;
631
primitiveArray[REG_INDEX(R9)] = state.__r9;
632
primitiveArray[REG_INDEX(R8)] = state.__r8;
633
primitiveArray[REG_INDEX(RDI)] = state.__rdi;
634
primitiveArray[REG_INDEX(RSI)] = state.__rsi;
635
primitiveArray[REG_INDEX(RBP)] = state.__rbp;
636
primitiveArray[REG_INDEX(RBX)] = state.__rbx;
637
primitiveArray[REG_INDEX(RDX)] = state.__rdx;
638
primitiveArray[REG_INDEX(RCX)] = state.__rcx;
639
primitiveArray[REG_INDEX(RAX)] = state.__rax;
640
primitiveArray[REG_INDEX(TRAPNO)] = 0; // trapno, not used
641
primitiveArray[REG_INDEX(ERR)] = 0; // err, not used
642
primitiveArray[REG_INDEX(RIP)] = state.__rip;
643
primitiveArray[REG_INDEX(CS)] = state.__cs;
644
primitiveArray[REG_INDEX(RFL)] = state.__rflags;
645
primitiveArray[REG_INDEX(RSP)] = state.__rsp;
646
primitiveArray[REG_INDEX(SS)] = 0; // We don't have SS
647
primitiveArray[REG_INDEX(FS)] = state.__fs;
648
primitiveArray[REG_INDEX(GS)] = state.__gs;
649
primitiveArray[REG_INDEX(ES)] = 0;
650
primitiveArray[REG_INDEX(DS)] = 0;
651
primitiveArray[REG_INDEX(FSBASE)] = 0;
652
primitiveArray[REG_INDEX(GSBASE)] = 0;
653
654
#elif defined(aarch64)
655
#define NPRGREG sun_jvm_hotspot_debugger_aarch64_AARCH64ThreadContext_NPRGREG
656
#define REG_INDEX(reg) sun_jvm_hotspot_debugger_aarch64_AARCH64ThreadContext_##reg
657
658
primitiveArray[REG_INDEX(R0)] = state.__x[0];
659
primitiveArray[REG_INDEX(R1)] = state.__x[1];
660
primitiveArray[REG_INDEX(R2)] = state.__x[2];
661
primitiveArray[REG_INDEX(R3)] = state.__x[3];
662
primitiveArray[REG_INDEX(R4)] = state.__x[4];
663
primitiveArray[REG_INDEX(R5)] = state.__x[5];
664
primitiveArray[REG_INDEX(R6)] = state.__x[6];
665
primitiveArray[REG_INDEX(R7)] = state.__x[7];
666
primitiveArray[REG_INDEX(R8)] = state.__x[8];
667
primitiveArray[REG_INDEX(R9)] = state.__x[9];
668
primitiveArray[REG_INDEX(R10)] = state.__x[10];
669
primitiveArray[REG_INDEX(R11)] = state.__x[11];
670
primitiveArray[REG_INDEX(R12)] = state.__x[12];
671
primitiveArray[REG_INDEX(R13)] = state.__x[13];
672
primitiveArray[REG_INDEX(R14)] = state.__x[14];
673
primitiveArray[REG_INDEX(R15)] = state.__x[15];
674
primitiveArray[REG_INDEX(R16)] = state.__x[16];
675
primitiveArray[REG_INDEX(R17)] = state.__x[17];
676
primitiveArray[REG_INDEX(R18)] = state.__x[18];
677
primitiveArray[REG_INDEX(R19)] = state.__x[19];
678
primitiveArray[REG_INDEX(R20)] = state.__x[20];
679
primitiveArray[REG_INDEX(R21)] = state.__x[21];
680
primitiveArray[REG_INDEX(R22)] = state.__x[22];
681
primitiveArray[REG_INDEX(R23)] = state.__x[23];
682
primitiveArray[REG_INDEX(R24)] = state.__x[24];
683
primitiveArray[REG_INDEX(R25)] = state.__x[25];
684
primitiveArray[REG_INDEX(R26)] = state.__x[26];
685
primitiveArray[REG_INDEX(R27)] = state.__x[27];
686
primitiveArray[REG_INDEX(R28)] = state.__x[28];
687
primitiveArray[REG_INDEX(FP)] = state.__fp;
688
primitiveArray[REG_INDEX(LR)] = state.__lr;
689
primitiveArray[REG_INDEX(SP)] = state.__sp;
690
primitiveArray[REG_INDEX(PC)] = state.__pc;
691
692
#else
693
#error UNSUPPORTED_ARCH
694
#endif /* amd64 */
695
696
print_debug("set registers\n");
697
(*env)->ReleaseLongArrayElements(env, registerArray, primitiveArray, 0);
698
699
return registerArray;
700
}
701
702
/*
703
* Class: sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal
704
* Method: translateTID0
705
* Signature: (I)I
706
*/
707
JNIEXPORT jint JNICALL
708
Java_sun_jvm_hotspot_debugger_macosx_MacOSXDebuggerLocal_translateTID0(
709
JNIEnv *env, jobject this_obj, jint tid)
710
{
711
print_debug("translateTID0 called on tid = 0x%x\n", (int)tid);
712
713
kern_return_t result;
714
thread_t foreign_tid, usable_tid;
715
mach_msg_type_name_t type;
716
717
foreign_tid = tid;
718
719
task_t gTask = getTask(env, this_obj);
720
result = mach_port_extract_right(gTask, foreign_tid,
721
MACH_MSG_TYPE_COPY_SEND,
722
&usable_tid, &type);
723
if (result != KERN_SUCCESS)
724
return -1;
725
726
print_debug("translateTID0: 0x%x -> 0x%x\n", foreign_tid, usable_tid);
727
728
return (jint) usable_tid;
729
}
730
731
732
static bool ptrace_continue(pid_t pid, int signal) {
733
// pass the signal to the process so we don't swallow it
734
int res;
735
if ((res = ptrace(PT_CONTINUE, pid, (caddr_t)1, signal)) < 0) {
736
print_error("attach: ptrace(PT_CONTINUE, %d) failed with %d\n", pid, res);
737
return false;
738
}
739
return true;
740
}
741
742
// waits until the ATTACH has stopped the process
743
// by signal SIGSTOP
744
static bool ptrace_waitpid(pid_t pid) {
745
int ret;
746
int status;
747
while (true) {
748
// Wait for debuggee to stop.
749
ret = waitpid(pid, &status, 0);
750
if (ret >= 0) {
751
if (WIFSTOPPED(status)) {
752
// Any signal will stop the thread, make sure it is SIGSTOP. Otherwise SIGSTOP
753
// will still be pending and delivered when the process is DETACHED and the process
754
// will go to sleep.
755
if (WSTOPSIG(status) == SIGSTOP) {
756
// Debuggee stopped by SIGSTOP.
757
return true;
758
}
759
if (!ptrace_continue(pid, WSTOPSIG(status))) {
760
print_error("attach: Failed to correctly attach to VM. VM might HANG! [PTRACE_CONT failed, stopped by %d]\n", WSTOPSIG(status));
761
return false;
762
}
763
} else {
764
print_error("attach: waitpid(): Child process exited/terminated (status = 0x%x)\n", status);
765
return false;
766
}
767
} else {
768
switch (errno) {
769
case EINTR:
770
continue;
771
break;
772
case ECHILD:
773
print_error("attach: waitpid() failed. Child process pid (%d) does not exist \n", pid);
774
break;
775
case EINVAL:
776
print_error("attach: waitpid() failed. Invalid options argument.\n");
777
break;
778
default:
779
print_error("attach: waitpid() failed. Unexpected error %d\n",errno);
780
break;
781
}
782
return false;
783
}
784
}
785
}
786
787
// attach to a process/thread specified by "pid"
788
static bool ptrace_attach(pid_t pid) {
789
int res;
790
if ((res = ptrace(PT_ATTACH, pid, 0, 0)) < 0) {
791
print_error("ptrace(PT_ATTACH, %d) failed with %d\n", pid, res);
792
return false;
793
} else {
794
return ptrace_waitpid(pid);
795
}
796
}
797
798
/*
799
* Class: sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal
800
* Method: attach0
801
* Signature: (I)V
802
*/
803
JNIEXPORT void JNICALL
804
Java_sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal_attach0__I(
805
JNIEnv *env, jobject this_obj, jint jpid)
806
{
807
print_debug("attach0 called for jpid=%d\n", (int)jpid);
808
809
JNF_COCOA_ENTER(env);
810
811
kern_return_t result;
812
task_t gTask = 0;
813
result = task_for_pid(mach_task_self(), jpid, &gTask);
814
if (result != KERN_SUCCESS) {
815
print_error("attach: task_for_pid(%d) failed: '%s' (%d)\n", (int)jpid, mach_error_string(result), result);
816
THROW_NEW_DEBUGGER_EXCEPTION("Can't attach to the process. Could be caused by an incorrect pid or lack of privileges.");
817
}
818
putTask(env, this_obj, gTask);
819
820
// use ptrace to stop the process
821
// on os x, ptrace only needs to be called on the process, not the individual threads
822
if (ptrace_attach(jpid) != true) {
823
mach_port_deallocate(mach_task_self(), gTask);
824
THROW_NEW_DEBUGGER_EXCEPTION("Can't attach to the process");
825
}
826
827
id symbolicator = nil;
828
id jrsSymbolicator = objc_lookUpClass("JRSSymbolicator");
829
if (jrsSymbolicator != nil) {
830
id (*dynamicCall)(id, SEL, pid_t) = (id (*)(id, SEL, pid_t))&objc_msgSend;
831
symbolicator = dynamicCall(jrsSymbolicator, @selector(symbolicatorForPid:), (pid_t)jpid);
832
}
833
if (symbolicator != nil) {
834
CFRetain(symbolicator); // pin symbolicator while in java heap
835
}
836
837
putSymbolicator(env, this_obj, symbolicator);
838
if (symbolicator == nil) {
839
THROW_NEW_DEBUGGER_EXCEPTION("Can't attach symbolicator to the process");
840
}
841
842
JNF_COCOA_EXIT(env);
843
}
844
845
/** For core file,
846
called from Java_sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal_attach0__Ljava_lang_String_2Ljava_lang_String_2 */
847
static void fillLoadObjects(JNIEnv* env, jobject this_obj, struct ps_prochandle* ph) {
848
int n = 0, i = 0;
849
850
// add load objects
851
n = get_num_libs(ph);
852
for (i = 0; i < n; i++) {
853
uintptr_t base;
854
const char* name;
855
jobject loadObject;
856
jobject loadObjectList;
857
jstring nameString;
858
859
base = get_lib_base(ph, i);
860
name = get_lib_name(ph, i);
861
nameString = (*env)->NewStringUTF(env, name);
862
CHECK_EXCEPTION;
863
loadObject = (*env)->CallObjectMethod(env, this_obj, createLoadObject_ID,
864
nameString, (jlong)0, (jlong)base);
865
CHECK_EXCEPTION;
866
loadObjectList = (*env)->GetObjectField(env, this_obj, loadObjectList_ID);
867
CHECK_EXCEPTION;
868
(*env)->CallBooleanMethod(env, loadObjectList, listAdd_ID, loadObject);
869
CHECK_EXCEPTION;
870
}
871
}
872
873
/*
874
* Class: sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal
875
* Method: attach0
876
* Signature: (Ljava/lang/String;Ljava/lang/String;)V
877
*/
878
JNIEXPORT void JNICALL
879
Java_sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal_attach0__Ljava_lang_String_2Ljava_lang_String_2(
880
JNIEnv *env, jobject this_obj, jstring execName, jstring coreName)
881
{
882
const char *execName_cstr;
883
const char *coreName_cstr;
884
jboolean isCopy;
885
struct ps_prochandle* ph;
886
887
execName_cstr = (*env)->GetStringUTFChars(env, execName, &isCopy);
888
CHECK_EXCEPTION;
889
coreName_cstr = (*env)->GetStringUTFChars(env, coreName, &isCopy);
890
CHECK_EXCEPTION;
891
892
print_debug("attach: %s %s\n", execName_cstr, coreName_cstr);
893
894
if ( (ph = Pgrab_core(execName_cstr, coreName_cstr)) == NULL) {
895
(*env)->ReleaseStringUTFChars(env, execName, execName_cstr);
896
(*env)->ReleaseStringUTFChars(env, coreName, coreName_cstr);
897
THROW_NEW_DEBUGGER_EXCEPTION("Can't attach to the core file");
898
}
899
(*env)->SetLongField(env, this_obj, p_ps_prochandle_ID, (jlong)(intptr_t)ph);
900
(*env)->ReleaseStringUTFChars(env, execName, execName_cstr);
901
(*env)->ReleaseStringUTFChars(env, coreName, coreName_cstr);
902
fillLoadObjects(env, this_obj, ph);
903
}
904
905
/*
906
* Class: sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal
907
* Method: detach0
908
* Signature: ()V
909
*/
910
JNIEXPORT void JNICALL
911
Java_sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal_detach0(
912
JNIEnv *env, jobject this_obj)
913
{
914
print_debug("detach0 called\n");
915
struct ps_prochandle* ph = get_proc_handle(env, this_obj);
916
if (ph != NULL && ph->core != NULL) {
917
Prelease(ph);
918
return;
919
}
920
JNF_COCOA_ENTER(env);
921
task_t gTask = getTask(env, this_obj);
922
923
// detach from the ptraced process causing it to resume execution
924
int pid;
925
kern_return_t k_res;
926
k_res = pid_for_task(gTask, &pid);
927
if (k_res != KERN_SUCCESS) {
928
print_error("detach: pid_for_task(%d) failed (%d)\n", pid, k_res);
929
}
930
else {
931
int res = ptrace(PT_DETACH, pid, 0, 0);
932
if (res < 0) {
933
print_error("detach: ptrace(PT_DETACH, %d) failed (%d)\n", pid, res);
934
}
935
}
936
937
mach_port_deallocate(mach_task_self(), gTask);
938
id symbolicator = getSymbolicator(env, this_obj);
939
if (symbolicator != nil) {
940
CFRelease(symbolicator);
941
}
942
JNF_COCOA_EXIT(env);
943
}
944
945