免费注册 查看新帖 |

Chinaunix

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

请教关于等待队列的问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2007-08-06 16:19 |只看该作者 |倒序浏览
在写一个AD的驱动程序。用到等待队列的地方如下所示。

结果:等待队列无法wake up。即可以打印出"will wake up",但进程无法被wake up,即无法打印“wake up”。


static wait_queue_head_t wait;  //声明了全局静态变量


static int adcdrv_open(struct inode*inode,struct file*filp)
{
        ....
        init_waitqueue_head(&wait);
        ....
}


static ssize_t adcdrv_read(struct file* file,char* buffer,size_t count,loff_t* ppos)
{
        ....
        interruptible_sleep_on(&wait);
        printk("wake up\n");
        ....
}


static void adcdone_int_handler(int irq, void *dev_id, struct pt_regs *reg)
{
        printk("will wake up\n");
        wake_up_interruptible(&wait);
}

请各位帮忙解答,谢谢~

论坛徽章:
0
2 [报告]
发表于 2007-08-06 16:31 |只看该作者
原帖由 spx 于 2007-8-6 16:19 发表
在写一个AD的驱动程序。用到等待队列的地方如下所示。

结果:等待队列无法wake up。即可以打印出"will wake up",但进程无法被wake up,即无法打印“wake up”。


static wait_queue_head_t wait;  //声 ...



sleep_on()系列的函数慢慢被淘汰了. 现在一般都用wait_event()系列的.
你试试这样可以不:

  1. int flag = 0;
  2. static DECLARE_WAIT_QUEUE_HEAD(wq);
  3. static ssize_t adcdrv_read(struct file* file,char* buffer,size_t count,loff_t* ppos)
  4. {
  5.         ....
  6.         wait_event_interruptible(wq, flag != 0);
  7.         printk("wake up\n");
  8.         ....
  9. }

  10. static void adcdone_int_handler(int irq, void *dev_id, struct pt_regs *reg)
  11. {
  12.         printk("will wake up\n");
  13.         flag = 1;
  14.         wake_up_interruptible(&wq);
  15. }


复制代码

论坛徽章:
0
3 [报告]
发表于 2007-08-06 17:05 |只看该作者
多谢2楼指点。但是我运行后出现如下错误:
是不是在声明和初始化的时候不对亚?

Unable to handle kernel NULL pointer dereferen                                            
Memory: 64MB =
sp : c01a1f1c  ip : 00000000  fp : c01a1f40 62540KB available (1528K code, 328K data,
r10: 00000001  r9 : c01a1f70  r8 : a0000093                        
r3 : 00000000  r2 : 00000001  r1 : c01a0000  r0 : ffffff9d
Flags: Nzcv  IRQs off  FIQs on  Mode SVC_32  Segment kernel
Control: C000317F  Table: 32B58000  DAC: 0000001D
Process swapper (pid: 0, stackpage=c01a1000)
Stack: (0xc01a1f0c to 0xc01a2000)
1f00:                            c032b3a0 c0025d7c 80000093 ffffffff c2cd25a0
1f20: 0000003e 20000000 c01b323c c01b2a7c c01bad80 c01a1f6c c01a1f44 c001a0cc
1f40: c0025d50 ea000000 c01a1fb8 00000001 c00228a8 60000013 ffffffff 30014c58
1f60: c01a1fd0 c01a1f70 c00194c0 c001a01c 00000000 00000032 0000001d c001acb0
1f80: c01a0000 c001acb0 c01a2e70 c01b32a4 c01a2c48 41129200 30014c58 c01a1fd0
1fa0: c01a1fb8 c01a1fb8 c001a5c4 c00228a8 60000013 ffffffff 00004000 c01df680
1fc0: c01b32bc c01a1fe0 c01a1fd4 c0019030 c001a588 c01a1ffc c01a1fe4 c00088bc
1fe0: c0019010 c01b36d0 c01e91f4 c01e91f4 00000000 c01a2000 c0008080 c000878c
Backtrace:
Function entered at [<c0025d40>] from [<c001a0cc>]
Function entered at [<c001a00c>] from [<c00194c0>]
Function entered at [<c001a578>] from [<c0019030>]
r6 = C01B32BC  r5 = C01DF680  r4 = 00004000
Function entered at [<c0019000>] from [<c00088bc>]
Function entered at [<c000877c>] from [<c0008080>]
Code: e1a07000 e1540007 0a000024 e514c004 (e59c3000)
Kernel panic: Aiee, killing interrupt handler!
In interrupt handler - not syncing

论坛徽章:
0
4 [报告]
发表于 2007-08-06 17:17 |只看该作者

回复 #1 spx 的帖子

我把初始化改回原来的
static wait_queue_head_t wait;
init_waitqueue_head(&wait);
就不会出现上面那样的错误,但是还是无法wake up进程。

论坛徽章:
0
5 [报告]
发表于 2007-08-06 18:06 |只看该作者
那你检查一下看是不是其它地方写错了吧, wait_event()的函数的确是这样用的.

论坛徽章:
0
6 [报告]
发表于 2007-08-06 18:24 |只看该作者
那可以能是那些地方的问题亚?
求教~~

论坛徽章:
0
7 [报告]
发表于 2007-08-06 18:36 |只看该作者
原帖由 spx 于 2007-8-6 18:24 发表
那可以能是那些地方的问题亚?
求教~~



以前使用wait_event()函数写的一个定时器, 你看看吧, 就是使用wait_event_interruptible()这个函数.
希望对你有用.


  1. #include<linux/init.h>
  2. #include<linux/sched.h>
  3. #include<linux/module.h>
  4. #include<linux/jiffies.h>
  5. #include<asm/io.h>
  6. #include<linux/timer.h>
  7. #include<linux/wait.h>

  8. MODULE_LICENSE("GPL");

  9. static DECLARE_WAIT_QUEUE_HEAD(wq);
  10. static int flag = 0;
  11. struct timer_list my_timer;

  12. void timer_test(unsigned long data)
  13. {
  14.         printk("this is a test\n");
  15.         flag = 1;
  16.         wake_up_interruptible(&wq);
  17.         printk("wake it up\n");
  18. }

  19. static int __init hello_init(void)
  20. {
  21.         init_timer(&my_timer);
  22.         my_timer.expires = jiffies + 5000;
  23.         my_timer.data = 0;
  24.         my_timer.function = timer_test;
  25.         add_timer(&my_timer);
  26.         wait_event_interruptible(wq, flag != 0);
  27.         printk("I am waked\n");
  28.         flag = 0;
  29.         return 0;
  30. }

  31. static void __exit hello_exit(void)
  32. {
  33.         del_timer(&my_timer);
  34.         printk(KERN_ALERT "Goodbye\n");
  35. }

  36. module_init(hello_init);
  37. module_exit(hello_exit);


复制代码

论坛徽章:
0
8 [报告]
发表于 2007-08-07 09:54 |只看该作者
还是没有办法wake up进程。下面是我的程序代码,请帮忙解决~~
#include <linux/config.h>
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>

#include <linux/sched.h>
#include <linux/irq.h>
#include <linux/delay.h>

#include <asm/hardware.h>
#include <asm/semaphore.h>
#include <asm/uaccess.h>


MODULE_LICENSE("GPL");
#define IOPORT_MAJOR 220

#define START_ADC_AIN(x) \
&nbsp;&nbsp;&nbsp;&nbsp;{ \
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;ADCCON = PRESCALE_EN | PRSCVL(255) | ADC_INPUT((x)) ; \
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;ADCCON |= ADC_START; \
&nbsp;&nbsp;&nbsp;&nbsp;}


static struct semaphore adc_lock;
static  wait_queue_head_t adc_wait;
static int flag=0;

static ssize_t adcdrv_read(struct file*,char*,size_t,loff_t*);
static int adcdrv_open(struct inode*,struct file*);

struct file_operations adcdrv_fops=
{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;open:adcdrv_open,
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;read:adcdrv_read,
};

//adcdone中断处理函数

static void adcdone_int_handler(int irq, void *dev_id, struct pt_regs *reg)
{
&nbsp;&nbsp;&nbsp;&nbsp;printk("will wake up\n");
&nbsp;&nbsp;&nbsp;&nbsp;flag=1;
&nbsp;&nbsp;&nbsp;&nbsp;wake_up_interruptible(&adc_wait);
}

int s3c2410_adc_read(int ain)
{
&nbsp;&nbsp;&nbsp;&nbsp;int ret = 0;

&nbsp;&nbsp;&nbsp;&nbsp;if (down_interruptible(&adc_lock))
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return -ERESTARTSYS;
&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;START_ADC_AIN(ain);  //设置ADC的参数


&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//当AD转换结束后会触发adcdone中断,在中断中唤醒adc_wait


&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;wait_event_interruptible(adc_wait,(flag!=0));

&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printk("wake up\n");
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;flag=0;


&nbsp;&nbsp;&nbsp;&nbsp;ret = ADCDAT0 ;

&nbsp;&nbsp;&nbsp;&nbsp;up(&adc_lock);
&nbsp;&nbsp;&nbsp;&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;printk("AIN[%d] = 0x%04x, %d\n", ain, ret, ADCCON & 0x80 ? 1:0);

&nbsp;&nbsp;&nbsp;&nbsp;return (ret & 0x3ff);
}

static int adcdrv_open(struct inode*inode,struct file*filp)
{
&nbsp;&nbsp;&nbsp;&nbsp;printk("open the adc device\n");
&nbsp;&nbsp;&nbsp;&nbsp;return 0;&nbsp;&nbsp;&nbsp;&nbsp;
}

static ssize_t adcdrv_read(struct file* file,char* buffer,size_t count,loff_t* ppos)
{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;int ret=0;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printk("in adcdrv_read function\n");
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;ret=s3c2410_adc_read(6);
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printk("the data is %d\n",ret);
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;copy_to_user(buffer,(char*)&ret,sizeof(int));
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return (sizeof(int));
}


static int __init adcdrv_init(void)
{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if(register_chrdev(IOPORT_MAJOR,"adc",&adcdrv_fops))
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printk("adc driver register failure\n");
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return -1;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;else
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printk("adc driver register success\n");
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if(request_irq(IRQ_ADC_DONE, adcdone_int_handler, SA_INTERRUPT,
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;"ADC", NULL) < 0)
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printk("request irq failure\n");
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return 1;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;else
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printk("request irq success\n");
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;init_MUTEX(&adc_lock);
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;init_waitqueue_head(&adc_wait);
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;ADCTSC = 0;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return 0;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return 0;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}
}

