- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
尝试使用 ptrace + seccomp 获取系统调用的返回值时我有点困惑。
FILTER MACHINE
A filter program is an array of instructions, with all branches forwardly
directed, terminated by a return instruction
man 2 ptrace 说:
PTRACE_O_TRACESECCOMP
While this triggers a PTRACE_EVENT stop, it is
similar to a syscall-enter-stop, in that the tracee has not yet
entered the syscall that seccomp triggered on. The seccomp event
message data (from the SECCOMP_RET_DATA portion of the seccomp filter
rule) can be retrieved with PTRACE_GETEVENTMSG.
PTRACE_GETEVENTMSG
For PTRACE_EVENT_SECCOMP, this is the seccomp(2)
filter's SECCOMP_RET_DATA associated with the triggered rule.
man 2 seccomp 说:
SECCOMP_RET_TRACE
The tracer will be notified of a
PTRACE_EVENT_SECCOMP and the SECCOMP_RET_DATA
portion of the filter's return value will be available to
the tracer via PTRACE_GETEVENTMSG
[...]
The seccomp check will not be run again after the tracer is notified.
事实证明,BPF 程序无法在 BPF_RET
语句之后执行进一步的操作。因此,当tracee在SECCOMP_RET_TRACE
上被中断时,它处于syscall-enter-stop
状态并且系统调用尚未进行,因此,返回代码肯定无处可取。我希望在后续调用 PTRACE_SYSCALL
后,tracee 将处于 syscall-exit-stop
状态,并且跟踪器将能够使用 获取系统调用的结果PTRACE_GETEVENTMSG
。但它在我的示例中不起作用。
#include <linux/filter.h>
#include <linux/seccomp.h>
#include <linux/unistd.h>
#include <stddef.h>
#include <stdio.h>
#include <sys/prctl.h>
#include <sys/ptrace.h>
#include <sys/wait.h>
#include <unistd.h>
int main(int argc, char **argv)
{
pid_t pid;
int status;
if (argc < 2) {
fprintf(stderr, "Usage: %s <prog> <arg1> ... <argN>\n", argv[0]);
return 1;
}
if ((pid = fork()) == 0) {
ptrace(PTRACE_TRACEME, 0, 0, 0);
struct sock_filter filter[] = {
BPF_STMT(BPF_LD | BPF_W | BPF_ABS, (offsetof(struct seccomp_data, nr))),
BPF_JUMP(BPF_JMP | BPF_JEQ | BPF_K, __NR_open, 1, 2),
BPF_JUMP(BPF_JMP | BPF_JEQ | BPF_K, __NR_openat, 0, 1),
BPF_STMT(BPF_RET | BPF_K, SECCOMP_RET_TRACE | SECCOMP_RET_DATA),
BPF_STMT(BPF_RET | BPF_K, SECCOMP_RET_ALLOW),
};
struct sock_fprog prog = {
.filter = filter,
.len = (unsigned short) (sizeof(filter)/sizeof(filter[0])),
};
if (prctl(PR_SET_NO_NEW_PRIVS, 1, 0, 0, 0) == -1)
return 2;
if (prctl(PR_SET_SECCOMP, SECCOMP_MODE_FILTER, &prog) == -1)
return 3;
kill(getpid(), SIGSTOP);
return execvp(argv[1], argv + 1);
} else {
waitpid(pid, &status, 0);
ptrace(PTRACE_SETOPTIONS, pid, 0, PTRACE_O_TRACESECCOMP);
ptrace(PTRACE_CONT, pid, 0, 0);
int status = 0;
unsigned long ret_data = 0;
while(1) {
while (1) {
waitpid(pid, &status, 0);
fprintf(stderr, "status = %08x\n", status);
if (status >> 8 == (SIGTRAP | (PTRACE_EVENT_SECCOMP << 8)))
break;
if (WIFEXITED(status))
return 0;
ptrace(PTRACE_CONT, pid, 0, 0);
}
// restart stopped tracee
ptrace(PTRACE_SYSCALL, pid, 0, 0);
// wait for SIGTRAP, when tracee will be in the syscall-exit-stop state
waitpid(pid, &status, 0);
ptrace(PTRACE_GETEVENTMSG, pid, 0, &ret_data);
fprintf(stderr, "retdat = %lu\n", ret_data);
ptrace(PTRACE_CONT, pid, 0, 0);
}
return 0;
}
}
我能够获取系统调用的返回代码来检查寄存器
// ptrace(PTRACE_GETEVENTMSG, pid, 0, &ret_data);
struct user_regs_struct regs;
ptrace(PTRACE_GETREGS, pid, 0, ®s);
fprintf(stderr, "retdat = %lu\n", regs.rax);
但我想知道如何按照文档中指定的方式进行操作。
最佳答案
How to get the return code of the syscall using SECCOMP_RET_DATA and PTRACE_GETEVENTMSG?
简单的答案是你不能。 seccomp 事件甚至在进入系统调用之前就被发送。您在那里看不到任何结果,因为还没有任何系统调用。要获得一个,您必须在收到 seccomp 事件后使用 PTRACE_SYSCALL
旋转该进程两次:
bool WaitForSyscallExit(const pid_t pid)
{
bool entered = false;
int status = 0;
while (true)
{
ptrace(PTRACE_SYSCALL, pid, 0, 0);
waitpid(pid, &status, 0);
if (WSTOPSIG(status) == SIGTRAP)
{
if (entered)
{
// If we had already entered before, then current SIGTRAP signal means exiting
break;
}
entered = true;
}
else if (WIFEXITED(status) || WIFSIGNALED(status) || WCOREDUMP(status))
{
std::cerr << "The child has unexpectedly exited." << std::endl;
return false;
}
}
return true;
}
由于使用了PTRACE_SYSCALL
,进程会被停止两次(第一次是进入系统调用后,下一次和最后一次是退出系统调用后)。只有在系统调用实际完成后(即第二个进程停止后)才能获取结果。是的,您只能通过手动读取寄存器来完成此操作,因为 seccomp 结构只能在该事件的 seccomp 跟踪处理程序中使用。甚至结构本身也不包含与系统调用结果相关的任何内容,并且手册页也没有提到获取结果值。
关于c - 如何使用 SECCOMP_RET_DATA 和 PTRACE_GETEVENTMSG 获取系统调用的返回码,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55006186/
为了让我的代码几乎完全用 Jquery 编写,我想用 Jquery 重写 AJAX 调用。 这是从网页到 Tomcat servlet 的调用。 我目前情况的类似代码: var http = new
我想使用 JNI 从 Java 调用 C 函数。在 C 函数中,我想创建一个 JVM 并调用一些 Java 对象。当我尝试创建 JVM 时,JNI_CreateJavaVM 返回 -1。 所以,我想知
环顾四周,我发现从 HTML 调用 Javascript 函数的最佳方法是将函数本身放在 HTML 中,而不是外部 Javascript 文件。所以我一直在网上四处寻找,找到了一些简短的教程,我可以根
我有这个组件: import {Component} from 'angular2/core'; import {UserServices} from '../services/UserService
我正在尝试用 C 实现一个简单的 OpenSSL 客户端/服务器模型,并且对 BIO_* 调用的使用感到好奇,与原始 SSL_* 调用相比,它允许一些不错的功能。 我对此比较陌生,所以我可能会完全错误
我正在处理有关异步调用的难题: 一个 JQuery 函数在用户点击时执行,然后调用一个 php 文件来检查用户输入是否与数据库中已有的信息重叠。如果是这样,则应提示用户确认是否要继续或取消,如果他单击
我有以下类(class)。 public Task { public static Task getInstance(String taskName) { return new
嘿,我正在构建一个小游戏,我正在通过制作一个数字 vector 来创建关卡,该数字 vector 通过枚举与 1-4 种颜色相关联。问题是循环(在 Simon::loadChallenge 中)我将颜
我有一个java spring boot api(数据接收器),客户端调用它来保存一些数据。一旦我完成了数据的持久化,我想进行另一个 api 调用(应该处理持久化的数据 - 数据聚合器),它应该自行异
首先,这涉及桌面应用程序而不是 ASP .Net 应用程序。 我已经为我的项目添加了一个 Web 引用,并构建了各种数据对象,例如 PayerInfo、Address 和 CreditCard。但问题
我如何告诉 FAKE 编译 .fs文件使用 fsc ? 解释如何传递参数的奖励积分,如 -a和 -target:dll . 编辑:我应该澄清一下,我正在尝试在没有 MSBuild/xbuild/.sl
我使用下划线模板配置了一个简单的主干模型和 View 。两个单独的 API 使用完全相同的配置。 API 1 按预期工作。 要重现该问题,请注释掉 API 1 的 URL,并取消注释 API 2 的
我不确定什么是更好的做法或更现实的做法。我希望从头开始创建目录系统,但不确定最佳方法是什么。 我想我在需要显示信息时使用对象,例如 info.php?id=100。有这样的代码用于显示 Game.cl
from datetime import timedelta class A: def __abs__(self): return -self class B1(A):
我在操作此生命游戏示例代码中的数组时遇到问题。 情况: “生命游戏”是约翰·康威发明的一种细胞自动化技术。它由一个细胞网格组成,这些细胞可以根据数学规则生存/死亡/繁殖。该网格中的活细胞和死细胞通过
如果我像这样调用 read() 来读取文件: unsigned char buf[512]; memset(buf, 0, sizeof(unsigned char) * 512); int fd;
我用 C 编写了一个简单的服务器,并希望调用它的功能与调用其他 C 守护程序的功能相同(例如使用 ./ftpd start 调用它并使用 ./ftpd stop 关闭该实例)。显然我遇到的问题是我不知
在 dos 中,当我粘贴此命令时它会起作用: "C:\Program Files (x86)\Google\Chrome\Application\chrome.exe" https://google.
在 dos 中,当我粘贴此命令时它会起作用: "C:\Program Files (x86)\Google\Chrome\Application\chrome.exe" https://google.
我希望能够从 cmd 在我的 Windows 10 计算机上调用 python3。 我已重新安装 Python3.7 以确保选择“添加到路径”选项,但仍无法调用 python3 并使 CMD 启动 P
我是一名优秀的程序员,十分优秀!