免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 9656 | 回复: 13
打印 上一主题 下一主题

请教个linux下aio的问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2010-05-17 15:17 |只看该作者 |倒序浏览
aio的异步完成通知有2种形式:
1)异步信号
2)系统回调函数

我想知道的是,当采用2)系统回调函数时,执行该函数的线程是内核线程么?应该不是发起aio的那个用户线程。

  1. void setup_io( ... )
  2. {
  3.   int fd;
  4.   struct aiocb my_aiocb;
  5.   ...
  6.   
  7.   bzero( (char *)&my_aiocb, sizeof(struct aiocb) );
  8.   my_aiocb.aio_fildes = fd;
  9.   my_aiocb.aio_buf = malloc(BUF_SIZE+1);
  10.   my_aiocb.aio_nbytes = BUF_SIZE;
  11.   my_aiocb.aio_offset = next_offset;
  12.   
  13.   my_aiocb.aio_sigevent.sigev_notify = SIGEV_THREAD;
  14.   my_aiocb.aio_sigevent.notify_function = aio_completion_handler;
  15.   my_aiocb.aio_sigevent.notify_attributes = NULL;
  16.   my_aiocb.aio_sigevent.sigev_value.sival_ptr = &my_aiocb;
  17.   ...
  18.   ret = aio_read( &my_aiocb );
  19. }

  20. void aio_completion_handler( sigval_t sigval )
  21. {
  22.   struct aiocb *req;
  23.   req = (struct aiocb *)sigval.sival_ptr;
  24.   
  25.   if (aio_error( req ) == 0) {
  26.      
  27.      ret = aio_return( req );
  28.   }
  29.   return;
  30. }
复制代码
也就是说,内核通知线程会执行aio_completion_handler?

论坛徽章:
0
2 [报告]
发表于 2010-05-17 16:28 |只看该作者
往底层一点儿说,回调函数,就是代码执行到一个地方,那里保存了一个函数的指针,然后直接就调用了那个函数指针了。跟你在那里直接调用一个函数是完全一样的效果。
所以,调用的地方是什么线程,函数执行时就是什么线程,任何回调函数,都与线程切换无关。

论坛徽章:
1
2015年辞旧岁徽章
日期:2015-03-03 16:54:15
3 [报告]
发表于 2010-05-17 16:56 |只看该作者
不是内核线程,就是 pthread_create 创建的普通线程

论坛徽章:
1
2015年辞旧岁徽章
日期:2015-03-03 16:54:15
4 [报告]
发表于 2010-05-17 16:57 |只看该作者
回复 1# wishel


  你在gnu 上下个glibc的代码,  aio的实现就在  glibc-2.9/sysdeps/pthread/

论坛徽章:
0
5 [报告]
发表于 2010-05-17 17:00 |只看该作者
回复 4# cookis


  ok,多谢!

论坛徽章:
0
6 [报告]
发表于 2010-05-17 20:28 |只看该作者
提示: 作者被禁止或删除 内容自动屏蔽

论坛徽章:
0
7 [报告]
发表于 2010-05-17 22:54 |只看该作者
从/proc/PID/status中可以看到执行这个函数时,是有两个线程的。

论坛徽章:
0
8 [报告]
发表于 2010-05-17 23:29 |只看该作者
用sleep,然后查看PS -AUXF

论坛徽章:
0
9 [报告]
发表于 2010-05-17 23:58 |只看该作者
郁闷...inux下的aio是不是问题很多?
大家帮我看下这段程序:

  1. #include <sys/epoll.h>
  2. #include <stdio.h>
  3. #include <string.h>
  4. #include <fcntl.h>
  5. #include <stdlib.h>
  6. #include <unistd.h>
  7. #include <aio.h>

  8. static const size_t BUF_SIZE = 512 * 1024;
  9. volatile int ret;
  10. int fds[2];

  11. void aio_completion_handler( sigval_t sigval ) {
  12.   struct aiocb *req;

  13.   req = (struct aiocb *)sigval.sival_ptr;

  14.   printf("begin ret in handler: %d\n", ret);

  15.   if (aio_error( req ) == 0) {
  16.     ret = aio_return( req );
  17.   } else {
  18.           perror("read error");
  19.   }

  20.   printf("end ret in handler: %d\n", ret);
  21.   write(fds[1], "hello world!\n", 13);
  22.   return;
  23. }

  24. void setup_io() {
  25.        char INPUTFILE[] = "inputfile";
  26.        int fin = open(INPUTFILE, O_RDONLY);
  27.        if (fin == -1)
  28.                perror("open file error");
  29.        int fd = fin;

  30.    int ret;
  31.    struct aiocb my_aiocb;

  32.    bzero( (char *)&my_aiocb, sizeof(struct aiocb) );

  33.   my_aiocb.aio_fildes = fd;
  34.   my_aiocb.aio_buf = malloc(BUF_SIZE+1);
  35.   my_aiocb.aio_nbytes = BUF_SIZE;
  36.   my_aiocb.aio_offset = 0;
  37.   my_aiocb.aio_sigevent.sigev_notify = SIGEV_THREAD;
  38.   my_aiocb.aio_sigevent.sigev_notify_function = aio_completion_handler;
  39.   my_aiocb.aio_sigevent.sigev_notify_attributes = NULL;
  40.   my_aiocb.aio_sigevent.sigev_value.sival_ptr = &my_aiocb;

  41.   ret = aio_read( &my_aiocb );
  42. }

  43. int main () {
  44.         if (pipe(fds) == -1)
  45.                 return -1;

  46.         int epollfd = epoll_create(10);
  47.         struct epoll_event* event = new epoll_event;
  48.         event->events = EPOLLIN;
  49.         event->data.fd = fds[0];
  50.         int n;
  51.         n = epoll_ctl(epollfd, EPOLL_CTL_ADD, fds[0], event);
  52.         printf("n= %d\n", n);

  53.         setup_io();
  54.         //sleep(1);

  55.                 epoll_event events[10];
  56.                 int nfds = epoll_wait(epollfd, events, 10 , -1);
  57.                 printf("nfds= %d\n", nfds);
  58.                 for (int i = 0; i < nfds; ++i) {
  59.                         epoll_event ev = events[i];
  60.                         if (ev.events & EPOLLIN) {
  61.                                 printf("fd %d, event EPOLLIN\n", ev.data.fd);
  62.                         }
  63.                         if (ev.events & EPOLLOUT) {
  64.                                 printf("fd %d, event EPOLLOUT\n", ev.data.fd);
  65.                         }
  66.                         if (ev.events & EPOLLHUP)
  67.                                 printf("fd %d, event EPOLLHUP\n", ev.data.fd);
  68.                         if (ev.events & EPOLLERR)
  69.                                 printf("fd %d, event EPOLLERR\n", ev.data.fd);
  70.                 }

  71.         //sleep(1);
  72.         printf("ret = %d!!!\n", ret);

  73.         return 0;
  74. }
复制代码
连续运行几次的结果:
wang@ubuntu:~/workspace/test/Debug$ ./test
n= 0
begin ret in handler: 0
end ret in handler: 8104
nfds= 1
fd 3, event EPOLLIN
ret = 8104!!!
wang@ubuntu:~/workspace/test/Debug$ ./test
n= 0
begin ret in handler: 0
end ret in handler: -1208869936
nfds= 1
fd 3, event EPOLLIN
ret = -1208869936!!!
wang@ubuntu:~/workspace/test/Debug$ ./test
n= 0
begin ret in handler: 0
end ret in handler: 8104
nfds= 1
fd 3, event EPOLLIN
ret = 8104!!!
wang@ubuntu:~/workspace/test/Debug$ ./test
n= 0
begin ret in handler: 0
end ret in handler: -1210160176
nfds= 1
fd 3, event EPOLLIN
ret = -1210160176!!!

论坛徽章:
0
10 [报告]
发表于 2010-05-18 00:02 |只看该作者
打开第68行的sleep后,基本上不出现ret为大负数的情况,但是有时候aio_completion_handler不会执行
程序卡住了,不知道是aio一直没完成还是完成后没能成功call aio_completion_handler

系统是ubuntu 8.10,明天去公司换个系统试试看。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP