{ .minor = MISC_DYNAMIC_MINOR, .name = DEVICE_NAME, .fops = &dev_fops, } //设备初始化(包括注册)函数 static int __init dev_init(void) { int ret int i for (i=0i<4i++) { s3c2410_gpio_cfgpin(gpio_table[i],gpio_cfg_table[i]) s3c2410_gpio_setpin(gpio_table[i],0) mdelay(500) s3c2410_gpio_setpin(gpio_table[i],1) } ret = misc_register(&misc) printk(DEVICE_NAME"MY_LED_DRIVER init ok\n") return ret } //设备注销函数 static void __exit dev_exit(void) { misc_deregister(&misc) } //与模块相关的函数 module_init(dev_init) module_exit(dev_exit) MODULE_LICENSE("GPL") MODULE_AUTHOR("blog.ednchina.com/itspy")
MODULE_DESCRIPTION("MY LED DRIVER") 到此,上面就完成了一个简单的驱动(别急,下面我们再会稍微增加点复杂的东西),以上代码的可以简单概括为:像自己写51单片机或者ARM的裸奔程序一样 *** 作IO函数,然后再linux系统中进行相关必须的函数关联和注册。 为什么要关联呢,为什么注册呢? 因为这是必须的,从以下这些结构体就知道了。 stuct file_operations{ struct module *owner loff_t (*llseek) (struct file *, loff_t, int) ssize_t (*read) (struct file *, char __user *, size_t, loff_t *) ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *) ssize_t (*aio_read) (struct kiocb *, const struct iovec *, unsigned long, loff_t) ssize_t (*aio_write) (struct kiocb *, const struct iovec *, unsigned long, loff_t) int (*readdir) (struct file *, void *, filldir_t)
unsigned int (*poll) (struct file *, struct poll_table_struct *) int (*ioctl) (struct inode *, struct file *, unsigned int, unsigned long) long (*unlocked_ioctl) (struct file *, unsigned int, unsigned long) … } file_operations 结构体中包括了很多与设备相关的函数指针,指向了驱动所提供的函数。 struct inode{ struct hlist_node i_hash struct list_head i_list struct list_head i_sb_list struct list_head i_dentry unsigned long i_ino atomic_t i_count unsigned int i_nlink uid_t i_uid gid_t i_gid dev_t i_rdev u64 i_version loff_t i_size … } inode 是 UNIX *** 作系统中的一种数据结构,它包含了与文件系统中各个文件相关的一些重要信息。在 UNIX 中创建文件系统时,同时将会创建大量的 inode 。通常,文件系统磁盘空间中大约百分之一空间分配给了 inode 表。 大略了解以上信息之后,我们只需把我们所要实现的功能和结构体关联起来。上例中已经完成IO写 *** 作的函数,现在我们再添加一个读的函数。基于这种原理,我们想实现各种功能的驱动也就很简单了。 //添加读函数示意, 用户层可以通过 read函数来 *** 作。 static int my_read(struct file* fp, char __user *dat,size_t cnt) { size_t i printk("now read the hardware...\n") for(i=0i<cnti++) dat[i] = 'A' dat[i] = '\0' return cnt } 这样,完成驱动编写。编译之后,本驱动可以通过直接嵌入内核中,也可以以模块的嵌入的形式加载到linux内核中去。 完成了驱动,写个应用程序了验证一下吧: int main(int argc,char ** argv) {
int on int led_no int fd char str[10] int cnt =0 fd = open("/dev/MY_LED_DRIVER",0) if (fd < 0) { printf("can't open dev\n") exit(1) } printf("read process\n") cnt = read(fd,str,10) printf("get data from driver:\n%s\ncount = %d\n",str,cnt) printf("read process end \n") cnt = 0 printf("running...\n") while(cnt++<1000) { ioctl(fd,0,0) //led off ioctl(fd,0,1) ioctl(fd,0,2) ioctl(fd,0,3) sleep(1) //printf("sdfdsfdsfdsfds...\n") ioctl(fd,1,0) //led on ioctl(fd,1,1) ioctl(fd,1,2) ioctl(fd,1,3) sleep(1) printf("%d\b",cnt) } close(fd) return 0 }
如何驱动业务流程学习了解linux驱动程序技巧学习的要领很主要,学习linux *** 作系统时,你可能会遇到关于驱动方面的疑问,这里将介绍学习linux驱动程序的要领,在这里拿出来和大家分享一下。
1.学会写基本的makefile
2.编一使用程序,可以用makefile跑起来
3.学会写linux驱动程序的makefile
4.写一基本char驱动,makefile编译通过,可以insmod,lsmod,rmmod.在驱动程序的init函数里打印helloworld,insmod后应该能够通过dmesg看到输出。
5.写一完整驱a动,加上read,write,ioctl,polling等各种函数的驱动实现。在ioctl里完成从用户空间向内核空间传递结构体的实现。
6.写一block驱动程序加上read,write,ioctl,poll等各种函数实现。
7.基本学习下内存管理,这个是最难的,明白各种memoryalloc的函数实现细节。这是linux开发的基本功。
8.学习锁机制的使用,这个不是最难的但是最容易犯错的,涉及到很多同步和并发的疑问。
9.看内核中实际使用的linux驱动程序代码。你会发觉最基本的你已经知道了,大的框架都是一样的,无非是read,write,ioctl等函数的实现,但里面包含了很多很多细小的实现细节是之前不知道的。这时候就要考虑到很多别的疑问而不仅仅是基本功能的实现。
推选您看2.6.20中integrated的一个驱动kvm,记得是在driver/lguest下,很好玩的,就是linux下的虚拟机驱动程序,代码不长,但功能强大。有能力的可以自己写一 *** 作系统按照要求做成硬盘镜像加载到虚拟机中,然后客户机可以有自己的4G虚拟地址空间。
10.看完linux驱动程序代码欢迎您进入Linuxkernel学习中来。
最基本的要领,跟着ldd(linuxdevivedriver)做一遍。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)