天嵌 ARM开发社区

 找回密码
 注册
查看: 3568|回复: 13

AD的驱动和触摸屏冲突

[复制链接]
imb1900220 发表于 2009-12-16 17:52:58 | 显示全部楼层 |阅读模式
做了一个ad的驱动,并且把采样的值显示在qt中,
但是遇到问题
1、AD采样用的是tq2440开发板上自带的可调电阻,对应的通道是通道2,
最后在qt中显示的时候,实际显示的数值并不是通道2 采样的数据,而是触摸屏的数据,


我在驱动中读取的数值是从DAT0中读取的,看了手册后知道这个是触摸屏的数据采样存储的寄存器,所以把我的AD采样的值给覆盖了,

想知道是不是这样的,
要想真实的显示自己的AD 数值要怎么办

还有,系统自带的触摸屏的驱动是编译在内核中的,是不是一上电启动开发板 触摸屏就会一直占用AD,


想读出通道2的数据,我要怎么办,,这个冲突怎么解决呢
亚瑟王 发表于 2009-12-17 18:51:51 | 显示全部楼层
ADC和触摸是不会冲突的,因为你想要读取哪个一路ADC时,你需要输入该路的通道号,然后再读取的,我认为你没有正确选择通道导致的。
 楼主| imb1900220 发表于 2009-12-18 10:21:15 | 显示全部楼层
那这样的可能就是我的驱动代码有问题了,

直接附上代码,请版主帮忙查查是哪错了吧


/*
* s3c2440_adc  
*  
* Author: SeonKon Choi <BUSHI@MIZI.COM>
* Date  : $Date: 2003/01/20 14:24:49 $  
*
* $Revision: 1.1.2.6 $
*
    2004-6-14 add a device by threewater<THREEWATER@UP-TECH.COM>

   Fri Dec 03 2002 SeonKon Choi <BUSHI@MIZI.COM>
   - initial

   modify by lxm<lxm650@163.com> 2009-02-16
*
* This file is subject to the terms and conditions of the GNU General Public
* License.  See the file COPYING in the main directory of this archive
* for more details.
*/  

#include <linux/module.h>
#include <linux/fs.h>           /* everything... */
#include <linux/cdev.h>
#include <linux/interrupt.h>    /* request_irq() */
#include <asm/arch/regs-adc.h>
#include <asm/hardware/clock.h>
#include <asm/io.h>
#include <asm/uaccess.h>        /* copy_to_user() */
#include "s3c2440_adc.h"   

#undef DEBUG   
#define DEBUG   
#ifdef DEBUG   
#define DPRINTK(x...) {printk(__FUNCTION__);printk("(%d): ",__LINE__);printk(x);}   
#else   
#define DPRINTK(x...) (void)(0)   
#endif   
  
#define VERSION_STRING  "ADC driver for s3c2440"
#define DEVICE_NAME "s3c2440_adc"   

#define ADC_CTL_BASE 0x58000000 /* 定义ADC寄存器基地址 */  
#define S3C2440ADC_MAJOR 240    /* 预设的S3C2440ADC的主设备号 */

static int s3c2440adc_major = S3C2440ADC_MAJOR;

struct s3c2440adc_dev {  
    struct cdev cdev;                /* cdev结构体 */
    struct semaphore lock;        /* 信号量 */
    wait_queue_head_t wait;        /* 定义等待队列头 */
    int channel;                /* 通道号 */  
    int prescale;                /* 预分频系数 */  
};  

struct s3c2440adc_dev s3c2440adc_dev;   /* 物理地址映射成虚拟地址后的基地址指针 */
struct s3c2440adc_dev *s3c2440adc_devp; /*设备结构体指针*/

static void __iomem *base_addr;         /* 物理地址映射成虚拟地址后的基地址指针 */
static struct clk   *adc_clock;         /* 时钟 */

/* ADC转换完成后中断将调用的函数*/
static irqreturn_t adcdone_int_handler(int irq, void *dev_id)  
{  
    wake_up(&s3c2440adc_dev.wait);  /* 唤醒等待队列 */
    return IRQ_HANDLED;
}  

/* ADC驱动的写函数 */  
static ssize_t s3c2440adc_write(struct file *file, const char *buffer, size_t count, loff_t * ppos)  
{  
    unsigned long data;  
  
    if(count!=sizeof(data)){  
    //错误的输入数据大小
            DPRINTK("the size of input data must be %d\n", sizeof(data));  
        return 0;  
    }  
  
    copy_from_user(&data, buffer, count);  
    s3c2440adc_dev.channel=ADC_WRITE_GETCH(data);       /* 得到需进行AD转换的通道号 */  
    s3c2440adc_dev.prescale=ADC_WRITE_GETPRE(data);     /* 得到预分频系数 */  
  
    DPRINTK("set adc channel=%d, prescale=0x%x\n", s3c2440adc_dev.channel, s3c2440adc_dev.prescale);  
  
    return count;  
}  

/* ADC驱动的读函数 */  
static ssize_t s3c2440adc_read(struct file *filp, char *buffer, size_t count, loff_t *ppos)  
{  
    unsigned long result = 0;  
    unsigned int data;
    if (down_interruptible(&s3c2440adc_dev.lock))  
        return -ERESTARTSYS;  
  
    /* 启动AD转换 */
    iowrite32(START_ADC(s3c2440adc_dev.channel, s3c2440adc_dev.prescale), base_addr + S3C2410_ADCCON);
    interruptible_sleep_on(&s3c2440adc_dev.wait);               /* 等待AD完成中断*/  
  
    data = ioread32(base_addr + S3C2410_ADCDAT0) & 0x3ff;       /* 保留转换数据寄存器的低10位 */
    result = ioread32(base_addr + S3C2410_ADCCON);
    DPRINTK("AIN[%d] = 0x%x, 0x%x\n", s3c2440adc_dev.channel, data, (unsigned int)result);
  
    copy_to_user(buffer, &data, sizeof(data));      /* 复制到用户空间 */  
  
    up(&s3c2440adc_dev.lock);  
  
    return sizeof(result);  
}  
  
/*文件打开函数*/  
static int s3c2440adc_open(struct inode *inode, struct file *filp)  
{  
    init_MUTEX(&s3c2440adc_dev.lock);  
    init_waitqueue_head(&(s3c2440adc_dev.wait));  
    filp->private_data = s3c2440adc_devp;  /*将设备结构体指针赋值给文件私有数据指针*/
    s3c2440adc_dev.channel=0;  
    s3c2440adc_dev.prescale=0xff;  

    base_addr = ioremap(ADC_CTL_BASE, 0x14);

        adc_clock = clk_get(NULL, "adc");
        if (!adc_clock) {
                DPRINTK("failed to get adc clock source\n");
                return -ENOENT;
        }
        clk_enable(adc_clock);

    DPRINTK( "adc opened\n");  
    return 0;  
}  
  
static int s3c2440adc_release(struct inode *inode, struct file *filp)  
{  
    DPRINTK( "adc closed\n");  
    return 0;  
}  
  
static struct file_operations s3c2440adc_fops = {  
    owner:  THIS_MODULE,  
    open:   s3c2440adc_open,  
    read:   s3c2440adc_read,     
    write:  s3c2440adc_write,  
    release:    s3c2440adc_release,  
};  
  
/*初始化并注册cdev*/
static void s3c2440adc_setup_cdev(struct s3c2440adc_dev *dev, int index)
{
    int err, devno = MKDEV(s3c2440adc_major, index);
    cdev_init(&dev->cdev, &s3c2440adc_fops);
    dev->cdev.owner = THIS_MODULE;
    dev->cdev.ops = &s3c2440adc_fops;
    err = cdev_add(&dev->cdev, devno, 1);
    if (err)
        DPRINTK("Error %d adding devno%d", err, index);
}

int __init s3c2440adc_init(void)  
{  
    int result;

    dev_t devno = MKDEV(s3c2440adc_major, 0);

    /* 申请中断 */
    result = request_irq(IRQ_ADC, adcdone_int_handler, SA_INTERRUPT, DEVICE_NAME, NULL);  

    /* 中断号被占用 */
    if (result) {  
        DPRINTK("IRQ%d already in use\n", IRQ_ADC);
              return result;  
    }  

    /* 申请设备号*/
    if (s3c2440adc_major)
        result = register_chrdev_region(devno, 1, "s3c2440adc");
    else  /* 动态申请设备号 */
    {
        result = alloc_chrdev_region(&devno, 0, 1, "s3c2440adc");
        s3c2440adc_major = MAJOR(devno);
    }  
    if (result < 0)
        return result;
   
    /* 动态申请设备结构体的内存*/
    s3c2440adc_devp = kmalloc(sizeof(struct s3c2440adc_dev), GFP_KERNEL);
    if (!s3c2440adc_devp)    /*申请失败*/
    {
        result = -ENOMEM;
        goto fail_malloc;
    }
    memset(s3c2440adc_devp, 0, sizeof(struct s3c2440adc_dev));
  
    s3c2440adc_setup_cdev(s3c2440adc_devp, 0);
    return 0;

    fail_malloc: unregister_chrdev_region(devno, 1);
    return result;
}

void __exit s3c2440adc_exit(void)  
{
        if (adc_clock) {
                clk_disable(adc_clock);
                //clk_unuse(adc_clock);
                clk_put(adc_clock);
                adc_clock = NULL;
        }

    cdev_del(&s3c2440adc_devp->cdev);                /*注销cdev*/
    kfree(s3c2440adc_devp);                             /*释放设备结构体内存*/
    unregister_chrdev_region(MKDEV(s3c2440adc_major, 0), 1); /*释放设备号*/
    iounmap(base_addr);                   /* 释放申请的虚拟地址 */
    free_irq(IRQ_ADC, NULL);                          /*释放中断*/
}  

MODULE_AUTHOR("lxm<lxm650@163.com>");
MODULE_LICENSE("Dual BSD/GPL");
MODULE_DESCRIPTION(VERSION_STRING);

module_init(s3c2440adc_init);
module_exit(s3c2440adc_exit);
亚瑟王 发表于 2009-12-18 18:40:58 | 显示全部楼层
在我们的2.6.30.4的内核中有一个ADC的驱动,我上传到附件中,你看看把。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有帐号?注册

x
upc_zongchao 发表于 2010-1-28 16:05:15 | 显示全部楼层
一直在头疼这个问题呢
亚瑟王 发表于 2010-1-28 21:08:33 | 显示全部楼层
你参考我们的那个驱动,要是每个客户都把自己出错的源码给我们帮忙纠错,我们的工程师别的事情都做不了了。
upc_zongchao 发表于 2010-5-26 16:14:12 | 显示全部楼层
想下载啊   老是贡献不足            呜呜
亚瑟王 发表于 2010-5-26 17:53:19 | 显示全部楼层
完成新手任务就行了。
guoyin 发表于 2010-5-29 12:47:48 | 显示全部楼层
呵呵,看来论坛改版给网友增加了难度了!
亚瑟王 发表于 2010-5-29 16:37:32 | 显示全部楼层
被广告贴逼的。没办法啊。现在的广告机器人太猛了。
www85522 发表于 2010-6-7 21:05:56 | 显示全部楼层
谢谢楼主 还不错啊
elitemouse 发表于 2010-6-15 21:16:37 | 显示全部楼层
:handshake  学习驱动中 非常感谢Arthur
dhb1988 发表于 2010-9-16 22:21:19 | 显示全部楼层
学习驱动中 非常感谢 顶
slesre 发表于 2010-11-13 23:27:20 | 显示全部楼层
同样在学习驱动 369471505QQ 广结驱动好友
您需要登录后才可以回帖 登录 | 注册

本版积分规则

关闭

i.MX8系列ARM cortex A53 M4 工控板上一条 /1 下一条

Archiver|手机版|小黑屋|天嵌 嵌入式开发社区 ( 粤ICP备11094220号-2 )

GMT+8, 2024-9-12 17:48 , Processed in 1.019570 second(s), 19 queries .

Powered by Discuz! X3.4 Licensed

Copyright © 2001-2020, Tencent Cloud.

快速回复 返回顶部 返回列表