i2clinux驱动
1. linux i2c驱动怎么写
1、使用linux系统i2c体系,包括设备驱动,总线驱动,一般总线驱动已经写好了,需要你写一个设备驱动
2、使用gpio模拟i2c协议
3、望采纳
4、谢谢
2. 请教i2c驱动测试 Linux交流区 ARM9之家论坛
这是IIC驱动中ioctl()的处理函数(源代码在drivers/i2c/i2c-dev.c)
static long i2cdev_ioctl(struct file *file, unsigned int cmd, unsigned long
arg)
{
struct i2c_client *client = (struct i2c_client *)file->private_data;
unsigned long funcs;
dev_dbg(&client->adapter->dev, "ioctl, cmd=0x%02x,
arg=0x%02lx\n",
cmd, arg);
switch ( cmd ) {
case I2C_SLAVE:
case I2C_SLAVE_FORCE:
/* NOTE: devices set up to work with "new style" drivers
* can't use I2C_SLAVE, even when the device node is not
* bound to a driver. Only I2C_SLAVE_FORCE will work.
*
* Setting the PEC flag here won't affect kernel drivers,
* which will be using the i2c_client node registered with
* the driver model core. Likewise, when that client has
* the PEC flag already set, the i2c-dev driver won't see
* (or use) this setting.
*/
if ((arg > 0x3ff) ||
(((client->flags & I2C_M_TEN) == 0) && arg >
0x7f))
return -EINVAL;
if (cmd == I2C_SLAVE &&
i2cdev_check_addr(client->adapter, arg))
return -EBUSY;
/* REVISIT: address could become busy later */
client->addr = arg;
return 0;
3. linux 怎么加载i2c驱动
假设手上有一块从淘宝上买来的开发板,我要在开发板的I2C总线上增加一个从设备(如at24c08),那么我要怎样写这个“I2C设备驱动”,让
应用程序可以访问at24c08呢?
先来看一个最简单的i2c设备驱动:
static struct i2c_board_info at24cxx_info = { //所支持的i2c设备的列表
I2C_BOARD_INFO("at24c08", 0x50), //一项代表一个支持的设备,它的名字叫做“at24c08”,器件地址是0x50
};
static struct i2c_client *at24cxx_client;
static int at24cxx_dev_init(void)
{
struct i2c_adapter *i2c_adap; //分配一个适配器的指针
i2c_adap = i2c_get_adapter(0); //调用core层的函数,获得一个i2c总线。这里我们已经知道新增的器件挂接在编号为0的i2c总线上
at24cxx_client = i2c_new_device(i2c_adap, &at24cxx_info); // 把i2c适配器和新增的I2C器件关联起来,这个用了i2c总线0,地址是0x50。这就组成了一个客户端
at24cxx_client i2c_put_adapter(i2c_adap);
return 0;
}
static void at24cxx_dev_exit(void)
{
i2c_unregister_device(at24cxx_client);
}
mole_init(at24cxx_dev_init);
mole_exit(at24cxx_dev_exit);
4. 用linux 调用内核中的统一I2C驱动 i2c总是 busy,求大神支招
希望能帮到你。
没这样用过,以前都是直接对/sys/bus/i2c/devices/0-0050/eeprom操作。
代码里有两次写,一次读,是在哪一次出错?
5. linux内核中i2c总线驱动对所有的i2c设备是否是通用的
i2C总线的驱动程序一般针对不同的CPU是不一样的,所以都位于arch目录下对应的cpu架构的common文件夹下。
对同一种架构的来看,I2C驱动仅实现底层的通信。故其是通用的。
6. 怎样在linux环境下轻松实现基于i2c总线的eeprom驱动程序
S3C2410X集成了一个LCD控制器(支持STN和TFT带有触摸屏的液晶显示屏)、SDRAM控制器、3个通道的UART、4个通道的DMA、4个具有PWM功能的计时器和一个内部时钟、8通道的10位ADC。S3C2410还有很多丰富的外部接口,例如触摸屏接口、I2C总线接口、I2S总...
7. 如何写linux的I2C驱动,更具体的是加密芯片at88sc0104c的驱动
直接在应用空间写吧,驱动的话要复杂点,会给你增加难度的。
在应用空间用/dev/i2cdev 来访问i2c 设备的例子,你直接 吧。
加密芯片的话,一般厂家都有支持代码的吧,不过不一定是linux 平台的,你把访问i2c 的那部分改改就成了。
8. 如何在Linux中让I2C驱动支持Sub Address的两种方法
【目的】
AS3527有一个模拟部分,称作AFE,其与数字部分通过i2c通信,此处AFE部分有很多寄存器供外界操作访问,如果想要访问这些寄存器,就要用到Sub Address,所以,要实现让i2c 驱动支持Sub Address的模式。
i2C本身的架构中,没有支持sub address,所以,我们只能想办法,让其I2C支持(方法1)或者用smbus的架构(方法2).
【方法】
方法1:
在i2c的message中传递一个2个字节的buffer,分别存放Sub Address和data
比如,对于读操作,就可以这么实现:
int afe_read_reg(int addr, u8 *pdata)
{
u8 msgbuf[2];
struct i2c_msg msg =
{
.addr = save_client->addr | ( << 8),
.flags = I2C_M_RD ,
.len = 2,
.buf = msgbuf,
};
msgbuf[0] = addr; //存放Sub Address,此处的Addr是寄存器地址,也就是Sub Address
msgbuf[1] = 0; //初始化
if (i2c_transfer(save_client->adapter, &msg, 1) < 0) {
dev_warn(&save_client->dev,
"can't read from afe /n");
return -ENOMEM;
}
*pdata = msgbuf[1];
return 0;
}
方法2:
使用SMBUS的框架,其支持Sub Address
在i2c读操作中,直接调用SMBUS架构中的函数i2c_smbus_read_byte_data:
int afe_read_reg(int addr, u8 *pdata)
{
int ret;
ret = i2c_smbus_read_byte_data(save_client, addr);
if (ret < 0)
return ret;
else {
*pdata = (u8)ret;
return 0;
}
}
然后函数调用顺序是
i2c_smbus_read_byte_data -> i2c_smbus_xfer ->
adapter->algo->smbus_xfer 或 i2c_smbus_xfer_emulated
(1)此处如果你自己的I2C驱动中没有实现
adapter->algo->smbus_xfer
那么就会去调用i2c_smbus_xfer_emulated,其会把I2C的读一个字节的操作,
分成2个message,然后
i2c_smbus_xfer_emulated -> i2c_transfer -> adap->algo->master_xfer(adap,msgs,num)
去调用底层自己的i2c传输的函数master_xfer去实现两个message的传输。
此处要注意的是,如果你的i2C的控制器和i2c设备,支持将此I2C的读一个字节操作分两个message传输,
那么此处此方法也是可以的。
而你的底层的master_xfer函数,只要负责将对应的message发送出去也就可以实现对应的功能了。
否则,就像我此处遇到的,我这里的AFE的i2c控制器,不支持读操作分成两次message,只支持一个I2C message的传输,
所以,只能是在底层特殊处理,将2个message自己整理成一个message,或者是用下面的办法。
(2)自己实现了adapter->algo->smbus_xfer
自己仿照i2c_smbus_xfer_emulated,在具体实现的时候,对于读和写都只是发送一个message,然后让底层代码
adap->algo->master_xfer去处理这个message,实现对应的读和写。
【注意】
1.以上,不论是1还是2,都是在实现了自己I2C驱动底层message传输的基本函数之后,才可以工作的。
而对于这个基本函数,即adap->algo->master_xfer,
都是要在实现的时候,注意上层传递过来的buffer的第一个字节是sub address,第二个字节才是要用于写入或读取的buffer。
2.对于方法2(2),在模拟i2c_smbus_xfer_emulated实现自己的xfer函数的时候,
不能直接调用i2c_transfer,因为i2c_transfer里面,去获得adapter->bus_lock,而i2c_smbus_xfer中,调用adapter->algo->smbus_xfer之前,已经进行了对于adapter->bus_lock锁定,而因此会形成死锁的的,办法是不要再去获得锁,而直接调用adapter->algo->master_xfer即可。
9. 关于linux的I2C驱动的to_i2c_client()问题
to_i2c_client(dev) 这个函数返回值是一个指针,这个指针是个struct i2c_client 类型的指针,这个指针指向块内存,内存中存放着 to_i2c_client(dev)这个函数产生的数据。。