共計 9268 個字符,預計需要花費 24 分鐘才能閱讀完成。
這篇文章主要介紹了 linux adc 設備指的是什么的相關知識,內容詳細易懂,操作簡單快捷,具有一定借鑒價值,相信大家閱讀完這篇 linux adc 設備指的是什么文章都會有所收獲,下面我們一起來看看吧。
linux adc 是混雜設備驅動;在 linux2.6.30.4 中,系統已經自帶有了 ADC 通用驅動文件“arch/arm/plat-s3c24xx/adc.c”,它是以平臺驅動設備模型的架構來編寫的,里面是一些比較通用穩定的代碼。
linux2.6.30.4 中,系統已經自帶有了 ADC 通用驅動文件 —arch/arm/plat-s3c24xx/adc.c,它是以平臺驅動設備模型的架構來編寫的,里面是一些比較通用穩定的代碼,但是 linux2.6.30.4 版本的 ADC 通用驅動文件并不完善,居然沒有讀函數。后來去看了 linux3.8 版本的 ADC 通用文件 —-arch/arm/plat-samsung/adc.c 才是比較完善的。
但是本節并不是分析這個文件,而是以另外一種架構來編寫 ADC 驅動,因為 ADC 驅動實在是比較簡單,就沒有使用平臺驅動設備模型為架構來編寫了,這次我們使用的是混雜 (misc) 設備驅動。
問:什么是 misc 設備驅動?
答:miscdevice 共享一個主設備號 MISC_MAJOR(10),但次設備號不同。所有的 miscdevice 設備形成一條鏈表,對設備訪問時內核根據設備號來查找對應的 miscdevice 設備,然后調用其 file_operations 結構體中注冊的文件操作接口進行操作。
struct miscdevice {
int minor; // 次設備號,如果設置為 MISC_DYNAMIC_MINOR 則系統自動分配
const char *name; // 設備名
const struct file_operations *fops; // 操作函數
struct list_head list;
struct device *parent;
struct device *this_device;
};
dev_init 入口函數分析:
static int __init dev_init(void)
int ret;
base_addr=ioremap(S3C2410_PA_ADC,0x20);
if (base_addr == NULL)
printk(KERN_ERR failed to remap register block\n
return -ENOMEM;
adc_clock = clk_get(NULL, adc
if (!adc_clock)
printk(KERN_ERR failed to get adc clock source\n
return -ENOENT;
clk_enable(adc_clock);
ADCTSC = 0;
ret = request_irq(IRQ_ADC, adcdone_int_handler, IRQF_SHARED, DEVICE_NAME, adcdev);
if (ret)
iounmap(base_addr);
return ret;
ret = misc_register(misc);
printk (DEVICE_NAME initialized\n
return ret;
}
首先是映射 ADC 寄存器地址將其轉換為虛擬地址,然后獲得 ADC 時鐘并使能 ADC 時鐘,接著申請 ADC 中斷,其中斷處理函數為
adcdone_int_handler,而 flags 為 IRQF_SHARED,即共享中斷,因為觸摸屏里也要申請 ADC 中斷,最后注冊一個混雜設備。
當應用程序 open (/dev/adc ,…)時,就會調用到驅動里面的 open 函數,那么我們來看看 open 函數做了什么?
static int tq2440_adc_open(struct inode *inode, struct file *filp)
/* 初始化等待隊列頭 */
init_waitqueue_head((adcdev.wait));
/* 開發板上 ADC 的通道 2 連接著一個電位器 */
adcdev.channel=2; // 設置 ADC 的通道
adcdev.prescale=0xff;
DPRINTK( ADC opened\n
return 0;
}
很簡單,先初始化一個等待隊列頭,因為入口函數里既然有申請 ADC 中斷,那么肯定要使用等待隊列,接著設置 ADC 通道,因為 TQ2440 的 ADC 輸入通道默認是 2,設置預分頻值為 0xff。
當應用程序 read 時,就會調用到驅動里面的 read 函數,那么我們來看看 read 函數做了些什么?
static ssize_t tq2440_adc_read(struct file *filp, char *buffer, size_t count, loff_t *ppos)
char str[20];
int value;
size_t len;
/* 嘗試獲得 ADC_LOCK 信號量,如果能夠立刻獲得,它就獲得信號量并返回 0
* 否則,返回非零,它不會導致調用者睡眠, 可以在中斷上下文使用
*/
if (down_trylock( ADC_LOCK) == 0)
/* 表示 A / D 轉換器資源可用 */
ADC_enable = 1;
/* 使能預分頻,選擇 ADC 通道,最后啟動 ADC 轉換 */
START_ADC_AIN(adcdev.channel, adcdev.prescale);
/* 等待事件,當 ev_adc = 0 時,進程被阻塞,直到 ev_adc 0 */
wait_event_interruptible(adcdev.wait, ev_adc);
ev_adc = 0;
DPRINTK(AIN[%d] = 0x%04x, %d\n , adcdev.channel, adc_data, ((ADCCON 0x80) ? 1:0));
/* 將在 ADC 中斷處理函數讀取的 ADC 轉換結果賦值給 value */
value = adc_data;
sprintf(str, %5d , adc_data);
copy_to_user(buffer, (char *) adc_data, sizeof(adc_data));
ADC_enable = 0;
up(ADC_LOCK);
else
/* 如果 A / D 轉換器資源不可用,將 value 賦值為 -1 */
value = -1;
/* 將 ADC 轉換結果輸出到 str 數組里,以便傳給應用空間 */
len = sprintf(str, %d\n , value);
if (count = len)
/* 從 str 數組里拷貝 len 字節的數據到 buffer,即將 ADC 轉換數據傳給應用空間 */
int r = copy_to_user(buffer, str, len);
return r ? r : len;
else
return -EINVAL;
}
tq2440_adc_read 函數首先嘗試獲得 ADC_LOCK 信號量,因為觸摸屏驅動也有使用 ADC 資源,兩者互有競爭關系,獲得 ADC 資源后,使能預分頻,選擇 ADC 通道,最后啟動 ADC 轉換,接著就調用 wait_event_interruptible 函數進行等待,直到 ev_adc 0 進程才會繼續往下跑,往下跑就會將 adc_data 數據讀出來,調用 copy_to_user 函數將 ADC 數據傳給應用空間,最后釋放 ADC_LOCK 信號量。
問:什么時候 ev_adc 0?默認 ev_adc = 0
答:在 adcdone_int_handler 中斷處理函數里,等數據讀出后,ev_adc 被設置為 1。
ADC 中斷處理函數 adcdone_int_handler
/* ADC 中斷處理函數 */
static irqreturn_t adcdone_int_handler(int irq, void *dev_id)
/* A/ D 轉換器資源可用 */
if (ADC_enable)
/* 讀 ADC 轉換結果數據 */
adc_data = ADCDAT0 0x3ff;
/* 喚醒標志位,作為 wait_event_interruptible 的喚醒條件 */
ev_adc = 1;
wake_up_interruptible(adcdev.wait);
return IRQ_HANDLED;
}
當 AD 轉換完成后就會觸發 ADC 中斷,就會進入 adcdone_int_handler,這個函數就會講 AD 轉換數據讀到 adc_data,接著將喚醒標志位 ev_adc 置 1,最后調用 wake_up_interruptible 函數喚醒 adcdev.wait 等待隊列。
總結一下 ADC 的工作流程:
一、open 函數里,設置模擬輸入通道,設置預分頻值
二、read 函數里,啟動 AD 轉換,進程休眠
三、adc_irq 函數里,AD 轉換結束后觸發 ADC 中斷,在 ADC 中斷處理函數將數據讀出,喚醒進程
四、read 函數里,進程被喚醒后,將 adc 轉換數據傳給應用程序
ADC 驅動參考源碼:
/*************************************
NAME:EmbedSky_adc.c
COPYRIGHT:www.embedsky.net
*************************************/
#include linux/errno.h
#include linux/kernel.h
#include linux/module.h
#include linux/slab.h
#include linux/input.h
#include linux/init.h
#include linux/serio.h
#include linux/delay.h
#include linux/clk.h
#include asm/io.h
#include asm/irq.h
#include asm/uaccess.h
#include mach/regs-clock.h
#include plat/regs-timer.h
#include plat/regs-adc.h
#include mach/regs-gpio.h
#include linux/cdev.h
#include linux/miscdevice.h
#include tq2440_adc.h
#undef DEBUG
//#define DEBUG
#ifdef DEBUG
#define DPRINTK(x...) {printk(KERN_DEBUG EmbedSky_adc: x);}
#else
#define DPRINTK(x...) (void)(0)
#endif
#define DEVICE_NAME adc /* 設備節點: /dev/adc */
static void __iomem *base_addr;
typedef struct
wait_queue_head_t wait; /* 定義等待隊列頭 */
int channel;
int prescale;
}ADC_DEV;
DECLARE_MUTEX(ADC_LOCK); /* 定義并初始化信號量, 并初始化為 1 */
static int ADC_enable = 0; /* A/ D 轉換器資是否可用標志位 */
static ADC_DEV adcdev; /* 用于表示 ADC 設備 */
static volatile int ev_adc = 0; /* 作為 wait_event_interruptible 的喚醒條件 */
static int adc_data;
static struct clk *adc_clock;
#define ADCCON (*(volatile unsigned long *)(base_addr + S3C2410_ADCCON)) //ADC control
#define ADCTSC (*(volatile unsigned long *)(base_addr + S3C2410_ADCTSC)) //ADC touch screen control
#define ADCDLY (*(volatile unsigned long *)(base_addr + S3C2410_ADCDLY)) //ADC start or Interval Delay
#define ADCDAT0 (*(volatile unsigned long *)(base_addr + S3C2410_ADCDAT0)) //ADC conversion data 0
#define ADCDAT1 (*(volatile unsigned long *)(base_addr + S3C2410_ADCDAT1)) //ADC conversion data 1
#define ADCUPDN (*(volatile unsigned long *)(base_addr + 0x14)) //Stylus Up/Down interrupt status
#define PRESCALE_DIS (0 14)
#define PRESCALE_EN (1 14)
#define PRSCVL(x) ((x) 6)
#define ADC_INPUT(x) ((x) 3)
#define ADC_START (1 0)
#define ADC_ENDCVT (1 15)
/* 使能預分頻,選擇 ADC 通道,最后啟動 ADC 轉換 */
#define START_ADC_AIN(ch, prescale) \
do{ ADCCON = PRESCALE_EN | PRSCVL(prescale) | ADC_INPUT((ch)) ; \
ADCCON |= ADC_START; \
}while(0)
/* ADC 中斷處理函數 */
static irqreturn_t adcdone_int_handler(int irq, void *dev_id)
/* A/ D 轉換器資源可用 */
if (ADC_enable)
/* 讀 ADC 轉換結果數據 */
adc_data = ADCDAT0 0x3ff;
/* 喚醒標志位,作為 wait_event_interruptible 的喚醒條件 */
ev_adc = 1;
wake_up_interruptible(adcdev.wait);
return IRQ_HANDLED;
static ssize_t tq2440_adc_read(struct file *filp, char *buffer, size_t count, loff_t *ppos)
char str[20];
int value;
size_t len;
/* 嘗試獲得 ADC_LOCK 信號量,如果能夠立刻獲得,它就獲得信號量并返回 0
* 否則,返回非零,它不會導致調用者睡眠, 可以在中斷上下文使用
*/
if (down_trylock( ADC_LOCK) == 0)
/* 表示 A / D 轉換器資源可用 */
ADC_enable = 1;
/* 使能預分頻,選擇 ADC 通道,最后啟動 ADC 轉換 */
START_ADC_AIN(adcdev.channel, adcdev.prescale);
/* 等待事件,當 ev_adc = 0 時,進程被阻塞,直到 ev_adc 0 */
wait_event_interruptible(adcdev.wait, ev_adc);
ev_adc = 0;
DPRINTK(AIN[%d] = 0x%04x, %d\n , adcdev.channel, adc_data, ((ADCCON 0x80) ? 1:0));
/* 將在 ADC 中斷處理函數讀取的 ADC 轉換結果賦值給 value */
value = adc_data;
sprintf(str, %5d , adc_data);
copy_to_user(buffer, (char *) adc_data, sizeof(adc_data));
ADC_enable = 0;
up(ADC_LOCK);
else
/* 如果 A / D 轉換器資源不可用,將 value 賦值為 -1 */
value = -1;
/* 將 ADC 轉換結果輸出到 str 數組里,以便傳給應用空間 */
len = sprintf(str, %d\n , value);
if (count = len)
/* 從 str 數組里拷貝 len 字節的數據到 buffer,即將 ADC 轉換數據傳給應用空間 */
int r = copy_to_user(buffer, str, len);
return r ? r : len;
else
return -EINVAL;
static int tq2440_adc_open(struct inode *inode, struct file *filp)
/* 初始化等待隊列頭 */
init_waitqueue_head((adcdev.wait));
/* 開發板上 ADC 的通道 2 連接著一個電位器 */
adcdev.channel=2; // 設置 ADC 的通道
adcdev.prescale=0xff;
DPRINTK( ADC opened\n
return 0;
static int tq2440_adc_release(struct inode *inode, struct file *filp)
DPRINTK( ADC closed\n
return 0;
static struct file_operations dev_fops = {
owner: THIS_MODULE,
open: tq2440_adc_open,
read: tq2440_adc_read,
release: tq2440_adc_release,
static struct miscdevice misc = {
.minor = MISC_DYNAMIC_MINOR,
.name = DEVICE_NAME,
.fops = dev_fops,
static int __init dev_init(void)
int ret;
base_addr=ioremap(S3C2410_PA_ADC,0x20);
if (base_addr == NULL)
printk(KERN_ERR failed to remap register block\n
return -ENOMEM;
adc_clock = clk_get(NULL, adc
if (!adc_clock)
printk(KERN_ERR failed to get adc clock source\n
return -ENOENT;
clk_enable(adc_clock);
ADCTSC = 0;
ret = request_irq(IRQ_ADC, adcdone_int_handler, IRQF_SHARED, DEVICE_NAME, adcdev);
if (ret)
iounmap(base_addr);
return ret;
ret = misc_register(misc);
printk (DEVICE_NAME initialized\n
return ret;
static void __exit dev_exit(void)
free_irq(IRQ_ADC, adcdev);
iounmap(base_addr);
if (adc_clock)
clk_disable(adc_clock);
clk_put(adc_clock);
adc_clock = NULL;
misc_deregister(misc);
EXPORT_SYMBOL(ADC_LOCK);
module_init(dev_init);
module_exit(dev_exit);
MODULE_LICENSE( GPL
MODULE_AUTHOR( www.embedsky.net
MODULE_DESCRIPTION(ADC Drivers for EmbedSky SKY2440/TQ2440 Board and support touch
ADC 應用測試參考源碼:
/*************************************
NAME:EmbedSky_adc.c
COPYRIGHT:www.embedsky.net
*************************************/
#include stdio.h
#include unistd.h
#include stdlib.h
#include sys/types.h
#include sys/stat.h
#include sys/ioctl.h
#include fcntl.h
#include linux/fs.h
#include errno.h
#include string.h
int main(void)
int fd ;
char temp = 1;
fd = open(/dev/adc , 0);
if (fd 0)
perror( open ADC device !
exit(1);
for( ; ; )
char buffer[30];
int len ;
len = read(fd, buffer, sizeof buffer -1);
if (len 0)
buffer[len] = \0
int value;
sscanf(buffer, %d , value);
printf(ADC Value: %d\n , value);
else
perror( read ADC device !
exit(1);
sleep(1);
adcstop:
close(fd);
}
測試結果:
[WJ2440]# ./adc_test
ADC Value: 693
ADC Value: 695
ADC Value: 694
ADC Value: 695
ADC Value: 702
ADC Value: 740
ADC Value: 768
ADC Value: 775
ADC Value: 820
ADC Value: 844
ADC Value: 887
ADC Value: 937
ADC Value: 978
ADC Value: 1000
ADC Value: 1023
ADC Value: 1023
ADC Value: 1023
關于“linux adc 設備指的是什么”這篇文章的內容就介紹到這里,感謝各位的閱讀!相信大家對“linux adc 設備指的是什么”知識都有一定的了解,大家如果還想學習更多知識,歡迎關注丸趣 TV 行業資訊頻道。