linux驱动-poll使用笔记

前言

一个项目中使用了赛灵思的FPGA,需要fpga这边和arm这边进行数据通讯,通讯方式使用的是一段fpga和arm共享的ddr内存,把这块内存做了一个fifo,并通过中断出发,我在arm这边实现一个驱动来接收处理中断,然后读取fifo.

驱动的结构体

c 复制代码
struct ddr_fifo_dev {
	int 			index;
	dev_t 			devid;
	struct cdev		cdev;
	struct miscdevice miscdev;
	struct device * device;
	struct device_node * nd;
	char 			name[64];
	struct fasync_struct * async_queue;
	wait_queue_head_t r_wait;
	struct file  *	file;
	int     		openflag;
	int    			irq_id;
	void __iomem  * regbase;
	phys_addr_t 	physreg;
	size_t			physreglen;

	phys_addr_t 	rxfifostart;
	size_t			rxfifolen;

	phys_addr_t 	txfifostart;
	size_t			txfifolen;

	atomic_t    	irq_flag;
};

poll处理函数

c 复制代码
static unsigned int ddr_fifo_poll(struct file * file, struct poll_table_struct * poll_table)
{
	unsigned int mask = 0;
	struct ddr_fifo_dev * df_devdata = 
		container_of(file->private_data, struct ddr_fifo_dev, miscdev);
	//添加这个文件结构到等待队列
	poll_wait(file, &df_devdata->r_wait, poll_table);
	if (atomic_read(&df_devdata->irq_flag)==0) {
		//没有数据,这个接口返回0,等待内核的下次调用
		mask = 0;
	}else{
		//有数据,返回POLLIN | POLLRDNORM后,调用poll的进程会被内核唤醒
		atomic_dec(&df_devdata->irq_flag);
		mask = POLLIN | POLLRDNORM;
		dev_dbg(df_devdata->device, "POLLIN irq_flag=%d\n",atomic_read(&df_devdata->irq_flag));
	}
	return mask;
}

中断处理函数

  1. 每次进入中断后,操作寄存器清除中断标记
  2. 唤醒等待队列上的进程
  3. 原子计数加一
c 复制代码
static irqreturn_t irq_interrupt(int irq, void * dev_id)
{
	struct ddr_fifo_dev * df_devdata = (struct ddr_fifo_dev *)dev_id;
	ddr_rxfifo_irq_clear(df_devdata->regbase);
	wake_up_interruptible(&df_devdata->r_wait);
	atomic_inc(&df_devdata->irq_flag);
	return IRQ_HANDLED;
}
相关推荐
迎風吹頭髮1 小时前
Linux内核架构浅谈8-Linux内核与UNIX的传承:设计思想与特性差异
linux·运维·架构
黑马金牌编程2 小时前
Linux 服务器常见的性能调优
linux·运维·服务器·性能优化
jieyu11192 小时前
网络、主机安全扫描工具
linux·安全·系统安全
tianyuanwo2 小时前
Linux进程管理中的T状态问题分析与解决体系
linux·运维·进程管理·t状态
汇能感知2 小时前
光谱相机的探测器阵列
经验分享·笔记·科技
CHHC18802 小时前
vSIM / SoftSIM笔记
笔记
liuyao_xianhui3 小时前
Linux_基本指令1
linux·运维·服务器
守望时空333 小时前
Linux挂载NTFS分区指南
linux
逆小舟4 小时前
【C/C++】指针
c语言·c++·笔记·学习
shan~~4 小时前
linux达梦数据库操作
linux·数据库·chrome