static void __exit adcdrv_exit(void)
{
&nbsp;&nbsp;&nbsp;&nbsp;free_irq(IRQ_ADC_DONE, NULL);
&nbsp;&nbsp;&nbsp;&nbsp;ret=unregister_chrdev(IOPORT_MAJOR,"adc");
&nbsp;&nbsp;&nbsp;&nbsp;if(ret)
&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printk("adc unregister failure\n");
&nbsp;&nbsp;&nbsp;&nbsp;}
&nbsp;&nbsp;&nbsp;&nbsp;else
&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printk("adc unregister success\n");
&nbsp;&nbsp;&nbsp;&nbsp;}
}

module_init(adcdrv_init);
module_exit(adcdrv_exit);

论坛徽章:
0
9 [报告]
发表于 2007-08-07 11:09 |只看该作者
我没有中断处理器,还是使用我我那个定时器的小程序来模拟你的中断.试了一下你的程序,没有发现什么问题.

  1. #include <linux/module.h>
  2. #include <linux/kernel.h>
  3. #include <linux/init.h>
  4. #include <linux/fs.h>
  5. #include <linux/sched.h>
  6. #include <linux/irq.h>
  7. #include <linux/delay.h>

  8. #include <asm/semaphore.h>
  9. #include <asm/uaccess.h>

  10. MODULE_LICENSE("GPL");

  11. static struct semaphore adc_lock;
  12. static  wait_queue_head_t adc_wait;
  13. static int flag=0;

  14. static int Major;
  15. struct timer_list my_timer;

  16. void adcdone_int_handler(unsigned long data)
  17. {
  18.     printk("will wake up\n");
  19.     flag=1;
  20.     wake_up_interruptible(&adc_wait);
  21.     printk("wait it up\n");
  22. }

  23. int s3c2410_adc_read(int ain)
  24. {
  25.     int ret = 0;

  26.     if (down_interruptible(&adc_lock))
  27.         return -ERESTARTSYS;
  28.     printk("waiting...\n");
  29.     wait_event_interruptible(adc_wait,(flag!=0));

  30.     printk("i am waked up\n");
  31.     flag=0;

  32.     up(&adc_lock);

  33.     return (10);
  34. }

  35. static int adcdrv_open(struct inode* inode,struct file* filp)
  36. {
  37.     printk("open the adc device\n");
  38.     return 0;
  39. }

  40. static ssize_t adcdrv_read(struct file* file,char* buffer,size_t count,loff_t* ppos)
  41. {
  42.      int ret=0;
  43.      printk("in adcdrv_read function\n");
  44.      init_timer(&my_timer);
  45.      my_timer.expires = jiffies + 5000;
  46.      my_timer.data = 0;
  47.      my_timer.function = adcdone_int_handler;
  48.      add_timer(&my_timer);

  49.      ret=s3c2410_adc_read(6);
  50.      printk("the data is %d\n",ret);
  51.      del_timer(&my_timer);
  52.      copy_to_user(buffer,(char*)&ret,sizeof(int));
  53.      return (sizeof(int));
  54. }


  55. static struct file_operations adcdrv_fops=
  56. {
  57.      .open=adcdrv_open,
  58.      .read=adcdrv_read
  59. };

  60. static int __init adcdrv_init(void)
  61. {
  62.        if((Major = register_chrdev(0,"adc",&adcdrv_fops)) < 0)
  63.        {
  64.           printk("adc driver register failure\n");
  65.           return -1;
  66.         }
  67.        else
  68.        {
  69.           printk("adc driver register success\n");
  70.                   printk("Major is %d\n", Major);
  71.           init_MUTEX(&adc_lock);
  72.           init_waitqueue_head(&adc_wait);



  73.                   return 0;
  74.       }
  75. }

  76. static void __exit adcdrv_exit(void)
  77. {
  78.     int ret;
  79.     ret=unregister_chrdev(Major,"adc");
  80.     if(ret)
  81.     {
  82.         printk("adc unregister failure\n");
  83.     }
  84.     else
  85.     {
  86.         printk("adc unregister success\n");
  87.     }
  88. }

  89. module_init(adcdrv_init);
  90. module_exit(adcdrv_exit);


