zoukankan      html  css  js  c++  java
  • poll机制

    poll机制的作用:使阻塞型函数超时返回,避免一直阻塞。

       

     

    实例:

    driver.c

    1 #include <linux/module.h>
    2 #include <linux/kernel.h>
    3 #include <linux/fs.h>
    4 #include <linux/init.h>
    5 #include <linux/delay.h>
    6 #include <linux/irq.h>
    7 #include <asm/uaccess.h>
    8 #include <asm/irq.h>
    9 #include <asm/io.h>
    10 #include <asm/arch/regs-gpio.h>
    11 #include <asm/hardware.h>
    12 #include <linux/poll.h>
    13
    14
    15 static int major;
    16
    17 static struct class *myKey_class;
    18 static struct class_device *myKey_class_dev;
    19
    20 volatile unsigned long *gpfcon;
    21 volatile unsigned long *gpfdat;
    22
    23 volatile unsigned long *gpgcon;
    24 volatile unsigned long *gpgdat;
    25
    26 static DECLARE_WAIT_QUEUE_HEAD(button_waitq);
    27 static volatile int ev_press = 0;
    28
    29 //normal:1; press:0;
    30 static unsigned char keyVal = 0;
    31
    32 struct pin_desc {
    33         unsigned int pin;
    34         unsigned int key_val;
    35 };
    36
    37 /*
    38 * 按键按下键值为0x01,...; 松开键值为0x81,...
    39 */
    40 struct pin_desc pins_desc[3] = {
    41         {S3C2410_GPF0, 0x01},
    42         {S3C2410_GPF2, 0x02},        
    43         {S3C2410_GPG11, 0x03},        
    44 };
    45
    46
    47 static int myKey_open(struct inode *inode, struct file *file);
    48 static int myKey_close(struct inode *inode, struct file *file);
    49 static ssize_t myKey_read(struct file *file, char __user *buf, size_t size, loff_t *ppos);
    50 static unsigned int myKey_poll(struct file *file, struct poll_table_struct *poll_table_struct);
    51
    52
    53 static struct file_operations myKey_fops = {
    54         .open = myKey_open,
    55         .read = myKey_read,
    56         .owner = THIS_MODULE,
    57         .release = myKey_close,
    58         .poll = myKey_poll,
    59 };
    60
    61
    62 static irqreturn_t handle_buttons(int irq, void *dev)
    63 {
    64         unsigned int kval;
    65         struct pin_desc *pinDesc = dev;
    66
    67         kval = s3c2410_gpio_getpin(pinDesc->pin);
    68         if (kval)        //松开
    69         {
    70                 keyVal = 0x80 | pinDesc->key_val;
    71         }
    72         else {                //按下
    73                 keyVal = pinDesc->key_val;
    74         }
    75
    76         //唤醒休眠进程
    77         ev_press = 1;        //中断发生标志
    78         wake_up_interruptible(&button_waitq);
    79
    80         return IRQ_RETVAL(IRQ_HANDLED);
    81 }
    82
    83
    84 static int myKey_open(struct inode *inode, struct file *file)
    85 {
    86         request_irq(IRQ_EINT0, handle_buttons, IRQT_BOTHEDGE, "S2", &pins_desc[0]);
    87         request_irq(IRQ_EINT2, handle_buttons, IRQT_BOTHEDGE, "S3", &pins_desc[1]);
    88         request_irq(IRQ_EINT19, handle_buttons, IRQT_BOTHEDGE, "S5", &pins_desc[2]);
    89         
    90         return 0;
    91 }
    92
    93
    94 static int myKey_close(struct inode *inode, struct file *file)
    95 {
    96         free_irq(IRQ_EINT0, &pins_desc[0]);
    97         free_irq(IRQ_EINT2, &pins_desc[1]);
    98         free_irq(IRQ_EINT19, &pins_desc[2]);
    99
    100         return 0;
    101 }
    102
    103
    104 static unsigned int myKey_poll(struct file *file, struct poll_table_struct *wait)
    105 {
    106         unsigned int mask = 0;
    107
    108         poll_wait(file, &button_waitq, wait);        //不会立即休眠
    109
    110         if (ev_press)
    111         {
    112                 mask |= POLLIN | POLLRDNORM;
    113         }
    114
    115         return mask;
    116 }
    117
    118
    119 static ssize_t myKey_read(struct file *file, char __user *buf, size_t size, loff_t *ppos)
    120 {
    121         //无中断进入休眠
    122         wait_event_interruptible(button_waitq, ev_press);
    123
    124         ev_press = 0;        //清除中断发生标志
    125         copy_to_user(buf, &keyVal, 1);
    126         return 0;
    127 }
    128
    129 static int __init myKey_init(void)
    130 {
    131         /* 物理地址映射成虚拟地址 */
    132         gpfcon = (volatile unsigned long*)ioremap(0x56000050, 16);
    133         gpfdat = gpfcon + 1;
    134
    135         gpgcon = (volatile unsigned long*)ioremap(0x56000060, 16);
    136         gpgdat = gpgcon + 1;
    137
    138         major = register_chrdev(0, "myKey", &myKey_fops);
    139         
    140         myKey_class = class_create(THIS_MODULE, "myKeyclass");
    141         myKey_class_dev = class_device_create(myKey_class, NULL, MKDEV(major, 0), NULL, "myKey");
    142
    143         return 0;
    144 }
    145
    146 static void __exit myKey_exit(void)
    147 {
    148         /* 释放虚拟地址映射 */
    149         iounmap(0x56000050);
    150         iounmap(0x56000060);
    151
    152         unregister_chrdev(major, "myKey");
    153
    154         class_device_unregister(myKey_class_dev);
    155         class_destroy(myKey_class);
    156         return;
    157 }
    158
    159 module_init(myKey_init);
    160 module_exit(myKey_exit);

       

    app.c

    1 #include <sys/types.h>
    2 #include <sys/stat.h>
    3 #include <fcntl.h>
    4 #include <stdio.h>
    5 #include <poll.h>
    6
    7 int main (void)
    8 {
    9         int fd;
    10         unsigned char keyVal;
    11         int ret;
    12         struct pollfd fds[1];
    13         
    14         printf("test app! ");
    15
    16         fd = open("/dev/myKey", O_RDWR);
    17         if(fd < 0)
    18         {
    19                 printf("open failed! %d ", fd);
    20                 return -1;
    21         }
    22
    23         fds[0].fd = fd;
    24         fds[0].events = POLLIN;
    25         
    26         while(1)
    27         {
    28                 ret = poll(fds, 1, 5000);
    29                 if (!ret)
    30                 {
    31                         printf("time out(5s) ");
    32                 }
    33                 else {
    34                         read(fd, &keyVal, 1);
    35                         printf("keyVal: 0x%x ", keyVal);
    36                 }
    37         }
    38         return 0;        
    39 }

       

    Makefile

    1 KERN_DIR = /work/system/linux-2.6.22.6
    2
    3 all:
    4         make -C $(KERN_DIR) M=`pwd` modules
    5

    6 clean:
    7         make -C $(KERN_DIR) M=`pwd` modules clean
    8
            rm -rf modules.order

    9
    10 obj-m += myKey_pollmcn.o

  • 相关阅读:
    iptables操作指令
    最怕问初衷,大梦成空
    kettle官网wiki
    Pentaho Spoon (Kettle) 出现 Timestamp :Unable to get timestamp from resultset at index 30 错误的解决
    win10安装vs2013失败解决方法
    2020 最新 kail 安装教程
    01_docker学习笔记(一)
    数据结构的基础概念
    BOS3
    配置文件添加druid控制台
  • 原文地址:https://www.cnblogs.com/lilto/p/11878055.html
Copyright © 2011-2022 走看看