在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

Linux驅動開發-編寫PCF8591(ADC)芯片驅動

DS小龍哥-嵌入式技術 ? 2022-09-17 15:16 ? 次閱讀

【摘要】 PCF8591是一個IIC總線接口的ADC/DAC轉換芯片,功能比較強大,這篇文章就介紹在Linux系統里如何編寫一個PCF8591的驅動,完成ADC數據采集,DAC數據輸出。

1. PCF8591介紹

PCF8591是一個IIC總線接口的ADC/DAC轉換芯片,功能比較強大,這篇文章就介紹在Linux系統里如何編寫一個PCF8591的驅動,完成ADC數據采集,DAC數據輸出。

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

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

PCF8591采用典型的I2C總線接口器件尋址方法,即總線地址由器件地址、引腳地址和方向位組成。飛利蒲公司規定A/D器件地址為1001。引腳地址為A2A1A0,其值由用戶選擇,因此I2C系統中最多可接8個具有I2C總線接口的A/D器件。地址的最后一位為方向位R/W,當主控器對A/D器件進行讀操作時為1,進行寫操作時為0。總線操作時,由器件地址、引腳地址和方向位組成的從地址為主控器發送的第一字節。

2. 硬件環境介紹

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

image-20220106091151114

下面是PCF8591的原理圖,介紹了每個引腳詳細功能:

image-20220106091509038image-20220106091635195

3. 驅動案例代碼

下面是PCF8591的驅動代碼,采用IIC子系統框架編程,驅動代碼分為設備端、驅動端兩部分。

驅動框架采用雜項字符設備完成注冊,給應用層提供訪問的設備節點,詳細的說明在代碼路寫了完整的注釋。

3.1 驅動端代碼

#include 
#include 
#include 
#include 
#include   /*注冊中斷相關*/
#include  		  /*中斷邊沿類型定義*/
#include  	  /*中斷IO口定義*/
#include   /*工作隊列相關*/
#include       /*互斥信號量頭文件*/
#include 
#include  /*雜項設備相關結構體*/
#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設備總線*/

/*讀取PCF8591  ADC數據*/
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;
}
/*定義一個文件操作集合結構體*/
static struct file_operations ops_PCF8591={
   .owner = THIS_MODULE,
   .read=PCF8591_read,       /*讀函數-被應用層read函數調用*/
   .write=PCF8591_write,     /*寫函數-被應用層write函數調用*/
   .open=PCF8591_open,       /*打開函數-被應用層open函數調用*/
   .release=PCF8591_release, /*釋放函數*/
};

/*定義一個雜項設備結構體*/
static struct miscdevice misce_PCF8591={
	.minor =MISC_DYNAMIC_MINOR, /*自動分配次設備號*/
	.name = "Tiny4412_PCF8591",  			/*名稱  在dev/目錄下邊可以找到*/
	.fops = &ops_PCF8591, 			/*文件操作集合*/
};

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

/*
IIC驅動端
*/
static const struct i2c_device_id i2c_id[] =
{
	{"Tiny4412_PCF8591",0},//設備端的名字為"my_PCF8591",后面的表示需要私有數據
	{}
};

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總線注冊一個驅動
	return 0;
}

static void __exit i2c_drv_exit(void)//平臺設備端的出口函數
{
	i2c_del_driver(&i2c_drv);
}

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

3.2 設備端代碼

#include 
#include 
#include 
#include 

/*獲取總線*/
struct i2c_adapter *i2c_adap;  //獲取到的總線存放在這個結構體
static struct i2c_client *i2cClient = NULL;

//PCF8591固定地址 b1001
//PCF8591硬件地址 b000
//組合:b1001000 = 0x48
//注意:IIC標準地址是7位
static unsigned short const i2c_addr_list[] = 
{ 
	0x48, I2C_CLIENT_END
};//地址隊列
static int __init i2c_dev_init(void)
{
	struct i2c_board_info i2c_info;//設備描述結構體,里面存放著欲設備的名字還有地址
	i2c_adap = i2c_get_adapter(0); //獲取0號總線
	if(i2c_adap==NULL)
	{
		printk("PCF8591--II總線0 獲取失敗!!\n");
	}
	
	memset(&i2c_info,0,sizeof(struct i2c_board_info));//把設備描述結構體清空結構體清空
	strlcpy(i2c_info.type,"Tiny4412_PCF8591",I2C_NAME_SIZE);//把設備的名字賦值給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)//平臺設備端的出口函數
{
	/*注銷設備*/
	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 應用層代碼

#include 
#include 
#include 
#include 
/*
PCF8591 應用層測試代碼
*/
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); //電壓= 采集的數字量*(參考電壓/分辨率);
		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;
}
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • adc
    adc
    +關注

    關注

    99

    文章

    6572

    瀏覽量

    546874
  • 轉換芯片
    +關注

    關注

    0

    文章

    75

    瀏覽量

    11488
  • PCF8591
    +關注

    關注

    3

    文章

    67

    瀏覽量

    33069
