Skip to content

Commit e7c7469

Browse files
committedNov 20, 2020
8246378: [Windows] assert on MethodHandle logging code
Reviewed-by: iklam, vlivanov
1 parent 98a5d5a commit e7c7469

11 files changed

+60
-144
lines changed
 

‎src/hotspot/cpu/aarch64/stubRoutines_aarch64.cpp

-1
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,6 @@
3333
// Implementation of the platform-specific part of StubRoutines - for
3434
// a description of how to extend it, see the stubRoutines.hpp file.
3535

36-
address StubRoutines::aarch64::_get_previous_fp_entry = NULL;
3736
address StubRoutines::aarch64::_get_previous_sp_entry = NULL;
3837

3938
address StubRoutines::aarch64::_f2i_fixup = NULL;

‎src/hotspot/cpu/aarch64/stubRoutines_aarch64.hpp

-6
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,6 @@ class aarch64 {
4343
friend class StubGenerator;
4444

4545
private:
46-
static address _get_previous_fp_entry;
4746
static address _get_previous_sp_entry;
4847

4948
static address _f2i_fixup;
@@ -77,11 +76,6 @@ class aarch64 {
7776

7877
public:
7978

80-
static address get_previous_fp_entry()
81-
{
82-
return _get_previous_fp_entry;
83-
}
84-
8579
static address get_previous_sp_entry()
8680
{
8781
return _get_previous_sp_entry;

‎src/hotspot/cpu/x86/methodHandles_x86.cpp

+35-34
Original file line numberDiff line numberDiff line change
@@ -528,6 +528,12 @@ void trace_method_handle_stub(const char* adaptername,
528528
}
529529
tty->cr();
530530

531+
// Note: We want to allow trace_method_handle from any call site.
532+
// While trace_method_handle creates a frame, it may be entered
533+
// without a PC on the stack top (e.g. not just after a call).
534+
// Walking that frame could lead to failures due to that invalid PC.
535+
// => carefully detect that frame when doing the stack walking
536+
531537
{
532538
// dumping last frame with frame::describe
533539

@@ -536,43 +542,38 @@ void trace_method_handle_stub(const char* adaptername,
536542
PRESERVE_EXCEPTION_MARK; // may not be needed but safer and inexpensive here
537543
FrameValues values;
538544

539-
// Note: We want to allow trace_method_handle from any call site.
540-
// While trace_method_handle creates a frame, it may be entered
541-
// without a PC on the stack top (e.g. not just after a call).
542-
// Walking that frame could lead to failures due to that invalid PC.
543-
// => carefully detect that frame when doing the stack walking
544-
545545
// Current C frame
546546
frame cur_frame = os::current_frame();
547547

548-
// Robust search of trace_calling_frame (independant of inlining).
549-
// Assumes saved_regs comes from a pusha in the trace_calling_frame.
550-
assert(cur_frame.sp() < saved_regs, "registers not saved on stack ?");
551-
frame trace_calling_frame = os::get_sender_for_C_frame(&cur_frame);
552-
while (trace_calling_frame.fp() < saved_regs) {
553-
trace_calling_frame = os::get_sender_for_C_frame(&trace_calling_frame);
554-
}
555-
556-
// safely create a frame and call frame::describe
557-
intptr_t *dump_sp = trace_calling_frame.sender_sp();
558-
intptr_t *dump_fp = trace_calling_frame.link();
559-
560-
bool walkable = has_mh; // whether the traced frame shoud be walkable
561-
562-
if (walkable) {
563-
// The previous definition of walkable may have to be refined
564-
// if new call sites cause the next frame constructor to start
565-
// failing. Alternatively, frame constructors could be
566-
// modified to support the current or future non walkable
567-
// frames (but this is more intrusive and is not considered as
568-
// part of this RFE, which will instead use a simpler output).
569-
frame dump_frame = frame(dump_sp, dump_fp);
570-
dump_frame.describe(values, 1);
571-
} else {
572-
// Stack may not be walkable (invalid PC above FP):
573-
// Add descriptions without building a Java frame to avoid issues
574-
values.describe(-1, dump_fp, "fp for #1 <not parsed, cannot trust pc>");
575-
values.describe(-1, dump_sp, "sp for #1");
548+
if (cur_frame.fp() != 0) { // not walkable
549+
550+
// Robust search of trace_calling_frame (independent of inlining).
551+
// Assumes saved_regs comes from a pusha in the trace_calling_frame.
552+
assert(cur_frame.sp() < saved_regs, "registers not saved on stack ?");
553+
frame trace_calling_frame = os::get_sender_for_C_frame(&cur_frame);
554+
while (trace_calling_frame.fp() < saved_regs) {
555+
trace_calling_frame = os::get_sender_for_C_frame(&trace_calling_frame);
556+
}
557+
558+
// safely create a frame and call frame::describe
559+
intptr_t *dump_sp = trace_calling_frame.sender_sp();
560+
intptr_t *dump_fp = trace_calling_frame.link();
561+
562+
if (has_mh) {
563+
// The previous definition of walkable may have to be refined
564+
// if new call sites cause the next frame constructor to start
565+
// failing. Alternatively, frame constructors could be
566+
// modified to support the current or future non walkable
567+
// frames (but this is more intrusive and is not considered as
568+
// part of this RFE, which will instead use a simpler output).
569+
frame dump_frame = frame(dump_sp, dump_fp);
570+
dump_frame.describe(values, 1);
571+
} else {
572+
// Stack may not be walkable (invalid PC above FP):
573+
// Add descriptions without building a Java frame to avoid issues
574+
values.describe(-1, dump_fp, "fp for #1 <not parsed, cannot trust pc>");
575+
values.describe(-1, dump_sp, "sp for #1");
576+
}
576577
}
577578
values.describe(-1, entry_sp, "raw top of stack");
578579

‎src/hotspot/cpu/x86/stubGenerator_x86_64.cpp

-21
Original file line numberDiff line numberDiff line change
@@ -566,26 +566,6 @@ class StubGenerator: public StubCodeGenerator {
566566
return start;
567567
}
568568

569-
// Support for intptr_t get_previous_fp()
570-
//
571-
// This routine is used to find the previous frame pointer for the
572-
// caller (current_frame_guess). This is used as part of debugging
573-
// ps() is seemingly lost trying to find frames.
574-
// This code assumes that caller current_frame_guess) has a frame.
575-
address generate_get_previous_fp() {
576-
StubCodeMark mark(this, "StubRoutines", "get_previous_fp");
577-
const Address old_fp(rbp, 0);
578-
const Address older_fp(rax, 0);
579-
address start = __ pc();
580-
581-
__ enter();
582-
__ movptr(rax, old_fp); // callers fp
583-
__ movptr(rax, older_fp); // the frame for ps()
584-
__ pop(rbp);
585-
__ ret(0);
586-
587-
return start;
588-
}
589569

590570
// Support for intptr_t get_previous_sp()
591571
//
@@ -6717,7 +6697,6 @@ address generate_avx_ghash_processBlocks() {
67176697
StubRoutines::_fence_entry = generate_orderaccess_fence();
67186698

67196699
// platform dependent
6720-
StubRoutines::x86::_get_previous_fp_entry = generate_get_previous_fp();
67216700
StubRoutines::x86::_get_previous_sp_entry = generate_get_previous_sp();
67226701

67236702
StubRoutines::x86::_verify_mxcsr_entry = generate_verify_mxcsr();

‎src/hotspot/cpu/x86/stubRoutines_x86.hpp

+1-6
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2013, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2013, 2020, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -42,7 +42,6 @@ class x86 {
4242

4343
#ifdef _LP64
4444
private:
45-
static address _get_previous_fp_entry;
4645
static address _get_previous_sp_entry;
4746

4847
static address _f2i_fixup;
@@ -57,10 +56,6 @@ class x86 {
5756

5857
public:
5958

60-
static address get_previous_fp_entry() {
61-
return _get_previous_fp_entry;
62-
}
63-
6459
static address get_previous_sp_entry() {
6560
return _get_previous_sp_entry;
6661
}

‎src/hotspot/cpu/x86/stubRoutines_x86_64.cpp

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2003, 2013, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2003, 2020, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -31,7 +31,6 @@
3131
// Implementation of the platform-specific part of StubRoutines - for
3232
// a description of how to extend it, see the stubRoutines.hpp file.
3333

34-
address StubRoutines::x86::_get_previous_fp_entry = NULL;
3534
address StubRoutines::x86::_get_previous_sp_entry = NULL;
3635

3736
address StubRoutines::x86::_f2i_fixup = NULL;

‎src/hotspot/os_cpu/windows_aarch64/os_windows_aarch64.cpp

+4-22
Original file line numberDiff line numberDiff line change
@@ -142,32 +142,14 @@ bool os::win32::get_frame_at_stack_banging_point(JavaThread* thread,
142142
return true;
143143
}
144144

145-
// By default, gcc always saves frame pointer rfp on this stack. This
146-
// may get turned off by -fomit-frame-pointer.
147145
frame os::get_sender_for_C_frame(frame* fr) {
148-
return frame(fr->link(), fr->link(), fr->sender_pc());
146+
ShouldNotReachHere();
147+
return frame();
149148
}
150149

151150
frame os::current_frame() {
152-
typedef intptr_t* get_fp_func ();
153-
get_fp_func* func = CAST_TO_FN_PTR(get_fp_func*,
154-
StubRoutines::aarch64::get_previous_fp_entry());
155-
if (func == NULL) return frame();
156-
intptr_t* fp = (*func)();
157-
if (fp == NULL) {
158-
return frame();
159-
}
160-
161-
frame myframe((intptr_t*)os::current_stack_pointer(),
162-
(intptr_t*)fp,
163-
CAST_FROM_FN_PTR(address, os::current_frame));
164-
if (os::is_first_C_frame(&myframe)) {
165-
166-
// stack is not walkable
167-
return frame();
168-
} else {
169-
return os::get_sender_for_C_frame(&myframe);
170-
}
151+
return frame(); // cannot walk Windows frames this way. See os::get_native_stack
152+
// and os::platform_print_native_stack
171153
}
172154

173155
////////////////////////////////////////////////////////////////////////////////

‎src/hotspot/os_cpu/windows_x86/os_windows_x86.cpp

+9-47
Original file line numberDiff line numberDiff line change
@@ -323,13 +323,6 @@ frame os::fetch_frame_from_context(const void* ucVoid) {
323323
return frame(sp, fp, epc);
324324
}
325325

326-
// VC++ does not save frame pointer on stack in optimized build. It
327-
// can be turned off by /Oy-. If we really want to walk C frames,
328-
// we can use the StackWalk() API.
329-
frame os::get_sender_for_C_frame(frame* fr) {
330-
return frame(fr->sender_sp(), fr->link(), fr->sender_pc());
331-
}
332-
333326
#ifndef AMD64
334327
// Ignore "C4172: returning address of local variable or temporary" on 32bit
335328
PRAGMA_DIAG_PUSH
@@ -390,49 +383,18 @@ bool os::win32::get_frame_at_stack_banging_point(JavaThread* thread,
390383
return true;
391384
}
392385

393-
#ifndef AMD64
394-
intptr_t* _get_previous_fp() {
395-
intptr_t **frameptr;
396-
__asm {
397-
mov frameptr, ebp
398-
};
399-
// ebp (frameptr) is for this frame (_get_previous_fp). We want the ebp for the
400-
// caller of os::current_frame*(), so go up two frames. However, for
401-
// optimized builds, _get_previous_fp() will be inlined, so only go
402-
// up 1 frame in that case.
403-
#ifdef _NMT_NOINLINE_
404-
return **(intptr_t***)frameptr;
405-
#else
406-
return *frameptr;
407-
#endif
386+
387+
// VC++ does not save frame pointer on stack in optimized build. It
388+
// can be turned off by /Oy-. If we really want to walk C frames,
389+
// we can use the StackWalk() API.
390+
frame os::get_sender_for_C_frame(frame* fr) {
391+
ShouldNotReachHere();
392+
return frame();
408393
}
409-
#endif // !AMD64
410394

411395
frame os::current_frame() {
412-
413-
#ifdef AMD64
414-
// apparently _asm not supported on windows amd64
415-
typedef intptr_t* get_fp_func ();
416-
get_fp_func* func = CAST_TO_FN_PTR(get_fp_func*,
417-
StubRoutines::x86::get_previous_fp_entry());
418-
if (func == NULL) return frame();
419-
intptr_t* fp = (*func)();
420-
if (fp == NULL) {
421-
return frame();
422-
}
423-
#else
424-
intptr_t* fp = _get_previous_fp();
425-
#endif // AMD64
426-
427-
frame myframe((intptr_t*)os::current_stack_pointer(),
428-
(intptr_t*)fp,
429-
CAST_FROM_FN_PTR(address, os::current_frame));
430-
if (os::is_first_C_frame(&myframe)) {
431-
// stack is not walkable
432-
return frame();
433-
} else {
434-
return os::get_sender_for_C_frame(&myframe);
435-
}
396+
return frame(); // cannot walk Windows frames this way. See os::get_native_stack
397+
// and os::platform_print_native_stack
436398
}
437399

438400
void os::print_context(outputStream *st, const void *context) {

‎src/hotspot/share/runtime/os.cpp

+6-2
Original file line numberDiff line numberDiff line change
@@ -1170,9 +1170,13 @@ void os::print_location(outputStream* st, intptr_t x, bool verbose) {
11701170
}
11711171

11721172
// Looks like all platforms can use the same function to check if C
1173-
// stack is walkable beyond current frame. The check for fp() is not
1174-
// necessary on Sparc, but it's harmless.
1173+
// stack is walkable beyond current frame.
11751174
bool os::is_first_C_frame(frame* fr) {
1175+
1176+
#ifdef _WINDOWS
1177+
return true; // native stack isn't walkable on windows this way.
1178+
#endif
1179+
11761180
// Load up sp, fp, sender sp and sender fp, check for reasonable values.
11771181
// Check usp first, because if that's bad the other accessors may fault
11781182
// on some architectures. Ditto ufp second, etc.

‎test/hotspot/jtreg/runtime/logging/RedefineClasses.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2014, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2014, 2020, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -31,7 +31,7 @@
3131
* java.instrument
3232
* @requires vm.jvmti
3333
* @run main RedefineClassHelper
34-
* @run main/othervm -Xmx256m -XX:MaxMetaspaceSize=64m -javaagent:redefineagent.jar -Xlog:all=trace:file=all.log RedefineClasses
34+
* @run main/othervm -Xmx256m -XX:MaxMetaspaceSize=64m -javaagent:redefineagent.jar -XX:+Verbose -Xlog:all=trace:file=all.log RedefineClasses
3535
*/
3636

3737
// package access top-level class to avoid problem with RedefineClassHelper

‎test/hotspot/jtreg/runtime/logging/TestMethodHandlesVerbose.java

+2-1
Original file line numberDiff line numberDiff line change
@@ -26,10 +26,11 @@
2626
* @bug 8244946
2727
* @summary Run simple test with -XX:+Verbose and -Xlog:methodhandles.
2828
*
29-
* @run main/othervm -XX:+IgnoreUnrecognizedVMOptions -XX:+Verbose -Xlog:methodhandles TestMethodHandlesVerbose
29+
* @run main/othervm -XX:+IgnoreUnrecognizedVMOptions -XX:+Verbose -Xlog:methodhandles TestMethodHandlesVerbose a b
3030
*/
3131

3232
public class TestMethodHandlesVerbose {
3333
public static void main(String[] args) {
34+
System.out.println(args[0] + args[1]);
3435
}
3536
}

0 commit comments

Comments
 (0)
Please sign in to comment.