mini2440 ADC可调电阻驱动程序开发源代码(杂项设备驱动框架)

发布者:SereneMeadow7最新更新时间:2024-06-20 来源: elecfans关键字:mini2440  ADC  可调电阻 手机看文章 扫描二维码
随时随地手机看文章

/*********************************************************/

/****s3c2440 ADC可调电阻驱动程序开发源代码(杂项设备驱动框架)****/

/********************************************************/

#include

#include

#include

 

#define DEVICE_NAME    'adc_driver'  /*设备名称*/

static void __iomem *adc_base;  /*定义了一个用来保存经过虚拟映射后的内存地址 */
static struct clk *adc_clk;    /*保存从平台时钟队列中获取ADC的时钟 */


DECLARE_MUTEX(ADC_LOCK);  /*申明并初始化一个信号量ADC_LOCK,对ADC资源进行互斥访问*/
static DECLARE_WAIT_QUEUE_HEAD(adc_waitq); /*定义并初始化一个等待队列adc_waitq,对ADC资源进行阻塞访问 */

 
static volatile int ev_adc = 0;  ///*用于标识AD转换后的数据是否可以读取,0表示不可读取 */
static int adc_data; /*用于保存读取的AD转换后的值,该值在ADC中断中读取*/

/*ADC中断服务程序,该服务程序主要是从ADC数据寄存器中读取AD转换后的值*/
static irqreturn_t adc_irq(int irq, void *dev_id)
{
 /*保证了应用程序读取一次这里就读取 AD转换的值一次,
 避免应用程序读取一次后发生多次中断多次读取AD转换值*/
 if(!ev_adc)
 {
 /*读取AD转换后的值保存到全局变量adc_data 中,S3C2410_ADCDAT0定义在regs-adc.h中,
 这里为什么要与上一个0x3ff,很简单,因为AD转换后的数据是保存在ADCDAT0的第0-9位,
 所以与上0x3ff(即:1111111111)后就得到第0-9位的数据,多余的位就都为0*/
  adc_data = readl(adc_base + S3C2410_ADCDAT0) & 0x3ff;
  ev_adc = 1; //将可读标识为1,并唤醒等待队列
  wake_up_interruptible(&adc_waitq);
 }
    return IRQ_HANDLED;
}

/*ADC设备驱动的打开接口函数*/
static int adc_open(struct inode *inode, struct file *file)
{
 int ret;

 /*申请ADC中断服务,这里使用的是共享中断:IRQF_SHARED,为什么要使用共享中断,因为在触摸屏驱动中
 也使用了这个中断号。中断服务程序为:adc_irq在下面实现,IRQ_ADC是ADC的中断号,这里注意:
 申请中断函数的最后一个参数一定不能为NULL,否则中断申请会失败,如果中断服务程序中用不到这个
 参数,就随便给个值就好了,我这里就给个1*/
 ret = request_irq(IRQ_ADC, adc_irq, IRQF_SHARED, DEVICE_NAME, 1);
 if (ret)
 {
  /*错误处理*/
  printk(KERN_ERR 'IRQ%d error %dn', IRQ_ADC, ret);
  return -EINVAL;
 }

    return 0;
}


/*设置ADC控制寄存器,开启AD转换*/
static void start_adc(void)
{
 unsigned int tmp;

 tmp = (1 << 14) | (255 << 6) | (0 << 3);/* 0 1 00000011 000 0 0 0 */
 writel(tmp, adc_base + S3C2410_ADCCON); /*AD预分频器使能、模拟输入通道设为AIN0*/

 tmp = readl(adc_base + S3C2410_ADCCON);
 tmp = tmp | (1 << 0);                /* 0 1 00000011 000 0 0 1 */
 writel(tmp, adc_base + S3C2410_ADCCON); /*AD转换开始*/
}


