坊間一直流傳著一個傳說~STM32的硬件I2C設(shè)計有BUG,最好不要用,用軟件I2C比較靠譜。長久以來,為了不必要的麻煩,我也一直沒有用過硬件I2C,主要是軟件I2C也比較方便,基本上任意端口都可以用。
最近畫了塊板子,正好用到了I2C,就順便來測試一下硬件I2C是不是真的像有些人說的不好用。
測試硬件:STM32F407VET6+AT24C64測試軟件:STM32CubeMX v6.1.1HAL庫:STM32CubeF4 Firmware Package V1.25.2
STM32CubeMX配置 使用STM32CubeMX配置很方便,時鐘等基礎(chǔ)配置不再詳細(xì)介紹,直接看I2C配置如下:
這里的速度模式選擇為標(biāo)準(zhǔn)模式,時鐘為100K。要求高的可以選擇Fast模式,400K時鐘。 配置完成后生成代碼。
編寫代碼 代碼生成后,直接調(diào)用讀寫數(shù)據(jù)的函數(shù)即可: HAL_I2C_Mem_Read HAL_I2C_Mem_Write 函數(shù)參數(shù)可參考代碼注釋。 24CXX系列的EEPROM進(jìn)行寫操作時需要注意,跨頁寫入時,要有一定的延時,否則會寫入不成功。不同容量的頁大小也不一樣。 另外,24C16以下容量的地址為8位,24C32以上容量的地址為16位,在調(diào)用讀寫函數(shù)時需要注意,選擇I2C_MEMADD_SIZE_8BIT或者I2C_MEMADD_SIZE_16BIT。測試使用的是24C64,所以選擇I2C_MEMADD_SIZE_16BIT。 為了方便操作,將讀寫函數(shù)再封裝一層,將跨頁寫入的各種情況都考慮到,實現(xiàn)任意地址連續(xù)寫入。程序如下:
#include “at24c64.h”#include “i2c.h”
#define AT24CXX_ADDR_READ
0xA1#define AT24CXX_ADDR_WRITE
0xA0#define PAGE_SIZE
32/** * @brief
AT24C64任意地址連續(xù)讀多個字節(jié)數(shù)據(jù) * @param
addr —— 讀數(shù)據(jù)的地址(0-65535) * @param
dat —— 存放讀出數(shù)據(jù)的地址 * @retval
成功 —— HAL_OK*/uint8_t At24cxx_Read_Amount_Byte(uint16_t addr, uint8_t* recv_buf, uint16_t size){
return HAL_I2C_Mem_Read(&hi2c2, AT24CXX_ADDR_READ, addr, I2C_MEMADD_SIZE_16BIT, recv_buf, size, 0xFFFFFFFF);}
/** * @brief
AT24C64任意地址連續(xù)寫多個字節(jié)數(shù)據(jù) * @param
addr —— 寫數(shù)據(jù)的地址(0-65535) * @param
dat —— 存放寫入數(shù)據(jù)的地址 * @retval
成功 —— HAL_OK*/uint8_t At24cxx_Write_Amount_Byte(uint16_t addr, uint8_t* dat, uint16_t size){
uint8_t i = 0; uint16_t cnt = 0;
//寫入字節(jié)計數(shù)
/* 對于起始地址,有兩種情況,分別判斷 */
if(0 == addr % PAGE_SIZE )
{
/* 起始地址剛好是頁開始地址 */
/* 對于寫入的字節(jié)數(shù),有兩種情況,分別判斷 */
if(size 《= PAGE_SIZE)
{
//寫入的字節(jié)數(shù)不大于一頁,直接寫入
return HAL_I2C_Mem_Write(&hi2c2, AT24CXX_ADDR_WRITE, addr, I2C_MEMADD_SIZE_16BIT, dat, size, 0xFFFFFFFF);
}
else
{
//寫入的字節(jié)數(shù)大于一頁,先將整頁循環(huán)寫入
for(i = 0;i 《 size/PAGE_SIZE; i++)
{
HAL_I2C_Mem_Write(&hi2c2, AT24CXX_ADDR_WRITE, addr, I2C_MEMADD_SIZE_16BIT, &dat[cnt], PAGE_SIZE, 0xFFFFFFFF);
HAL_Delay(3);
addr += PAGE_SIZE;
cnt += PAGE_SIZE;
}
//將剩余的字節(jié)寫入
return HAL_I2C_Mem_Write(&hi2c2, AT24CXX_ADDR_WRITE, addr, I2C_MEMADD_SIZE_16BIT, &dat[cnt], size - cnt, 0xFFFFFFFF);
}
}
else
{
/* 起始地址偏離頁開始地址 */
/* 對于寫入的字節(jié)數(shù),有兩種情況,分別判斷 */
if(size 《= (PAGE_SIZE - addr%PAGE_SIZE))
{
/* 在該頁可以寫完 */
return HAL_I2C_Mem_Write(&hi2c2, AT24CXX_ADDR_WRITE, addr, I2C_MEMADD_SIZE_16BIT, dat, size, 0xFFFFFFFF);
}
else
{
/* 該頁寫不完 */
//先將該頁寫完
cnt += PAGE_SIZE - addr%PAGE_SIZE;
HAL_I2C_Mem_Write(&hi2c2, AT24CXX_ADDR_WRITE, addr, I2C_MEMADD_SIZE_16BIT, dat, cnt, 0xFFFFFFFF);
addr += cnt;
HAL_Delay(3);
//循環(huán)寫整頁數(shù)據(jù)
for(i = 0;i 《 (size - cnt)/PAGE_SIZE; i++)
{
HAL_I2C_Mem_Write(&hi2c2, AT24CXX_ADDR_WRITE, addr, I2C_MEMADD_SIZE_16BIT, &dat[cnt], PAGE_SIZE, 0xFFFFFFFF);
HAL_Delay(3);
addr += PAGE_SIZE;
cnt += PAGE_SIZE;
}
//將剩下的字節(jié)寫入
return HAL_I2C_Mem_Write(&hi2c2, AT24CXX_ADDR_WRITE, addr, I2C_MEMADD_SIZE_16BIT, &dat[cnt], size - cnt, 0xFFFFFFFF);
}
}}
測試結(jié)果經(jīng)過測試硬件I2C讀寫EEPROM正常。沒有發(fā)現(xiàn)所謂的BUG,當(dāng)然這只是M4內(nèi)核的針對EEPROM一種器件的測試,對于其它內(nèi)核(M3等)和其它I2C器件,還有待驗證。
總結(jié)硬件I2C使用起來比較簡單,不需要自己去調(diào)節(jié)時序,但是只能使用固定的幾個引腳。軟件模擬I2C可以使用任意引腳,針對不同的MCU,移植起來比較方便,但對于不同頻率的MCU,時序調(diào)節(jié)比較麻煩。
兩者各有其優(yōu)缺點,需要根據(jù)實際需求去選擇。
審核編輯 :李倩
-
STM32
+關(guān)注
關(guān)注
2295文章
11032瀏覽量
365217 -
I2C
+關(guān)注
關(guān)注
28文章
1541瀏覽量
127930 -
函數(shù)
+關(guān)注
關(guān)注
3文章
4381瀏覽量
64945
原文標(biāo)題:STM32的硬件I2C有BUG嗎?
文章出處:【微信號:mcu168,微信公眾號:硬件攻城獅】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
STM32H7CubeMX配置硬件I2C,讀寫失敗是什么問題呀?
I2C總線通信原理 如何設(shè)計I2C總線電路
I2C總線的優(yōu)缺點分析
I2C總線與Arduino的接口示例
I2C總線的工作模式介紹
I2C總線故障排除技巧
I2C總線設(shè)備地址設(shè)置方法
I2C總線應(yīng)用實例分析
RISC V的I2C操作
I2C協(xié)議的基礎(chǔ)知識

評論