LinuxLinux设备驱动开发设备驱动开发(LinuxIDC.com)Ubuntu,Fedora,SUSEITLinuxLinuxLinuxIDC.com2006925LinuxIDCLinuxIDCLinuxLinuxIDC.comUbuntuFedoraSUSEITLinuxGoogle-LinuxLinuxLinuxIDC.comLinuxUbuntuFedoraRedHatSUSELinuxAndroidLinux----LinuxIDC[6688.CC]Copyright©2006-2011LinuxAllrightsreserved(LinuxIDC.com)Ubuntu,Fedora,SUSEITLinux 今天的内容Linux设备驱动的现状从nonos驱动到Linux驱动内核设施自旋锁、信号量、互斥量、完成量异步通知、信号阻塞与非阻塞内存与I/O操作,DMA中断,top half/bottom half字符设备驱动复杂设备驱动的框架LCD设备FRAMEBUFFERFLASH设备MTDTTY设备块设备用户空间的设备驱动设备驱动开发流程开发环境建设调试手段用户空间测试设备驱动的学习方法(LinuxIDC.com)Ubuntu,Fedora,SUSEITLinux Linux设备驱动的现状高需求Linux内核的绝大多数代码为设备驱动新设备、新芯片、新驱动的需求高门槛涉及到大量硬件操作涉及到内核基础知识涉及到并发控制与同步复杂的软件结构框架高回报(LinuxIDC.com)Ubuntu,Fedora,SUSEITLinux 从nonos驱动到Linux驱动nonos驱动单刀直入简单直接提供APILinux驱动兵团战役复杂间接提供API应用软件SerialSendSerialRecv设备驱动LightOnLightOffFlashWrFlashRd硬件串口LEDFLASH硬件操作系统API操作系统驱动中独立于设备的接口驱动中的硬件操作用户应用程序nonos驱动与应用onos驱动与应用(LinuxIDC.com)Ubuntu,Fedora,SUSEITLinux 并发和竞态并发和竞态:对称多处理器(SMP)的多个CPU单CPU内进程与抢占它的进程中断(硬中断、软中断、Tasklet、底半部)与进程之间处理思路:lock() //锁定,拿虎符. . .critical section //临界区,调动军队. . .unlock() //解锁定,归还虎符常用方法:中断屏蔽原子操作自旋锁信号量互斥体(LinuxIDC.com)Ubuntu,Fedora,SUSEITLinux 原子变量接口•整型原子操作• 设置原子变量的值•void atomic_set(atomic_t *v, int i); //设置原子变量的值为i•atomic_t v = ATOMIC_INIT(0); //定义原子变量v并初始化为0•̅获取原子变量的值•atomic_read(atomic_t *v); //返回原子变量的值• 原子变量加/减•void atomic_add(int i, atomic_t *v); //原子变量增加i•void atomic_sub(int i, atomic_t *v); //原子变量减少i•ξ原子变量自增/自减•void atomic_inc(atomic_t *v); //原子变量增加1•void atomic_dec(atomic_t *v); //原子变量减少1•η操作并测试•int atomic_inc_and_test(atomic_t *v);•int atomic_dec_and_test(atomic_t *v);•int atomic_sub_and_test(int i, atomic_t *v);•⎝操作并返回•int atomic_add_return(int i, atomic_t *v);•int atomic_sub_return(int i, atomic_t *v);•int atomic_inc_return(atomic_t *v);•int atomic_dec_return(atomic_t *v);•位原子操作•⎝设置/清除/反转位•void set_bit(nr, void *addr);•void clear_bit(nr, void *addr);•void change_bit(nr, void *addr);• 测试位•test_bit(nr, void *addr);•η测试并操作位•int test_and_set_bit(nr, void *addr);•int test_and_clear_bit(nr, void *addr);•int test_and_change_bit(nr, void *addr);(LinuxIDC.com)Ubuntu,Fedora,SUSEITLinux 自旋锁VS 信号量自旋锁:忙等待,无调度开销进程抢占被禁止锁定期间不能睡觉•spinlock_t lock;•spin_lock_init(&lock);•spin_lock (&lock) ; //获取自旋锁,保护临界区•. . ./ /临界区•spin_unlock (&lock) ; //解锁信号量拿不到就切换进程,有调度开销锁定期间可以睡觉,不用于中断上下文•//定义信号量•DECLARE_MUTEX(mount_sem);•down(&mount_sem);//获取信号量,保护临界区•. . .•critical section //临界区•. . .•up(&mount_sem);//释放信号量(LinuxIDC.com)Ubuntu,Fedora,SUSEITLinux 设备访问方式读写函数读写读写系统调用资源不可获得资源可获得阻塞阻塞I/O返回返回用户空间poll()或select()poll系统调用资源状态变更非阻塞轮询I/O返回xxx_func()唤醒读写yyy_func()中断唤醒资源可获得资源状态变更唤醒读写函数阻塞读写系统调用返回返回读写函数读写系统调用资源可获得signal信号处理函数(进行读写)异步通知系统调用内核空间中断signal(LinuxIDC.com)Ubuntu,Fedora,SUSEITLinux 阻塞非阻塞等待队列:进程等待被唤醒的一种机制阻塞与非阻塞使用模板1 static ssize_t xxx_write(struct file *file, const char *buffer, size_t count,2 loff_t *ppos)3 {4 ...5 DECLARE_WAITQUEUE(wait, current); //定义等待队列6 add_wait_queue(&xxx_wait, &wait); //添加等待队列7 8 ret = count;9 /* 等待设备缓冲区可写*/10 do11 {12 avail = device_writable(...);13 if (avail 0)14 __set_current_state(TASK_INTERRUPTIBLE);//改变进程状态15 16 if (avail 0)17 {18 if (filef_flags &O_NONBLOCK) //非阻塞19 {20 if (!ret)21 ret = EAGAIN;22 goto out;23 }24 schedule(); //调度其他进程执行25 if (signal_pending(current))//如果是因为信号唤醒26 {27 if (!ret)28 ret = ERESTARTSYS;29 goto out;30 }31 }32 }while (avail 0);33 34 /* 写设备缓冲区*/35 device_write(...)36 out:37 remove_wait_queue(&xxx_wait, &wait);//将等待队列移出等待队列头38 set_current_state(TASK_RUNNING);//设置进程状态为TASK_RUNNING39 return ret;40 } (LinuxIDC.com)Ubuntu,Fedora,SUSEITLinux polling驱动中POLL模板•1 static unsigned int xxx_poll(struct file *filp, poll_table *wait)•2 {•3 unsigned int mask = 0;•4 struct xxx_dev *dev = filpprivate_data; /*获得设备结构体指针*/•6 ... •8 poll_wait(filp, &devwait, wait);•9 •10 if (...)//可读•11 {•12 mask |= POLLIN | POLLRDNORM; /*标示数据可获得*/•13 } •15 if (...)//可写•16 {•17 mask |= POLLOUT | POLLWRNORM; /*标示数据可写入*/•18 }•19 •20 ...•21 return mask;•22 }用户空间POLL模板•fd_set fds; •FD_ZERO(&fds);•FD_SET(fd, &fds);•select(fd + 1, &rfds, &wfds, NULL, NULL);•if (FD_ISSET(fd, &fds))•{• printf(Poll monitor:can be access\n);•}(LinuxIDC.com)Ubuntu,Fedora,SUSEITLinux 异步I/O信号:软件意义上的“中断”驱动发出信号•kill_fasync(&devasync_queue, SIGIO, POLL_IN);用户空间应用程序处理信号•24 signal(SIGIO, input_handler);•25 fcntl(STDIN_FILENO, F_SETOWN, getpid());•26 oflags = fcntl(STDIN_FILENO, F_GETFL);•27 fcntl(STDIN_FILENO, F_SETFL, oflags | FASYNC);•8 void input_handler(int num)•9 { ...•14 len = read(STDIN_FILENO, &data, MAX_LEN);•15 ... }fcntl(fd, F_SETOWN,