嵌入式Linux之我行——S3C2440上 ADC驱动实例开发讲解 .

2019-07-12 16:28发布

  转自http://www.rosoo.net/a/linux/201006/9648.html   TAG: 驱动开发  S3C2440  ADC驱动     嵌入式Linux之我行,主要讲述和总结了本人在学习嵌入式linux中的每个步骤。一为总结经验,二希望能给想入门嵌入式Linux的朋友提 供方便。如有错误之处,谢请指正。
一、开发环境
  • 主  机:VMWare--Fedora 9
  • 开发板:Mini2440--64MB Nand, Kernel:2.6.30.4
  • 编译器:arm-linux-gcc-4.3.2
二、硬件原理分析
S3C2440内部ADC结构图 我们从上面的结构图和数据手册可以知道,该ADC模块总共有8个通道可以进行模拟信号的输入,分别是AIN0、AIN1、AIN2、AIN3、 YM、YP、XM、XP。那么ADC是怎么实现模拟信号到数字信号的转换呢?首先模拟信号从任一通道输入,然后设定寄存器中预分频器的值来确定AD转换器 频率,最后ADC将模拟信号转换为数字信号保存到ADC数据寄存器0中(ADCDAT0),然后ADCDAT0中的数据可以通过中断或查询的方式来访问。 对于ADC的各寄存器的操作和注意事项请参阅数据手册。 上图是mini2440上的ADC应用实例,开发板通过一个10K的电位器(可变电阻)来产生电压模拟信号,然后通过第一个通道(即:AIN0)将 模拟信号输入ADC。 三、实现步骤 ADC设备在Linux中可以看做是简单的字符设备,也可以当做是一混杂设备(misc设备),这里我们就看做是misc设备来实现ADC的驱动。 注意:这里我们获取AD转换后的数据将采用中断的方式,即当AD转换完成后产生AD中断,在中断服务程序中来读取ADCDAT0的第0-9位的值(即AD 转换后的值)。 1、建立驱动程序文件my2440_adc.c,实现驱动的初始化和退出,代码如下:
  1. #include  
  2. #include  
  3. #include  
  4. #include  
  5. #include  
  6. #include  
  7. #include  
  8. #include  
  9. #include  
  10. #include  
  11. #include  
  12.  
  13. /*定义了一个用来保存经过虚拟映射后的内存地址*/ 
  14. static void __iomem *adc_base; 
  15.  
  16. /*保存从平台时钟队列中获取ADC的时钟*/ 
  17. static struct clk *adc_clk; 
  18.  
  19. /*申明并初始化一个信号量ADC_LOCK,对ADC资源进行互斥访问*/ 
  20. DECLARE_MUTEX(ADC_LOCK); 
  21.  
  22. static int __init adc_init(void
  23.     int ret; 
  24.  
  25. /*从平台时钟队列中获取ADC的时钟,这里为什么要取得这个时钟,因为ADC的转换频率跟时钟有关。 
  26. 系统的一些时钟定义在arch/arm/plat-s3c24xx /s3c2410-clock.c中*/ 
  27.     adc_clk = clk_get(NULL, "adc"); 
  28.     if (!adc_clk) 
  29.     { 
  30. /*错误处理*/ 
  31.         printk(KERN_ERR "failed to find adc clock source/n"); 
  32.         return -ENOENT; 
  33.     } 
  34.  
  35. /*时钟获取后要使能后才可以使用,clk_enable定义在arch/arm/plat-s3c/clock.c中*/ 
  36.     clk_enable(adc_clk); 
  37.  
  38. /*将ADC的IO端口占用的这段 IO空间映射到内存的虚拟地址,ioremap定义在io.h中。 
  39. 注意:IO空间要映射后才能使用,以后对虚拟地址的操作就是对IO空间的操作, S3C2410_PA_ADC
  40. 是ADC控制器的基地址,定义在mach-s3c2410/include/mach/map.h中,0x20是虚拟地址长度大小*/ 
  41.     adc_base = ioremap(S3C2410_PA_ADC, 0x20); 
  42.     if (adc_base == NULL) 
  43.     { 
  44. /*错误处理*/ 
  45.         printk(KERN_ERR "Failed to remap register block/n"); 
  46.         ret = -EINVAL; 
  47.         goto err_noclk; 
  48.     } 
  49.  
  50. /*把看ADC注册成为misc设备,misc_register定义在miscdevice.h中adc_miscdev结构体定义 
  51. 及内部接口函数在第②步中讲,MISC_DYNAMIC_MINOR是次设备号,定义在miscdevice.h中*/ 
  52.     ret = misc_register(&adc_miscdev); 
  53.     if (ret) 
  54.     { 
  55. /*错误处理*/ 
  56.         printk(KERN_ERR "cannot register miscdev on minor=%d (%d)/n",
  57. MISC_DYNAMIC_MINOR, ret); 
  58.         goto err_nomap; 
  59.     } 
  60.  
  61.     printk(DEVICE_NAME " initialized!/n"); 
  62.  
  63.     return 0; 
  64.  
  65. //以下是上面错误处理的跳转点 
  66. err_noclk: 
  67.     clk_disable(adc_clk); 
  68.     clk_put(adc_clk); 
  69.  
  70. err_nomap: 
  71.     iounmap(adc_base); 
  72.  
  73.     return ret; 
  74.  
  75. static void __exit adc_exit(void
  76.     free_irq(IRQ_ADC, 1);/*释放中断*/ 
  77.     iounmap(adc_base);/*释放虚拟地址映射空间*/ 
  78.  
  79.     if (adc_clk)/*屏蔽和销毁时钟*/ 
  80.     { 
  81.         clk_disable(adc_clk);     
  82.         clk_put(adc_clk); 
  83.         adc_clk = NULL; 
  84.     } 
  85.  
  86.     misc_deregister(&adc_miscdev);/*注销misc设备*/ 
  87.  
  88. /*导出信号量ADC_LOCK在触摸屏驱动中使用,因为触摸屏驱动和ADC驱动公用 
  89. 相关的寄存器,为了不产生资源竞态,就用信号量来保证资源的互斥访问*/ 
  90. EXPORT_SYMBOL(ADC_LOCK); 
  91.  
  92. module_init(adc_init); 
  93. module_exit(adc_exit); 
  94.  
  95. MODULE_LICENSE("GPL"); 
  96. MODULE_AUTHOR("Huang Gang"); 
  97. MODULE_DESCRIPTION("My2440 ADC Driver"); 
2、adc_miscdev结构体定义及内部各接口函数的实现,代码如下:
  1. #include  
  2.  
  3. /*设备名称*/ 
  4. #define DEVICE_NAME    "my2440_adc" 
  5.  
  6. /*定义并初始化一个等待队列adc_waitq,对ADC资源进行阻塞访问*/ 
  7. static DECLARE_WAIT_QUEUE_HEAD(adc_waitq); 
  8.  
  9. /*用于标识AD转换后的数据是否可以读取,0表示不可读取*/ 
  10. static volatile int ev_adc = 0; 
  11.  
  12. /*用于保存读取的AD转换后的值,该值在ADC中断中读取*/ 
  13. static int adc_data; 
  14.  
  15. /*misc设备结构体实现*/ 
  16. static struct miscdevice adc_miscdev = 
  17.     .minor   = MISC_DYNAMIC_MINOR, /*次设备号,定义在 miscdevice.h中,为255*/ 
  18.     .name    = DEVICE_NAME,        /* 设备名称*/ 
  19.     .fops    = &adc_fops,          /*对ADC设备文件操作*/ 
  20. }; 
  21.  
  22. /*字符设备的相关操作实现*/ 
  23. static struct file_operations adc_fops = 
  24.     .owner    = THIS_MODULE, 
  25.     .open     = adc_open, 
  26.     .read     = adc_read,     
  27.     .release  = adc_release, 
  28. }; 
  29.  
  30. /*ADC设备驱动的打开接口函数*/ 
  31. static int adc_open(struct inode *inode, struct file *file) 
  32.     int ret; 
  33.  
  34. /*申请ADC中断服务,这里使用的是共享中断:IRQF_SHARED,为什么要使用共享中断,因为在触摸屏驱动中 
  35. 也使用了这个中断号。中断服务程序为:adc_irq在下面实现,IRQ_ADC是ADC的中断号,这里注意: 
  36. 申请中断函数的最后一个参数一定不能为NULL,否则中断申请会失败,如果中断服务程序中用不到这个 
  37. 参数,就随便给个值就好了,我这里就给个1*/ 
  38.     ret = request_irq(IRQ_ADC, adc_irq, IRQF_SHARED, DEVICE_NAME, 1); 
  39.     if (ret) 
  40.     { 
  41. /*错误处理*/ 
  42.         printk(KERN_ERR "IRQ%d error %d/n", IRQ_ADC, ret); 
  43.         return -EINVAL; 
  44.     } 
  45.  
  46.     return 0; 
  47.  
  48. /*ADC中断服务程序,该服务程序主要是从ADC数据寄存器中读取AD转换后的值*/ 
  49. static irqreturn_t adc_irq(int irq, void *dev_id) 
  50. /*保证了应用程序读取一次这里就读取 AD转换的值一次, 
  51. 避免应用程序读取一次后发生多次中断多次读取AD转换值*/ 
  52.     if(!ev_adc) 
  53.     { 
  54. /*读取AD转换后的值保存到全局变量adc_data 中,S3C2410_ADCDAT0定义在regs-adc.h中, 
  55. 这里为什么要与上一个0x3ff,很简单,因为AD转换后的数据是保存在ADCDAT0的第0-9位, 
  56. 所以与上0x3ff(即:1111111111)后就得到第0-9位的数据,多余的位就都为0*/ 
  57.         adc_data = readl(adc_base + S3C2410_ADCDAT0) & 0x3ff; 
  58.  
  59. /*将可读标识为1,并唤醒等待队列*/ 
  60.         ev_adc = 1; 
  61.         wake_up_interruptible(&adc_waitq); 
  62.     } 
  63.  
  64.     return IRQ_HANDLED; 
  65.  
  66. /*ADC设备驱动的读接口函数*/ 
  67. static ssize_t adc_read(struct file *filp, char *buffer, size_t count, loff_t *ppos) 
  68. /*试着获取信号量(即:加锁)*/ 
  69.     if (down_trylock(&ADC_LOCK)) 
  70.     { 
  71.         return -EBUSY; 
  72.     } 
  73.  
  74.     if(!ev_adc)/*表示还没有AD转换后的数据,不可读取*/ 
  75.     { 
  76.         if(filp->f_flags & O_NONBLOCK) 
  77.         { 
  78. /*应用程序若采用非阻塞方式读取则返回错误*/ 
  79.             return -EAGAIN; 
  80.         } 
  81.         else/*以阻塞方式进行读取*/ 
  82.         { 
  83. /*设置ADC控制寄存器,开启AD转换*/ 
  84.             start_adc(); 
  85.  
  86. /*使等待队列进入睡眠*/ 
  87.             wait_event_interruptible(adc_waitq, ev_adc); 
  88.         } 
  89.     } 
  90.  
  91. /*能到这里就表示已有AD转换后的数据,则标识清0,给下一次读做判断用*/ 
  92.     ev_adc = 0; 
  93.  
  94. /*将读取到的AD转换后的值发往到上层应用程序*/ 
  95.     copy_to_user(buffer, (char *)&adc_data, sizeof(adc_data)); 
  96.  
  97. /*释放获取的信号量(即:解锁)*/ 
  98.     up(&ADC_LOCK); 
  99.  
  100.     return sizeof(adc_data); 
  101.  
  102. /*设置ADC控制寄存器,开启AD转换*/ 
  103. static void start_adc(void
  104.     unsigned int tmp; 
  105.  
  106.     tmp = (1 << 14) | (255 << 6) | (0 << 3);/* 0 1 00000011 000 0 0 0 */ 
  107.     writel(tmp, adc_base + S3C2410_ADCCON); /*AD预分频器使能、模拟输入通道设为AIN0*/ 
  108.  
  109.     tmp = readl(adc_base + S3C2410_ADCCON); 
  110.     tmp = tmp | (1 << 0);                 /* 0 1 00000011 000 0 0 1 */ 
  111.     writel(tmp, adc_base + S3C2410_ADCCON); /*AD转换开始*/ 
  112.  
  113. /*ADC设备驱动的关闭接口函数*/ 
  114. static int adc_release(struct inode *inode, struct file *filp) 
  115.     return 0; 
注意:在上面实现的每步中,为了让代码逻辑更加有条理和容易理解,就没有考虑代码的顺序,比如函数要先定义后调用。如果要编译此代码,请严格按照C 语言的规范来调整代码的顺序。 3、编写用户应用程序测试my2440_adc驱动。建立应用程序adc_test.c,代码如下:
  1. #include  
  2. #include  
  3. #include  
  4.  
  5. int main(int argc, char **argv) 
  6.     int fd; 
  7.  
  8.     //以阻塞方式打开设备文件,非阻塞时flags=O_NONBLOCK 
  9.     fd = open("/dev/my2440_adc", 0); 
  10.  
  11.     if(fd < 0) 
  12.     { 
  13.         printf("Open ADC Device Faild!/n"); 
  14.         exit(1); 
  15.     } 
  16.  
  17.     while(1) 
  18.     { 
  19.         int ret; 
  20.         int data; 
  21.          
  22.         //读设备 
  23.         ret = read(fd, &data, sizeof(data)); 
  24.  
  25.         if(ret != sizeof(data)) 
  26.         { 
  27.             if(errno != EAGAIN) 
  28.             { 
  29.                 printf("Read ADC Device Faild!/n"); 
  30.             } 
  31.  
  32.             continue
  33.         } 
  34.         else 
  35.         { 
  36.             printf("Read ADC value is: %d/n", data); 
  37.         } 
  38.     } 
  39.     close(fd); 
  40.     return 0; 
4、将驱动程序下载挂载到内核,下载应用程序到开发板上后,运行应用程序,扭动mini2440开发板上的定位器,可以观察到ADC转换值的变化, 证明驱动程序工作正常。效果图如下: