gpt4 book ai didi

linux - fio启动时如何加载各种io引擎?

转载 作者:太空狗 更新时间:2023-10-29 12:11:43 24 4
gpt4 key购买 nike

fio 支持一大堆 io 引擎 - 所有支持的引擎都在这里:https://github.com/axboe/fio/tree/master/engines

我一直在尝试了解 fio 的工作原理,但一直被困在 fio 如何加载所有 io 引擎上。

例如,我看到每个引擎都有一个注册和注销自身的方法,例如 sync.c 使用以下方法注册和注销

fio_syncio_register:https://github.com/axboe/fio/blob/master/engines/sync.c#L448

fio_syncio_unregister: https://github.com/axboe/fio/blob/master/engines/sync.c#L461

我的问题是谁调用这些方法?

为了找到答案,我尝试在 gdb 下运行 fio - 在 fio_syncio_register 和 main 函数中放置了一个断点,fio_syncio_register 甚至在 main 之前就被调用了,这告诉我它与 __libc_csu_init 有关回溯确认

(gdb) bt
#0 fio_syncio_register () at engines/sync.c:450
#1 0x000000000047fb9d in __libc_csu_init ()
#2 0x00007ffff6ee27bf in __libc_start_main (main=0x40cd90 <main>, argc=2, argv=0x7fffffffe608, init=0x47fb50 <__libc_csu_init>, fini=<optimized out>, rtld_fini=<optimized out>, stack_end=0x7fffffffe5f8)
at ../csu/libc-start.c:247
#3 0x000000000040ce79 in _start ()

我花了一些时间阅读关于 __libc_csu_init__libc_csu_fini 的每一个描述都谈到了用 __attribute__((constructor)) 装饰的方法将是在 main 之前调用,但在 fio sync.c 的情况下,我没有看到用 __attribute__

修饰的 fio_syncio_register

有人可以帮我理解这个流程是如何工作的吗?我应该阅读其他 Material 来理解这一点吗?

谢谢

最佳答案

有趣的问题。我在查看源代码时找不到答案,所以这是我采取的步骤:

$ make
$ find . -name 'sync.o'
./engines/sync.o

$ readelf -WS engines/sync.o | grep '\.init'
[12] .init_array INIT_ARRAY 0000000000000000 0021f0 000008 00 WA 0 0 8
[13] .rela.init_array RELA 0000000000000000 0132a0 000018 18 36 12 8

这告诉我们全局初始值设定项存在于该对象中。这些在程序启动时调用。它们是什么?

$ objdump -Dr engines/sync.o | grep -A4 '\.init'
Disassembly of section .init_array:

0000000000000000 <.init_array>:
...
0: R_X86_64_64 .text.startup

很有趣。显然有一个特殊的 .text.startup 部分。里面有什么?

$ objdump -dr engines/sync.o | less
...
Disassembly of section .text.startup:

0000000000000000 <fio_syncio_register>:
0: 48 83 ec 08 sub $0x8,%rsp
4: bf 00 00 00 00 mov $0x0,%edi
5: R_X86_64_32 .data+0x380
9: e8 00 00 00 00 callq e <fio_syncio_register+0xe>
a: R_X86_64_PC32 register_ioengine-0x4
...

为什么,这正是我们正在寻找的功能。但是它怎么会出现在这个特殊的部分呢?要回答这个问题,我们可以查看经过预处理的源代码(回想起来,我应该开始的)。

我们如何获得它?编译 sync.o 的命令行是隐藏的。查看 Makefile,我们可以使用 QUIET_CC='' 取消隐藏命令行。

$ rm engines/sync.o && make QUIET_CC=''
gcc -o engines/sync.o -std=gnu99 -Wwrite-strings -Wall -Wdeclaration-after-statement -g -ffast-math -D_GNU_SOURCE -include config-host.h -I. -I. -O3 -U_FORTIFY_SOURCE -D_FORTIFY_SOURCE=2 -DBITS_PER_LONG=64 -DFIO_VERSION='"fio-2.16-5-g915ca"' -D_LARGEFILE_SOURCE -D_FILE_OFFSET_BITS=64 -DFIO_INTERNAL -DFIO_INC_DEBUG -c engines/sync.c
LINK fio

现在我们知道命令行了,可以生成预处理文件了:

$ gcc -E -dD -std=gnu99 -ffast-math  -D_GNU_SOURCE -include config-host.h -I. -I. -O3 -U_FORTIFY_SOURCE -D_FORTIFY_SOURCE=2 -DBITS_PER_LONG=64 -DFIO_VERSION='"fio-2.16-5-g915ca"' -D_LARGEFILE_SOURCE -D_FILE_OFFSET_BITS=64 -DFIO_INTERNAL -DFIO_INC_DEBUG engines/sync.c -o /tmp/sync.i

查看 /tmp/sync.i,我们看到:

static void __attribute__((constructor)) fio_syncio_register(void)
{
register_ioengine(&ioengine_rw);
register_ioengine(&ioengine_prw);
...

嗯,它毕竟是__attribute__((constructor))。但它是如何到达那里的呢?啊哈!我错过了 this line 上的 fio_init :

static void fio_init fio_syncio_register(void)

fio_init 代表什么?再次在 /tmp/sync.i 中:

#define fio_init __attribute__((constructor))

所以 这就是它的工作原理。

关于linux - fio启动时如何加载各种io引擎?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41356074/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com