91欧美超碰AV自拍|国产成年人性爱视频免费看|亚洲 日韩 欧美一厂二区入|人人看人人爽人人操aV|丝袜美腿视频一区二区在线看|人人操人人爽人人爱|婷婷五月天超碰|97色色欧美亚州A√|另类A√无码精品一级av|欧美特级日韩特级

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Linux驅(qū)動(dòng)開發(fā)-編寫PCF8591(ADC)芯片驅(qū)動(dòng)

DS小龍哥-嵌入式技術(shù) ? 2022-09-17 15:16 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

【摘要】 PCF8591是一個(gè)IIC總線接口的ADC/DAC轉(zhuǎn)換芯片,功能比較強(qiáng)大,這篇文章就介紹在Linux系統(tǒng)里如何編寫一個(gè)PCF8591的驅(qū)動(dòng),完成ADC數(shù)據(jù)采集,DAC數(shù)據(jù)輸出。

1. PCF8591介紹

PCF8591是一個(gè)IIC總線接口的ADC/DAC轉(zhuǎn)換芯片,功能比較強(qiáng)大,這篇文章就介紹在Linux系統(tǒng)里如何編寫一個(gè)PCF8591的驅(qū)動(dòng),完成ADC數(shù)據(jù)采集,DAC數(shù)據(jù)輸出。

下面是PCF8591的官方介紹-摘自中文手冊(cè):

PCF8591是具有I2C總線接口的8位A/D及D/A轉(zhuǎn)換器。有4路A/D轉(zhuǎn)換輸入,1路D/A模擬輸出。這就是說(shuō),它既可以作A/D轉(zhuǎn)換也可以作D/A轉(zhuǎn)換,A/D轉(zhuǎn)換為逐次比較型。

PCF8591采用典型的I2C總線接口器件尋址方法,即總線地址由器件地址、引腳地址和方向位組成。飛利蒲公司規(guī)定A/D器件地址為1001。引腳地址為A2A1A0,其值由用戶選擇,因此I2C系統(tǒng)中最多可接8個(gè)具有I2C總線接口的A/D器件。地址的最后一位為方向位R/W,當(dāng)主控器對(duì)A/D器件進(jìn)行讀操作時(shí)為1,進(jìn)行寫操作時(shí)為0??偩€操作時(shí),由器件地址、引腳地址和方向位組成的從地址為主控器發(fā)送的第一字節(jié)。

2. 硬件環(huán)境介紹

當(dāng)前的開發(fā)板采用友善之臂Tiny4412開發(fā)板,采用三星的exynos-4412芯片,下面是開發(fā)板與PCF8591的硬件連線圖:

image-20220106091151114

下面是PCF8591的原理圖,介紹了每個(gè)引腳詳細(xì)功能:

image-20220106091509038image-20220106091635195

3. 驅(qū)動(dòng)案例代碼

下面是PCF8591的驅(qū)動(dòng)代碼,采用IIC子系統(tǒng)框架編程,驅(qū)動(dòng)代碼分為設(shè)備端、驅(qū)動(dòng)端兩部分。

驅(qū)動(dòng)框架采用雜項(xiàng)字符設(shè)備完成注冊(cè),給應(yīng)用層提供訪問(wèn)的設(shè)備節(jié)點(diǎn),詳細(xì)的說(shuō)明在代碼路寫了完整的注釋。

3.1 驅(qū)動(dòng)端代碼

#include 
#include 
#include 
#include 
#include   /*注冊(cè)中斷相關(guān)*/
#include  		  /*中斷邊沿類型定義*/
#include  	  /*中斷IO口定義*/
#include   /*工作隊(duì)列相關(guān)*/
#include       /*互斥信號(hào)量頭文件*/
#include 
#include  /*雜項(xiàng)設(shè)備相關(guān)結(jié)構(gòu)體*/
#include          /*文件操作集合頭文件*/
#include     /*使用copy_to_user和copy_from_user*/

#define AIN0 0x40
#define AIN1 0x41
#define AIN2 0x42
#define AIN3 0x43

static struct i2c_client *PCF8591_client; /*IIC設(shè)備總線*/

/*讀取PCF8591  ADC數(shù)據(jù)*/
unsigned char PCF8591_ReadADC(unsigned char ch)
{
	return i2c_smbus_read_byte_data(PCF8591_client,ch); 
}
static int PCF8591_open(struct inode *my_inode, struct file *my_file)
{
	return 0;
}