复制代码


这个代码是去打开这个程序的代码.
test.c

  1. #include<stdio.h>
  2. #include<stdlib.h>
  3. #include<unistd.h>
  4. #include<fcntl.h>
  5. #include<sys/types.h>

  6. int main()
  7. {
  8.         int fd;
  9.         int i;
  10.         fd = open("abc", O_RDONLY);
  11.         if (fd < 0)
  12.         {
  13.                 printf("error\n");
  14.                 return -1;
  15.         }

  16.         if (read(fd, &i, sizeof(int)) < 0)
  17.         {
  18.                 printf("read error\n");
  19.                 return -1;
  20.         }
  21.         printf("i = %d\n", i);
  22.         return 0;
  23. }

复制代码


运行之后dmesg可以查看到如下结果.

adc driver register success
Major is 253
open the adc device
in adcdrv_read function
waiting...
will wake up
wait it up
i am waked up
the data is 10


这个模拟程序与你那个应该是差不多的. 我这儿运行是没有问题的.

[ 本帖最后由 scutan 于 2007-8-7 11:54 编辑 ]

论坛徽章:
0
10 [报告]
发表于 2007-08-07 15:21 |只看该作者
我把你的程序拿去运行,结果还是不行,只运行到“will wake up”就停住了。
我是在嵌入式linux下运行,内核版本是2.4.18

open the adc device
in adcdrv_read function
waiting...
will wake up
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP