I2C驱动框架之probe方式

基于Linux的I2C驱动,采用probe方式。按照如下这个框架可以写出任何支持I2C总线的器件的Linux驱动。

I2C器件连接至cpu的特定的i2c的接口,也就是挂载在cpu的i2c适配器上,i2c器件要和cpu进行信息交换必须要通过cpu操作适配器来交互。cpu上有1个或多个适配器,每个适配器上可以挂载256个设备地址不一样的i2c器件,通过i2c驱动就可以让cpu和适配器上的多个不一样的i2c器件通信而不会产生冲突。

驱动包括两个文件,dev.c和drive.c,其中dev.c是构建I2C设备,即创建I2C_Client结构体,而driver.c是在probe中获取dev.c中构建的i2c_client,然后构建fileoperation;具体步骤结合代码如下。

1、在dev的入口函数中构建I2C-CLient

static unsigned short addr_list[] = {
		0x60, 0x50, I2C_CLIENT_END   //这个数组包含了设备地址,以I2C_CLIENT_END为数组结尾</span>

};
static struct i2c_client *at24cxx_client;    //创建I2C_Client结构体
static int at24cxx_dev_init(void)
{
	/*构建I2C_Client*/
	struct i2c_adapter *adapter;
	struct i2c_board_info info;
	adapter=i2c_get_adapter(0);         //获取适配器,因为有些cpu有多个I2C适配器,参数0为适配器编号

	memset(&info, 0, sizeof(struct i2c_board_info));
	strlcpy(info.type, "at24cxx", I2C_NAME_SIZE);    //这个是I2C_NAME,是和i2c_driver匹配的关键字
	at24cxx_client=i2c_new_probed_device(adapter, &info, addr_list);//如果数组中地址的设备存在则成功返回i2c_client结构体
//这个函数最终会调用device_create创建设备

	i2c_put_adapter(adapter);
	if(at24cxx_client)
		return 0;
	else
		return -ENODEV;
}

2、在driver.c的入口函数中注册I2C_driver结构体

static const struct i2c_device_id = {
	{ "at24cxx", 0 },
	{ }
};


static struct i2c_driver at24cxx_driver = {
		.driver = {
			.name = "100ask",
			.owner = THIS_MODULE,
		},
		.probe = at24cxx_probe,
		.remove = __devexit_p(at24cxx_remove),
		.id_table = at24cxx_ids,//id_table中name是和dev创建i2c_client的name匹配,若匹配则会调用probe设备方法
};

static int at24cxx_drv_init(void)
{
	/*×¢²ái2c_driver*/

	return i2c_add_driver(&at24cxx_driver);

}

3、在probe设备方法中构建file_operation注册字符设备

static struct file_operations at24cxx_fops =
{
	.owner = THIS_MODULE,
	.read = at24cxx_read,
	.write = at24cxx_write,

};

static int at24cxx_probe(struct i2c_client *client, const struct i2c_device_id *id)
{
		printk("%s %s %d.\n",__FILE__,__FUNCTION__,__LINE__);
		at24cxx_client = client;//将dev中构建的i2c_client结构体传递过来,因为这个结构体在read,write等设备方法中传递信息时需要用到
		major = register_chrdev(0,"at24cxx",&at24cxx_fops);//注册字符设备
		class = class_create(THIS_MODULE,"at24cxx");
		device_create(class,NULL,MKDEV(major,0),NULL,"at24cxx");
		return 0;
}

4、构建write和read等设备方法中传递I2C消息

static ssize_t at24cxx_read(struct file *file, char __user *buf, size_t count, loff_t *off)
{
	unsigned char addr,data;
	copy_from_user(&addr,buf,1);
	data = i2c_smbus_read_byte_data(at24cxx_client,addr);//这里用i2c_smbus_read_byte_data函数来和实际的I2C设备进行信息交互</span>
//这里具体用哪个函数来传递消息需要结合具体的器件的时序,不同的时序有不同的传递函数,查看Linux源码中的说明文档有关于传递函数的说明。
	copy_to_user(buf,&data,1);
	return 0;
}
static ssize_t at24cxx_write(struct file *file, const char __user *buf, size_t count , loff_t *off)
{
	unsigned char ker_buf[2];
	unsigned char addr,data;
	copy_from_user(ker_buf,buf,2);
	addr = ker_buf[0];
	data = ker_buf[1];

	if(!i2c_smbus_write_byte_data(at24cxx_client,addr,data))
		return 2;
	else
		return -EIO;

}

如果i2c总线中挂载了实际的i2c设备,而且设备地址在以上的addr_list中,则无论是先加载dev.ko还是先加载driver.ko都会成功的执行probe函数,然后创建字符设备,在应用程序中open设备,并调用read和write则会相应的调用driver中的read和write设备方法。

测试应用程序

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

/* i2c_test r addr
 * i2c_test w addr val
 */

void print_usage(char *file)
{
	printf("%s r addr\n", file);
	printf("%s w addr val\n", file);
}

int main(int argc, char **argv)
{
	int fd;
	unsigned char buf[2];

	if ((argc != 3) && (argc != 4))
	{
		print_usage(argv[0]);
		return -1;
	}

	fd = open("/dev/at24cxx", O_RDWR);
	if (fd < 0)
	{
		printf("can't open /dev/at24cxx\n");
		return -1;
	}

	if (strcmp(argv[1], "r") == 0)
	{

		buf[0] = strtoul(argv[2], NULL, 0);
		printf("before data: %c, %d, 0x%2x\n", buf[0], buf[0], buf[0]);
		read(fd, buf, 1);
		printf("data: %c, %d, 0x%2x\n", buf[0], buf[0], buf[0]);
	}
	else if ((strcmp(argv[1], "w") == 0) && (argc == 4))
	{
		buf[0] = strtoul(argv[2], NULL, 0);
		buf[1] = strtoul(argv[3], NULL, 0);
		if (write(fd, buf, 2) != 2)
			printf("write err, addr = 0x%02x, data = 0x%02x\n", buf[0], buf[1]);
	}
	else
	{
		print_usage(argv[0]);
		return -1;
	}

	return 0;
}

2014--12--17

征途开始

后续补充

时间: 2024-10-05 07:14:36

I2C驱动框架之probe方式的相关文章

I2C驱动框架(四)

参考:I2C子系统之platform_driver初始化--I2C_adap_s3c_init() 在完成platform_device的添加之后,i2c子系统将进行platform_driver的注册过程.platform_driver的注册通过调用初始化函数i2c_adapter_s3c_init函数来完成. static struct platform_device_id s3c24xx_driver_ids[] = { { .name = "s3c2410-i2c", .dri

I2C驱动框架

以用i2c通信的实时时钟为例 框架入口源文件:i2c_m41t11.c (可根据入口源文件,再按着框架到内核走一遍) 内核版本:linux_2.6.22.6   硬件平台:JZ2440 以下是驱动框架: 以下是驱动代码 i2c_m41t11.c : #include <linux/kernel.h> #include <linux/init.h> #include <linux/module.h> #include <linux/slab.h> #inclu

I2C驱动框架(二)

在linux内核启动的时候最先执行的和I2C子系统相关的函数应该是driver/i2c/i2c-core.c文件中的i2c_init()函数. 1 static int __init i2c_init(void) 2 { 5 retval = bus_register(&i2c_bus_type); 9 i2c_adapter_compat_class = class_compat_register("i2c-adapter"); 15 retval = i2c_add_dri

spi驱动框架全面分析,从master驱动到设备驱动

内核版本:linux2.6.32.2 硬件资源:s3c2440 参考:  韦东山SPI视频教程 内容概括: 1.I2C 驱动框架回顾 2.SPI 框架简单介绍 3.master 驱动框架 3.1 驱动侧 3.2 设备侧 4.SPI 设备驱动框架 4.1 设备册 4.2 驱动侧 5.设备驱动程序实例 1.I2C 驱动框架回顾 在前面学习 I2C 驱动程序的时候我们知道,I2C 驱动框架分为两层,一层是控制器驱动程序 i2c_adapter,它一般是由芯片厂商写好的,主要提供一个 algorithm

I2C驱动详解

I2C讲解: 在JZ2440开发板上,I2C是由两条数据线构成的SCL,SDA:SCL作为时钟总线,SDA作为数据总线:两条线上可挂载I2C设备,如:AT24C08 两条线连接ARM9 I2C控制器,通过控制来控制I2C设备的识别设备地址.读.写操作:如图所示 从中所知:I2C线上可以挂载很多个I2C设备:挂载简单,只需要一根数据线和一根时钟线就可以挂载上去,通过地址来去别每个设备的区别: I2C操作: 对I2C操作主要思想为:1.找到设备  2.进行读写操作 主要原理为: 1.发送开始信号S

TQ2440学习笔记——Linux上I2C驱动的两种实现方法(1)

作者:彭东林 邮箱:[email protected] 内核版本:Linux-3.14 u-boot版本:U-Boot 2015.04 硬件:TQ2440 (NorFlash:2M   NandFlash:256M  内存:64M) 摘要 这里并不深入分析Linux下I2C驱动的实现,只是以TQ2440硬件平台为例分析I2C驱动的两种方法. 第一种方法: 使用S3C2440自带的I2C控制器实现,这个kernel已经支持,我们只需要配置即可. 第二种方法: 使用GPIO模拟,这个在kernel中

Linux I2C驱动编写要点

继续上一篇博文没讲完的内容“针对 RepStart 型i2c设备的驱动模型”,其中涉及的内容有:i2c_client 的注册.i2c_driver 的注册.驱动程序的编写. 一.i2c 设备的注册分析:在新版本内核的i2c驱动模型中,支持多种方式来注册 i2c 设备,在Documentation/i2c/instantiating-devices文件中有讲到,在内核中对应的抽象数据结构就是 struct i2c_client. (1)Declare the I2C devices by bus

3.4.2内核下的I2C驱动

1. 框架1.1 硬件协议简介1.2 驱动框架1.3 bus-drv-dev模型及写程序a. 设备的4种构建方法a.1 定义一个i2c_board_info, 里面有:名字, 设备地址 然后i2c_register_board_info(busnum, ...) (把它们放入__i2c_board_list链表) list_add_tail(&devinfo->list, &__i2c_board_list); 链表何时使用: i2c_register_adapter > i2

Linux USB驱动框架分析 【转】

转自:http://blog.chinaunix.net/uid-11848011-id-96188.html 初次接触与OS相关的设备驱动编写,感觉还挺有意思的,为了不至于忘掉看过的东西,笔记跟总结当然不可缺,更何况我决定为嵌入式卖命了.好,言归正传,我说一说这段时间的收获,跟大家分享一下Linux的驱动开发.但这次只先针对Linux的USB子系统作分析,因为周五研讨老板催货.当然,还会顺带提一下其他的驱动程序写法. 事实上,Linux的设备驱动都遵循一个惯例——表征驱动程序(用driver更