×
嵌入式 > 技术百科 > 详情

STM8S 硬件SPI驱动74HC595

发布时间:2023-12-18 发布时间:
|

简介:一直对STM8S的硬件SPI感兴趣,但没有真正使用过,以往都是用IO口模拟,这次刚好有个板子上有两个595驱动的8位LED数码管,就在上面试了一把,把过程记录一下。


595是数据接收器件,所以MCU只用了3条线与之相连,分别是MOSI接SER,SCK接CLK,NSS接RCK。其实这里RCK可以用其他的IO口的。


相关的代码:


LOCAL uint8 DISP_BUF[8]={0,1,2,3,4,5,6,7}; //显示缓冲区

PUBLIC uint8 DISP_TAB[]= //显示码表

{

0x14,0xD7,0x4C,0x45,0x87,0x25,0x24,0x57,0x04,0x05,0x06,0xA4,0x3C


};


相关的IO口设为上拉输出


LOCAL void gpio_init(void)

{

GPIO_DeInit(GPIOC);

GPIO_Init(GPIOC,GPIO_PIN_5|GPIO_PIN_6,GPIO_MODE_OUT_PP_LOW_FAST);

GPIO_DeInit(GPIOE);

GPIO_Init(GPIOE,GPIO_PIN_5,GPIO_MODE_OUT_PP_LOW_FAST);

}


SPI初始化


LOCAL void spi_init(void)

{

SPI_DeInit();

CLK_PeripheralClockConfig(CLK_PERIPHERAL_SPI, ENABLE);


SPI_Init(SPI_FIRSTBIT_LSB, SPI_BAUDRATEPRESCALER_256, SPI_MODE_MASTER,SPI_CLOCKPOLARITY_LOW, SPI_CLOCKPHASE_1EDGE, SPI_DATADIRECTION_1LINE_TX, SPI_NSS_SOFT, 0x07);

SPI_Cmd(ENABLE);

}


显示程序


PUBLIC void DISP_Display(void)

{

uint8 i,dig = 0x80;


for (i=0;i<8;i++)

{

GPIO_WriteLow(GPIOE,GPIO_PIN_5);

while (SPI_GetFlagStatus(SPI_FLAG_TXE) == RESET);

SPI_SendData(DISP_TAB[DISP_BUF[i]]);

while (SPI_GetFlagStatus(SPI_FLAG_TXE) == RESET);

SPI_SendData(dig);

while (SPI_GetFlagStatus(SPI_FLAG_TXE) == RESET);

GPIO_WriteHigh(GPIOE,GPIO_PIN_5);

delay(20);

dig >>= 1;

}

}



用示波器看了下,MOSI脚和SCK脚的波形正常。猜想是硬件SPI的速度太快,595跟不上。于是修改代码,在两次发送数据之后都延时一下。


PUBLIC void DISP_Display(void)

{

uint8 i,dig = 0x80;


for (i=0;i<8;i++)

{

GPIO_WriteLow(GPIOE,GPIO_PIN_5);

while (SPI_GetFlagStatus(SPI_FLAG_TXE) == RESET);

SPI_SendData(DISP_TAB[DISP_BUF[i]]);


delay(500);

while (SPI_GetFlagStatus(SPI_FLAG_TXE) == RESET);

SPI_SendData(dig);


delay(500);

while (SPI_GetFlagStatus(SPI_FLAG_TXE) == RESET);

GPIO_WriteHigh(GPIOE,GPIO_PIN_5);

delay(20);

dig >>= 1;

}

}



结论:


1. STM8S 硬件SPI的速度较快,实际应用的时候需要考虑外部设备的响应速度问题。


2. while (SPI_GetFlagStatus(SPI_FLAG_TXE) == RESET);只能保证每次SPI发送数据前,发送缓冲区是空的,即上次数据已经发送完成,但不能保证外部设备正确接收了数据。


『本文转载自网络,版权归原作者所有,如有侵权请联系删除』

热门文章 更多
变废为宝.电话亭将变多功能5G基站?破解5G BBU集中机房建设难题.原来可以如此简单!