/*ADC设备驱动的读接口函数*/
static ssize_t adc_read(struct file *filp, char *buffer, size_t count, loff_t *ppos)
{
 /*试着获取信号量(即:加锁)*/
 if (down_trylock(&ADC_LOCK))
 {
  return -EBUSY;
 }

 if(!ev_adc)/*表示还没有AD转换后的数据,不可读取*/
 {
  if(filp->f_flags & O_NONBLOCK) //应用程序若采用非阻塞方式读取则返回错误
  {
    return -EAGAIN;
  }
  else/*以阻塞方式进行读取*/
  {
  start_adc();  /*设置ADC控制寄存器,开启AD转换*/
  wait_event_interruptible(adc_waitq, ev_adc); /*使等待队列进入睡眠*/
  }
 }
 /*能到这里就表示已有AD转换后的数据,则标识清0,给下一次读做判断用*/
    ev_adc = 0;
 /*将读取到的AD转换后的值发往到上层应用程序*/
 copy_to_user(buffer, (char *)&adc_data, sizeof(adc_data));
 up(&ADC_LOCK); /*释放获取的信号量(即:解锁)*/

 return sizeof(adc_data);
}
 


/*ADC设备驱动的关闭接口函数*/
static int adc_release(struct inode *inode, struct file *filp)
{
    return 0;
}


/*字符设备的相关操作实现*/
static struct file_operations adc_fops =
{
 .owner    = THIS_MODULE,
 .open    = adc_open,
 .read    = adc_read,   
 .release  = adc_release,
};

/*misc设备结构体实现*/
static struct miscdevice adc_miscdev =
{
 .minor  = MISC_DYNAMIC_MINOR, /*次设备号,定义在 miscdevice.h中,为255*/
 .name    = DEVICE_NAME,        /* 设备名称*/
 .fops    = &adc_fops,          /*对ADC设备文件操作*/
};


static int __init adc_init(void)
{
    int ret; 
 /*从平台时钟队列中获取ADC的时钟,这里为什么要取得这个时钟,因为ADC的转换频率跟时钟有关。
 系统的一些时钟定义在arch/arm/plat-s3c24xx /s3c2410-clock.c中*/
    adc_clk = clk_get(NULL, 'adc');
 if (!adc_clk)
 {
  /*错误处理*/
  printk(KERN_ERR 'failed to find adc clock sourcen');
  return -ENOENT;
 }

 /*时钟获取后要使能后才可以使用,clk_enable定义在arch/arm/plat-s3c/clock.c中*/
 clk_enable(adc_clk);

 /*将ADC的IO端口占用的这段 IO空间映射到内存的虚拟地址,ioremap定义在io.h中。
 注意:IO空间要映射后才能使用,以后对虚拟地址的操作就是对IO空间的操作, S3C2410_PA_ADC
 是ADC控制器的基地址,定义在mach-s3c2410/include/mach/map.h中,0x20是虚拟地址长度大小*/
  adc_base = ioremap(S3C2410_PA_ADC, 0x20);
 if (adc_base == NULL)
 {

  printk(KERN_ERR 'Failed to remap register blockn'); /*错误处理*/
    ret = -EINVAL;
    goto err_noclk;
 }
 
 /*把看ADC注册成为misc设备,misc_register定义在miscdevice.h中adc_miscdev结构体定义
 及内部接口函数在第②步中讲,MISC_DYNAMIC_MINOR是次设备号,定义在miscdevice.h中*/
    ret = misc_register(&adc_miscdev);
 if (ret)
 {
  /*错误处理*/
  printk(KERN_ERR 'cannot register miscdev on minor=%d (%d)n',
  MISC_DYNAMIC_MINOR, ret);
  goto err_nomap;
 }

  printk(DEVICE_NAME ' initialized!n');

    return 0;
 //以下是上面错误处理的跳转点
err_noclk:
  clk_disable(adc_clk);
    clk_put(adc_clk);
 
err_nomap:
    iounmap(adc_base);
 
    return ret;
}
 
static void __exit adc_exit(void)
{
    free_irq(IRQ_ADC, 1);/*释放中断*/
    iounmap(adc_base);/*释放虚拟地址映射空间*/
 
 if (adc_clk)/*屏蔽和销毁时钟*/
 {
    clk_disable(adc_clk);   
    clk_put(adc_clk);
    adc_clk = NULL;
 }
 
  misc_deregister(&adc_miscdev);/*注销misc设备*/
}
 
