如何编写Linux的驱动程序

如何编写Linux的驱动程序,第1张

}  //IO功能选项,硬件上拉输出  static unsigned int gpio_cfg_table[] = {      S3C2410_GPB5_OUTP,    S3C2410_GPB6_OUTP,    S3C2410_GPB7_OUTP,    S3C2410_GPB8_OUTP, }  //编写一个ioctl函数,这个函数提供给用户端使用(也就是用户态使用)  static int my_ioctl(struct inode *inode,struct file* file,unsigned int cmd,           unsigned long arg) {                 if (arg > 4)        {            return -EINVAL        }         if (cmd == 1) //led ON        {             s3c2410_gpio_setpin(gpio_table[arg],0)            return 0        }         if (cmd == 0) //led OFF        {            s3c2410_gpio_setpin(gpio_table[arg],1)           return 0        }        else        {             return -EINVAL        }  }  //一个和文件设备相关的结构体。  static struct file_operations dev_fops =  {         .owner = THIS_MODULE,        .ioctl = my_ioctl,         //.read  = my_read,   //这个暂时屏蔽,一会我们再加入一个读 *** 作的函数 }  //linux中设备的注册结构体 static struct miscdevice misc = 

{         .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)做一遍。


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/8763830.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-04-20
下一篇 2023-04-20

发表评论

登录后才能评论

评论列表(0条)

保存