static ssize_t PCF8591_read(struct file *my_file, char __user *buf, size_t my_len, loff_t * my_loff)
{
	unsigned char data=PCF8591_ReadADC(AIN0);
	copy_to_user(buf,&data,1);
	
	data=PCF8591_ReadADC(AIN1);
	printk("1:%d\r\n",data);
	data=PCF8591_ReadADC(AIN2);
	printk("2:%d\r\n",data);
	data=PCF8591_ReadADC(AIN3);
	printk("3:%d\r\n",data);
	return 0;
}
static  ssize_t PCF8591_write(struct file *my_file, const char __user *buf, size_t my_len, loff_t *my_loff)
{
	//DAC輸出
	i2c_smbus_write_byte_data(PCF8591_client,0x40,100);
	return 0;
}
static int  PCF8591_release(struct inode *my_inode, struct file *my_file)
{
	return 0;
}
/*定義一個(gè)文件操作集合結(jié)構(gòu)體*/
static struct file_operations ops_PCF8591={
   .owner = THIS_MODULE,
   .read=PCF8591_read,       /*讀函數(shù)-被應(yīng)用層read函數(shù)調(diào)用*/
   .write=PCF8591_write,     /*寫函數(shù)-被應(yīng)用層write函數(shù)調(diào)用*/
   .open=PCF8591_open,       /*打開函數(shù)-被應(yīng)用層open函數(shù)調(diào)用*/
   .release=PCF8591_release, /*釋放函數(shù)*/
};

/*定義一個(gè)雜項(xiàng)設(shè)備結(jié)構(gòu)體*/
static struct miscdevice misce_PCF8591={
	.minor =MISC_DYNAMIC_MINOR, /*自動(dòng)分配次設(shè)備號(hào)*/
	.name = "Tiny4412_PCF8591",  			/*名稱  在dev/目錄下邊可以找到*/
	.fops = &ops_PCF8591, 			/*文件操作集合*/
};

static int i2c_probe(struct i2c_client *client, const struct i2c_device_id *device_id)//匹配成功時(shí)調(diào)用
{
	PCF8591_client=client;
	printk("<1>""驅(qū)動(dòng)端IIC匹配的地址=0x%x\n",client->addr);
	
	/* 檢測(cè)適配器是否支持smbus字節(jié)讀寫函數(shù) */
    if(i2c_check_functionality(client->adapter, I2C_FUNC_SMBUS_BYTE_DATA)) 
	{
		printk("適配器支持smbus字節(jié)讀寫函數(shù)\n");
	}
	
	/*注冊(cè)*/
	misc_register(&misce_PCF8591);
	return 0;
}
static int i2c_remove(struct i2c_client *client)
{
	misc_deregister(&misce_PCF8591);/*注銷*/
	printk("i2c_驅(qū)動(dòng)端卸載成功!!!\n");
	return 0;
}

/*
IIC驅(qū)動(dòng)端
*/
static const struct i2c_device_id i2c_id[] =
{
	{"Tiny4412_PCF8591",0},//設(shè)備端的名字為"my_PCF8591",后面的表示需要私有數(shù)據(jù)
	{}
};

struct i2c_driver i2c_drv =
{
	.driver=
	{
		.name = "PCF8591",   
		.owner = THIS_MODULE,
	},	
	.probe = i2c_probe,
	.remove = i2c_remove,
	.id_table = i2c_id,
};
static int __init i2c_drv_init(void)
{
	i2c_add_driver(&i2c_drv);//向iic總線注冊(cè)一個(gè)驅(qū)動(dòng)
	return 0;
}

static void __exit i2c_drv_exit(void)//平臺(tái)設(shè)備端的出口函數(shù)
{
	i2c_del_driver(&i2c_drv);
}

module_init(i2c_drv_init);
module_exit(i2c_drv_exit);
MODULE_LICENSE("GPL");

3.2 設(shè)備端代碼

#include 
#include 
#include 
#include 

/*獲取總線*/
struct i2c_adapter *i2c_adap;  //獲取到的總線存放在這個(gè)結(jié)構(gòu)體
static struct i2c_client *i2cClient = NULL;

