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
    +关注

    关注

    100

    文章

    7407

    浏览量

    553816
  • 转换芯片
    +关注

    关注

    0

    文章

    80

    浏览量

    11966
  • PCF8591
    +关注

    关注

    3

    文章

    67

    浏览量

    33638
收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    Linux驱动开发的必备知识

    、内核模块编程: 掌握内核模块的编写、编译、加载和卸载方法。 了解内核模块的初始化和清理函数的编写。 5、设备驱动框架: 熟悉字符设备、块设备、网络设备等驱动框架。 能够根据
    发表于 12-04 07:58

    迅为iTOP-3568开发板 Linux驱动开发实战:menuconfig图形化配置实验

    迅为iTOP-3568开发板 Linux驱动开发实战:menuconfig图形化配置实验
    的头像 发表于 11-24 15:29 518次阅读
    迅为iTOP-3568<b class='flag-5'>开发</b>板 <b class='flag-5'>Linux</b><b class='flag-5'>驱动</b><b class='flag-5'>开发</b>实战:menuconfig图形化配置实验

    【迅为工业RK3568稳定可靠】itop-3568开发Linux驱动开发实战:RK3568内核模块符号导出详解

    【迅为工业RK3568稳定可靠】itop-3568开发Linux驱动开发实战:RK3568内核模块符号导出详解
    的头像 发表于 11-21 13:25 664次阅读
    【迅为工业RK3568稳定可靠】itop-3568<b class='flag-5'>开发</b>板<b class='flag-5'>Linux</b><b class='flag-5'>驱动</b><b class='flag-5'>开发</b>实战:RK3568内核模块符号导出详解

    【免费送书】成为硬核Linux开发者:《Linux 设备驱动开发(第 2 版)》

    Linux系统的设备驱动开发,一直给人门槛较高的印象,主要因内核机制抽象、需深度理解硬件原理、开发调试难度大所致。2021年,一本讲解驱动
    的头像 发表于 11-18 08:06 450次阅读
    【免费送书】成为硬核<b class='flag-5'>Linux</b><b class='flag-5'>开发</b>者:《<b class='flag-5'>Linux</b> 设备<b class='flag-5'>驱动</b><b class='flag-5'>开发</b>(第 2 版)》

    【书籍评测活动NO.67】成为硬核Linux开发者:《Linux 设备驱动开发(第 2 版)》

    开发设备驱动和板级支持包。截至目前,Linux仍是嵌入式系统领域的主流内核,几乎广泛应用于工业界的所有领域,这主要得益于其功能强大的子系统。因此,约翰·马迪厄编写本书,向广大
    发表于 11-17 17:52

    itop-3568开发驱动开发指南-实验程序的编写

    本实验对应的网盘路径为:iTOP-RK3568 开发板【底板 V1.7 版本】\\\\03_【iTOP-RK3568开发板】指南教程\\\\02_Linux 驱动配套资料\\\\04_
    发表于 05-19 10:26

    纳祥科技NX6806中文规格书,8位AD和DA单片机拓展,国产替代PCF8591

    连接到I2C总线,不带额外的硬件。NX6806通过双线串行传输设备之间的地址,控制和数据双向I2C总线,设备的功能包括模拟输入多路复用,片上跟踪和保持功能。性能上,NX6806可国产替代PCF8591
    发表于 04-18 17:30 1次下载

    硬核升级!华清远见STM32MP157驱动开发课程助力嵌入式Linux底层开发入门进阶

    在嵌入式Linux系统开发中,驱动程序开发是一项关键技术,它作为硬件与软件之间的桥梁,实现了操作系统对硬件设备的控制。相较于嵌入式Linux
    的头像 发表于 04-03 16:40 707次阅读
    硬核升级!华清远见STM32MP157<b class='flag-5'>驱动</b><b class='flag-5'>开发</b>课程助力嵌入式<b class='flag-5'>Linux</b>底层<b class='flag-5'>开发</b>入门进阶

    RTC芯片Linux PCA2131驱动程序吗?

    RTC 芯片Linux PCA2131驱动程序吗? 1) 如果没有,我可以使用任何兼容的驱动程序来驱动这个 RTC
    发表于 03-31 06:22

    如何使用FPGA驱动并行ADC和DAC芯片,使用不同编码方式的ADC与DAC时的注意事项

    ADC和DAC是FPGA与外部信号的接口,从数据接口类型的角度划分,有低速的串行接口和高速的并行接口。FPGA经常用来采集中高频信号,因此使用并行ADC和DAC居多。本文将介绍如何使用FPGA驱动并行
    的头像 发表于 03-14 13:54 1831次阅读
    如何使用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 1129次阅读
    恩智浦为无线连接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 654次阅读
    纳祥科技NX6806,8位A/D和D/A转换器,国产替代<b class='flag-5'>PCF8591</b>

    迅为RK3568开发驱动指南Linux中通用SPI设备驱动

    迅为RK3568开发驱动指南Linux中通用SPI设备驱动
    的头像 发表于 01-23 11:02 3433次阅读
    迅为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 1509次阅读
    迅为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 1215次阅读
    纳祥科技NX6806,低功耗8位A/D和D/A转换器,国产替代<b class='flag-5'>PCF8591</b>