- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
正如标题所说 - ucontext_t
中的 rip
不指向引发 SIGTRAP
的 int3
。相反,它指向下一条指令。
这偏离了我(幼稚)的预期,即每条错误指令都将在从信号处理程序返回时重试(如果上下文未更改并且进程在处理程序中时未明确终止)。
另一方面,当获取 SIGILL
时 - 上下文指向错误指令。同样在 ARM 和 Aarch64 上 - SIGTRAP
上下文也指向相关的 bkpt #0
/bpt #0
。
测试程序:
/* sigtest.c */
#define _GNU_SOURCE
#include <stdio.h>
#include <signal.h>
#include <ucontext.h>
extern void do_int3(void);
extern void do_ud2(void);
void sighandler(int signo, siginfo_t* info, void* context) {
struct ucontext_t* uctx = context;
/* Yes, printf() here is bad. I promise to never ever do this in real programs */
printf("Got signal %d:\n"
"\tsi_addr: %p\n"
"\tcontext RIP: %p\n",
signo,
info->si_addr,
(void*)uctx->uc_mcontext.gregs[REG_RIP]);
}
int main(int argc, char** argv) {
struct sigaction sa = {};
sa.sa_flags = SA_SIGINFO | SA_ONESHOT;
sa.sa_sigaction = sighandler;
sigemptyset(&sa.sa_mask);
sigaction(SIGTRAP, &sa, NULL);
sigaction(SIGILL, &sa, NULL);
void (*fn)(void) = 0;
if (argv[1][0] == 't') {
fn = do_int3;
}
if (argv[1][0] == 'u') {
fn = do_ud2;
}
printf("call function at %p\n", fn);
fn();
printf("call returned\n");
}
; do_int3.S
.text
.globl do_int3
.type do_int3, @function
do_int3:
int3
ret
.size do_int3, .-do_int3
; do_ud2.S
.text
.globl do_ud2
.type do_ud2, @function
do_ud2:
ud2
ret
.size do_ud2, .-do_ud2
编译运行:
$ cc sigtest.c do_int3.S do_ud2.S
$ ./a.out t
call function at 0x5620b87c6908
Got signal 5:
si_addr: (nil)
context RIP: 0x5620b87c6909
call returned
$ ./a.out u
call function at 0x557d1bc1590a
Got signal 4:
si_addr: 0x557d1bc1590a
context RIP: 0x557d1bc1590a
Illegal instruction (core dumped)
很容易注意到,对于 SIGTRAP
,rip
值指向下一条指令,而不是指向 int3
。
为什么在SIGTRAP
上下文中调整指令指针不重试相关的int3
?
最佳答案
Why is instruction pointer adjusted in SIGTRAP context to not retry related int3?
它不是由内核调整的;硬件推送的异常返回地址是int
指令之后的地址,包括int3
。
请记住 int3
与 the normal case of int n
仅略有不同例如 int 0x80
。 int
被设计为类似于系统调用或远程调用,因此(异常)返回地址是int
之后的地址。否则,除非内核在 iret
之前编辑了异常返回信息,否则 int 0x80
系统调用将永远重新运行。
那么为什么 Linux 的 int3
/int 3
处理程序不将保存的 RIP 减 1?一方面,如果调试人员愿意,他们可以在软件中做到这一点,并且保持内核简单更好。 (对于确实想知道硬件推送的地址的软件来说,维护、效率和更少的分解。)
另一方面,1 字节的固定偏移量并不总是正确的:2 字节 CD 03 int 3
引发与 1 字节 CC int3
相同的异常.即使您想尝试,x86 机器代码也不会唯一地向后解码,因此混淆后的机器代码可能会给出一个地址,该地址并不是所执行指令的实际开始位置。 (尽管如果从那里解码,它将作为 int 3
或 int3
运行)。例如如果向后看,2 字节 rep int3
与 add al, 0xf3
/int3
无法区分。
像 GDB 这样插入 int3
的软件会知道它插入了什么,并且需要知道目标机器的详细信息,因此可以处理偏移量。
关于c - x86, amd64 : Why SIGTRAP' ucontext instruction pointer does not point to related int3,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/72973683/
我问的平台是linux。我说的线程本地存储是gcc中的pthread_xxx或__thread。 我想在我的程序中使用 ucontext 东西,即 makecontext、swapcontext 等。
我读到 ucontext 用于在 linux 中保存多个线程之间的上下文。既然OS做的是不同线程之间的上下文切换,为什么linux要提供这个头文件(ucontext.h)来进行上下文切换呢? 最佳答案
Boost v1.59 中的 Boost.Context 文档报告了以下性能比较结果: +----------+----------------------+-------------------+-
使用 gcc,我在编译使用 ucontext.h 的东西时遇到这些错误 /usr/include/sys/ucontext.h: At top level: /usr/include/
cygwin 似乎不支持用户上下文切换(由 ucontext.h 支持)。 鉴于我需要在 cygwin 下开发一个用户上下文切换应用程序,有什么解决方案吗? 最佳答案 您可能可以编写自己的实现,环绕
我正在尝试将使用 ucontext 的库移植到支持 pthread 但不支持 ucontext 的平台。代码写得很好,因此用对 pthread 例程的调用替换对 ucontext API 的所有调用应
中的用户线程函数已弃用,因为它们使用了已弃用的 C 功能(它们 use a function declaration with empty parentheses for an argument )
所以我正在尝试解除分配上下文。 这是我创建上下文的方式 ucontext* uPtr = new ucontext; getcontext(uPtr); char* stack = new char[
当我在下面编译我的代码时,它会打印 I am running :) 永远(直到我向程序发送 KeyboardInterrupt 信号), 但是当我取消注释 //printf("done:%d\n",
这里是代码,不知道每个detain是什么意思,比如REG_RIP,什么是“RIP” //ucontext.h enum { //... REG_RIP, //... } here是完整代码 最佳答
我正在为操作系统类(class)做一个项目。任务是实现一个处理线程的库,类似于 pthreads,但要简单得多。它的目的是练习调度算法。最终产品是一个 .a 文件。类(class)结束,一切正常(在功
作为更大的线程调度 api 的一部分,这是我想做的事情。 我想创建一个线程,当主线程(创建线程的线程)退出时,我刚创建的线程应该执行。我正在尝试使用 ucontext 和 uc_link 执行此操作,
我去年学习了操作系统,期间我使用用户上下文(在头文件 ucontext.h 中定义)为一个项目实现线程调度程序(其中每个线程模拟一个进程)。我正在参加一个讲座,将讨论用户上下文,我突然想到,尽管去年完
嗨,我需要从一个地方跳到另一个地方... 但我想知道哪个更好用,setjmp 或 ucontext,例如: setjmp 和 ucontext 是否可移植? 我的代码使用这些库是线程安全的吗? 为什么
据我所知,ucontext 提供了比 setjmp 更好的东西。但它已被弃用,现在已从 POSIX 规范中删除。那么它为什么会出现,又为什么会被移除? 最佳答案 makecontext的签名来自 uc
我正在设计一个具有以下功能的调度算法: 在一个进程中有 2 个用户线程(上下文)(我应该做 3 个线程,但在 osx 上还不起作用,所以我决定现在让 2 个工作) 使用每 1 秒关闭一次的 SIGAL
我正在使用 ucontext 开发一个简单的、具有教育意义的绿色(合作)基于量子的线程示例。但我面临着问题。下面提供的示例非常容易理解,我将感谢您为我提供的任何帮助: TestApp.java: pu
我的代码在这里: #define _GNU_SOURCE #include "lib-ult.h" #include #include #include #include #define TR
我有这个最小的 helloworld,扩展了 ucontext.h: #include #include int main(int argc, char** argv) { printf ("
我正在尝试使用 ucontext 例程来实现多线程库。运行此代码时出现“浮点异常(核心已转储)”。 #include #include #include typedef struct {
我是一名优秀的程序员,十分优秀!