/*导出信号量ADC_LOCK在触摸屏驱动中使用,因为触摸屏驱动和ADC驱动公用
93.相关的寄存器,为了不产生资源竞态,就用信号量来保证资源的互斥访问*/
//EXPORT_SYMBOL(ADC_LOCK);
module_init(adc_init);
module_exit(adc_exit);
MODULE_LICENSE('GPL');
MODULE_AUTHOR('youshaohui 2012.10.31 in ruanjianyuan');
MODULE_DESCRIPTION('s3c2440 ADC Driver');

 

 

//=======================================================================

            应用程序测试代码开发如下:

//=======================================================================


#include

#define MYADC '/dev/adc_driver'

void Delay_MS( unsigned int time)  //50 ns
{
 unsigned int i,j;
 
 for ( i=0; i {
    for(j=0;j<30000;j++)
    {
   
    }
  }
}
//adc可调电阻
int main(void)
{
 int fd;
 int i=0;
 unsigned int value = -1;
 char buf[30]={0};

      fd = open(MYADC,O_RDWR,0666);
 if (fd < 0)
 {
  perror('open device adc_driver errorn');
  exit(1);
 }
 printf('open /dev/adc_driver success!n');

 
 while(1)
 {
  read(fd,&value,4);

  printf('result value=%dn',value);
 
  Delay_MS(1000);
 }
 
 if(close(fd)<0)
 {
  perror('close errorn');
  exit(1);
 }
 
 return 0;
}


关键字:mini2440  ADC  可调电阻 引用地址:mini2440 ADC可调电阻驱动程序开发源代码(杂项设备驱动框架)

上一篇:mini2440下调试IIC时序问题总结与分析
下一篇:把驱动编译进内核的3种方法(mini2440 key按键为例)

推荐阅读最新更新时间:2024-11-07 09:28

模数转换IC ADC0809的使用制做之一……基础知识
太罗嗦的内容我就不说了,只讲些跟制做有关的最精炼的知识。 ADC0809是可以将我们要测量的模拟电压信号量转换为数字量从而可以进行存储或显示的一种转换IC。 下面是它的管脚图和逻辑图: 管脚功能说明: IN0-IN7:模拟量输入通道。就是说它可以分时地分别对八个模拟量进行测量转换。 ADDA-C:地址线。也就是通过这三根地址线的不同编码来选择对哪个模拟量进行测量转换。 ALE:地址锁存允许信号。在低电平时向ADDA-C写地址,当ALE跳至高电平后ADDA-C上的数据被锁存 START:启动转换信号。当它为上升沿后,将内部寄存器清0。当它为下降沿后,开始A/D转换。 D0-D7:数据输出口。转换后的数字数据量
[单片机]
模数转换IC <font color='red'>ADC</font>0809的使用制做之一……基础知识
STM32学习——ROV遇到的坑
(1)USART 1.使能串口前必须清除TC位,否则后面多路采集ADC时就凉了。。。 USART_ClearFlag(USART2, USART_FLAG_TC); USART_Cmd(USART2, ENABLE); //使能串口2 2.查询等待方式发送数据时需要先清TC,否则会丢失第一个数据 void Send_Buf(u8* Data_Buf) { u32 i; USART_ClearFlag(USART1,USART_FLAG_TC); for(i=0;i USART2_REC_LEN;i++) { USART_SendData(USART1,Data_Buf
[单片机]
STM32的ADC输入通道配置
STM32中最多有3个ADC模块,每个模块对应的通道不完全重叠。 下图是STM32F103CDE数据手册中的总框图的左下角,图中可以看出有8个外部ADC管脚分别接到了3个ADC模块,有8个外部ADC管脚只分别接到了2个ADC模块,还有5个外部ADC管脚只接到了ADC3模块,这样总共是21个通道。 下表是这些ADC管脚与每个ADC模块的对应关系,表中可以看出ADC1还有2个内部通道,分别接到内部的温度传感器和内部的参照电压:
[单片机]
STM32的<font color='red'>ADC</font>输入通道配置
串行LVDS接口ADC改善电路板的布线设计
当共模信号较难处理或对系统有负面影响的时候,需要进行信号调理。部分系统的设计会将模拟变换器输出的单端信号转为全差分信号,然后将这些信号传送到差分输入ADC。这种设计的优点是,大部分混入差分线路的噪声会同时出现在两条线路上 (假设差分线路都是按差分方式平衡布局)。 输入信号转为数字信号之后,便必须传送到DSP或ASIC/FPGA,以便进行处理。全差分输出信号电路通过两条对称的线路输出及吸收电流。低电压差分信号 (LVDS) 便是这种信号。ADC12QS065 芯片就采用了 LVDS 技术,可解决系统这方面的问题 (见图 1)。 图1  ADC12QS065芯片的简化框图 图2 CMOS 与 LVDS 两种电路板布局的比较
[嵌入式]
【stm32f407】ADC实验
1. ADC简介 STM32F4xx系列一般都有3个ADC,这些ADC可以独立使用,也可以使用双重/三重模式(提高采样率)。STM32F4的ADC是12位逐次逼近型的模拟数字转换器。它有19个通道,可测量16个外部源、2个内部源和Vbat通道的信号。这些通道的A/D转换可以单次、连续、扫描或间断模式执行。ADC的结果可以左对齐或右对齐方式存储在16位数据寄存器中。 模拟看门狗特性允许应用程序检测输入电压是否超出用户定义的高/低阀值。 STM32F407VGT6包含有3个ADC。STM32F4的ADC最大的转换速率为2.4Mhz,也 就 是转换时间为0.41us(在ADCCLK=36M,采样周期为3个ADC时钟下得到),不要
[单片机]
【stm32f407】<font color='red'>ADC</font>实验
根据韦东山修改的mini2440 nand flash裸机程序
说明 开发板:mini2440 Nand flash:K9F2G08U0B2,56M 本程序实现功能:我是参考韦东山的第8章的裸机程序,但是他的nand flash型号是64MB的K9F12080M,所以我依照他的程序加以修改,下载到nand flash,从nand flash实现将main.c程序复制到sdram中执行。 参考书:《嵌入式应用开发完全手册》 《ARM处理器裸机开发实战---机制而非策略》 知识点梳理 页的绝对地址与相对地址 在nand.c中有一个函数void RdNF2SDRAM(),它的功能
[单片机]
根据韦东山修改的<font color='red'>mini2440</font> nand flash裸机程序
STM32学习记录19 定时器触发ADC
关于定时器出发ADC的问题一直比较迷惑,看各种资料讲的也很少。这里就那官方的例程来仔细分析一下吧。 1:下面为官方例程, int main(void) { #ifdef DEBUG debug(); #endif /* System clocks configuration ---------------------------------------------*/ RCC_Configuration(); /* NVIC configuration ------------------------------------------------------*/ NVIC_Configurat
[单片机]
STM32学习记录19 定时器触发<font color='red'>ADC</font>
利用24位Σ-Δ型ADC实现精密电子秤设计
电路功能与优势   本电路为采用AD7192构建的电子秤系统。AD7192是一款超低噪声、低漂移24位Σ-Δ转换器,内置PGA。该器件将大多数系统构建模块置于芯片内,因此能够简化电子秤设计。该器件可在4.7 Hz至4.8 kHz的完整输出数据速率范围内工作,并保持良好的性能,因此可用于以较低速度工作的电子秤系统,以及料斗秤等较高速电子秤系统。 图1:采用AD7192的电子秤系统(原理示意图,未显示所有连接)    电路描述   AD7192提供一种集成式电子秤解决方案,可以直接与称重传感器接口。只需在模拟输入端用一些滤波器,在基准电压引脚上配置一些电容等外部元件,便可满足电磁屏蔽(EMC)要求。来自称重传感器的
[模拟电子]
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved