linux嵌入式系统驱动程序的阻塞与异步

时间:2022-02-18 11:04:39

对于那些需要进程独占的设备,需要使用linux提供的阻塞编程。步骤如下:

1.在设备驱动程序中定义该设备的进程等待列多,并将其初始化

static wait_queue_head_t wait_queue;

init_waitqueue_head(&wait_queue);

2.在设备驱动程序的读操作中,调用函数wait_event 实现阻塞访问

int mixled_read(struct file * filp,char__user * buffer,size_t size,loff_t * ppos)

{

wait_event(wait_queue,Elmixled_dev->full_flag!=0);

elmixled_dev->full_flag = 0;

if(copy_to_user(buffer,Elmixled_dev->buf,read_lea))

{

printk(“copy to user err!”);

}

return lead_len;

}

在写操作中,调用wake_up函数唤醒该设备等待进程列队上的进程

int mixled_write(struct file * filp, const char__user * buffer,size_t size,loff_t * ppos)

{

int write_len =size;

if(copy_from_user(Elmixled_dev->buf,buffer,write_len))

{

printk(“copy from user err”);

}

elmixled_dev->full_flag = 1;

wake_up(&wait_queue);

return write_len;

}

对于异步驱动编程,需要做的步骤如下:

1.定义设备驱动程序的异步通知列队

struct fasync_struct * fasync_queue;

实现设备驱动程序的异步操作函数fasync,并在异步操作函数fasync中调用fasync_helper函数将当前进程添加到设备驱动程序的异步通知列队

int mixled_fasync(int fd,struct file * file ,int on)

{

printk(“enter mixled_fasync function”);

return fasync_helper(fd,file,on,&Elmixled_dev->fasync_queue);

}

struct file_operations mixled_ops={

.fasync = mixled_fasync

}

3.在设备驱动程序中,当检测到设备状态信息发生变化时,如数据到达或者按键被按下就需要通知应用程序,调用函数kill_fasync想应用程序发送异步通知消息

irqeturn_t mixled_isr(int irq,void * dev_id)

{

printk(“key interrupt take place”);

if(Elmixled_dev-》fasync_queue)

{

kill_fasync(&Elmixled_dev->fasync_queue,SIGIO,POLLMSG)

}

return IRQ_HANDLED;

}

4.在设备驱动的close函数中,调用mixled_fasync函数将当前进程从设备的异步通知列队中移除

mixled_fasync(-1,file,0)