大家好: 我用STM32F10C8T6 连接IIC 接口的OLED。 STM32cubeMX进行初始化设置,以及修改了一下DMA发送函数,发现不能显示 接口驱动是没问题的,以前用3.5库(没使用DMA)是能正常显示的。 使用板上的LED灯检查卡在哪,发现卡在第二次发送上。 下面是精简后的程序,还请大神指导一下。 int main(void) { HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); MX_DMA_Init(); MX_I2C1_Init(); OLED_Init(); while (1) { } } void I2C_Master_Transmit_DMA(uint16_t DevAddress, uint8_t *pData, uint16_t Size) { while (HAL_I2C_Master_Transmit_DMA(&hi2c1,DevAddress,pData,Size)!= HAL_OK) { if (HAL_I2C_GetError(&hi2c1)!= HAL_I2C_ERROR_AF) { Error_Handler(); } } } void OLED_WR_Byte(uint8_t Byte,uint8_t DC) { uint8_t SendBuff[2]; SendBuff[0] = DC; SendBuff[1] = Byte; I2C_Master_Transmit_DMA(OLED_ADDRESS,SendBuff,2); } void OLED_Init(void) { delay_ms(200); // OLED_WR_Byte(0xAE,OLED_CMD);//--display off关闭显示 OLED_WR_Byte(0x00,OLED_CMD);//---set low column address 设置开始低列地址为SEG0 /*************目的测试卡在哪*********************/ HAL_GPIO_WritePin(LED_GPIO_Port,LED_Pin,GPIO_PIN_SET); OLED_WR_Byte(0x10,OLED_CMD);//---set high column address设置高列地址为0000b OLED_WR_Byte(0x40,OLED_CMD);//--set start line address 设置开始行地址 OLED_WR_Byte(0xB0,OLED_CMD);//--set page address 设置开始页地址PAGE0 OLED_WR_Byte(0x81,OLED_CMD); // contrast control对比度控制,双字节命令 OLED_WR_Byte(0xFF,OLED_CMD);//--对比度为256 OLED_WR_Byte(0xA1,OLED_CMD);//set segment remap SEG0列地址为127 OLED_WR_Byte(0xA6,OLED_CMD);//--设置为正常显示(正常 / 反相显示) OLED_WR_Byte(0xA8,OLED_CMD);//--set multiplex ratio(1 to 64)设置驱动路数 OLED_WR_Byte(0x3F,OLED_CMD);//--驱动路数为1/32 duty占空比 OLED_WR_Byte(0xC8,OLED_CMD);//Com scan direction: Scan from COM[N-1] to COM0 OLED_WR_Byte(0xD3,OLED_CMD);//-set display offset OLED_WR_Byte(0x00,OLED_CMD);// OLED_WR_Byte(0xD5,OLED_CMD);//set osc division OLED_WR_Byte(0x80,OLED_CMD);// OLED_WR_Byte(0xD8,OLED_CMD);//set area color mode off OLED_WR_Byte(0x05,OLED_CMD);// OLED_WR_Byte(0xD9,OLED_CMD);//Set Pre-Charge Period OLED_WR_Byte(0xF1,OLED_CMD);// OLED_WR_Byte(0xDA,OLED_CMD);//set com pin configuartion OLED_WR_Byte(0x12,OLED_CMD);// OLED_WR_Byte(0xDB,OLED_CMD);//set Vcomh设置电压 OLED_WR_Byte(0x30,OLED_CMD);// OLED_WR_Byte(0x8D,OLED_CMD);//set charge pump enable设置电荷泵 OLED_WR_Byte(0x14,OLED_CMD);//开电荷泵 OLED_WR_Byte(0xAF,OLED_CMD);//--turn on oled panel,开始显示 } |
方法一:加延时,固定延时,程序效率比较低,代码如下
void OLED_WR_Byte(uint8_t Byte,uint8_t DC)
{
uint8_t SendBuff[2];
SendBuff[0] = DC;
SendBuff[1] = Byte;
I2C_Master_Transmit_DMA(OLED_ADDRESS,SendBuff,2);
HAL_Delay(1);
}
方法二:加I2C总线状态判断,程序不卡,但没显示,不知什么原因?
void I2C_Master_Transmit_DMA(uint16_t DevAddress, uint8_t *pData, uint16_t Size)
{
while (HAL_I2C_GetState(&hi2c1)!=HAL_I2C_STATE_READY) {}
while (HAL_I2C_Master_Transmit_DMA(&hi2c1,DevAddress,pData,Size)!= HAL_OK)
{
if (HAL_I2C_GetError(&hi2c1)!= HAL_I2C_ERROR_AF)
{
Error_Handler();
}
}
}
1、发送控制命令没有使用DMA方式,直接采用HAL_I2C_Master_Transmit
2、发送显示内容采用以下方式
void I2C_Master_Transmit_DMA(uint16_t DevAddress, uint8_t *pData, uint16_t Size)
{
while (HAL_I2C_GetState(&hi2c1)!=HAL_I2C_STATE_READY) {}
while (HAL_I2C_Master_Transmit_DMA(&hi2c1,DevAddress,pData,Size)!= HAL_OK)
{
if (HAL_I2C_GetError(&hi2c1)!= HAL_I2C_ERROR_AF)
{
Error_Handler();
}
}
}
void OLED_ShowChar(uint8_t chr)
{
uint8_t c=0,i;
c = chr - ' '; //得到字库地址码
uint8_t SendBuff[7];
SendBuff[0] = OLED_DATA;
for(i=0;i<6;i++)
{
SendBuff[i+1] = F6x8[i + 5 * c];
}
I2C_Master_Transmit_DMA(OLED_ADDRESS,SendBuff,7);
}
结果:能显示,但连续显示字符时,第一个字符显示不出来,后续字符显示均正常,在第一个字符前加延时也不行。
刚开始使用硬件I2C,在开发板上试是好的,项目中的板子就不行,所不同的是一个是Microchip的24LCXX系列片子,一个是AT24CXX系列片子,对比了数据资料,参数没发现有明显不同。
调试跟踪,发现死在了一个while中,ST的固件库中是死等标志。后来的做法是,把while循环做了超时退出,实际操作是成功了,但就是标志没置过来。(有人说这种现象是ST I2C的Bug)。
单步调试是好的。后来调试的时候,在两个写操作之前加了一条Printf打印信息,想用来定位程序跑到什么地方出错的。神奇的事情发生了,竟然一路通畅跑过去了。一删Printf,就容易出错。
仔细看EEPROM手册,发现有一个时间非常关键,是前一个“停止”到下一个“起始”之前的延时最大需要5ms,即Twr。见下图时序参数表:
时序波形示意:
后来我在程序中增加了Twr这个延时,保证不少于5ms,后来I2C再没有挂过,产品也持续生产使用了五六年了。
个人猜想,在一个I2C写周期后,硬件I2C立即去读取标志,造成了硬件I2C内部某些机制紊乱。或许与外部I2C器件逻辑及管脚状态也有一定关系。
以上个人经验体会,仅供参考。
评分
查看全部评分
何时怎样停掉?
http://community.st.com/thread/3 ... p-in-hali2cgetstate
使用仿真,卡在了
HAL_Init();—— HAL_MspInit(); —— __HAL_AFIO_REMAP_SWJ_DISABLE();
看看是驱动接口是否开启了功能复用。
评分
查看全部评分
就发送两个uint8_t的数据,数据量太少了,估计还不如直接发送来的快。
不知道楼主有没有检测到DMA传输完成中断?还有就是楼主说卡在了__HAL_AFIO_REMAP_SWJ_DISABLE();吗?怎么会卡在这里?楼主可以把STM32cubeMX进行初始化设置截图贴出来,便于分析
评分
查看全部评分
http://community.st.com/thread/4 ... ith-dma-not-working
http://community.st.com/thread/27652
这个问题已解决了,原来不知怎样在cubemx中设置SWD接口
评分
查看全部评分
1、为了学习研究,所以使用DMA
2、初始化时的确使用率比较低,两个数据。但用于显示时会使用一次传输显示整个字符(英文6字节数据,中文16字节数据)。
3、__HAL_AFIO_REMAP_SWJ_DISABLE();问题已解决了。
谢谢你的经验。
我的现象也是单步DEBUG能正常运行,一起执行就卡。
我查查相关手册看看有没有类似间隔问题。
谢谢,我英文也不行,慢慢看ing
我在发送前增加检查DMA是否完成,结果还是一样的卡
void I2C_Master_Transmit_DMA(uint16_t DevAddress, uint8_t *pData, uint16_t Size)
{
while ((HAL_DMA_GetState(&hdma_i2c2_tx) != HAL_DMA_STATE_READY) && (HAL_DMA_GetState(&hdma_i2c2_rx) != HAL_DMA_STATE_READY)) {} while (HAL_I2C_Master_Transmit_DMA(&hi2c1,DevAddress,pData,Size)!= HAL_OK)
{
if (HAL_I2C_GetError(&hi2c1)!= HAL_I2C_ERROR_AF)
{
Error_Handler();
}
}
}