因为想跟踪下在新建进程时,如何处理新建进程的vruntime,所以跟踪了下fork。

以glic-2.17中ARM为例(unicore架构的没找到),实际上通过寄存器向系统调用传递的参数为:

r7: __NR_clone 120

r0: CLONE_CHILD_SETTID | CLONE_CHILD_CLEARTID | SIGCHLD

r1: NULL

r2: NULL

r3: NULL

r4: &THREAD_SELF->tid

fork()---->__fork()---->__libc_fork()

__libc_fork()---->INLINE_SYSCALL (clone, , \
CLONE_CHILD_SETTID | \
CLONE_CHILD_CLEARTID | \
SIGCHLD, \
NULL, NULL, NULL, &THREAD_SELF->tid)
#define INLINE_SYSCALL(name, nr, args...)  \
({ unsigned int _sys_result = \
                  INTERNAL_SYSCALL (name, , nr, args); \
if (__builtin_expect \
                (INTERNAL_SYSCALL_ERROR_P (_sys_result, ), )) \
{ \
__set_errno (INTERNAL_SYSCALL_ERRNO \
                            (_sys_result, )); \
_sys_result = (unsigned int) -; \
} \
(int) _sys_result; })
#define INTERNAL_SYSCALL(name, err, nr, args...)   \
INTERNAL_SYSCALL_RAW(SYS_ify(name), err, nr, args) //#define __NR_clone 120
#define SYS_ify(syscall_name) (__NR_##syscall_name)
//err没用使用,也没有传递宏参,第一次见到
# define INTERNAL_SYSCALL_RAW(name, err, nr, args...) \
({ \
register int _a1 asm ("r0"), _nr asm ("r7"); \
LOAD_ARGS_##nr (args) \
_nr = name; \
asm volatile ("swi 0x0 @ syscall " #name \
: "=r" (_a1) \
: "r" (_nr) ASM_ARGS_##nr \
: "memory"); \
_a1; })

关于宏INTERAL_SYSCALL_RAW还是满有意思的,写了个函数测试下,真实的看下库如何向系统调用传参:

#define LOAD_ARGS_0()
#define ASM_ARGS_0 #define LOAD_ARGS_1(a1) \
int _a1tmp = (int)(a1); \
LOAD_ARGS_0 () \
_a1 = _a1tmp; #define ASM_ARGS_1 ASM_ARGS_0, "r" (_a1) #define LOAD_ARGS_2(a1, a2) \
int _a2tmp = (int)(a2); \
LOAD_ARGS_1(a1) \
register int _a2 asm ("a2") = _a2tmp; #define ASM_ARGS_2 ASM_ARGS_1, "r" (_a2) #define LOAD_ARGS_3(a1, a2, a3) \
int _a3tmp = (int)(a3); \
LOAD_ARGS_2 (a1, a2) \
register int _a3 asm ("a3") = _a3tmp; #define ASM_ARGS_3 ASM_ARGS_2, "r" (_a3) #define LOAD_ARGS_4(a1, a2, a3, a4) \
int _a4tmp = (int) (a4); \
LOAD_ARGS_3 (a1, a2, a3) \
register int _a4 asm ("a4") = _a4tmp; #define ASM_ARGS_4 ASM_ARGS_3, "r" (_a4) #define LOAD_ARGS_5(a1, a2, a3, a4, a5) \
int _v1tmp = (int)(a5); \
LOAD_ARGS_4 (a1, a2, a3, a4) \
register int _v1 asm ("v1") = _v1tmp; #define ASM_ARGS_5 ASM_ARGS_4, "r" (_v1) #define INTERNAL_SYSCALL_RAW(name, err, nr, args...) \
({ \
register int _a1 asm("r0"), _nr asm("r7"); \
LOAD_ARGS_5 (args) \
_nr = name; \
asm volatile ("swi 0x0" \
: "=r" (_a1) \
: "r" (_nr) ASM_ARGS_##nr \
: "memory" \
); \
_a1; \
}) void test(void)
{
INTERNAL_SYSCALL_RAW(, , , , , , , );
}

预处理后即为:

void test(void)
{
({
register int _a1 asm("r0"), _nr asm("r7");
int _v1tmp = (int)();
int _a4tmp = (int) ();
int _a3tmp = (int)();
int _a2tmp = (int)();
int _a1tmp = (int)();
_a1 = _a1tmp;
register int _a2 asm ("a2") = _a2tmp;
register int _a3 asm ("a3") = _a3tmp;
register int _a4 asm ("a4") = _a4tmp;
register int _v1 asm ("v1") = _v1tmp;
_nr = ;
asm volatile (
"swi 0x0"
: "=r" (_a1)
: "r" (_nr) , "r" (_a1), "r" (_a2),
                       "r" (_a3), "r" (_a4), "r" (_v1)
: "memory"
);
_a1;
});
}

相应的汇编代码:

