stm32 I2C自收发测试例程

发布者:星辰小鹿最新更新时间:2019-04-02 来源: eefocus关键字:stm32  I2C  自收发测试 手机看文章 扫描二维码
随时随地手机看文章

define BufferSize 4

define I2C2_SLAVE_ADDRESS7 0x30

/* Private macro ————————————————————-*/ 

/* Private variables ———————————————————*/


vu8 I2C1_Buffer_Tx[BufferSize] = {1, 2, 3, 4}; /* I2C1待发送字节数组 */ 

vu8 I2C2_Buffer_Rx[BufferSize] = {0, 0, 0, 0}; /* I2C2待接收字节缓冲 */ 

vu8 Tx_Idx = 0; /* I2C1数据发送计数变量 */ 

vu8 Rx_Idx = 0; /* I2C2数据接收计数变量 */


/* 自定义参数宏 ——————————————————-*/


define I2C1_SLAVE_ADDRESS7 0x30 /* 定义 I2C1 本地地址为0x30 */

define I2C2_SLAVE_ADDRESS7 0x30 /* 定义 I2C2 本地地址为0x30 */

define BufferSize 4

/* 自定义函数宏 ——————————————————-*/


/* 自定义变量 ——————————————————-*/


/* 用户函数声明 ———————————————————*/


void RCC_Configuration(void); 

void GPIO_Configuration(void); 

void NVIC_Configuration(void); 

void USART_Configuration(void); 

void I2c_Configuration(void);


int main(void) 

/* 设置系统时钟 */ 

RCC_Configuration(); 

/* 设置 NVIC */ 

NVIC_Configuration(); 

/* 设置GPIO端口 */ 

GPIO_Configuration(); 

/* 设置USART */ 

USART_Configuration(); 

/* 设置 IIC */ 

I2c_Configuration();


/* I2C1产生开始条件 */

I2C_GenerateSTART(I2C1, ENABLE);


while(1);


}


void RCC_Configuration(void) 

ErrorStatus HSEStartUpStatus;


RCC_DeInit();

RCC_HSEConfig(RCC_HSE_ON);

HSEStartUpStatus = RCC_WaitForHSEStartUp();

if(HSEStartUpStatus == SUCCESS)

{

    FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);

    FLASH_SetLatency(FLASH_Latency_2);

    RCC_HCLKConfig(RCC_SYSCLK_Div1); 

    RCC_PCLK2Config(RCC_HCLK_Div1); 

    RCC_PCLK1Config(RCC_HCLK_Div2);

    RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);    

    RCC_PLLCmd(ENABLE);

    while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET);

    RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);

    while(RCC_GetSYSCLKSource() != 0x08);

}


/* 开启 I2C1、I2C2 设备时钟 */

RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1 | RCC_APB1Periph_I2C2, ENABLE);

/* 开启 GPIOA、GPIOB和 USART 设备时钟 */

RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB, ENABLE);

}


void GPIO_Configuration(void) 

/* 定义 GPIO 初始化结构体 GPIO_InitStructure */ 

GPIO_InitTypeDef GPIO_InitStructure;


/* 配置 I2C1 设备的引脚为复用开漏模式 */

GPIO_InitStructure.GPIO_Pin =  GPIO_Pin_6 | GPIO_Pin_7;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_OD;

GPIO_Init(GPIOB, &GPIO_InitStructure);


/* 配置 I2C2 设备的引脚为复用开漏模式 */

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10 | GPIO_Pin_11;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_OD;

GPIO_Init(GPIOB, &GPIO_InitStructure);


/* 配置 USART 设备引脚 */ 

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_Init(GPIOA, &GPIO_InitStructure);


GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;

GPIO_Init(GPIOA, &GPIO_InitStructure);



}


void I2c_Configuration(void) 

/* 定义 I2C 初始化结构体 I2C_InitStructure */ 

I2C_InitTypeDef I2C_InitStructure;


/*  

*   I2C模式;

*   占空比50%;

*   本地地址(前面宏定义定义为0x30)

*   使能应答;

*   应答7位地址;

*   速率200K;

*/

I2C_InitStructure.I2C_Mode = I2C_Mode_I2C;//设置I2C接口为I2C模式

I2C_InitStructure.I2C_DutyCycle = I2C_DutyCycle_2;  //设置占空比为2/1

I2C_InitStructure.I2C_OwnAddress1 = I2C1_SLAVE_ADDRESS7;     //设置第一个设备自身地址可以是7位也可以是10位

I2C_InitStructure.I2C_Ack = I2C_Ack_Enable;//使能ACk应答

I2C_InitStructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit; //设置应答7位地址

I2C_InitStructure.I2C_ClockSpeed = 200000;  //

I2C_Init(I2C1, &I2C_InitStructure);


I2C_InitStructure.I2C_Mode = I2C_Mode_I2C;

I2C_InitStructure.I2C_DutyCycle = I2C_DutyCycle_2;

I2C_InitStructure.I2C_OwnAddress1 = I2C2_SLAVE_ADDRESS7;

I2C_InitStructure.I2C_Ack = I2C_Ack_Enable;

I2C_InitStructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit;

I2C_InitStructure.I2C_ClockSpeed = 200000;

I2C_Init(I2C2, &I2C_InitStructure);


/* 开启I2C1、I2C2 的事件、缓存中断 */

I2C_ITConfig(I2C1, I2C_IT_EVT | I2C_IT_BUF, ENABLE);

I2C_ITConfig(I2C2, I2C_IT_EVT | I2C_IT_BUF, ENABLE);


/* 使能I2C1、I2C2 接口 */

I2C_Cmd(I2C1, ENABLE);

I2C_Cmd(I2C2, ENABLE);


}


void NVIC_Configuration(void) 

/* 定义 NVIC 初始化结构体 NVIC_InitStructure */ 

NVIC_InitTypeDef NVIC_InitStructure;


/* 选择 NVIC 优先级分组1 */

NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);


/* 设置并使能 I2C1 中断 */

NVIC_InitStructure.NVIC_IRQChannel = I2C1_EV_IRQChannel;

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;

NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;

NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

NVIC_Init(&NVIC_InitStructure);


/* 设置并使能 I2C2 中断 */  

NVIC_InitStructure.NVIC_IRQChannel = I2C2_EV_IRQChannel;

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;

NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;

NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

NVIC_Init(&NVIC_InitStructure);


}


void USART_Configuration(void) 

/* 定义USART初始化结构体 USART_InitStructure */ 

USART_InitTypeDef USART_InitStructure; 

/* 定义USART初始化结构体 USART_ClockInitStructure */ 

USART_ClockInitTypeDef USART_ClockInitStructure;


/*  

*   波特率为115200bps;

*   8位数据长度;

*   1个停止位,无校验;

*   禁用硬件流控制;

*   禁止USART时钟;

*   时钟极性低;

*   在第2个边沿捕获数据

*   最后一位数据的时钟脉冲不从 SCLK 输出; 

*/

USART_ClockInitStructure.USART_Clock = USART_Clock_Disable;

USART_ClockInitStructure.USART_CPOL = USART_CPOL_Low;

USART_ClockInitStructure.USART_CPHA = USART_CPHA_2Edge;

USART_ClockInitStructure.USART_LastBit = USART_LastBit_Disable;

USART_ClockInit(USART1 , &USART_ClockInitStructure);


USART_InitStructure.USART_BaudRate = 115200;

USART_InitStructure.USART_WordLength = USART_WordLength_8b;

USART_InitStructure.USART_StopBits = USART_StopBits_1;

USART_InitStructure.USART_Parity = USART_Parity_No ;

USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;

USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;

USART_Init(USART1 , &USART_InitStructure);


/* 使能USART1 */

USART_Cmd(USART1 , ENABLE);


}


void I2C1_EV_IRQHandler(void) 

switch (I2C_GetLastEvent(I2C1)) 

case I2C_EVENT_MASTER_MODE_SELECT: /* 已发送起始条件 */ 

/* 发送从机地址 */ 

I2C_Send7bitAddress(I2C1, I2C2_SLAVE_ADDRESS7, I2C_Direction_Transmitter);


        break;

    }


case I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED:        /* 从机地址已发送 */

    {

        /* 发送第一个数据 */

        printf("rn The I2C1 has send data 0x0%xrn", I2C1_Buffer_Tx[Rx_Idx]);

        I2C_SendData(I2C1, I2C1_Buffer_Tx[Tx_Idx++]);  


        break;

    }



case I2C_EVENT_MASTER_BYTE_TRANSMITTED:                 /* 第一个数据已发送 */

    {

        if(Tx_Idx < BufferSize)

        {

            /* 继续发送剩余数据... ... */

            printf("rn The I2C1 has send data 0x0%xrn", I2C1_Buffer_Tx[Rx_Idx]);

            I2C_SendData(I2C1, I2C1_Buffer_Tx[Tx_Idx++]);

        }

        else

        {

            /* 剩余数据发送完毕,发送结束条件 */

            I2C_GenerateSTOP(I2C1, ENABLE);

            /* 禁止 I2C1 中断*/

            I2C_ITConfig(I2C1, I2C_IT_EVT | I2C_IT_BUF, DISABLE);

        }


        break;

    }



default:

    {

        break;

    }

}


}


/********************************************************************* 

* Function Name : I2C1_ER_IRQHandler 

* Description : This function handles I2C1 Error interrupt request. 

* Input : None 

* Output : None 

* Return : None 

*********************************************************************/ 

void I2C1_ER_IRQHandler(void) 

}


/********************************************************************* 

* Function Name : I2C2_EV_IRQHandler 

* Description : This function handles I2C2 Event interrupt request. 

* Input : None 

* Output : None 

* Return : None 

*********************************************************************/ 

void I2C2_EV_IRQHandler(void) 

switch (I2C_GetLastEvent(I2C2)) 

case I2C_EVENT_SLAVE_RECEIVER_ADDRESS_MATCHED: /* 收到匹配的地址数据 */ 

break; 

}


case I2C_EVENT_SLAVE_BYTE_RECEIVED:             /* 收到数据 */

    {

        if (Rx_Idx < BufferSize)

        {

            I2C2_Buffer_Rx[Rx_Idx] = I2C_ReceiveData(I2C2);

            printf("rn The I2C2 has received data 0x%xrn", I2C2_Buffer_Rx[Rx_Idx++]);

        }


        break;

    }



case I2C_EVENT_SLAVE_STOP_DETECTED:             /* 收到结束条件 */

    {       

        I2C_ClearFlag(I2C2, I2C_FLAG_STOPF);

        I2C_ITConfig(I2C2, I2C_IT_EVT | I2C_IT_BUF, DISABLE);


        break;

    }


default:

    {

        break;

    }

}


}


int fputc(int ch, FILE *f) 

USART_SendData(USART1, (u8) ch); 

while(USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET); 

return ch; 

}


这里写图片描述

注意事项: 

1、注意在配置I2C设备的引脚时,一定要配置成复用开漏模式。原因一方面在于I2C总线要求总线具备线“与”的特性,若配置成推挽模式则无法提供这种特性;而另一方面输出低电平的时刻,对芯片内部表现为VCC直接和GND连接,此时I/O口将烧毁。 

2、I/O口接上拉电阻是必须的,典型的取值是4.7k。 

3、I2C总线定义标准模式最高可工作在100KHZ下,快速模式可工作在400KHZ下。则程序中I2C设备的初始化参数I2C_InitStructure.I2C_ClockSpeed不能大于400 000.

关键字:stm32  I2C  自收发测试 引用地址:stm32 I2C自收发测试例程

上一篇:stm32中i2c之学习浅谈
下一篇:STM32 模拟IIC完整代码

推荐阅读最新更新时间:2024-03-16 16:26

STM32开发 -- Jlink常用命令
h speed 4000 device = STM32F429ZI erase loadbin BOOT.bin 0x8000000 loadbin 正式版本.bin 0x08004000 loadbin 工厂测试.bin 0x08020000 savebin 工厂生产.bin 0x08000000 0x00080000 rx 3 qc exit 一直在用这个脚本烧写程序,但是,细细一看有点懵。 里面 h,rx 3,qc 等的含义不太清楚。 接下来详细总结一下,Jlink都还有哪些常用的命令。 一、查看jlink命令 使用的jlink版本如下图: jlink命令: J-Link ? Available comma
[单片机]
<font color='red'>STM32</font>开发 -- Jlink常用命令
基于STM32的半导体制冷片控制系统设计
一些医疗检测仪器在检测时需要模拟人体温度环境以确保检测的精确性,本文以STM32为主控制器,电机驱动芯片DRV8834 为驱动器,驱动半导体致冷器(帕尔贴)给散热片加热或者制冷。但由于常规的温度控制存在惯性温度误差的问题,无法兼顾高精度和高速性的严格要求,所以采用模糊自适应PID控制方法在线实时调整PID参数,计算PID参数Kp、Ki、Kd调整控制脉冲来控制驱动器的使能。从simulink仿真的和实验结果来看模糊PID控制系统精度高、响应速度快,能达到预期效果。 温度参数是工业生产中常用的被控对象之一,在化工生产、冶金工业、电力工程和食品加工等领域广泛应用,在医疗检测设备中时常需要模拟人体温度进行成分检测。采用直流电机驱动芯片DRV
[单片机]
基于<font color='red'>STM32</font>的半导体制冷片控制系统设计
毕业设计| 两轮平衡小车
# 硬件介绍 # 主控芯片:100脚的STM32F103VET6; 陀螺仪:MPU6050; 电机驱动:TB6612; 蓝牙:汇承的HC05邮票孔封装的; WIFI:济南有人科技的USR-WIFI232-S; 小车底盘:平衡小车之家的某一款带编码器的(不是我买的,同学的); 电池:一节7.2的镍镉电池; 液晶屏:中景园电子1.3寸IIC接口的OLED; 5V降压:ASM1117-5.0; 3.3V降压:SP6203; 开关:三脚纽子开关; 电池接口:T插; 其他器件还有:拨码开关用的是4P贴片式2.54mm角距的,电阻电容这些用的基本上是0603封装按键是两脚贴片,microusb接口用的是5针 7.2四脚插板牛角母座,超声波是
[单片机]
毕业设计| 两轮<font color='red'>自</font>平衡小车
STM32连接射频si4438模块
SI4438射频模块参数: 1、频率范围:425-525MHz 2、数字接收信号强度指示(RSSI) 3、64字节收发数据寄存器(FIFO) 4、跳频功能 等! 使用SI的WDS工具生成代码 1、 选择仿真模式 2、芯片选择si4438 B1模式 3、 Radio Configuration Application 4、 Select Application 1、 Select Project 选择Bidirectional packet ,双向通信模式 2、 Configure project 配置工程 Frequency and power: 频率和功率的设置, ba
[单片机]
STM32单片机特性解析
STM32单片机 由ST厂商推出的STM32系列单片机,行业的朋友都知道,这是一款性价比超高的系列单片机,应该没有之一,功能及其强大。其基于专为要求高性能、 低成本、低功耗的嵌入式应用专门设计的ARM Cortex-M内核,同时具有一流的外设:1μs的双12位ADC,4兆位/秒的UART,18兆位/秒的SPI等等,在功耗和集成度方面也有不俗的表现,当然和MSP430的功耗比起来是稍微逊色的一些,但这并不影响工程师们对它的热捧程度,由于其简单的结构和易用的工具再配合其强大的功能在行业中赫赫有名。其强大的功能主要表现在: 特性 1、内核:ARM32位Cortex-M3CPU,最高工作频率72MHz,1.25DMIPS/MHz,单周
[单片机]
STM32引脚JTDO、JNTRST与JTDI作为普通IO口使用配置
使用Jlink向STM32烧录程序时,需要使用6个芯片的引脚(以STM32F103C8T6为例),分别是PB4/JNTRST、PB3/JTDO、PA13/JTMS、PA14/JTCK、PA15/JTDI、NRST。标准的20针JLink接口如下图所示。 当芯片IO口资源比较紧张时,可选择SW模式烧录程序。SWD只需用到PA13/JTMS、PA14/JTCK两根线,NREST可以接可不接,而剩下的PB4/JNTRST、PB3/JTDO和PA15/JTDI就可以当做普通IO使用。但是这三个口当做普通IO使用时需要先配置。配置方法: void GPIOInit(void) { GPIO_InitTypeDef GPIO_Init
[单片机]
基于I2C总线的DS1629型智能温度传感器性能特点及应用
DS1629是美国DALLAS公司新推出的一种带实时日历时钟和I2C总线接口的智能 温度传感器 ,可广泛用于温度传感系统、数据记录器、个人计算机、个人数字助理和移动电话等设备中。
[嵌入式]
stm32的串口调试卡死问题
且看这次出现的bug是一直等待, while(USART_GetFlagStatus(USE_Usart,USART_FLAG_TC)==RESET); 这个有很多说法 不过本人处理的方法是加入了一个一行代码就好了 NVIC_SetVectorTable(NVIC_VectTab_FLASH,0);
[单片机]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
热门活动
换一批
更多
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

更多精选电路图
换一换 更多 相关热搜器件
更多每日新闻
随便看看
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved