SHT20 IIC驱动 返回能接受值,值是无效的,波形分析求助

2019-07-21 04:37发布

#include "iic.h"

void delay_us(uint16_t nus);


void iic_Delay(uint16_t value )
{
    uint16_t i;
    for (i = 0; i < value; i++);
}

void IIC_Init(void)
{
       
       
        //使能GPIOB
        RCC->IOPENR |= 1 << 1;
               
        //重置GPIO端口B
       
        //配置SCL管脚   PB12   
        GPIOB->MODER &= ~(0x3 << 2 * 12);
        GPIOB->MODER |= 1 << 2 * 12;
        GPIOB->OTYPER &= ~(1 << 12);  //推挽输出
        GPIOB->OSPEEDR |= 0x3 << 2 * 12;
       
        //配置SDA管脚   PB13
        GPIOB->MODER &= ~(0x3 << 2 * 13);
        GPIOB->MODER |= 0 << 2 * 13;
//        GPIOB->OTYPER |= (1 << 13);  //开漏输出
        GPIOB->OTYPER &= ~(1 << 13);  //推挽输出
        GPIOB->OSPEEDR |= 0x3 << 2 * 13;
       
        GPIOB->ODR |= 0x3 << 12;
       
}

//产生IIC起始信号
void IIC_Start(void)
{
        Set_SDA_OUTPUT();     //sda线输出
        Set_SDA_HIGH();                    
        Set_SCL_HIGH();
        iic_Delay(20);
        Set_SDA_LOW();//START:when CLK is high,DATA change form high to low
        iic_Delay(20);
        Set_SCL_LOW();//钳住I2C总线,准备发送或接收数据
}          
//产生IIC停止信号
void IIC_Stop(void)
{
        Set_SDA_OUTPUT();//sda线输出
        Set_SCL_LOW();
        Set_SDA_LOW();//STOP:when CLK is high DATA change form low to high
        iic_Delay(20);
        Set_SCL_HIGH();
        iic_Delay(20);
        Set_SDA_HIGH();//发送I2C总线结束信号
        iic_Delay(20);                                                                  
}
//等待应答信号到来
//返回值:1,接收应答失败
//        0,接收应答成功
uint8_t  IIC_Wait_Ack(void)
{
        uint8_t  ucErrTime=0;
        Set_SDA_INPUT();      //SDA设置为输入  
        Set_SDA_HIGH();iic_Delay(10);          
        Set_SCL_HIGH();iic_Delay(10);         
        while(Read_SDA())
        {
                ucErrTime++;
                if(ucErrTime>250)
                {
                        IIC_Stop();
                        return 1;
                }
        }
        Set_SCL_LOW();//时钟输出0           
        return 0;  
}
//产生ACK应答
void IIC_Ack(void)
{
        Set_SCL_LOW();
        Set_SDA_OUTPUT();
        Set_SDA_LOW();
        iic_Delay(20);
        Set_SCL_HIGH();
        iic_Delay(20);
        Set_SCL_LOW();
        iic_Delay(20);
        Set_SDA_HIGH();
}
//不产生ACK应答                    
void IIC_NAck(void)
{
        Set_SCL_LOW();
        Set_SDA_OUTPUT();
        Set_SDA_HIGH();
        iic_Delay(20);
        Set_SCL_HIGH();
        iic_Delay(20);
        Set_SCL_LOW();
}                                                                              
//IIC发送一个字节
//返回从机有无应答
//1,有应答
//0,无应答                          
void IIC_Send_Byte(uint8_t txd)
{                        
    uint8_t  t;   
        Set_SDA_OUTPUT();            
    Set_SCL_LOW();//拉低时钟开始数据传输
    for(t=0;t<8;t++)
    {              
        if((txd&0x80)>>7)
                {
                        Set_SDA_HIGH();
                }
                else
                {
                        Set_SDA_LOW();
                }
        txd<<=1;           
                iic_Delay(20);   //对TEA5767这三个延时都是必须的
                Set_SCL_HIGH();
                iic_Delay(20);
                Set_SCL_LOW();       
                iic_Delay(20);
    }
        Set_SCL_LOW();         
}             
//读1个字节,ack=1时,发送ACK,ack=0,发送nACK   
uint8_t  IIC_Read_Byte(unsigned char ack)
{
        unsigned char i,receive=0;
        Set_SDA_INPUT();//SDA设置为输入
    for(i=0;i<8;i++ )
        {
                Set_SCL_LOW();
                iic_Delay(20);
                Set_SCL_HIGH();
        receive<<=1;
        if(Read_SDA())receive++;   
                iic_Delay(20);
    }                                         
    if (!ack)
        IIC_NAck();//发送nACK
    else
        IIC_Ack(); //发送ACK   
    return receive;
}

uint8_t  i2c_CheckDevice(uint8_t  _Address)
{
        uint8_t  ucAck;

        if ( Read_SDA() && Read_SCL())
        {
                IIC_Start( );                        /* 发送启动信号 */

                /* 发送设备地址+读写控制bit(0 = w, 1 = r) bit7 先传 */
                IIC_Send_Byte( _Address | 1);
                ucAck = IIC_Wait_Ack();                        /* 检测设备的ACK应答 */

                IIC_Stop();                                /* 发送停止信号 */

                return ucAck;
        }
        return 1;        /* I2C总线异常 */
}

友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
5条回答
peng1554
1楼-- · 2019-07-21 06:18
质子哥
2楼-- · 2019-07-21 07:39
peng1554 发表于 2019-3-6 16:05
这两个代码可以参考
http://www.openedv.com/forum.php?mod=viewthread&tid=82737&highlight=SHT20
http: ...

谢谢,我参考的代码就是咱们论坛上的,有应答但数据不对,第一个数据始终为0
秋天
3楼-- · 2019-07-21 07:42
 精彩回答 2  元偷偷看……
tgw860910
4楼-- · 2019-07-21 12:47
IIC一般的读写操作是这样的:

写:
start + 设备写地址 + 写的位置 + 写的内容 + stop

读:
start + 设备写地址 + 读的位置 + start + 设备读地址 + 读回来的数据 + stop

其中哪里需要ACK,你就自己详细看协议吧
sammy902583
5楼-- · 2019-07-21 14:40
您用Command 0xE5, 这个command, SHT20自己会强制拉I2C Bus, 可能就是您读到0的状态.
Hold master的方式感觉不好用, 我比较建议用command 0xF5.

自己拿出SHT20的规格书, 比对一下流程吧.

一周热门 更多>