test:
stmfd sp!, {r4, r7, fp}
add fp, sp, #
sub sp, sp, # mov r3, #
str r3, [fp, #-] mov r3, #
str r3, [fp, #-] mov r3, #
str r3, [fp, #-] mov r3, #
str r3, [fp, #-] mov r3, #
str r3, [fp, #-] ldr r0, [fp, #-]
ldr r1, [fp, #-]
ldr r2, [fp, #-]
ldr r3, [fp, #-]
ldr r4, [fp, #-]
mov r7, #
#APP
@ "go.c"
swi 0x0
@ ""
sub sp, fp, #
ldmfd sp!, {r4, r7, fp}
bx lr

glibc中fork系统调用传参的更多相关文章

  1. 用 set follow-fork-mode child即可。这是一个 gdb 命令,其目的是告诉 gdb 在目标应用调用fork之后接着调试子进程而不是父进程,因为在 Linux 中fork系统调用成功会返回两次,一次在父进程,一次在子进程

    GDB的那些奇淫技巧 evilpan 收录于 Security  2020-09-13  约 5433 字   预计阅读 11 分钟  709 次阅读  gdb也用了好几年了,虽然称不上骨灰级玩家,但 ...

  2. 在ListBoxItem的样式中的button传参,把当前选中项传递到命令的方法

    原文:在ListBoxItem的样式中的button传参,把当前选中项传递到命令的方法 前端页面: <Style x:Key="ThumbItemStyle" TargetT ...

  3. Angular 项目开发中父子组件传参

    在项目开发中经常会遇到 组件之间传参的问题.今天总结下在使用angular的项目中父子组件传参的问题: 1.父组件向子组件传参: 然后在父组件中 然后在父组件的html中 然后就可以在子组件中使用了 ...

  4. jq中的ajax传参

        一.   jq中的Ajax传参有两种           1.通过url地址来传参    2.通过data来传递参数 1. url来传递参数 function GetQuery(id) { | ...

  5. Python中的引用传参

    Python中函数参数是引用传递(注意不是值传递).对于不可变类型,因变量不能修改,所以运算不会影响到变量自身:而对于可变类型来说,函数体中的运算有可能会更改传入的参数变量. 引用传参一: >& ...

  6. JS中onclick事件传参

    美术馆案例中,需要将“增加一个img标签,放入大图片”这样一个函数封装,但是在调用的时候需要传参. <script type="text/javascript"> on ...

  7. springMVC中controller的传参的几种案例

    1.springmvc的controller方法不指定method时,默认get/post都支持 //@RequestMapping(value="test") //@Reques ...

  8. python中给函数传参是传值还是传引用

    首先还是应该科普下函数参数传递机制,传值和传引用是什么意思? 函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题.基本的参数传递机制有两种:值传递和引用传 ...

  9. vue中的路由传参及跨组件传参

    路由跳转   this.$router.push('/course'); this.$router.push({name: course}); this.$router.go(-1); this.$r ...

随机推荐

  1. vue-router "path" is required in a route configuration

    启用了动态路由,一直提示这个错误,页面打开也是空白,后来发现原来是component参数错误. 正确的写法为: component: () => import ('@/views/own-spa ...

  2. java算法---五家共井

    古代数学巨著<九章算数>中有这么一道题叫“五家共井,甲二绠(汲水用的井绳)不足,如(接上)乙一绠:乙三绠不足,如丙一绠: 丙四绠不足,如丁一绠:丁五绠不足,如戊一绠:戊六绠不足,如甲一绠, ...

  3. [转]mysql 行转列 列转行

    原文地址:http://www.cnblogs.com/xiaoxi/p/7151433.html 一.行转列 即将原本同一列下多行的不同内容作为多个字段,输出对应内容. 建表语句 DROP TABL ...

  4. [转]JSTL 与 JSP 或者 Java 相互传递变量的代码

    原文地址:http://blog.csdn.net/joyous/article/details/6689861 两种方式 <c:set var="s1" value=&qu ...

  5. C#2.0 Socket套接字编程之实例初探 200

    首先从原理上解释一下采用Socket接口的网络通讯,这里以最常用的C/S模式作为范例,首先,服务端有一个进程(或多个进程)在指定的端口等待客户来连接,服务程序等待客户的连接信息,一旦连接上之后,就可以 ...

  6. 【WPF/WAF】使用System.Windows.Interactivity交互事件

    下载System.Windows.Interactivity.dll文件,并引入项目中(在VS项目的引用列表中可以看到).可在Nuget搜索System.Windows.Interactivity下载 ...

  7. java.io.BufferedOutputStream 源码分析

    BufferedOutputStream  是一个带缓冲区的输出流,通过设置这种输出流,应用程序就可以字节写入到缓冲区中,当缓冲区满了以后再调用底层系统,而不必针对每次字节写入调用底层系统,从而提高系 ...

  8. hadoop rebalance

    之前一直没做过rebalance,以为速度很快,结果大意了,等到磁盘达到90%的时候,才开始做rebalance. 默认的从日志中可以看到总共需要迁移1.89T,但是每次只移动40G大小的量. 然后查 ...

  9. Extjs4.x (MVC)Controller中refs以及Ext.ComponentQuery解析

    refs : Object[]5 Array of configs to build up references to views on page. For example: Ext.define(& ...

  10. Java知多少(17)强调一下编程风格

    讲完了Java的基础语法,大家就可以编写简单的程序代码了,这里有必要强调一下编程风格. 代码风格虽然不影响程序的运行,但对程序的可读性却非常重要.自己编写的程序要让别人看懂,首先在排版方面要非常注意. ...