当FIFO中有数据时,读进程才能把数据读出,而且读取数据后会从globalfifo中拿掉;只有当globalfifo不是满的时,写进程才能往FIFO中写数据
.// globalfifo设备结构体
struct globalfifo_dev {
struct cdev cdev;
unsigned int current_len;
unsigned char mem [GLOBALFIFO_SIZE];
struct mutex mutex;
wait_queue_head_t r_wait;
wait_queue_head_t w_wait;
}核心
结构的变量,其定义:
linux-2.6.22 / include / linux / cdev.h
struct cdev {
struct kobject k,//每个cdev都是一个kobject对象
struct module * owner,指向实现驱动的模块
const struct file_operation * ops,//操纵这个字符设备的方法
struct list_head list,//与cdev对应的字符设备文件的inode-> i_devices的链表头
dev_t dev; //起始设备编号
unsigned int count,//设备范围号大小
};
一个CDEV一般有两种方式初始化:静态的和动态的:1静态内存定义初始化结构的CDEV我_CDEV; cdev _ init(&my _ cdev,&FOPS); my _ cdev.owner = THIS _ MODULE; 2.动态内存定义初始化:struct cdev * my _ cdev = cdev _ alloc(); my _ cdev-> ops =&fops; my _ cdev-> owner = THIS _ MODULE; 使用两种方式的功能是一样的,只是使用的内存区不一样,一般视实际的数据结构需求而定.struct cdev * cdev _ alloc(void){struct cdev * p = kzalloc(sizeof(struct cdev) ,FGP _ KERNEL); if(p){INIT _ LIST _ HEAD(对 - >列表); kobject _ init(对 - > Kobj专有,&ktype _ cdev _ default,_ cdev _ dynamic); }返回磷; } void cdev _ init(struct cdev * cdev,const struct file _ operation * * fops){memset的(CDEV,0,的sizeof(* CDEV)); INIT _ LIST _ HEAD(cdev->列表); kobject _ init(&cdev-> kobj,&ktype _ cdev _ default); cdev-> ops = fops; }初始化CDEV后,需要把它添加到系统中去。可以调用cdev_add()函数。传入的cdev结构指针,起始设备编号,以及设备编号范围.int cdev _
add(struct cdev * p,dev _ t dev,unsigned count){p-> dev = dev; p-> count = count; return kobj _ map(cdev _ map,dev,count,NULL,exact _ match,exact _ lock,p); }内核中所有字符设备都会记录在一个kobj _ map结构的cdev _ map变量中。这个结构的变量中包含一个散列表用来快速存取所有的对象.kobj _ map()函数就是用来把字符设备编号和CDEV结构变量一起保存到cdev _ map这个散列表中。当后续要打开一个字符设备文件时,通过调用kobj _ lookup()函数,根据设备编号就可以找到的cdev结构变量,从而取出其中的OPS字段。当一个字符设备不在需要时,用cdev _ dev()函数来释放CDEV占用的内存void cdev _ dev(struct cdev * p){cdev _ unmap(p-> cdev,p-> count) ; konject _ put(对 - > Kobj专有); }当前_ LEN成员用以表征目前FIFO中所有有效数据的长度,当前_ LEN = 0,意味着FIFO为空,当前_ LEN =的globalfifo _ SIZE意味着FIFO满[这里写链接内容](HTTPS: // blog .youkuaiyun.com / yusiguyuan / article / details / 11354467)