前一篇是讲用IO口模拟74HC595的时序来控制的,因为74HC595刚好也是串行输入的,所以直接用串行总线SPI来控制,大概说下电路的连接,SPI2的SCK引脚连接74HC595的SHCP引脚,SPI2的CS引脚连接74HC595的STCP引脚,SPI2的MOSI引脚连接74HC595的DS引脚,因为74HC595是没有数据返回给MCU的,所以SPI2的MISO引脚并不需要连接。
下面就把代码放上来,代码很简单,直接使用SPI发送数据就可以了。
开发平台:MDK5.14
MCU:STM32F103ZET6
SPI2头文件
#ifndef __SPI_H
#define __SPI_H
#include "stm32f10x.h"
/*SPI接口定义-开头****************************/
//#define SPIx SPI1
//#define SPI_APBxClock_FUN RCC_APB2PeriphClockCmd
//#define SPI_CLK RCC_APB2Periph_SPI1
#define SPIx SPI2
#define SPI_APBxClock_FUN RCC_APB1PeriphClockCmd
#define SPI_CLK RCC_APB1Periph_SPI2
//CS(NSS)引脚 片选选普通GPIO即可
#define SPI_CS_APBxClock_FUN RCC_APB2PeriphClockCmd
#define SPI_CS_CLK RCC_APB2Periph_GPIOB
#define SPI_CS_PORT GPIOB
#define SPI_CS_PIN GPIO_Pin_12
//SCK引脚
#define SPI_SCK_APBxClock_FUN RCC_APB2PeriphClockCmd
#define SPI_SCK_CLK RCC_APB2Periph_GPIOB
#define SPI_SCK_PORT GPIOB
#define SPI_SCK_PIN GPIO_Pin_13
//MISO引脚
#define SPI_MISO_APBxClock_FUN RCC_APB2PeriphClockCmd
#define SPI_MISO_CLK RCC_APB2Periph_GPIOB
#define SPI_MISO_PORT GPIOB
#define SPI_MISO_PIN GPIO_Pin_14
//MOSI引脚
#define SPI_MOSI_APBxClock_FUN RCC_APB2PeriphClockCmd
#define SPI_MOSI_CLK RCC_APB2Periph_GPIOB
#define SPI_MOSI_PORT GPIOB
#define SPI_MOSI_PIN GPIO_Pin_15
#define SPI_CS_LOW() GPIO_ResetBits( SPI_CS_PORT, SPI_CS_PIN )
#define SPI_CS_HIGH() GPIO_SetBits( SPI_CS_PORT, SPI_CS_PIN )
/*SPI接口定义-结尾****************************/
void SPI_Config(void);
u8 SPI_Read_Send_Byte(u8 byte);
u16 SPI_Read_Send_HalfWord(u16 HalfWord);
void SPI_Send_Byte(u8 byte);
void SPI_Send_Multi_Byte(u8 *data, u16 len);
#endif /* __SPI_H */
SPI2
#include "spi.h"
/**
* @brief SPI_FLASH初始化
* @param 无
* @retval 无
*/
void SPI_Config(void)
{
SPI_InitTypeDef SPI_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
/* 使能SPI时钟 */
SPI_APBxClock_FUN ( SPI_CLK, ENABLE );
/* 使能SPI引脚相关的时钟 */
SPI_CS_APBxClock_FUN ( SPI_CS_CLK|SPI_SCK_CLK|SPI_MISO_PIN|SPI_MOSI_PIN, ENABLE );
/* 配置SPI的 CS引脚,普通IO即可 */
GPIO_InitStructure.GPIO_Pin = SPI_CS_PIN;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_Init(SPI_CS_PORT, &GPIO_InitStructure);
/* 配置SPI的 SCK引脚*/
GPIO_InitStructure.GPIO_Pin = SPI_SCK_PIN;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_Init(SPI_SCK_PORT, &GPIO_InitStructure);
/* 配置SPI的 MISO引脚*/
GPIO_InitStructure.GPIO_Pin = SPI_MISO_PIN;
GPIO_Init(SPI_MISO_PORT, &GPIO_InitStructure);
/* 配置SPI的 MOSI引脚*/
GPIO_InitStructure.GPIO_Pin = SPI_MOSI_PIN;
GPIO_Init(SPI_MOSI_PORT, &GPIO_InitStructure);
/* 停止信号CS引脚高电平*/
SPI_CS_HIGH();
SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;
SPI_InitStructure.SPI_Mode = SPI_Mode_Master;
SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;
SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;
SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;
SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;
SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_64;
SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;
SPI_InitStructure.SPI_CRCPolynomial = 7;
SPI_Init(SPIx , &SPI_InitStructure);
SPI_Cmd(SPIx , ENABLE);
}
u8 SPI_Read_Send_Byte(u8 byte)
{
while (SPI_I2S_GetFlagStatus(SPIx , SPI_I2S_FLAG_TXE) == RESET);
SPI_I2S_SendData(SPIx , byte);
while (SPI_I2S_GetFlagStatus(SPIx , SPI_I2S_FLAG_RXNE) == RESET);
return SPI_I2S_ReceiveData(SPIx);
}
u16 SPI_Read_Send_HalfWord(u16 HalfWord)
{
while (SPI_I2S_GetFlagStatus(SPIx , SPI_I2S_FLAG_TXE) == RESET);
SPI_I2S_SendData(SPIx , HalfWord);
while (SPI_I2S_GetFlagStatus(SPIx , SPI_I2S_FLAG_RXNE) == RESET);
return SPI_I2S_ReceiveData(SPIx );
}
// 在驱动74HC595只需要SPI的发送函数,并没用到读取函数
void SPI_Send_Byte(u8 byte) //发送一个字节
{
while (SPI_I2S_GetFlagStatus(SPIx , SPI_I2S_FLAG_TXE) == RESET){}
SPI_I2S_SendData(SPIx , byte);
while (SPI_I2S_GetFlagStatus(SPIx , SPI_I2S_FLAG_RXNE) == RESET);
SPI_I2S_ReceiveData(SPIx);
}
void SPI_Send_Multi_Byte(u8 *data, u16 len) //连续发送 len 个字节
{
u8 i;
SPI_CS_LOW();
for (i = 0; i < len; i ++ )
{
SPI_Send_Byte(data[i]);
}
SPI_CS_HIGH(); // STCP产生上升沿
}
/*********************************************END OF FILE**********************/
主函数
#include "stm32f10x.h"
#include "spi.h"
//#include "595.h"
#include "SysTick.h"
u8 pos; // led位置
u8 Led_Pos_Buf[2] = {0x00, 0x00}; //存储要发送的指令字节
//第pos个led亮:1 2 3 4 5 6 7 8
u8 Led[32] = {0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, //控制第二级74HC595
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, //第一级的led全灭
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, //第二级的led全灭
0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80};//控制第一级74HC595
int main(void)
{
SPI_Config();
// HC595_GPIO_Config();
SysTick_Init();
while(1)
{
for (pos = 0; pos < 16; pos ++) //第pos个灯,实现流水灯效果
{
SysTick_Delay_Ms(500); //延时500毫秒
Led_Pos_Buf[1] = Led[pos]; //存放第一级74HC595的数据,因为先进先出,所以第一级放在Led_Pos_Buf[1],而不是Led_Pos_Buf[0]
Led_Pos_Buf[0] = Led[pos+16]; //存放第二级74HC595的数据
SPI_Send_Multi_Byte(Led_Pos_Buf,2);//将当前数据发送到595
}
}
}
好了,74HC595的文章就暂时写到这里,有时间可能会再写一篇用stm32产生一路PWM波来控制74HC595实现对LED亮度调节,也就是PWM调光功能。