129 lines
4.6 KiB
C
129 lines
4.6 KiB
C
/**
|
||
******************************************************************************
|
||
* @文件 spi.c
|
||
* @作者 阜阳师范大学物电学院
|
||
* @版本 V0.1
|
||
* @日期 2025-03-28
|
||
* @简介 硬件 SPI 驱动程序
|
||
*******
|
||
**/
|
||
|
||
#include "spi.h"
|
||
|
||
|
||
// 定义一个SPI句柄结构体变量,用于存储SPI的初始化参数和状态信息
|
||
SPI_HandleTypeDef SPI2_Handle;
|
||
|
||
|
||
/**
|
||
* @brief 初始化SPI2外设
|
||
*
|
||
* 该函数用于初始化SPI2外设,配置SPI的工作模式、数据大小、时钟极性等参数,
|
||
* 并使能SPI2外设。
|
||
*/
|
||
void SPI_Init(void)
|
||
{
|
||
// 使能SPI2的时钟,确保SPI2外设能够正常工作
|
||
SPI2_SPI_CLK_ENABLE();
|
||
|
||
// 指定SPI实例为SPI2
|
||
SPI2_Handle.Instance = SPI2;
|
||
// 设置SPI工作模式为主机模式,由该SPI设备控制通信过程
|
||
SPI2_Handle.Init.Mode = SPI_MODE_MASTER;
|
||
// 设置SPI通信方向为单线模式,即只使用一条数据线进行通信
|
||
SPI2_Handle.Init.Direction = SPI_DIRECTION_1LINE;
|
||
// 设置SPI数据大小为8位,每次传输8位数据
|
||
SPI2_Handle.Init.DataSize = SPI_DATASIZE_8BIT;
|
||
// 设置SPI时钟极性为高电平,即空闲状态下时钟信号为高电平
|
||
SPI2_Handle.Init.CLKPolarity = SPI_POLARITY_HIGH;
|
||
// 设置SPI时钟相位为第二个边沿采样数据
|
||
SPI2_Handle.Init.CLKPhase = SPI_PHASE_2EDGE; /*上升沿采样数据*/
|
||
// 设置SPI片选信号为软件控制
|
||
SPI2_Handle.Init.NSS = SPI_NSS_SOFT;
|
||
// 设置SPI波特率预分频器为256,降低SPI通信速率
|
||
SPI2_Handle.Init.BaudRatePrescaler = SPI_BAUDRATEPRESCALER_256;
|
||
// 设置SPI数据传输的起始位为高位在前
|
||
SPI2_Handle.Init.FirstBit = SPI_FIRSTBIT_MSB;
|
||
// 禁用SPI的TI模式
|
||
SPI2_Handle.Init.TIMode = SPI_TIMODE_DISABLE;
|
||
// 禁用SPI的CRC计算功能
|
||
SPI2_Handle.Init.CRCCalculation = SPI_CRCCALCULATION_DISABLE;
|
||
// 设置CRC多项式为7(虽然已禁用CRC计算,但仍需设置该参数)
|
||
SPI2_Handle.Init.CRCPolynomial = 7;
|
||
|
||
// 根据上述配置初始化SPI2外设
|
||
HAL_SPI_Init(&SPI2_Handle);
|
||
// 使能SPI2外设,开始工作
|
||
__HAL_SPI_ENABLE(&SPI2_Handle);
|
||
}
|
||
|
||
/**
|
||
* @brief SPI外设的底层硬件初始化函数
|
||
*
|
||
* 该函数用于初始化SPI外设所使用的GPIO引脚,包括时钟线、数据线等。
|
||
*
|
||
* @param hspi 指向SPI句柄结构体的指针,用于指定要初始化的SPI实例
|
||
*/
|
||
void HAL_SPI_MspInit(SPI_HandleTypeDef *hspi)
|
||
{
|
||
// 定义一个GPIO初始化结构体变量,用于配置GPIO引脚的参数
|
||
GPIO_InitTypeDef gpio_init_struct;
|
||
// 判断要初始化的SPI实例是否为SPI2
|
||
if (hspi->Instance == SPI2)
|
||
{
|
||
// 使能SPI2的MOSI引脚的时钟
|
||
SPI_MOSI_GPIO_CLK_ENABLE();
|
||
// 注释掉的代码,原本可能用于使能SPI2的MISO引脚的时钟
|
||
// SPI_MISO_GPIO_CLK_ENABLE();
|
||
// 使能SPI2的时钟线引脚的时钟
|
||
SPI_CLK_GPIO_CLK_ENABLE();
|
||
|
||
// 配置SPI2的时钟线引脚
|
||
gpio_init_struct.Pin = SPI_CLK_GPIO_PIN;
|
||
// 设置引脚模式为复用推挽输出,用于连接SPI时钟信号
|
||
gpio_init_struct.Mode = GPIO_MODE_AF_PP;
|
||
// 设置引脚速度为高频
|
||
gpio_init_struct.Speed = GPIO_SPEED_FREQ_HIGH;
|
||
// 根据上述配置初始化SPI2的时钟线引脚
|
||
HAL_GPIO_Init(SPI_CLK_GPIO_PORT, &gpio_init_struct);
|
||
|
||
// 注释掉的代码,原本可能用于配置SPI2的MISO引脚
|
||
// MISO引脚模式设置(复用输出)
|
||
// gpio_init_struct.Pin = SPI_MISO_GPIO_PIN;
|
||
// HAL_GPIO_Init(SPI_MISO_GPIO_PORT, &gpio_init_struct);
|
||
|
||
// 配置SPI2的MOSI引脚
|
||
gpio_init_struct.Pin = SPI_MOSI_GPIO_PIN;
|
||
// 设置引脚模式为复用推挽输出,用于连接SPI数据输出信号
|
||
gpio_init_struct.Mode = GPIO_MODE_AF_PP;
|
||
// 设置引脚速度为高频
|
||
gpio_init_struct.Speed = GPIO_SPEED_FREQ_HIGH;
|
||
// 根据上述配置初始化SPI2的MOSI引脚
|
||
HAL_GPIO_Init(SPI_MOSI_GPIO_PORT, &gpio_init_struct);
|
||
}
|
||
}
|
||
|
||
|
||
/**
|
||
* @简介 SPI 写入一个字节8位
|
||
* @输入参数 pData:要写入的数据指针
|
||
* @输入参数 Size:要写入的数据大小
|
||
* @输入参数 dir:1 数据从后面往前发送 0:数据从前往后发送
|
||
* @注意事项 此函数是高位优先写入
|
||
* @返回值 无
|
||
*/
|
||
void Hard_SPI_Transmit(SPI_HandleTypeDef *hspi, uint8_t *pData, uint16_t Size, uint8_t dir)
|
||
{
|
||
uint16_t i = 0;
|
||
|
||
dir ? pData += (Size-1): pData;
|
||
|
||
for(i =0; i < Size; i++)
|
||
{
|
||
HAL_SPI_Transmit(hspi, pData, 1, 1000);
|
||
dir ? pData--: pData++;
|
||
}
|
||
}
|
||
|
||
|