JDK-8186780: clang fastdebug assertion failure in os_linux_x86:os::verify_stack_alignment()
Martin Buchholz
martinrb at google.com
Mon Mar 11 19:37:56 UTC 2019
Let's try to get the last straggler clang linux failure in OpenJDK resolved.
https://bugs.openjdk.java.net/browse/JDK-8186780
This is not a RFR because we need to get consensus on strategy.
I'm hoping a hotspot developer takes ownership.
---
We should stop trying to deduce stack alignment using non-portable
assembly. Instead, we should use more portable constructs that ensure that
stack objects are actually correctly aligned. We can get rid
of os::verify_stack_alignment() and replace it with a single portable
function. Here's a proof of concept:
(I was surprised to see no pre-existing use of alignof)
(we will soon be using C++14, where presence of alignof is guaranteed)
// Are C++11 alignof, alignas and std::max_align_t available?
// This could instead be implemented via a configure check.
// The __cplusplus macro is not reliable on Microsoft Visual C++:
//
https://devblogs.microsoft.com/cppblog/msvc-now-correctly-reports-__cplusplus/
#if __cplusplus >= 201103L || (defined(_MSC_VER) && _MSC_VER >= 1900)
#define HAVE_ALIGNOF 1
#endif
#ifndef PRODUCT
void os::verify_stack_alignment() {
#ifdef HAVE_ALIGNOF
STATIC_ASSERT((StackAlignmentInBytes & (alignof(std::max_align_t) - 1))
== 0);
{
char jiggle_alignment;
std::max_align_t object;
assert(((uintptr_t)&object & (alignof(std::max_align_t) - 1)) == 0,
"incorrect stack alignment");
}
{
char jiggle_alignment;
alignas(StackAlignmentInBytes) char object;
assert(((uintptr_t)&object & (StackAlignmentInBytes - 1)) == 0,
"incorrect stack alignment");
}
#endif
}
#endif
---
If we really really want to check the alignment of the stack frame itself
(why?!) then we should not be checking the stack pointer (esp) but instead
using the gcc/clang API __builtin_frame_address(0)
+// TODO: this should be a configure check
+#if defined(__clang__) || defined(__GNUC__)
+#define HAVE_BUILTIN_FRAME_ADDRESS 1
+#endif
+
#ifndef PRODUCT
void os::verify_stack_alignment() {
#ifdef AMD64
- assert(((intptr_t)os::current_stack_pointer() &
(StackAlignmentInBytes-1)) == 0, "incorrect stack alignment");
+#if defined(HAVE_BUILTIN_FRAME_ADDRESS)
+ void *address = __builtin_frame_address(0);
+#else
+ void *address = os::current_stack_pointer();
+#endif
+ assert(((intptr_t)address & (StackAlignmentInBytes-1)) == 0, "incorrect
stack alignment");
#endif
}
#endif
---
Many other solutions exist - we can simply delete the check, or we can
prevent inlining of verify_stack_alignment using NOINLINE.
More information about the hotspot-runtime-dev
mailing list