收藏 人收藏

    評論

    相關推薦

    如何使用FPGA驅動并行ADC和DAC芯片,使用不同編碼方式的ADC與DAC時的注意事項

    ADC和DAC是FPGA與外部信號的接口,從數據接口類型的角度劃分,有低速的串行接口和高速的并行接口。FPGA經常用來采集中高頻信號,因此使用并行ADC和DAC居多。本文將介紹如何使用FPGA驅動并行
    的頭像 發表于 03-14 13:54 ?283次閱讀
    如何使用FPGA<b class='flag-5'>驅動</b>并行<b class='flag-5'>ADC</b>和DAC<b class='flag-5'>芯片</b>,使用不同編碼方式的<b class='flag-5'>ADC</b>與DAC時的注意事項

    恩智浦為無線連接SoC開發的統一WiFi驅動程序多芯片多接口驅動(MXM)

    本文將重點介紹恩智浦為無線連接SoC開發的統一Wi-Fi驅動程序——多芯片多接口驅動 (MXM),詳細說明其架構設計如何簡化基于恩智浦無線連接SoC和i.MX應用處理器的
    發表于 02-28 09:13 ?319次閱讀
    恩智浦為無線連接SoC<b class='flag-5'>開發</b>的統一WiFi<b class='flag-5'>驅動</b>程序多<b class='flag-5'>芯片</b>多接口<b class='flag-5'>驅動</b>(MXM)

    納祥科技NX6806,8位A/D和D/A轉換器,國產替代PCF8591

    NX6806是一種單芯片、單電源、低功耗的8位CMOS數據采集設備,具有四個模擬輸入、一個模擬輸出和一個串行I2C總線接口。 三個地址引腳A0、A1和A2用于對硬件地址進行編程,允許使用最多八個設備連接到I2C總線不帶額外的硬件。 性能上,NX6806可國產替代PCF8591
    的頭像 發表于 02-05 17:25 ?120次閱讀
    納祥科技NX6806,8位A/D和D/A轉換器,國產替代<b class='flag-5'>PCF8591</b>

    迅為RK3568開發驅動指南Linux中通用SPI設備驅動

    迅為RK3568開發驅動指南Linux中通用SPI設備驅動
    的頭像 發表于 01-23 11:02 ?2071次閱讀
    迅為RK3568<b class='flag-5'>開發</b>板<b class='flag-5'>驅動</b>指南<b class='flag-5'>Linux</b>中通用SPI設備<b class='flag-5'>驅動</b>

    迅為RK3568開發板SPI驅動指南-mcp2515驅動編寫:讀寄存器函數

    迅為RK3568開發板SPI驅動指南-mcp2515驅動編寫:讀寄存器函數
    的頭像 發表于 01-20 14:43 ?586次閱讀
    迅為RK3568<b class='flag-5'>開發</b>板SPI<b class='flag-5'>驅動</b>指南-mcp2515<b class='flag-5'>驅動</b><b class='flag-5'>編寫</b>:讀寄存器函數

    納祥科技NX6806,低功耗8位A/D和D/A轉換器,國產替代PCF8591

    PCF8591是一款具有I2C接口的8位A/D和D/A轉換器,它采用了先進的CMOS工藝技術制造而成。隨著國內國產化的興起,這一經典芯片可被不少國內芯片所替代,如NX6806,它在原版的性能、功能上
    的頭像 發表于 01-04 08:56 ?345次閱讀
    納祥科技NX6806,低功耗8位A/D和D/A轉換器,國產替代<b class='flag-5'>PCF8591</b>

    基于恩智浦FRDM-MCXA153開發板實現RT-Thread的ADC驅動

    數轉換器(ADC)是現代嵌入式系統中不可或缺的組件,它能將連續的模擬信號轉換為離散的數字信號。本文將深入探討如何在NXP的FRDM-MCXA153開發板上實現和使用RT-Thread的ADC
    的頭像 發表于 11-01 12:29 ?653次閱讀
    基于恩智浦FRDM-MCXA153<b class='flag-5'>開發</b>板實現RT-Thread的<b class='flag-5'>ADC</b><b class='flag-5'>驅動</b>

    北京迅為RK3568開發板嵌入式學習之Linux驅動全新更新-CAN+

    北京迅為RK3568開發板嵌入式學習之Linux驅動全新更新-CAN+
    的頭像 發表于 09-04 15:29 ?732次閱讀
    北京迅為RK3568<b class='flag-5'>開發</b>板嵌入式學習之<b class='flag-5'>Linux</b><b class='flag-5'>驅動</b>全新更新-CAN+

    linux系統的設備驅動一般分幾類

    Linux系統的設備驅動是操作系統與硬件設備之間的橋梁,負責實現操作系統與硬件設備之間的通信和控制。Linux系統的設備驅動可以分為以下幾類: 字符設備
    的頭像 發表于 08-30 15:13 ?665次閱讀

    Linux設備驅動程序分類有哪些

    Linux設備驅動程序是操作系統與硬件設備之間的橋梁,負責實現硬件設備與操作系統之間的通信和控制。Linux設備驅動程序的分類繁多,可以根據不同的標準進行分類。 按硬件類型分類
    的頭像 發表于 08-30 15:11 ?839次閱讀

    linux驅動程序如何加載進內核

    Linux系統中,驅動程序是內核與硬件設備之間的橋梁。它們允許內核與硬件設備進行通信,從而實現對硬件設備的控制和管理。 驅動程序的編寫 驅動
    的頭像 發表于 08-30 15:02 ?741次閱讀

    linux驅動程序的編譯方法是什么

    Linux驅動程序的編譯方法主要包括兩種: 與內核一起編譯 和 編譯成獨立的內核模塊 。以下是對這兩種方法的介紹: 一、與內核一起編譯 與內核一起編譯意味著將驅動程序的源代碼直接集成到Linu
    的頭像 發表于 08-30 14:46 ?875次閱讀

    Linux 驅動開發與應用開發,你知道多少?

    一、Linux驅動開發與應用開發的區別開發層次不同:Linux
    的頭像 發表于 08-30 12:16 ?1094次閱讀
    <b class='flag-5'>Linux</b> <b class='flag-5'>驅動</b><b class='flag-5'>開發</b>與應用<b class='flag-5'>開發</b>,你知道多少?

    SPI+DMA方式驅動ADC芯片

    有SPI+DMA方式驅動ADC芯片的例程嗎
    發表于 04-26 17:53

    ArmSoM系列板卡 嵌入式Linux驅動開發實戰指南 之 字符設備驅動

    字符設備驅動 本章,我們將學習字符設備使用、字符設備驅動相關的概念,理解字符設備驅動程序的基本框架,并從源碼上分析字符設備驅動實現和管理等。 主要分為下面五部分:
    的頭像 發表于 04-10 09:53 ?1187次閱讀
    ArmSoM系列板卡 嵌入式<b class='flag-5'>Linux</b><b class='flag-5'>驅動</b><b class='flag-5'>開發</b>實戰指南 之 字符設備<b class='flag-5'>驅動</b>
    主站蜘蛛池模板: 手机看片福利在线 | 成人看的午夜免费毛片 | 全部在线播放免费毛片 | 一级做a爱免费观看视频 | www久久久| 国产精品免费看久久久久 | 午夜精品福利影院 | 国产午夜一区二区在线观看 | 唐人社电亚洲一区二区三区 | 国产香港三级理论在线 | 国产精品一久久香蕉产线看 | 玖玖在线精品 | 久久国内精品视频 | 国产大片免费观看资源 | 丁香六月婷婷激情 | 男人的天堂视频网站清风阁 | 亚洲线精品一区二区三区 | 黄色免费网站视频 | 黄色免费小视频 | 丁香六月婷婷七月激情 | 国产特黄 | 狠狠色狠色综合曰曰 | 美女扒开尿口给男人爽的视频 | 国产成都一二三四区 | 爆操欧美 | 美女被草视频 | 久久国产乱子伦精品免费看 | 永久黄色免费网站 | 操美女视频网站 | 国产理论在线 | 一本大道一卡二卡四卡 | v天堂网| h视频在线播放 | 亚洲理论片在线观看 | 一个人看aaaa免费中文 | 劳拉淫欲护士bd字幕 | 免费看黄色小视频 | 欧美在线伊人 | 午夜神马福利 | 国产欧美精品一区二区色综合 | 久久久噜噜噜www成人网 |