//PCF8591固定地址 b1001
//PCF8591硬件地址 b000
//組合:b1001000 = 0x48
//注意:IIC標(biāo)準(zhǔn)地址是7位
static unsigned short const i2c_addr_list[] = 
{ 
	0x48, I2C_CLIENT_END
};//地址隊(duì)列
static int __init i2c_dev_init(void)
{
	struct i2c_board_info i2c_info;//設(shè)備描述結(jié)構(gòu)體,里面存放著欲設(shè)備的名字還有地址
	i2c_adap = i2c_get_adapter(0); //獲取0號(hào)總線
	if(i2c_adap==NULL)
	{
		printk("PCF8591--II總線0 獲取失敗!!\n");
	}
	
	memset(&i2c_info,0,sizeof(struct i2c_board_info));//把設(shè)備描述結(jié)構(gòu)體清空結(jié)構(gòu)體清空
	strlcpy(i2c_info.type,"Tiny4412_PCF8591",I2C_NAME_SIZE);//把設(shè)備的名字賦值給i2c_info
	
	i2cClient = i2c_new_probed_device(i2c_adap,&i2c_info,i2c_addr_list,NULL);
	if(i2cClient==NULL)
	{
		printk("PCF8591 0x%x:地址不可用!!\n",i2c_addr_list[0]);
	}
	i2c_put_adapter(i2c_adap);
	printk("PCF8591_dev_init初始化成功!!\n");
	return 0;
}
static void __exit i2c_dev_exit(void)//平臺(tái)設(shè)備端的出口函數(shù)
{
	/*注銷設(shè)備*/
	i2c_unregister_device(i2cClient);
	i2c_release_client(i2cClient);
	printk("PCF8591_dev_exit ok!!\n");
}

module_init(i2c_dev_init);
module_exit(i2c_dev_exit);
MODULE_LICENSE("GPL");

3.3 應(yīng)用層代碼

#include 
#include 
#include 
#include 
/*
PCF8591 應(yīng)用層測(cè)試代碼
*/
int main(int argc,char **argv)
{
	unsigned char data=0;
    int fp;
	float tmp; //  tmp=5.34v   0.34
	int a;
	int b;	
	fp=open("/dev/Tiny4412_PCF8591",O_RDWR);
	if(fp<0)      /*判斷文件是否打開成功*/
	{
		printf("PCF8591 driver open error!\n");
		return -1;
	}
	while(1)
	{
		read(fp,&data,1);
		write(fp,&data,1);
		printf("ADC1=%d\n",data);
		tmp=(float)data*(5.0/255); //電壓= 采集的數(shù)字量*(參考電壓/分辨率);
		a=tmp;  //a=5  tmp=5.3
		b=(int)((tmp-a)*1000);    //b=0.34
		printf("ADC1=%d.%dV\r\n",(int)a,(int)b);  
		sleep(1);
	}
	close(fp);
	return 0;
}
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • adc
    adc
    +關(guān)注

    關(guān)注

    100

    文章

    7511

    瀏覽量

    555927
  • 轉(zhuǎn)換芯片
    +關(guān)注

    關(guān)注

    0

    文章

    85

    瀏覽量

    12163
  • PCF8591
    +關(guān)注

    關(guān)注

    3

    文章

    67

    瀏覽量

    33770
收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    深度解析ES8389/ES8390/音頻芯片Linux驅(qū)動(dòng)Linux6.1內(nèi)核)

    基于 Linux6.1 內(nèi)核,從驅(qū)動(dòng)架構(gòu)、寄存器配置、核心函數(shù)、數(shù)據(jù)流走向四個(gè)維度,完整拆解 ES8389 的 Linux 驅(qū)動(dòng)實(shí)現(xiàn),幫你吃透這款
    的頭像 發(fā)表于 02-02 11:37 ?1329次閱讀
    深度解析ES8389/ES8390/音頻<b class='flag-5'>芯片</b><b class='flag-5'>Linux</b><b class='flag-5'>驅(qū)動(dòng)</b>(<b class='flag-5'>Linux</b>6.1內(nèi)核)

    RK?平臺(tái)?SPI?開發(fā)完全指南(驅(qū)動(dòng)?+?配置?+?測(cè)試?+?優(yōu)化)

    ? ? ? 在嵌入式 Linux 開發(fā)中, SPI (串行外設(shè)接口)憑借高速同步通信特性,廣泛應(yīng)用于傳感器、存儲(chǔ)芯片、顯示模塊等外設(shè)交互場(chǎng)景。瑞芯微( Rockchip )作為主流嵌入式芯片
    的頭像 發(fā)表于 01-30 22:35 ?629次閱讀
    RK?平臺(tái)?SPI?<b class='flag-5'>開發(fā)</b>完全指南(<b class='flag-5'>驅(qū)動(dòng)</b>?+?配置?+?測(cè)試?+?優(yōu)化)

    迅為Hi3403開發(fā)驅(qū)動(dòng)教程全面上線,開啟嵌入式Linux開發(fā)新篇章!

    迅為iTOP-Hi3403開發(fā)linux驅(qū)動(dòng)教程全面上線,開啟嵌入式Linux開發(fā)新篇章!
    的頭像 發(fā)表于 12-18 14:23 ?677次閱讀
    迅為Hi3403<b class='flag-5'>開發(fā)</b>板<b class='flag-5'>驅(qū)動(dòng)</b>教程全面上線,開啟嵌入式<b class='flag-5'>Linux</b><b class='flag-5'>開發(fā)</b>新篇章!

    Linux驅(qū)動(dòng)開發(fā)的必備知識(shí)

    、內(nèi)核模塊編程: 掌握內(nèi)核模塊的編寫、編譯、加載和卸載方法。 了解內(nèi)核模塊的初始化和清理函數(shù)的編寫。 5、設(shè)備驅(qū)動(dòng)框架: 熟悉字符設(shè)備、塊設(shè)備、網(wǎng)絡(luò)設(shè)備等驅(qū)動(dòng)框架。 能夠根據(jù)
    發(fā)表于 12-04 07:58

    迅為iTOP-3568開發(fā)板?Linux驅(qū)動(dòng)開發(fā)實(shí)戰(zhàn):menuconfig圖形化配置實(shí)驗(yàn)

    迅為iTOP-3568開發(fā)板?Linux驅(qū)動(dòng)開發(fā)實(shí)戰(zhàn):menuconfig圖形化配置實(shí)驗(yàn)
    的頭像 發(fā)表于 11-24 15:29 ?766次閱讀
    迅為iTOP-3568<b class='flag-5'>開發(fā)</b>板?<b class='flag-5'>Linux</b><b class='flag-5'>驅(qū)動(dòng)</b><b class='flag-5'>開發(fā)</b>實(shí)戰(zhàn):menuconfig圖形化配置實(shí)驗(yàn)

    【迅為工業(yè)RK3568穩(wěn)定可靠】itop-3568開發(fā)Linux驅(qū)動(dòng)開發(fā)實(shí)戰(zhàn):RK3568內(nèi)核模塊符號(hào)導(dǎo)出詳解

    【迅為工業(yè)RK3568穩(wěn)定可靠】itop-3568開發(fā)Linux驅(qū)動(dòng)開發(fā)實(shí)戰(zhàn):RK3568內(nèi)核模塊符號(hào)導(dǎo)出詳解
    的頭像 發(fā)表于 11-21 13:25 ?1185次閱讀
    【迅為工業(yè)RK3568穩(wěn)定可靠】itop-3568<b class='flag-5'>開發(fā)</b>板<b class='flag-5'>Linux</b><b class='flag-5'>驅(qū)動(dòng)</b><b class='flag-5'>開發(fā)</b>實(shí)戰(zhàn):RK3568內(nèi)核模塊符號(hào)導(dǎo)出詳解

    【免費(fèi)送書】成為硬核Linux開發(fā)者:《Linux 設(shè)備驅(qū)動(dòng)開發(fā)(第 2 版)》

    Linux系統(tǒng)的設(shè)備驅(qū)動(dòng)開發(fā),一直給人門檻較高的印象,主要因內(nèi)核機(jī)制抽象、需深度理解硬件原理、開發(fā)調(diào)試難度大所致。2021年,一本講解驅(qū)動(dòng)
    的頭像 發(fā)表于 11-18 08:06 ?1394次閱讀
    【免費(fèi)送書】成為硬核<b class='flag-5'>Linux</b><b class='flag-5'>開發(fā)</b>者:《<b class='flag-5'>Linux</b> 設(shè)備<b class='flag-5'>驅(qū)動(dòng)</b><b class='flag-5'>開發(fā)</b>(第 2 版)》

    【書籍評(píng)測(cè)活動(dòng)NO.67】成為硬核Linux開發(fā)者:《Linux 設(shè)備驅(qū)動(dòng)開發(fā)(第 2 版)》

    ,開發(fā)設(shè)備驅(qū)動(dòng)和板級(jí)支持包。截至目前,Linux仍是嵌入式系統(tǒng)領(lǐng)域的主流內(nèi)核,幾乎廣泛應(yīng)用于工業(yè)界的所有領(lǐng)域,這主要得益于其功能強(qiáng)大的子系統(tǒng)。因此,約翰·馬迪厄編寫本書,向廣大
    發(fā)表于 11-17 17:52

    迅為3568開發(fā)板從零學(xué)習(xí)Linux驅(qū)動(dòng)開發(fā):迅為一站式資料包如何讓我效率翻倍

    迅為3568開發(fā)板從零學(xué)習(xí)Linux驅(qū)動(dòng)開發(fā):迅為一站式資料包如何讓我效率翻倍
    的頭像 發(fā)表于 11-05 11:16 ?400次閱讀
    迅為3568<b class='flag-5'>開發(fā)</b>板從零學(xué)習(xí)<b class='flag-5'>Linux</b><b class='flag-5'>驅(qū)動(dòng)</b><b class='flag-5'>開發(fā)</b>:迅為一站式資料包如何讓我效率翻倍

    itop-3568開發(fā)驅(qū)動(dòng)開發(fā)指南-實(shí)驗(yàn)程序的編寫

    本實(shí)驗(yàn)對(duì)應(yīng)的網(wǎng)盤路徑為:iTOP-RK3568 開發(fā)板【底板 V1.7 版本】\\\\03_【iTOP-RK3568開發(fā)板】指南教程\\\\02_Linux 驅(qū)動(dòng)配套資料\\\\04_
    發(fā)表于 05-19 10:26

    納祥科技NX6806中文規(guī)格書,8位AD和DA單片機(jī)拓展,國(guó)產(chǎn)替代PCF8591

    連接到I2C總線,不帶額外的硬件。NX6806通過(guò)雙線串行傳輸設(shè)備之間的地址,控制和數(shù)據(jù)雙向I2C總線,設(shè)備的功能包括模擬輸入多路復(fù)用,片上跟蹤和保持功能。性能上,NX6806可國(guó)產(chǎn)替代PCF8591。
    發(fā)表于 04-18 17:30 ?1次下載

    硬核升級(jí)!華清遠(yuǎn)見(jiàn)STM32MP157驅(qū)動(dòng)開發(fā)課程助力嵌入式Linux底層開發(fā)入門進(jìn)階

    在嵌入式Linux系統(tǒng)開發(fā)中,驅(qū)動(dòng)程序開發(fā)是一項(xiàng)關(guān)鍵技術(shù),它作為硬件與軟件之間的橋梁,實(shí)現(xiàn)了操作系統(tǒng)對(duì)硬件設(shè)備的控制。相較于嵌入式Linux
    的頭像 發(fā)表于 04-03 16:40 ?1008次閱讀
    硬核升級(jí)!華清遠(yuǎn)見(jiàn)STM32MP157<b class='flag-5'>驅(qū)動(dòng)</b><b class='flag-5'>開發(fā)</b>課程助力嵌入式<b class='flag-5'>Linux</b>底層<b class='flag-5'>開發(fā)</b>入門進(jìn)階

    迅為RK3568開發(fā)板helloworld 驅(qū)動(dòng)實(shí)驗(yàn)-驅(qū)動(dòng)編寫

    在學(xué)習(xí) C 語(yǔ)言或者其他語(yǔ)言的時(shí)候,我們通常是打印一句“helloworld”來(lái)開啟編程世界的大門。學(xué)習(xí)驅(qū)動(dòng)程序編程亦可以如此,使用 helloworld 作為我們的第一個(gè)驅(qū)動(dòng)程序。接下來(lái)開始編寫
    發(fā)表于 04-01 15:20

    RTC芯片Linux PCA2131驅(qū)動(dòng)程序嗎?

    RTC 芯片Linux PCA2131驅(qū)動(dòng)程序嗎? 1) 如果沒(méi)有,我可以使用任何兼容的驅(qū)動(dòng)程序來(lái)驅(qū)動(dòng)這個(gè) RTC
    發(fā)表于 03-31 06:22

    如何使用FPGA驅(qū)動(dòng)并行ADC和DAC芯片,使用不同編碼方式的ADC與DAC時(shí)的注意事項(xiàng)

    ADC和DAC是FPGA與外部信號(hào)的接口,從數(shù)據(jù)接口類型的角度劃分,有低速的串行接口和高速的并行接口。FPGA經(jīng)常用來(lái)采集中高頻信號(hào),因此使用并行ADC和DAC居多。本文將介紹如何使用FPGA驅(qū)動(dòng)并行
    的頭像 發(fā)表于 03-14 13:54 ?2190次閱讀
    如何使用FPGA<b class='flag-5'>驅(qū)動(dòng)</b>并行<b class='flag-5'>ADC</b>和DAC<b class='flag-5'>芯片</b>,使用不同編碼方式的<b class='flag-5'>ADC</b>與DAC時(shí)的注意事項(xiàng)