怪异的I2C命令求解!
是这样的,这个芯片的datasheet上写的I2C帧格式如下:
Start => I 2C ID addr_Wr (0x90) => Sub addr (0x02) => Sequence command (0x64) => Control command (0x04) => Stop
Start => I 2C ID addr_Rd (0x91) => Sub addr (0x11) => Read 12 byte data => Stop.
紅色表示 Host to client, 綠色表示 client to Host.
开始都是用的标准I2C函数去读写,但是发现能写,不能读,原来这个芯片的READ帧和一般的不一样
一般的READ帧有两个start,第一个start后面是写入寄存器地址,第二个start后面才是读
求个方法改造i2c bus 函数,让发送能满足这样的格式!
[解决办法]
你研究下I2C_RDWR吧
[解决办法]
什么操作系统,可以抛开总线驱动,自己摸一个就OK了
[解决办法]
这个片子着实有趣。如果真是这样,它本身是不符合I2C标准的。不知楼主是用什么方式进行I2C传输的,即是用处理器的硬件I2C设备还是软件实现的。如果是硬件I2C设备,估计不能支持这种非I2C标准的操作。比如说Cortex-m3的I2C设备,就没这种用法。
所以,楼主还是用软件方法通过两IO口实现吧。下面给出一个51通过IO口软件实现的示例,它是通过I2C协议读取L3G4200D(陀螺仪)的数据,这程序的I2C部分绝对没有问题,我试过的。不过由于只是借鉴这程序来写你的程序,所以并没必要全部看懂。只需要看I2C部分的函数即可,建议从main顺藤摸瓜。
/*
* L3G4200D模块
*
* 用途:GY-50 L3G4200D IIC测试程序
*
* 作者日期备注
* Huafeng Lin2010/12/10新增
* Huafeng Lin2010/12/11修改
*
*/
#include <REG51.H>
#include <math.h> //Keil library
#include <stdio.h> //Keil library
#include <INTRINS.H>
#define uchar unsigned char
#define uint unsigned int
#define DataPort P2//LCD1602数据端口
sbit SCL=P1^0; //IIC时钟引脚定义
sbit SDA=P1^1; //IIC数据引脚定义
sbitLCM_RS=P0^2; //LCD1602命令端口
sbit LCM_RW=P0^1; //LCD1602命令端口
sbit LCM_EN=P0^0; //LCD1602命令端口
//********************
#define WHO_AM_I 0x0F
#define CTRL_REG1 0x20
#define CTRL_REG2 0x21
#define CTRL_REG3 0x22
#define CTRL_REG4 0x23
#define CTRL_REG5 0x24
#define REFERENCE 0x25
#define OUT_TEMP 0x26
#define STATUS_REG 0x27
#define OUT_X_L 0x28
#define OUT_X_H 0x29
#define OUT_Y_L 0x2A
#define OUT_Y_H 0x2B
#define OUT_Z_L 0x2C
#define OUT_Z_H 0x2D
#define FIFO_CTRL_REG 0x2E
#define FIFO_SRC_REG 0x2F
#define INT1_CFG 0x30
#define INT1_SRC 0x31
#define INT1_TSH_XH 0x32
#define INT1_TSH_XL 0x33
#define INT1_TSH_YH 0x34
#define INT1_TSH_YL 0x35
#define INT1_TSH_ZH 0x36
#define INT1_TSH_ZL 0x37
#define INT1_DURATION 0x38
//****************************
#defineSlaveAddress 0xD2 //定义器件在IIC总线中的从地址,根据ALT ADDRESS地址引脚不同修改
typedef unsigned char BYTE;
typedef unsigned short WORD;
BYTE BUF[8]; //接收数据缓存区
uchar ge,shi,bai,qian,wan; //显示变量
int dis_data; //变量
uchar devid;
void delay(unsigned int k);
void InitLcd(); //初始化lcd1602
void InitL3G4200D(); //初始化L3G4200D
void WriteDataLCM(uchar dataW);
void WriteCommandLCM(uchar CMD,uchar Attribc);
void DisplayOneChar(uchar X,uchar Y,uchar DData);
void conversion(uint temp_data);
void Single_WriteL3G4200D(uchar REG_Address,uchar REG_data); //单个写入数据
uchar Single_ReadL3G4200D(uchar REG_Address); //单个读取内部寄存器数据
void Multiple_ReadL3G4200D(); //连续的读取内部寄存器数据
//------------------------------------
void Delay5us();
void Delay5ms();
void L3G4200D_Start();
void L3G4200D_Stop();
void L3G4200D_SendACK(bit ack);
bit L3G4200D_RecvACK();
void L3G4200D_SendByte(BYTE dat);
BYTE L3G4200D_RecvByte();
void L3G4200D_ReadPage();
void L3G4200D_WritePage();
void display_x();
void display_y();
void display_z();
//-----------------------------------
//*********************************************************
void conversion(uint temp_data)
{
wan=temp_data/10000+0x30 ;
temp_data=temp_data%10000; //取余运算
qian=temp_data/1000+0x30 ;
temp_data=temp_data%1000; //取余运算
bai=temp_data/100+0x30 ;
temp_data=temp_data%100; //取余运算
shi=temp_data/10+0x30 ;
temp_data=temp_data%10; //取余运算
ge=temp_data+0x30;
}
/*******************************/
void delay(unsigned int k)
{
unsigned int i,j;
for(i=0;i<k;i++)
{
for(j=0;j<121;j++)
{;}
}
}
/*******************************/
void WaitForEnable(void)
{
DataPort=0xff;
LCM_RS=0;LCM_RW=1;_nop_();
LCM_EN=1;_nop_();_nop_();
while(DataPort&0x80);
LCM_EN=0;
}
/*******************************/
void WriteCommandLCM(uchar CMD,uchar Attribc)
{
if(Attribc)WaitForEnable();
LCM_RS=0;LCM_RW=0;_nop_();
DataPort=CMD;_nop_();
LCM_EN=1;_nop_();_nop_();LCM_EN=0;
}
/*******************************/
void WriteDataLCM(uchar dataW)
{
WaitForEnable();
LCM_RS=1;LCM_RW=0;_nop_();
DataPort=dataW;_nop_();
LCM_EN=1;_nop_();_nop_();LCM_EN=0;
}
/***********************************/
void InitLcd()
{
WriteCommandLCM(0x38,1);
WriteCommandLCM(0x08,1);
WriteCommandLCM(0x01,1);
WriteCommandLCM(0x06,1);
WriteCommandLCM(0x0c,1);
}
/***********************************/
void DisplayOneChar(uchar X,uchar Y,uchar DData)
{
Y&=1;
X&=15;
if(Y)X
[解决办法]
=0x40;
X
[解决办法]
=0x80;
WriteCommandLCM(X,0);
WriteDataLCM(DData);
}
/**************************************
延时5微秒(STC90C52RC@12M)
不同的工作环境,需要调整此函数,注意时钟过快时需要修改
当改用1T的MCU时,请调整此延时函数
**************************************/
void Delay5us()
{
_nop_();_nop_();_nop_();_nop_();
_nop_();_nop_();_nop_();_nop_();
_nop_();_nop_();_nop_();_nop_();
_nop_();_nop_();_nop_();_nop_();
_nop_();_nop_();_nop_();_nop_();
_nop_();_nop_();_nop_();_nop_();
}
/**************************************
延时5毫秒(STC90C52RC@12M)
不同的工作环境,需要调整此函数
当改用1T的MCU时,请调整此延时函数
**************************************/
void Delay5ms()
{
WORD n = 560;
while (n--);
}
/**************************************
起始信号
**************************************/
void L3G4200D_Start()
{
SDA = 1; //拉高数据线
SCL = 1; //拉高时钟线
Delay5us(); //延时
SDA = 0; //产生下降沿
Delay5us(); //延时
SCL = 0; //拉低时钟线
}
/**************************************
停止信号
**************************************/
void L3G4200D_Stop()
{
SDA = 0; //拉低数据线
SCL = 1; //拉高时钟线
Delay5us(); //延时
SDA = 1; //产生上升沿
Delay5us(); //延时
}
/**************************************
发送应答信号
入口参数:ack (0:ACK 1:NAK)
**************************************/
void L3G4200D_SendACK(bit ack)
{
SDA = ack; //写应答信号
SCL = 1; //拉高时钟线
Delay5us(); //延时
SCL = 0; //拉低时钟线
Delay5us(); //延时
}
/**************************************
接收应答信号
**************************************/
bit L3G4200D_RecvACK()
{
SCL = 1; //拉高时钟线
Delay5us(); //延时
CY = SDA; //读应答信号
SCL = 0; //拉低时钟线
Delay5us(); //延时
return CY;
}
/**************************************
向IIC总线发送一个字节数据
**************************************/
void L3G4200D_SendByte(BYTE dat)
{
BYTE i;
for (i=0; i<8; i++) //8位计数器
{
dat <<= 1; //移出数据的最高位
SDA = CY; //送数据口
SCL = 1; //拉高时钟线
Delay5us(); //延时
SCL = 0; //拉低时钟线
Delay5us(); //延时
}
L3G4200D_RecvACK();
}
/**************************************
从IIC总线接收一个字节数据
**************************************/
BYTE L3G4200D_RecvByte()
{
BYTE i;
BYTE dat = 0;
SDA = 1; //使能内部上拉,准备读取数据,
for (i=0; i<8; i++) //8位计数器
{
dat <<= 1;
SCL = 1; //拉高时钟线
Delay5us(); //延时
dat
[解决办法]
= SDA; //读数据
SCL = 0; //拉低时钟线
Delay5us(); //延时
}
return dat;
}
//单字节写入*******************************************
void Single_WriteL3G4200D(uchar REG_Address,uchar REG_data)
{
L3G4200D_Start(); //起始信号
L3G4200D_SendByte(SlaveAddress); //发送设备地址+写信号
L3G4200D_SendByte(REG_Address); //内部寄存器地址,请参考中文pdf22页
L3G4200D_SendByte(REG_data); //内部寄存器数据,请参考中文pdf22页
L3G4200D_Stop(); //发送停止信号
}
//单字节读取*****************************************
uchar Single_ReadL3G4200D(uchar REG_Address)
{ uchar REG_data;
L3G4200D_Start(); //起始信号
L3G4200D_SendByte(SlaveAddress); //发送设备地址+写信号
L3G4200D_SendByte(REG_Address); //发送存储单元地址,从0开始
L3G4200D_Start(); //起始信号
L3G4200D_SendByte(SlaveAddress+1); //发送设备地址+读信号
REG_data=L3G4200D_RecvByte(); //读出寄存器数据
L3G4200D_SendACK(1);
L3G4200D_Stop(); //停止信号
return REG_data;
}
//*************************************************
//
//连续读出内部数据
//
//**************************************************
void Multiple_readL3G4200D(void)
{ uchar i;
L3G4200D_Start(); //起始信号
L3G4200D_SendByte(SlaveAddress); //发送设备地址+写信号
L3G4200D_SendByte(0x28); //发送存储单元地址
L3G4200D_Start(); //起始信号
L3G4200D_SendByte(SlaveAddress+1); //发送设备地址+读信号
for (i=0; i<6; i++) //连续读取6个地址数据,存储中BUF
{
BUF[i] = L3G4200D_RecvByte()&0xFF; //BUF存储数据
if (i == 5)
{
L3G4200D_SendACK(1); //最后一个数据需要回NOACK
}
else
{
L3G4200D_SendACK(0); //回应ACK
}
}
L3G4200D_Stop(); //停止信号
Delay5ms();
}
//*****************************************************************
//初始化L3G4200D,根据需要请参考pdf,第27页,进行修改************************
void InitL3G4200D()
{
Single_WriteL3G4200D(CTRL_REG1, 0x0f); //
Single_WriteL3G4200D(CTRL_REG2, 0x00); //
Single_WriteL3G4200D(CTRL_REG3, 0x08); //
Single_WriteL3G4200D(CTRL_REG4, 0x00); //
Single_WriteL3G4200D(CTRL_REG5, 0x00);
}
//***********************************************************************
//显示x轴
void display_x()
{ // float temp;
BUF[0]= Single_ReadL3G4200D(0x28);
BUF[1]= Single_ReadL3G4200D(0x29);//数值计算,请参考L3G4200D_AN3393 PDF 第12页
dis_data=(BUF[1]<<8)+BUF[0]; //合成数据
if(dis_data<0){
dis_data=-dis_data;
DisplayOneChar(2,0,'-'); //显示正负符号位
}
else DisplayOneChar(2,0,' '); //显示空格
conversion(dis_data); //转换出显示需要的数据
DisplayOneChar(0,0,'X'); //第0行,第0列 显示X
DisplayOneChar(1,0,':');
DisplayOneChar(3,0,qian);
DisplayOneChar(4,0,'.');
DisplayOneChar(5,0,bai);
DisplayOneChar(6,0,shi);
DisplayOneChar(7,0,ge);
}
//***********************************************************************
//显示y轴
void display_y()
{ // float temp;
BUF[2]= Single_ReadL3G4200D(0x2a);
BUF[3]= Single_ReadL3G4200D(0x2b); //数值计算,请参考L3G4200D_AN3393 PDF 第12页
dis_data=(BUF[3]<<8)+BUF[2]; //合成数据
if(dis_data<0){
dis_data=-dis_data;
DisplayOneChar(2,1,'-'); //显示正负符号位
}
else DisplayOneChar(2,1,' '); //显示空格
conversion(dis_data); //转换出显示需要的数据
DisplayOneChar(0,1,'Y'); //第1行,第0列 显示y
DisplayOneChar(1,1,':');
DisplayOneChar(3,1,qian);
DisplayOneChar(4,1,'.');
DisplayOneChar(5,1,bai);
DisplayOneChar(6,1,shi);
DisplayOneChar(7,1,ge);
}
//***********************************************************************
//显示z轴
void display_z()
{ //float temp;
BUF[4]= Single_ReadL3G4200D(0x2c);
BUF[5]= Single_ReadL3G4200D(0x2d); //数值计算,请参考L3G4200D_AN3393 PDF 第12页
dis_data=(BUF[5]<<8)+BUF[4]; //合成数据
if(dis_data<0){
dis_data=-dis_data;
DisplayOneChar(10,1,'-'); //显示负符号位
}
else DisplayOneChar(10,1,' '); //显示空格
conversion(dis_data); //转换出显示需要的数据
DisplayOneChar(10,0,'Z'); //第0行,第10列 显示Z
DisplayOneChar(11,0,':');
DisplayOneChar(11,1,qian);
DisplayOneChar(12,1,'.');
DisplayOneChar(13,1,bai);
DisplayOneChar(14,1,shi);
DisplayOneChar(15,1,ge);
}
//*********************************************************
//******主程序********
//*********************************************************
void main()
{
delay(500); //上电延时
InitLcd(); //液晶初始化
InitL3G4200D(); //初始化L3G4200D
devid=Single_ReadL3G4200D(0X26); //读取温度,温度地址0x26
while(1) //循环
{
display_x(); //---------显示X轴
display_y(); //---------显示Y轴
display_z(); //---------显示Z轴
delay(1000); //延时
}
}