linux内核GPIO模拟I2C实例
Linux I2C(一)之常用的几种实例化(i2c_client )
* struct i2c_board_info - template for device creation * @type: chip type, to initialize i2c_ * @flags: to initialize i2c_client.flags * @addr: stored in i2c_client.addr * @platform_data: stored in i2c_client.dev.platform_data * @archdata: copied into i2c_client.dev.archdata * @of_node: pointer to OpenFirmware device node * @acpi_node: ACPI device node * @irq: stored in i2c_client.irq * * I2C doesn't actually support hardware probing, although controllers and * devices may be able to use I2C_SMBUS_QUICK to tell whether or not there's * a device at a given address. Drivers commonly need more information than * that, such as chip type, configuration, associated IRQ, and so on. * * i2c_board_info is used to build tables of information listing I2C devices * that are present. This information is used to grow the driver model tree. * For mainboards this is done statically using i2c_register_board_info(); * bus numbers identify adapters that aren't yet available. For add-on boards, * i2c_new_device() does this dynamically with the adapter already known. */ struct i2c_board_info { char type[I2C_NAME_SIZE]; unsigned short flags; unsigned short addr; void *platform_data; struct dev_archdata *archdata; struct device_node *of_node; struct acpi_dev_node acpi_node; int irq; }; i2c_register_board_info: /** * i2c_register_board_info - statically declare I2C devices * @busnum: identifies the bus to which these devices belong * @info: vector of i2c device descriptors * @len: how many descriptors in the vector; may be zero to reserve * the specified bus number. * * Systems using the Linux I2C driver stack can declare tables of board info * while they initialize. This should be done in board-specific init code * near arch_initcall() time, or equivalent, before any I2C adapter driver is * registered. For example, mainboard init code could define several devices,
LinuxI2C驱动--用户态驱动简单示例
LinuxI2C驱动--⽤户态驱动简单⽰例1. Linux内核⽀持I2C通⽤设备驱动(⽤户态驱动:由应⽤层实现对硬件的控制可以称之为⽤户态驱动),实现⽂件位于drivers/i2c/i2c-dev.c,设备⽂件为/dev/i2c-02. I2C通⽤设备驱动以字符设备注册进内核的static const struct file_operations i2cdev_fops = {.owner = THIS_MODULE,.llseek = no_llseek,.read = i2cdev_read,.write = i2cdev_write,.unlocked_ioctl = i2cdev_ioctl,.open = i2cdev_open,.release = i2cdev_release,};res = register_chrdev(I2C_MAJOR, "i2c", &i2cdev_fops);3. 对设备⽂件进⾏读写时,可以调⽤read、write或者ioctl等⽅法,他们都是通过调⽤函数i2c_transfer来实现对I2C设备的操作的int i2c_transfer(struct i2c_adapter * adap, struct i2c_msg *msgs, int num){int ret;/* REVISIT the fault reporting model here is weak:** - When we get an error after receiving N bytes from a slave,* there is no way to report "N".** - When we get a NAK after transmitting N bytes to a slave,* there is no way to report "N" ... or to let the master* continue executing the rest of this combined message, if* that's the appropriate response.** - When for example "num" is two and we successfully complete* the first message but get an error part way through the* second, it's unclear whether that should be reported as* one (discarding status on the second message) or errno* (discarding status on the first one).*/if (adap->algo->master_xfer) {#ifdef DEBUGfor (ret = 0; ret < num; ret++) {dev_dbg(&adap->dev, "master_xfer[%d] %c, addr=0x%02x, ""len=%d%s\n", ret, (msgs[ret].flags & I2C_M_RD)'R' : 'W', msgs[ret].addr, msgs[ret].len,(msgs[ret].flags & I2C_M_RECV_LEN) ? "+" : "");}#endifif (in_atomic() || irqs_disabled()) {ret = mutex_trylock(&adap->bus_lock);if (!ret)/* I2C activity is ongoing. */return -EAGAIN;} else {mutex_lock_nested(&adap->bus_lock, adap->level);}ret = adap->algo->master_xfer(adap,msgs,num);mutex_unlock(&adap->bus_lock);return ret;} else {dev_dbg(&adap->dev, "I2C level transfers not supported\n");return -EOPNOTSUPP;}}4. i2c_transfer通过代码可以看出,i2c_transfer 通过调⽤相应的 adapter 的 master_xfer ⽅法实现的,⽽ master_xfer 主要是根据 struct i2c_msg 类型的msgs来进⾏处理的。
文档:、linux-308内核I2C触摸屏移植
Linux3.0.8平台搭建移植文档——I2C触摸屏移植1.I2C子系统goodix电容屏采用I2C接口与处理器连接,我们要首先确保linux内核拥有对I2C子系统的支持,下面我们从子系统的配置和电容屏驱动两个方面进行移植修改1)配置I2C子系统支持#make menuconfigDevice Drivers ---><*> I2C support --->[*] Enable compatibility bits for old user-space (NEW)<*> I2C device interface< > I2C bus multiplexing support (NEW)[*] Autoselect pertinent helper modules (NEW)I2C Hardware Bus support --->进入I2C Hardware Bus support选项,选中以下内容:*** I2C system bus drivers (mostly embedded / system-on-chip) ***< > Synopsys DesignWare (NEW)<*> GPIO-based bitbanging I2C< > OpenCores I2C Controller (NEW)< > PCA9564/PCA9665 as platform device (NEW)<*> S3C2410 I2C Driver< > Simtec Generic I2C interface (NEW)...2.GOODIX电容屏移植1)添加goodix电容屏驱动将“goodix_touch.c”文件copy到drivers/input/touchscreen/目录下,并将"goodix_touch.h"、"goodix_queue.h"文件copy到include/linux/目录下,并修改Kconfig文件及Makefile文件支持触摸屏驱动的配置和编译#vi driver/input/touchscreen/Kconfig在config TOUCHSCREEN_TPS6507X选项的后面添加以下内容:config TOUCHSCREEN_GOODIXtristate "GOODIX based touchscreen"depends on I2ChelpIt is a android driver to support Gooidx's touchscreen whose nameis guitar on s5pv210 platform. The touchscreen can support multi-touch not more than two fingers.Say Y here to enable the driver for the touchscreen on theS5V SMDK board.If unsure, say N.To compile this driver as a module, choose M here:the module will be called goodix_touch.ko.#vi driver/input/touchscreen/Makefile在文件最后添加如下内容:obj-$(CONFIG_TOUCHSCREEN_GOODIX) += goodix_touch.o2)添加i2c_board_info资源(电容屏)#vi arch/arm/mach-s5pv210/mach-smdkv210.c在smdkv210_i2c_devs0结构体数组定义中添加以下内容:...{ I2C_BOARD_INFO("24c08", 0x50), }, /* Samsung S524AD0XD1 */ { I2C_BOARD_INFO("wm8580", 0x1b), },{I2C_BOARD_INFO("Goodix-TS", 0x55),.irq = IRQ_EINT(4),},...3)配置电容屏选项#make menuconfigDevice Drivers --->Input device support --->...[*] Touchscreens ---><*> GOODIX based touchscreen4)make将在arch/arm/boot/下生成编译好的可执行程序zImage下载到开发板即可,执行命令“cat /dev/input/event0”,然后用手触摸屏幕会在终端看到输出打印的乱码信息,表示移植成功,如果没有打印信息或没有“/dev/input/event0”这个设备说明移植失败。
linux gpio模拟i2c的使用用GPIO模拟I2C总线
linux gpio模拟i2c的使用用GPIO模拟I2C总线这个结构专门用于数据传输相关的addr为I2C设备地址,flags为一些标志位,len为数据的长度,buf为数据。
这里宏定义的一些标志还是需要了解一下。
I2C_M_TEN表示10位设备地址I2C_M_RD读标志I2C_M_NOSTART无起始信号标志I2C_M_IGNORE_NAK忽略应答信号标志回到for,这里的num代表有几个structi2c_msg,进入for语句,接下来是个if语句,判断这个设备是否定义了I2C_M_NOSTART标志,这个标志主要用于写操作时,不必重新发送起始信号和设备地址,但是对于读操作就不同了,要调用i2c_repstart这个函数去重新发送起始信号,调用bit_doAddress函数去重新构造设备地址字节,来看这个函数。
static int bit_doAddress(struct i2c_adapter *i2c_adap, struct i2c_msg *msg) { unsigned short flags = msg->flags; unsigned short nak_ok = msg->flags& I2C_M_IGNORE_NAK; struct i2c_algo_bit_data *adap = i2c_adap->algo_data; unsigned charaddr; int ret, retries; retries = nak_ok ? 0 :i2c_adap->retries; if (flags & I2C_M_TEN) { /* a ten bit address */ addr = 0xf0 | ((msg->addr >> 7) & 0x03);bit_dbg(2, &i2c_adap->dev, "addr0: %d\n", addr); /* try extended address code...*/ ret =try_address(i2c_adap, addr, retries); if ((ret != 1) && !nak_ok){ dev_err(&i2c_adap->dev,"died at extended address code\n"); return -EREMOTEIO; } /* the remaining 8 bit address */ ret = i2c_outb(i2c_adap, msg->addr & 0x7f); if ((ret != 1)&& !nak_ok) { /* the chip did not ack / xmission error occurred */dev_err(&i2c_adap->dev, "died at 2nd address code\n"); return -EREMOTEIO; } if (flags & I2C_M_RD) { bit_dbg(3,&i2c_adap->dev, "emitting repeated ""start condition\n"); i2c_repstart(adap);/* okay, now switch into reading mode */addr |= 0x01; ret = try_address(i2c_adap,addr, retries); if ((ret != 1)&& !nak_ok){ dev_err(&i2c_adap->dev, "died at repeated address code\n");return -EREMOTEIO; } } } else { /* normal 7bit address */ addr = msg->addr << 1; if (flags &I2C_M_RD) addr |= 1; if (flags& I2C_M_REV_DIR_ADDR) addr ^= 1; ret = try_address(i2c_adap, addr, retries); if ((ret != 1) && !nak_ok) return -ENXIO; } return 0; }这里先做了一个判断,10位设备地址和7位设备地址分别做不同的处理,通常一条I2C总线上不会挂那么多I2C设备,所以10位地址不常用,直接看对7位地址的处理。
STM32用GPIO模拟IIC通讯C语言源码实测可用
STM32用GPIO模拟IIC通讯C语言源码实测可用以下是一段用于STM32的GPIO模拟IIC(I2C)通讯的C语言源码:```c#include "stm32f10x.h"//定义SDA和SCL的GPIO端口和引脚#define SDA_GPIO GPIOB#define SDA_PIN GPIO_Pin_7#define SCL_GPIO GPIOB#define SCL_PIN GPIO_Pin_6//定义IIC的延时函数void IIC_Delayvolatile int i = 100;while (i--);//IIC开始信号void IIC_Start//先置SDA和SCL为高电平GPIO_SetBits(SDA_GPIO, SDA_PIN);GPIO_SetBits(SCL_GPIO, SCL_PIN);IIC_Delay(;//再拉低SDA,产生开始信号GPIO_ResetBits(SDA_GPIO, SDA_PIN); IIC_Delay(;GPIO_ResetBits(SCL_GPIO, SCL_PIN); //IIC停止信号void IIC_Stop//先置SDA为低电平GPIO_ResetBits(SDA_GPIO, SDA_PIN); IIC_Delay(;//再拉高SDA,产生停止信号GPIO_SetBits(SDA_GPIO, SDA_PIN); IIC_Delay(;GPIO_SetBits(SCL_GPIO, SCL_PIN); //IIC发送一个字节数据void IIC_WriteByte(uint8_t data) uint8_t i;for (i = 0; i < 8; i++)//先将SCL拉低GPIO_ResetBits(SCL_GPIO, SCL_PIN);//根据发送数据的当前位决定SDA的电平if ((data<<i) & 0x80)GPIO_SetBits(SDA_GPIO, SDA_PIN); elseGPIO_ResetBits(SDA_GPIO, SDA_PIN); IIC_Delay(;//拉高SCL,使得IIC设备接收数据GPIO_SetBits(SCL_GPIO, SCL_PIN);IIC_Delay(;}//最后拉低SCL,等待应答GPIO_ResetBits(SCL_GPIO, SCL_PIN); //IIC读取一个字节数据uint8_t IIC_ReadByte(uint8_t ack) uint8_t i, data = 0;for (i = 0; i < 8; i++)//先将SCL拉低GPIO_ResetBits(SCL_GPIO, SCL_PIN);//等待IIC设备准备好数据GPIO_SetBits(SCL_GPIO, SCL_PIN);IIC_Delay(;//读取SDA的电平if (GPIO_ReadInputDataBit(SDA_GPIO, SDA_PIN)) data ,= 0x01;if (i < 7)data <<= 1;}//最后拉低SCL,进行应答GPIO_ResetBits(SCL_GPIO, SCL_PIN);if (ack)GPIO_ResetBits(SDA_GPIO, SDA_PIN);elseGPIO_SetBits(SDA_GPIO, SDA_PIN);IIC_Delay(;GPIO_SetBits(SCL_GPIO, SCL_PIN);IIC_Delay(;GPIO_ResetBits(SCL_GPIO, SCL_PIN);return data;int main//使能GPIO端口的时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Pin = SDA_PIN;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(SDA_GPIO, &GPIO_InitStructure);GPIO_InitStructure.GPIO_Pin = SCL_PIN;GPIO_Init(SCL_GPIO, &GPIO_InitStructure);//执行IIC通讯的操作IIC_Start(;IIC_WriteByte(0xA0);IIC_WriteByte(0x01);IIC_Stop(;while (1);```这段代码使用STM32的GPIO模拟IIC通讯,定义了SDA和SCL的GPIO端口和引脚,并实现了IIC的开始信号、停止信号、发送字节和接收字节的函数。
linuxgpio模拟i2c的使用用GPIO模拟I2C总线
linuxgpio模拟i2c的使用用GPIO模拟I2C总线这个结构专门用于数据传输相关的addr为I2C设备地址,flags 为一些标志位,len为数据的长度,buf为数据。
这里宏定义的一些标志还是需要了解一下。
I2C_M_TEN表示10位设备地址I2C_M_RD读标志I2C_M_NOSTART无起始信号标志I2C_M_IGNORE_NAK忽略应答信号标志回到for,这里的num代表有几个struct i2c_msg,进入for语句,接下来是个if语句,判断这个设备是否定义了I2C_M_NOSTART 标志,这个标志主要用于写操作时,不必重新发送起始信号和设备地址,但是对于读操作就不同了,要调用i2c_repstart这个函数去重新发送起始信号,调用bit_doAddress函数去重新构造设备地址字节,来看这个函数。
1.static int bit_doAddress(struct i2c_adapter *i2c_adap, stru ct i2c_msg *msg)2.{3.unsigned short flags = msg->flags;4.unsigned short nak_ok = msg->flags & I2C_M_IGNORE_ NAK;5.struct i2c_algo_bit_data *adap = i2c_adap->algo_data;6.7.unsigned char addr;8.int ret, retries;9.10.retries = nak_ok ? 0 : i2c_adap->retries;11.12.if (flags & I2C_M_TEN) {13./* a ten bit address */14.addr = 0xf0 | ((msg->addr >> 7) & 0x03);15.bit_dbg(2, &i2c_adap->dev, "addr0: %d\n", addr);16./* try extended address code...*/17.ret = try_address(i2c_adap, addr, retries);18.if ((ret != 1) && !nak_ok) {19.dev_err(&i2c_adap->dev,20."died at extended address code\n");21.return -EREMOTEIO;22.}23./* the remaining 8 bit address */24.ret = i2c_outb(i2c_adap, msg->addr & 0x7f);25.if ((ret != 1) && !nak_ok) {26./* the chip did not ack / xmission error occurred */27.dev_err(&i2c_adap->dev, "died at 2nd address code\n ");28.return -EREMOTEIO;29.}30.if (flags & I2C_M_RD) {31.bit_dbg(3, &i2c_adap->dev, "emitting repeated "32."start condition\n");33.i2c_repstart(adap);34./* okay, now switch into reading mode */35.addr |= 0x01;36.ret = try_address(i2c_adap, addr, retries);37.if ((ret != 1) && !nak_ok) {38.dev_err(&i2c_adap->dev,39."died at repeated address code\n");40.return -EREMOTEIO;41.}42.}43.} else { /* normal 7bit address */44.addr = msg->addr << 1;45.if (flags & I2C_M_RD)46.addr |= 1;47.if (flags & I2C_M_REV_DIR_ADDR)48.addr ^= 1;49.ret = try_address(i2c_adap, addr, retries);50.if ((ret != 1) && !nak_ok)51.return -ENXIO;52.}53.54.return 0;55.}这里先做了一个判断,10位设备地址和7位设备地址分别做不同的处理,通常一条I2C总线上不会挂那么多I2C设备,所以10位地址不常用,直接看对7位地址的处理。
GPIO模拟I2C应用设计说明书
GPIO模拟I2C应用设计说明书1简介1.1目的介绍I2C的特性,工作原理以及通过GPIO(GeneralPurpose Input /Output)模拟I2C总线的时序实现数据传输;该文档描述的设计思想可供GPIO驱动开发者进行设计参考。
本文以Hi3510为例,通过具体的应用实例来描述GPIO模拟I2C实现数据传输,本文的读者假定是对GPIO的特点以及工作方式等非常熟悉,涉及到GPIO的内容请参考相关的文档。
2I2C2.1概述I2C总线(The Inter-Integrated Circuit)是一种通用的串行总线,是用于IC器件之间连接的二线制总线。
他通过串行数据或者地址线(Serial Data Line,SDA)及串行时钟线(Serial ClockLine,SCL)两线在连接到总线上的器件之间传送信息,并根据地址识别每个器件。
一个或多个微控制器以及外围器件可以通过I2C总线接口非常方便的连接在一起构成系统。
这种总线结构的连线和连接引脚少,器件间总线简单。
结构紧凑,因此其构成系统的成本较低;并且在总线上增加器件不会影响系统的正常工作,所有的IC器件共用一套总线,因此其系统修改和可扩展性好。
即使有不同时钟速度的器件连接到总线上,时间同步机制也能够很方便地确定总线时钟,在Hi3510中,I2C仅支持Master功能,遵守I2C总线协议2.1版本,在Hi3510中可用于Video Encoder、Video Decoder、Digital Camera的控制接口。
2.2特点I2C接口有以下特点:I2C接口在I2C总线上可以作为接收器,也可以作为发送器,具体由数据的传输方向确定支持标准传输模式和快速传输模式提供TX FIFO 、RX FIFO,支持DMA数据传输支持中断上报和初始化中断状态,屏蔽后中断状态的查询2.3信号描述本节描述了I2C单元的输入输出管脚信号,如下表所示:2.4工作原理I2C总线是由数据线SDA和时钟线SCL构成的串行总线,可发送和接收数据。
linux i2c案例
linux i2c案例Linux中的I2C是一种串行通信协议,用于连接微控制器、传感器和其他外设。
本文将介绍10个关于Linux I2C的案例。
1. 使用I2C读取温度传感器数据在Linux中,可以使用I2C接口读取连接到系统的温度传感器的数据。
通过使用相应的设备文件和I2C工具,可以轻松地读取传感器数据并进行处理。
2. 使用I2C控制LED灯通过I2C总线可以控制连接到系统的外设,例如LED灯。
在Linux 中,可以使用I2C设备文件和相应的命令来控制LED的亮灭。
3. 使用I2C读取加速度传感器数据加速度传感器是一种常见的外设,用于检测物体的加速度。
在Linux中,可以使用I2C接口读取连接到系统的加速度传感器的数据,并用于各种应用,如运动检测和姿态测量。
4. 使用I2C读取压力传感器数据压力传感器是一种常见的外设,用于测量气体或液体的压力。
在Linux中,可以使用I2C接口读取连接到系统的压力传感器的数据,并用于各种应用,如气象监测和液位检测。
5. 使用I2C读取光照传感器数据光照传感器是一种常见的外设,用于测量环境中的光照强度。
在Linux中,可以使用I2C接口读取连接到系统的光照传感器的数据,并用于各种应用,如自动调光和环境光控制。
6. 使用I2C读取触摸屏控制器数据触摸屏控制器是一种常见的外设,用于检测触摸屏的触摸位置和手势。
在Linux中,可以使用I2C接口读取连接到系统的触摸屏控制器的数据,并用于各种应用,如触摸输入和手势识别。
7. 使用I2C读取电池监控芯片数据电池监控芯片是一种常见的外设,用于监测设备的电池电量和充放电状态。
在Linux中,可以使用I2C接口读取连接到系统的电池监控芯片的数据,并用于电池管理和电源优化。
8. 使用I2C读取电容触摸芯片数据电容触摸芯片是一种常见的外设,用于检测触摸面板的触摸位置和手势。
在Linux中,可以使用I2C接口读取连接到系统的电容触摸芯片的数据,并用于触摸输入和手势识别。
gpio模拟i2c 自适应频率
gpio模拟i2c 自适应频率GPIO模拟I2C自适应频率是一种通过软件模拟I2C通信协议的方法,通过利用单片机的GPIO口来模拟I2C总线上的数据通信。
在I2C通信中,主要有两种频率,标准模式(100KHz)和快速模式(400KHz),而在模拟I2C通信时,我们需要实现自适应频率的功能,即根据外部设备的实际工作频率自动调整模拟I2C通信的速率。
实现GPIO模拟I2C自适应频率的关键是根据外部设备的响应时间来动态调整通信速率,从而保证通信的稳定性和可靠性。
下面是一些参考内容,可帮助你实现GPIO模拟I2C自适应频率:1. 定义I2C通信的起始和停止信号:在模拟I2C通信时,首先需要定义起始和停止信号的信号序列,用于区分数据的开始和结束。
2. 设计GPIO口的控制程序:利用单片机的GPIO口模拟I2C通信时,需要编写相应的控制程序,包括发送和接收数据的函数,起始和停止信号的生成,时钟信号的控制等。
3. 自适应频率算法:设计一个自适应频率的算法,根据外部设备的响应时间动态调整通信速率。
可以通过在通信过程中计算通信时间,根据通信时间的长短来自动调整通信速率。
4. 考虑通信的稳定性和可靠性:在设计GPIO模拟I2C自适应频率的过程中,需要考虑通信的稳定性和可靠性,尽量减少通信的误码率,确保数据的准确传输。
5. 调试和优化:在实际应用中,需要对设计的GPIO模拟I2C自适应频率的程序进行调试和优化,确保通信的正常运行,可以通过逐步测试的方式逐步完善功能。
通过以上的参考内容,你可以实现GPIO模拟I2C自适应频率的功能,根据外部设备的实际工作频率来自动调整通信速率,从而实现稳定可靠的通信。
希望以上内容对你有所帮助,祝你实现成功!。
gpio模拟i2c 自适应频率
gpio模拟i2c 自适应频率GPIO 模拟 I2C 自适应频率GPIO 模拟 I2C 是一种利用通用输入/输出 (GPIO) 引脚模拟I2C 总线的技术,无需专用的 I2C 硬件接口。
这种方法具有一定的优势,包括:低成本:无需额外的 I2C 硬件,降低了成本。
灵活性:可以通过软件配置 GPIO 引脚,允许在不同设备上灵活实现。
可扩展性:可以用额外的 GPIO 引脚扩展 I2C 总线,增加连接的设备数量。
自适应频率自适应频率是指 GPIO 模拟 I2C 能够根据设备的响应动态调整I2C 总线频率。
这对于与时钟速率不同的设备进行通信非常有用,确保稳定可靠的数据传输。
实现自适应频率实现自适应频率 GPIO 模拟 I2C 时,需要考虑以下因素:硬件设置:确保 GPIO 引脚正确配置为 I2C 模式,并且具有合适的时钟源。
软件算法:使用自适应频率算法来监测设备响应并相应调整总线频率。
时序控制:精确控制 I2C 总线时序,包括时钟高低电平时间和数据传输时间。
自适应频率算法常用的自适应频率算法包括:二分法:通过连续调整总线频率并监测设备响应,缩小到合适的频率范围。
PID 控制器:基于设备响应的误差值,使用比例积分微分(PID) 控制器动态调整频率。
神经网络:训练神经网络来预测设备的最佳频率,并根据实时反馈进行调整。
优点GPIO 模拟 I2C 自适应频率具有以下优点:提高通信可靠性:通过使用设备的最佳频率,确保稳定可靠的数据传输。
优化性能:根据设备的实际情况调整频率,优化 I2C 总线性能。
减少功耗:通过使用较低的频率与设备通信,可以节省功耗。
应用GPIO 模拟 I2C 自适应频率广泛应用于各种领域,包括:嵌入式系统:在受限于成本和空间的嵌入式系统中,实现 I2C连接。
物联网:连接各种物联网设备,实现数据传输和控制。
工业自动化:控制和监控工业设备,实现可靠的通信。
注意事项在使用 GPIO 模拟 I2C 自适应频率时,需要注意以下事项:总线长度限制:GPIO 模拟 I2C 的时序控制能力有限,因此总线长度受到限制。
i2c-gpio的设备树写法
i2c-gpio是一种可以通过通用输入输出(GPIO)来模拟I2C总线的技术。
在嵌入式系统中,我们经常会遇到需要使用I2C设备的情况。
而在一些特定的硬件评台上,可能并没有专门的I2C总线控制器,这时就需要通过i2c-gpio来模拟I2C总线了。
在Linux系统中,设备的初始化和配置通常是通过设备树(Device Tree)来完成的。
那么,在设备树中,我们应该如何编写i2c-gpio设备呢?1. 理解i2c-gpio我们需要理解i2c-gpio的工作原理。
i2c-gpio通过配置GPIO引脚的输入输出状态来模拟I2C总线的时钟和数据线。
在设备树中,我们需要指定哪些GPIO引脚用作时钟线,哪些用作数据线,以及它们的引脚编号、极性等信息。
2. 查找设备树文档在编写设备树之前,我们需要先查找硬件评台的设备树文档。
这些文档通常会包含设备树的语法规则、i2c-gpio的使用方法以及硬件评台特定的配置信息。
设备树文档对于正确配置i2c-gpio非常重要,因为不同的硬件评台可能会有不同的GPIO引脚分配、极性要求等。
3. 编写i2c-gpio节点在设备树中,每个设备都会有一个对应的节点来描述它的属性和连接关系。
对于i2c-gpio设备,我们需要编写一个对应的节点来描述这个设备。
这个节点包括了设备的类型、引脚信息、中断信息等。
4. 设置GPIO引脚在i2c-gpio节点中,我们需要指定哪些GPIO引脚用作时钟线,哪些用作数据线。
通常情况下,这些信息会包括引脚编号、极性等信息。
这些信息需要根据硬件评台的实际情况进行配置。
5. 配置i2c-gpio属性除了GPIO引脚的设置,我们还需要配置i2c-gpio设备的一些属性,比如时钟频率、位置区域长度等。
这些属性也需要在设备树中进行配置,以确保i2c-gpio可以正常工作。
6. 编译和加载设备树完成设备树的编写后,我们需要将设备树文件编译成二进制格式,并将其加载到硬件评台上。
linuxgpio模拟i2c的使用用GPIO模拟I2C总线
linuxgpio模拟i2c的使用用GPIO模拟I2C总线在drivers/i2c/busses下包含各种I2C总线驱动,如S3C2440的I2C总线驱动i2c-s3c2410.c,使用GPIO模拟I2C总线的驱动i2c-gpio.c,这里只分析i2c-gpio.c。
i2c-gpio.c它是gpio模拟I2C总线的驱动,总线也是个设备,在这里将总线当作平台设备处理,那驱动当然是平台设备驱动,看它的驱动注册和注销函数。
[html] view plaincopyprint?1.1. static int __init i2c_gpio_init(void)2.2. {3.3. int ret;4.4.5.5. ret = platform_driver_register(&i2c_gpio_driver);6.6. if (ret)7.7. printk(KERN_ERR "i2c-gpio: probe failed: %d\n", ret);8.8.9.9. return ret;10.10. }11.11. module_init(i2c_gpio_init);12.12.13.13. static void __exit i2c_gpio_exit(void)14.14. {15.15. platform_driver_unregister(&i2c_gpio_driver);16.16. }17.17. module_exit(i2c_gpio_exit);没有什么好说的,它的初始化和注销函数就是注册和注销一个平台设备驱动,直接看它的platform_driver结构i2c_gpio_driver[html] view plaincopyprint?1.1. static struct platform_driver i2c_gpio_driver = {2.2. .driver = {3.3. .name = "i2c-gpio",4.4. .owner = THIS_MODULE,5.5. },6.6. .probe = i2c_gpio_probe,7.7. .remove = __devexit_p(i2c_gpio_remove),8.8. };平台驱动设备放在arch/arm/mach-xxxx/board-xxx.c中[html] view plaincopyprint?1.1. #if defined(CONFIG_I2C_GPIO) | \2.2. defined(CONFIG_I2C_GPIO_MODULE)3.3. static struct i2c_gpio_platform_data i2c_gpio_adapter_d ata = {4.4. .sda_pin = PINID_GPMI_D05,5.5. .scl_pin = PINID_GPMI_D04,6.6. .udelay = 5, //100KHz7.7. .timeout = 100,8.8. .sda_is_open_drain = 1,9.9. .scl_is_open_drain = 1,10.10. };11.11.12.12. static struct platform_device i2c_gpio = {13.13. .name = "i2c-gpio",14.14. .id = 0,15.15. .dev = {16.16. .platform_data = &i2c_gpio_adapter_data,17.17. .release = mxs_nop_release,18.18. },19.19. };20.20. #endif在这里struct platform_device结构中的name字段要和struct platform_driver中driver字段中name字段要相同,因为平台总线就是通过这个来判断设备和驱动是否匹配的。
GPIO驱动I2C接口案例
9.4 应用案例(一):GPIO软件模拟I2C协议本案例主要研究利用STM32F10X的GPIO端口引脚,模拟I2C协议,读写A T24C01芯片,实现MCU与从器件接口,完成数据通信功能。
在介绍功能实现前,先介绍I2C总线协议规范,然后介绍应用驱动分层模型、接口电路设计、接口协议软件的详细实现与具体步骤等。
9.4.1 I2C总线协议规范为了更好说明如何用GPIO软件实现I2C协议,编写代码前,先对I2C协议做详细介绍,描述功能特点,包括:电器接口、总线时序、数据帧格式等。
(1) I2C总线规范简介I2C BUS(Inter IC BUS)是NXP半导体公司推出的芯片间串行传输总线,它以2根连线实现了完善的全双工同步数据传送,可以极方便地构成多机系统和外围器件扩展系统。
I2C总线采用了器件地址的硬件设置方法,通过软件寻址完全避免了器件的片选线寻址方法,从而使硬件系统具有最简单而灵活的扩展方法。
I2C总线的2根线(SDA:串行数据,SCL:串行时钟)连接到总线上的任何一个器件,每个器件都应有一个唯一的地址,而且都可以作为一个发送器或接收器。
此外,器件在执行数据传输时也可以被看作是主机或从机。
发送器:本次数据传送中发送数据(不包括地址和命令)到总线的器件;接收器:本次数据传送中从总线接收数据(不包括地址和命令)的器件;主控制器(主机):初始化发送、产生时钟信号和终止发送的器件,它可以是发送器或接收器,主控制器通常是微控制器;从控制器(从机):被主机寻址的器件,它可以是发送器或接收器。
(2)I2C电气接口SDA和SCL都是双向线路。
如图9-3-1所示,I2C总线接口均为开漏或开集电极输出,因此需要为总线增加外部上拉电阻R1、R2实现“线与”功能。
当总线空闲时,这2条线路都是高电平。
I2C 接口的标准传输速率为100Kbit/s,最高传输速率可达400Kbit/s。
总线速率越高,总线上拉电阻就越小,100Kbit/s总线速率,通常使用5.1K欧姆的上拉电阻。
模拟I2C怎么用--教你使用GPIO口模拟I2C总线协议
模拟I2C怎么⽤--教你使⽤GPIO⼝模拟I2C总线协议所谓模拟I2C是指使⽤普通GPIO⼝的输⼊输出功能来模拟I2C总线的时序,⽤来通过I2C总线进⾏通信。
I2C的基本知识:1、I2C总线有两条线:SCL是时钟线,SDA是数据线;2、I2C总线通信⽅式是主从模式,即由主设备发起通信,从设备响应通信;3、I2C从设备具有I2C地址,从设备只有收到⾃⼰的地址信息后才会被唤醒;4、具有不同地址的从设备可以挂载到同⼀个I2C总线上;5、从设备地址的最后⼀个Bit表⽰读写,0表⽰写操作,1表⽰读操作;6、I2C总线地址有7Bit表⽰⽅法和8Bit表⽰⽅法,7Bit表⽰⽅法是地址中不包含表⽰读写的最后⼀个Bit;7、当SCL=1时,SDA产⽣下降沿来启动I2C;8、当SCL=1时,SDA产⽣上升沿来停⽌I2C;9、I2C启动后,当SCL=1时,SDA的电平不允许有变化;10、I2C启动后,只有当SCL=0时,数据发送⽅才能在SDA上改变发送电平;11、I2C总线上数据接收⽅在接收完⼀个字节数据(8Bit)后,要在下⼀个SCL的上升沿,通过SDA响应ACK(SDA=0)或NACK(SDA=1)信号;12、I2C外部需根据传输速率匹配上拉电阻,速率越⾼,上拉电阻越⼩,否则会影响时序;13、I2C引脚作为输出时需是开漏输出,作为输⼊时需是浮空输⼊,不能匹配内部上拉或下拉电阻;话不多说,直接上代码:⼀、基本接⼝定义为了提⾼代码的可移植性和使⽤⽅便性,我定义了⼀些宏和结构体,下⾯介绍⼀下这些宏和结构体。
1、结构体I2C总线有SDA和SCL两个引脚,所以我构造了⼀个结构体来定义表⽰这两个引脚的基本信息,我是在STM32平台做的例程,所以是这样定义的:typedef struct {uint32_t SDA_RCC_APB2Periph;// SDA脚时钟GPIO_TypeDef* SDA_Port;//SDA脚Portuint16_t SDA_Pin;//SDA脚Pinuint32_t SCL_RCC_APB2Periph;//SCL脚时钟GPIO_TypeDef* SCL_Port;//SCL脚Portuint16_t SCL_Pin;//SCL脚Pin} sw_i2c_gpio_t;当然如果你使⽤的是其他平台,可以对结构体进⾏重新构造。
使用GPIO模拟I2C总线进行通信
使⽤GPIO模拟I2C总线进⾏通信I2C是由Philips公司发明的⼀种串⾏数据通信协议,仅使⽤两根信号线:SerialClock(简称SCL)和SerialData(简称SDA)。
I2C是总线结构,1个Master,1个或多个Slave,各Slave设备以7位地址区分,地址后⾯再跟1位读写位,表⽰读(=1)或者写(=0),所以我们有时也可看到8位形式的设备地址,此时每个设备有读、写两个地址,⾼7位地址其实是相同的。
I2C数据格式如下:⽆数据:SCL=1,SDA=1;开始位(Start):当SCL=1时,SDA由1向0跳变;停⽌位(Stop):当SCL=1时,SDA由0向1跳变;数据位:当SCL由0向1跳变时,由发送⽅控制SDA,此时SDA为有效数据,不可随意改变SDA;当SCL保持为0时,SDA上的数据可随意改变;地址位:定义同数据位,但只由Master发给Slave;应答位(ACK):当发送⽅传送完8位时,发送⽅释放SDA,由接收⽅控制SDA,且SDA=0;否应答位(NACK):当发送⽅传送完8位时,发送⽅释放SDA,由接收⽅控制SDA,且SDA=1。
当数据为单字节传送时,格式为:开始位,8位地址位(含1位读写位),应答,8位数据,应答,停⽌位。
当数据为⼀串字节传送时,格式为:开始位,8位地址位(含1位读写位),应答,8位数据,应答,8位数据,应答,……,8位数据,应答,停⽌位。
需要注意的是:1,SCL⼀直由Master控制,SDA依照数据传送的⽅向,读数据时由Slave控制SDA,写数据时由Master控制SDA。
当8位数据传送完毕之后,应答位或者否应答位的SDA控制权与数据位传送时相反。
2,开始位“Start”和停⽌位“Stop”,只能由Master来发出。
3,地址的8位传送完毕后,成功配置地址的Slave设备必须发送“ACK”。
否则否则⼀定时间之后Master视为超时,将放弃数据传送,发送“Stop”。
gpio模拟i2c串联电阻
gpio模拟i2c串联电阻在嵌入式系统中,I2C(Inter-Integrated Circuit)是一种通信协议,用于在芯片之间传输数据。
它是一种串行通信协议,可以通过两根信号线(SDA和SCL)实现多个设备之间的通信。
然而,在某些情况下,我们可能需要在电路中模拟一个串联电阻来模拟I2C总线上的电阻。
GPIO(General Purpose Input/Output)是通用输入输出引脚,可以通过编程控制其输入和输出状态。
在一些嵌入式系统中,我们可以利用GPIO模拟I2C总线上的电阻。
本文将介绍如何使用GPIO来模拟I2C串联电阻。
我们需要了解I2C的电气特性。
I2C总线上的电阻主要有两种:上拉电阻和下拉电阻。
上拉电阻通常连接到VCC电源,而下拉电阻通常连接到地。
这些电阻的作用是在总线上产生高电平和低电平。
要模拟上拉电阻,我们可以使用GPIO输出高电平。
GPIO输出高电平时,相当于连接到了VCC电源,从而模拟了上拉电阻。
同样,要模拟下拉电阻,我们可以使用GPIO输出低电平,相当于连接到了地。
接下来,我们需要了解GPIO的工作模式。
在模拟I2C串联电阻时,我们需要将GPIO设置为输出模式。
通过编程,我们可以控制GPIO 的输出状态,从而模拟I2C总线上的电阻。
在开始模拟之前,我们需要确定哪些GPIO引脚可以用作模拟串联电阻。
通常,芯片手册会提供GPIO引脚的详细信息,包括其编号和功能。
我们可以根据芯片手册选择合适的GPIO引脚进行模拟。
一旦确定了GPIO引脚,我们可以使用编程语言(如Python)来控制GPIO的输出状态。
在Python中,我们可以使用第三方库(如RPi.GPIO)来操作GPIO。
具体的代码如下所示:```pythonimport RPi.GPIO as GPIO# 设置GPIO引脚SDA_PIN = 18SCL_PIN = 19# 初始化GPIOGPIO.setmode(GPIO.BCM)GPIO.setup(SDA_PIN, GPIO.OUT)GPIO.setup(SCL_PIN, GPIO.OUT)# 模拟上拉电阻GPIO.output(SDA_PIN, GPIO.HIGH)GPIO.output(SCL_PIN, GPIO.HIGH)# 模拟下拉电阻GPIO.output(SDA_PIN, GPIO.LOW)GPIO.output(SCL_PIN, GPIO.LOW)# 关闭GPIOGPIO.cleanup()```在上述代码中,我们首先通过`GPIO.setmode(GPIO.BCM)`设置GPIO 的编号模式为BCM模式。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
linux内核GPIO模拟I2C实例
2010-10-11 作者:cvip302814 来源:cvip302814的blog
前言:
在许多情况下,我们并没有足够的I2C总线,本文主在介绍如何利用Linux 内核中的i2c-gpio模块,利用2条GPIO线模拟i2c总线,并挂载设备。
思路:
先通过对i2c-gpio所定义的结构体初始化(包括初始化i2c的2条线,频率,timeout等)并将i2c-gpio模块编译进内核,实现用GPIO_X,GPIO_Y 2条GPIO线注册新的i2c总线。
此时这个模块对i2c设备是透明的,及挂在这2条GPIO线的i2c设备可以直接使用Linux内核通用的i2c设备注册,传输和注销等方法。
步骤:
首先确认在注册i2c-gpio模块前,所要用到的2条GPIO口是没有被系统其它地方所调用的。
在每个系统平台启动时,都会打开一系列的设备,他们通常实现在arch/目录下相应的平台子目录中的例如setup.c,devices.c文件中,在这里我们进行i2c总线的注册以及设备的挂载。
i2c-gpio定义的结构在
include/linux/i2c-gpio.h中:
/**
* struct i2c_gpio_platform_data - Platform-dependent data for i2c-gpio
* @sda_pin: GPIO pin ID to use for SDA
* @scl_pin: GPIO pin ID to use for SCL
* @udelay: signal toggle delay. SCL frequency is (500 / udelay) kHz
* @timeout: clock stretching timeout in jiffies. If the slave keeps
* SCL low for longer than this, the transfer will time out.
* @sda_is_open_drain: SDA is configured as open drain, i.e. the pin
* isn't actively driven high when setting the output value high.
* gpio_get_value() must return the actual pin state even if the
* pin is configured as an output.
* @scl_is_open_drain: SCL is set up as open drain. Same requirements
* as for sda_is_open_drain apply.
* @scl_is_output_only: SCL output drivers cannot be turned off.
*/
struct i2c_gpio_platform_data {
unsigned int sda_pin;
;
此时我们就可以在i2c设备的驱动程序中通过遍历所在i2c总线,得到其所在的地址i2c_device_addr。
在i2c驱动中,需要注册一个i2c_driver的结构体,例如:
static const struct i2c_device_id lis35de_id[] = {
{ "lis35de", 0 },
{ }
};
static struct i2c_driver st_lis35de_driver = {
.probe = st_lis35de_probe,
.remove = st_lis35de_remove,
.suspend = st_lis35de_suspend,
.resume = st_lis35de_resume,
.id_table = lis35de_id,
.driver = {
.name = "lis35de",
},
};
static int __init st_lis35de_init(void)
{
printk(KERN_INFO "st_lis35de_init\n");
return i2c_add_driver(&st_lis35de_driver);
}
在init时用i2c_add_driver(&st_lis35de_driver),此时将会对所在i2c总线进行遍历并得到该设备的适配器等信息,主要目的即是使驱动得到自己的i2c_client,在这个i2c_client中,已经有了该i2c设备的地址等信息,我们在驱动中定义一个新的i2c_client全局变量,把得到的这个i2c_client传给这个全局变量,从而可以继续后面的i2c操作。
此时我们就可以使用通用的i2c读写操作了。
总结:
直接用GPIO口模拟I2C时序和利用内核模块i2c-gpio虚拟i2c总线的区别:
1.用GPIO口模拟I2C时序不需要在系统启动时注册I2C总线,只需要在I2C设备驱动中单独实现。
用i2c-gpio模块虚拟i2c总线需要在系统启动时注册新的I2C总线,并将i2c设备挂载到新的i2c总线,涉及的范围较广。
2.用GPIO口模拟I2C时序,代码操作较繁琐,且不方便挂载多个i2c
设备。
用i2c-gpio模块可以完全模拟i2c总线,可以挂载多个设备。
3.在i2c读写操作时,用GPIO口模拟I2C时序需要每次根据读/写操作
发送器件地址<<1+1/0,然后再发送寄存器地址。
用i2c-gpio模块相当于直接在i2c总线上操作,在系统启动挂载i2c设备时已经告诉了i2c总线它的地址,在该设备自己的驱动中,只需要通过i2c_add_driver操作即可以得到其地址等诸多信息,读写操作只需要发送寄存器地址即可。
附:i2c一般的读写操作
#include <linux/i2c.h>
/*
读操作:
*/
static int i2c_RxData(char *rxData, int length)
{
struct i2c_msg msgs[] = {
/* 把1个字节的i2c设备寄存器地址告诉总线 */
{
.addr = client->addr,
.flags = 0, //写操作
.len = 1,
.buf = rxData,
},
/* 从总线读取length个字节的数据,存入rxData */
{
.addr =client ->addr,
.flags = I2C_M_RD, //I2C_M_RD在i2c.h中被定义为1,读操作 .len = length,
.buf = rxData,
},
};
if (i2c_transfer(client->adapter, msgs, 2) < 0) { /* 传输并判断是否传输
错误 */
printk(KERN_ERR "I2C_RxData: transfer error\n");
return -EIO;
} else
return 0;
}
/*
写操作
*/
static int i2c_TxData(char *txData, int length)
{
struct i2c_msg msg[] = {
/* 第1个字节是器件寄存器地址,后面的字节是写入的数据 */ {
.addr = client->addr,
.flags = 0,
.len = length,
.buf = txData,
},
};
if (i2c_transfer(client->adapter, msg, 1) < 0) {
printk(KERN_ERR "I2C_TxData: transfer error\n");
return -EIO;
} else
return 0;
}。