- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我在尝试使用 gdb 在 Alpine Linux 上调试 OpenJDK java 时遇到了困难 - 有人成功做到了吗?
当尝试在 gdb 中调试 java 时,例如,gdb java
和 r -version
,它立即失败并显示:
Thread 1 "java" recieved signal ?, Unknown signal.
__cp_end () at src/thread/x86_64/syscall_cp.s:29
我搜索了又搜索,但找不到任何有关在 Alpine 上调试 OpenJDK 的引用资料或解决方案。
在其他平台(macOS Sierra、MinGW)上看到的处理相同 gdb 错误的其他线程表明 recieved signal ?, Unknown signal
可能由各种原因引起,包括 gdb bug , uncaught exception , stack overflow ,以及其他应用程序错误。
在 gdb 之外,java 可以正常工作,gdb 可以很好地调试一个简单的 C++ 程序。我正在运行 Alpine V3.8。
我尝试过的事情:
8.0.1-r6
、8.0.1-r3
、7.12.1-r1
)。1.8.0_171
、1.7.0_181
)。/bin/ash
、/bin/bash
)运行,使用和不使用 sudo
。.gdbinit
中禁用信号停止:handle SIGSEGV nostop noprint pass
,同样适用于 SIGPIPE
、SIGHUP
,SIGFPE
,SIG34
。set startup-with-shell off
添加到 .gdbinit
。感谢您的帮助!
编辑:
这是抛出未知信号的完整堆栈,它导致 JVMInit 失败:
(gdb) r -version
Starting program: /usr/lib/jvm/java-1.8-openjdk/bin/java -version
process 16214 is executing new program: /usr/lib/jvm/java-1.8-openjdk/bin/java
[New LWP 16219]
Thread 1 "java" received signal ?, Unknown signal.
__cp_end () at src/thread/x86_64/syscall_cp.s:29
29 src/thread/x86_64/syscall_cp.s: No such file or directory.
(gdb) info threads
Id Target Id Frame
* 1 LWP 16214 "java" __cp_end () at src/thread/x86_64/syscall_cp.s:29
2 LWP 16219 "java" __synccall (func=func@entry=0x7ffff7da2662 <do_setrlimit>, ctx=ctx@entry=0x7ffff7ff4720)
at src/thread/synccall.c:143
(gdb) where
#0 __cp_end () at src/thread/x86_64/syscall_cp.s:29
#1 0x00007ffff7dbed2d in __syscall_cp_c (nr=202, u=<optimized out>, v=<optimized out>, w=<optimized out>, x=<optimized out>,
y=<optimized out>, z=0) at src/thread/pthread_cancel.c:35
#2 0x00007ffff7dbe350 in __timedwait_cp (addr=addr@entry=0x7ffff7ff4b20, val=16219, clk=clk@entry=0, at=at@entry=0x0, priv=priv@entry=0)
at src/thread/__timedwait.c:31
#3 0x00007ffff7dbfdc4 in __pthread_timedjoin_np (t=0x7ffff7ff4ae8, res=res@entry=0x7fffffffa348, at=at@entry=0x0)
at src/thread/pthread_join.c:16
#4 0x00007ffff7dbfe02 in __pthread_join (t=<optimized out>, res=res@entry=0x7fffffffa348) at src/thread/pthread_join.c:27
#5 0x00007ffff7b6695e in ContinueInNewThread0 (continuation=continuation@entry=0x7ffff7b61a60 <JavaMain>, stack_size=1048576,
args=args@entry=0x7fffffffa3e0)
at /home/buildozer/aports/community/openjdk8/src/icedtea-3.8.0/openjdk/jdk/src/solaris/bin/java_md_solinux.c:1046
#6 0x00007ffff7b634a4 in ContinueInNewThread (ifn=ifn@entry=0x7fffffffa4f0, threadStackSize=<optimized out>, argc=1,
argv=<optimized out>, mode=mode@entry=841574793, what=what@entry=0x0, ret=0)
at /home/buildozer/aports/community/openjdk8/src/icedtea-3.8.0/openjdk/jdk/src/share/bin/java.c:2024
#7 0x00007ffff7b66a08 in JVMInit (ifn=ifn@entry=0x7fffffffa4f0, threadStackSize=<optimized out>, argc=<optimized out>,
argv=<optimized out>, mode=841574793, mode@entry=0, what=what@entry=0x0, ret=<optimized out>)
at /home/buildozer/aports/community/openjdk8/src/icedtea-3.8.0/openjdk/jdk/src/solaris/bin/java_md_solinux.c:1093
#8 0x00007ffff7b63e30 in JLI_Launch (argc=<optimized out>, argv=<optimized out>, jargc=<optimized out>, jargv=<optimized out>,
appclassc=1, appclassv=0x0, fullversion=0x555555554843 "1.8.0_171-b11", dotversion=0x55555555483f "1.8", pname=0x55555555483a "java",
lname=0x555555554832 "openjdk", javaargs=0 '\000', cpwildcard=1 '\001', javaw=0 '\000', ergo=0)
at /home/buildozer/aports/community/openjdk8/src/icedtea-3.8.0/openjdk/jdk/src/share/bin/java.c:304
#9 0x0000555555554691 in main (argc=<optimized out>, argv=<optimized out>)
at /home/buildozer/aports/community/openjdk8/src/icedtea-3.8.0/openjdk/jdk/src/share/bin/main.c:125
(gdb)
匹配此堆栈跟踪的 musl 源文件:
OpenJDK 源代码:
JVMInit
尝试通过调用 ContinueInNewThread
创建 JavaMain
native 线程,后者调用 ContinueInNewThread0(JavaMain, threadStackSize, (void *)&args)
,然后它就爆炸了。
最佳答案
TL;DR:问题是 GDB 缺乏对内部 musl 信号的支持,在此 gdb ticket 中报告.
此处提供了快速修补的 GDB:
https://github.com/shaharv/alpine-gdb-builds/releases/tag/v0.1
补丁提交: shaharv/binutils-gdb@0ca9c66 .
通过补丁,信号正确识别为 SIGSYNCCALL。
然后,可以使用 handle SIGSYNCCALL nostop noprint pass
屏蔽它。
幸运的是,我能够想出一个解决方法!
调试 Alpine OpenJDK java 时的 gdb 崩溃可以通过以下方式解决:
中断 os::init_2
设置 MaxFDLimit=0
我已经使用 OpenJDK 8 和 11 早期访问测试了变通方法,因此它也可能适用于 OpenJDK 9 和 10。
不幸的是,此解决方法的范围非常有限:
openjdk8-dbg
调试符号包。总结:
当在 gdb 中调用 setrlimit
函数时发生崩溃。 musl 的 setrlimit
实现使用 SIGSYNCCALL
向线程发出信号,gdb 不支持它,并导致 Unknown signal
错误。为避免该错误,通过关闭MaxFDLimit
全局变量来禁用JavaMain
的相关初始化代码。
完整解释:
在 JVM 初始化期间,创建了一个 JavaMain
native 线程,并创建了 VM。在 VM 创建过程中,存在特定于操作系统的初始化,其中调用了 setrlimit
。这是堆栈跟踪的相关部分:
#0 __synccall (func=func@entry=0x7ffff7da2662 <do_setrlimit>, ctx=ctx@entry=0x7ffff7ff4720) at src/thread/synccall.c:48
#1 0x00007ffff7da26a1 in setrlimit (resource=resource@entry=7, rlim=rlim@entry=0x7ffff7ff4750) at src/misc/setrlimit.c:42
#2 0x00007ffff73bd1fe in os::init_2 ()
at /home/buildozer/aports/community/openjdk8/src/icedtea-3.8.0/openjdk/hotspot/src/os/linux/vm/os_linux.cpp:5096
#3 0x00007ffff746177d in Threads::create_vm (args=0x7ffff7ff4a20, canTryAgain=canTryAgain@entry=0x7ffff7ff4987)
at /home/buildozer/aports/community/openjdk8/src/icedtea-3.8.0/openjdk/hotspot/src/share/vm/runtime/thread.cpp:3361
#4 0x00007ffff729cd48 in JNI_CreateJavaVM (vm=0x7ffff7ff4a10, penv=0x7ffff7ff4a18, args=<optimized out>)
at /home/buildozer/aports/community/openjdk8/src/icedtea-3.8.0/openjdk/hotspot/src/share/vm/prims/jni.cpp:5221
#5 0x00007ffff7b61b0b in InitializeJVM (ifn=<synthetic pointer>, penv=0x7ffff7ff4a18, pvm=0x7ffff7ff4a10)
at /home/buildozer/aports/community/openjdk8/src/icedtea-3.8.0/openjdk/jdk/src/share/bin/java.c:1231
#6 JavaMain (_args=<optimized out>)
罪魁祸首是 setrlimit
函数调用。musl 的 setrlimit
实现是 AS-Safe ,这意味着,从异步信号处理程序调用它是安全的。同步部分正在通过调用 __synccall
( setrlimit.c ) 处理:
int setrlimit(int resource, const struct rlimit *rlim)
{
struct ctx c = { .res = resource, .rlim = rlim, .err = -1 };
__synccall(do_setrlimit, &c);
if (c.err) {
if (c.err>0) errno = c.err;
return -1;
}
return 0;
}
__synccall
( synccall.c ) 阻塞所有信号,然后迭代进程的所有线程并向它们发送 SIGSYNCCALL
信号(并且仅当所有线程都确认该信号时, do_setrlimit
被执行):
r = -__syscall(SYS_tgkill, pid, tid, SIGSYNCCALL);
但是,SIGSYNCCALL
信号是 musl 的内部信号,不由 gdb 处理。 gdb 显式处理所有信号类型,但 SIGSYNCCALL
不包含在处理的信号中(参见 gdb 的 signals.c )。因此,当发出信号时,gdb 将终止并出现Unknown signal
错误。
解决方法:
解决方法是在 OpenJDK 中即时禁用对 setrlimit
的调用。相关代码在os::init_2
函数中(os_linux.cpp):
if (MaxFDLimit) {
// set the number of file descriptors to max. print out error
// if getrlimit/setrlimit fails but continue regardless.
struct rlimit nbr_files;
int status = getrlimit(RLIMIT_NOFILE, &nbr_files);
if (status != 0) {
if (PrintMiscellaneous && (Verbose || WizardMode))
perror("os::init_2 getrlimit failed");
} else {
nbr_files.rlim_cur = nbr_files.rlim_max;
status = setrlimit(RLIMIT_NOFILE, &nbr_files);
if (status != 0) {
if (PrintMiscellaneous && (Verbose || WizardMode))
perror("os::init_2 setrlimit failed");
}
}
}
通过将MaxFDLimit
设置为0,上面的代码不会执行,VM初始化可以正常进行。有一个用于切换此变量的命令行选项 -XX:-MaxFDLimit
,但它在 Solaris only 上可用。 ,所以我们别无选择,只能在 gdb 中手动关闭此变量。
MaxFDLimit
背后的原因是历史性的,旨在增加默认 FD 限制 (256) 非常低的古代系统的文件描述符默认限制,如 JDK-8010126 中所述。 . Alpine V3.8 的默认限制为 1024,因此禁用此代码应该是安全的 - 如果需要,可以使用 ulimit
增加限制,而不是通过 JVM 本身。
关于java - gdb 在 Alpine Linux 上调试 OpenJDK java 失败,出现 "Thread recieved signal ?, Unknown signal",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52119176/
我在使用以下代码时遇到问题: function http_file_exists($url){ $f=fopen($url,"r"); if($f){ fclose($f); retu
我已经通过 Git 部署到 Azure 几个月了,没有出现重大问题,但现在我似乎遇到了一个无法克服的错误。 我创建了一个新的 Azure 网站,为正在开发的项目创建单独的预览链接。我在新站点上设置了
我已经通过flutter创建了一个App并完成了它,我想在flutter文档中阅读时进行部署。 我收到此错误: FAILURE: Build failed with an exception. * W
我在Windows 10中使用一些简单的Powershell代码遇到了这个奇怪的问题,我认为这可能是我做错了,但我不是Powershell的天才。 我有这个: $ix = [System.Net.Dn
我正在尝试使用 RapidJSON 解析从服务器接收到的数据。以下是收到的确切字符串: [ { "Node": "9478149a08f9", "Address": "172.17
我尝试为 ios 编译 OpenCV。我总是收到这些错误。我用不同版本的opencv试了一下,结果都是一样的。 我运行这个:python 平台/ios/build_framework.py ios_o
我在一台机器上做基本的发布/订阅,我的客户端是 StackExchange-Redis 的 C# 客户端,我在同一台机器上运行基于 Windows 的 Redis 服务器(服务器版本 2.8.4) 当
我有这段代码,但无法执行,请帮我解决这个问题 连接 connect_error) { die ("connection failed: " . $terhubung->connect_erro
我在 tomcat 上运行并由 maven 编译的 Web 应用程序给出了以下警告和错误。我可以在本地存储库中看到所有 JAR,但有人可以帮忙吗。 WARNING: Failed to scan JA
我正在 Windows 8 上使用 Android Studio 开发一个 android 应用程序,我正在使用一些 native 代码。突然间我无法编译我的 C 文件。当我运行 ndk-build
下面的代码对类和结构的成员进行序列化和反序列化。序列化工作正常,但我在尝试使用 oarch >> BOOST_SERIALIZATION_NVP(outObj); 反序列化时遇到了以下错误; 代码中是
如果我运行此命令“rspec ./spec/requests/api/v1/password_reset_request_spec.rb”,此文件中的所有测试都会通过。 但是,当我运行“rspec”时
我在尝试执行测试以使用 Protractor 上传文件时出错,我的代码是这个 it('it should be possible to upload a file', function() {
System.loadLibrary("nativefaceswap"); 当我运行我的应用程序时,我在 Android Studio 中发现了此类错误。在logcat中显示: java.lang.U
我希望有人能帮助我!使用任何方法或命令行的任何 SSL/HTTPS 调用均无效。 我在 Windows 10 中使用 Ubuntu Server 18.04 作为子系统。我的问题是昨天才开始出现的,因
通过删除这两个值将日期字段从 null=True 和 Blank=True 更改为 required 时,使用 db.alter 命令时遇到问题。 当以下行被注释掉时,迁移运行不会出现问题。
我第一次使用 Heroku 尝试创建应用程序(使用 SendGrid 的 Inbound Parse Webhook"和 Twilio SMS 通过电子邮件发送和接收 SMS 消息)。通过 Virtu
我正在将我的 swift 项目更新到 Xcode 7 上的 Swift 2.0。xcode 在构建项目时报告了以下错误: 命令/Applications/Xcode.app/Contents/Deve
在我的代码中,SSL 库函数 SSL_library_init() 没有按预期返回 1。我如何才能看到它返回了什么错误? 我在 SSL_library_init() 之后调用了 SSL_load_er
我正在尝试运行在以下链接中找到的答案: Asynchronously Load the Contents of a Div 但是当我这样做时,我会遇到我不太理解的错误。 我的代码: $(documen
我是一名优秀的程序员,十分优秀!