OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 3378|回复: 5

为什么我的I2C跑不出来,求助

[复制链接]

2

主题

8

帖子

0

精华

新手入门

积分
4
金钱
4
注册时间
2021-4-17
在线时间
5 小时
发表于 2022-2-28 21:46:33 | 显示全部楼层 |阅读模式
12金钱
本帖最后由 OUYTYI 于 2022-3-1 19:31 编辑

.h文件内容如下

#ifndef _IIC_EEPROM
#define _IIC_EEPROM

#include "stm32f10x.h"

#define IIC_EEPROM_ADDRESS                        0XA0
#define EEPROM_PAGE                                        8
        
#define IIC_GPIO_SCL_PIN                        GPIOB        
#define IIC_GPIO_SDA_PIN                        GPIOB
#define IIC_GPIO_SCL_PORT                        GPIO_Pin_6        
#define IIC_GPIO_SDA_PORT                        GPIO_Pin_7
        
#define IIC_EEPROM                                        IIC1
#define IIC_EEPROM_BAUDRATE                        400000
                                                               
#define IIC_GPIO_CLK                                RCC_APB2Periph_GPIOB
#define IIC_CLK                                                RCC_APB1Periph_IIC

#define IIC_SCL_0                                        GPIO_ResetBits(GPIOB,GPIO_Pin_6)
#define IIC_SCL_1                                        GPIO_SetBits(GPIOB,GPIO_Pin_6)

#define IIC_SDA_0                                        GPIO_ResetBits(GPIOB,GPIO_Pin_7)
#define IIC_SDA_1                                        GPIO_SetBits(GPIOB,GPIO_Pin_7)


#define IIC_Read_SDA_status                                (IIC_GPIO_SDA_PIN->IDR & IIC_GPIO_SDA_PORT)

void IIC_EEPROM_GPIO_Init(void);

u8 IIC_Byte_Read(void);
void IIC_Byte_Write(u8 data);
u8 IIC_ReadOneByte(u16 address);
u8 IIC_WriteOneByte(u16 address,u8 write_data);
void IIC_Read(u16 ReadAddr,u8 *pBuffer,u16 NumToRead);
void IIC_Write(u16 WriteAddr,u8 *pBuffer,u16 NumToWrite);
void IIC_Start(void);
void IIC_Stop(void);
u8 AT24CXX_Check(void);

#endif


.c文件如下
#include "I2C_EEPROM.h"
#include "my_usart.h"
#include "delay.h"

u32 IIC_TIME;


void IIC_EEPROM_GPIO_Init(void)
{
        GPIO_InitTypeDef GPIO_InitStructure;
        RCC_APB2PeriphClockCmd(        RCC_APB2Periph_GPIOB, ENABLE );        //使能GPIOB时钟
           
        GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6|GPIO_Pin_7;
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD ;   
        GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
        GPIO_Init(GPIOB, &GPIO_InitStructure);
        GPIO_SetBits(GPIOB,GPIO_Pin_6|GPIO_Pin_7);
}


void IIC_Byte_Send(u8 data)
{
        u8 i;
        for(i=0;i<8;i++)
        {
                if(data&0x80)
                {
                        IIC_SDA_1;
                }
                else
                {
                        IIC_SDA_0;
                }
                delay_us(5);
                IIC_SCL_1;
                delay_us(5);
                IIC_SCL_0;
               
                if(i==7)
                {
                        IIC_SDA_1;
                }
                data<<=1;
                delay_us(5);
               
        }        
}

u8 IIC_Byte_Read(void)
{
        u8 i;
        u8 data=0;
        for(i=0;i<8;i++)
        {
                data<<=1;
                delay_us(5);
                IIC_SCL_1;
               
                if(IIC_Read_SDA_status)
                {
                        data++;
                }
               
                delay_us(5);
                IIC_SCL_0;
        }
        return data;
}

u8 IIC_Wait_ack(void)
{
        u8 IIC_DEVICE_RETURN;
        IIC_SDA_1;
        delay_us(5);
        IIC_SCL_1;
        delay_us(5);
        IIC_SCL_0;
        if(IIC_Read_SDA_status==0)
        {
                IIC_DEVICE_RETURN=1;
        }
        else
        {
                IIC_DEVICE_RETURN=0;
        }
        return IIC_DEVICE_RETURN;
}


void IIC_Start(void)
{
        IIC_SCL_1;
        IIC_SDA_1;
        delay_ms(5);
        IIC_SDA_0;
        delay_ms(5);
        IIC_SCL_0;
        
}

void IIC_Stop(void)
{
        IIC_SCL_1;
        IIC_SDA_0;
        delay_us(5);
        IIC_SDA_1;
        delay_us(5);
        IIC_SCL_0;
}

void IIC_Ack(void)
{
        IIC_SDA_0;
        IIC_SCL_1;
        delay_us(5);
        IIC_SCL_0;
        delay_us(5);
}

void IIC_Nack(void)
{
        IIC_SDA_1;
        IIC_SCL_1;
        delay_us(5);
        IIC_SCL_0;
        delay_us(5);
}





u8 IIC_WriteOneByte(u16 address,u8 write_data)
{
        IIC_Start();
        if(IIC_Wait_ack()==0)
        {
                goto IIC_FAIL;
        }
        
        IIC_Byte_Send(0XA0);
        if(IIC_Wait_ack()==0)
        {
                goto IIC_FAIL;
        }
        
        IIC_Byte_Send(address);
        if(IIC_Wait_ack()==0)
        {
                goto IIC_FAIL;
        }
        
        IIC_Byte_Send(write_data);
        if(IIC_Wait_ack()==0)
        {
                goto IIC_FAIL;
        }
        IIC_Stop();
        return 1;
        
        IIC_FAIL:
        IIC_Stop();
        return 0;
}


u8 IIC_ReadOneByte(u16 address)
{
        u8 DATA;
        
        IIC_Start();
        if(IIC_Wait_ack()==0)
        {
                goto IIC_FAIL;
        }
        
        IIC_Byte_Send(0XA0);
        if(IIC_Wait_ack()==0)
        {
                goto IIC_FAIL;
        }
        
        IIC_Byte_Send(address);
        if(IIC_Wait_ack()==0)
        {
                goto IIC_FAIL;
        }
        
        IIC_Start();
        if(IIC_Wait_ack()==0)
        {
                goto IIC_FAIL;
        }
        
        IIC_Byte_Send(0XA1);
        if(IIC_Wait_ack()==0)
        {
                goto IIC_FAIL;
        }
        
        DATA =IIC_Byte_Read();
        
        
        IIC_Stop();
        return DATA;
        
        IIC_FAIL:
        IIC_Stop();
        return 0;
}

void IIC_Write(u16 WriteAddr,u8 *pBuffer,u16 NumToWrite)
{
        while(NumToWrite--)
        {
                IIC_WriteOneByte(WriteAddr,*pBuffer);
                WriteAddr++;
                pBuffer++;
        }
}

void IIC_Read(u16 ReadAddr,u8 *pBuffer,u16 NumToRead)
{
        while(NumToRead)
        {
                *pBuffer++=IIC_ReadOneByte(ReadAddr++);        
                NumToRead--;
        }
}




u8 AT24CXX_Check(void)
{
        u8 temp;
        temp=IIC_ReadOneByte(255);//避免每次开机都写AT24CXX                           
        if(temp==0X55)return 0;                  
        else//排除第一次初始化的情况
        {
                IIC_WriteOneByte(255,0X55);
            temp=IIC_ReadOneByte(255);         
                if(temp==0X55)return 0;
        }
        return 1;                                                                                          
}


main函数如下:

#include "stm32f10x.h"
#include "led.h"
#include "delay.h"
#include "my_usart.h"
#include "I2C_EEPROM.h"

int main(void)
{        
        u8 IIC_WriteData[10]={1,1,1,1,1,1,1,1,1,1};
        u8 IIC_ReadData[10]={0};
        delay_init();
        my_usart_init();
        
        IIC_EEPROM_GPIO_Init();
        

        while(AT24CXX_Check()==1)
        {
                delay_ms(500);
                printf("error\r\n");
        }
        
        printf("IIC实验\r\n");
        IIC_EEPROM_GPIO_Init();

        IIC_Write(0,(u8*)IIC_WriteData,8);
        IIC_Read(0,IIC_ReadData,8);
        
        for(i=0;i<10;i++)
        {
                printf("%d\r\n",IIC_WriteData);
                printf("%d",IIC_ReadData);
        }
        
        while(1)
        {
               
        }
               
}        








I2C_SOFT.zip

7.16 MB, 下载次数: 1

最佳答案

查看完整内容[请看2#楼]

为什么原子的例程里要在sda线设置输入输出状态的切换呢?我看野火的例程就没有这种操作,正常配置推挽的话,没必要切换输入输出状态吧,我把我的start函数放进原子的例程,把原子原先的例程注释,发现程序就不行了,真的好奇怪,有大佬能帮忙看一看吗,万分感谢
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

2

主题

8

帖子

0

精华

新手入门

积分
4
金钱
4
注册时间
2021-4-17
在线时间
5 小时
 楼主| 发表于 2022-2-28 21:46:34 | 显示全部楼层
为什么原子的例程里要在sda线设置输入输出状态的切换呢?我看野火的例程就没有这种操作,正常配置推挽的话,没必要切换输入输出状态吧,我把我的start函数放进原子的例程,把原子原先的例程注释,发现程序就不行了,真的好奇怪,有大佬能帮忙看一看吗,万分感谢
回复

使用道具 举报

14

主题

821

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
2070
金钱
2070
注册时间
2021-7-17
在线时间
647 小时
发表于 2022-3-1 10:42:15 | 显示全部楼层
帮顶   
回复

使用道具 举报

2

主题

8

帖子

0

精华

新手入门

积分
4
金钱
4
注册时间
2021-4-17
在线时间
5 小时
 楼主| 发表于 2022-3-3 22:03:21 | 显示全部楼层
我就想知道解答自己的贴子能不能把自己悬赏的金钱拿回来
回复

使用道具 举报

2

主题

8

帖子

0

精华

新手入门

积分
4
金钱
4
注册时间
2021-4-17
在线时间
5 小时
 楼主| 发表于 2022-3-3 22:07:03 | 显示全部楼层
自己解答自己问题,程序问题出在IIC 的起始信号后面跟了ack检测,违反了IIC协议的时序,所以跑不出来,然后这边需要说一下原子的例程,个人感觉原子没有很好的用IIC的原理写程序,首先IIC协议是外部接上拉电阻的,本身的协议其实就靠开漏输出就能解决,原子的操作寄存器输入输出模式切换完全是多余的一步,然后目前还在检测原子的写时序最后加延时的必要性,感觉没必要,但是初步检测如果跟着原子例程写的程序,是必须加这个延时的,实验结束再回这个帖子
回复

使用道具 举报

2

主题

8

帖子

0

精华

新手入门

积分
4
金钱
4
注册时间
2021-4-17
在线时间
5 小时
 楼主| 发表于 2022-3-4 22:54:54 | 显示全部楼层
本帖最后由 OUYTYI 于 2022-3-19 16:12 编辑

。。。。。。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-2-28 03:08

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表