问题描述
我正在使用W25Q16开发STM32L432KC。每当我得到0xFF
时。根据数据表,要获取芯片制造商代码,我需要发送0x90
和3个虚拟字节。芯片应返回0xEF
,0x17
。但是,由于某种原因,我收到了2个字节的0xFF
。
SPI_HandleTypeDef hspi1;
static void MX_SPI1_Init(void)
{
hspi1.Instance = SPI1;
hspi1.Init.Mode = SPI_MODE_MASTER;
hspi1.Init.Direction = SPI_DIRECTION_2LInes;
hspi1.Init.DataSize = SPI_DATASIZE_8BIT;
hspi1.Init.CLKPolarity = SPI_POLARITY_LOW;
hspi1.Init.CLKPhase = SPI_PHASE_1EDGE;
hspi1.Init.NSS = SPI_NSS_SOFT;
hspi1.Init.BaudratePrescaler = SPI_BAUdratEPRESCALER_32;
hspi1.Init.FirstBit = SPI_FirsTBIT_MSB;
hspi1.Init.TIMode = SPI_TIMODE_disABLE;
hspi1.Init.CRCCalculation = SPI_CRCCALculaTION_disABLE;
hspi1.Init.CRCpolynomial = 10;
hspi1.Init.CRCLength = SPI_CRC_LENGTH_DATASIZE;
hspi1.Init.NsspMode = SPI_NSS_pulse_ENABLE;
if (HAL_SPI_Init(&hspi1) != HAL_OK)
{
_Error_Handler(__FILE__,__LINE__);
}
}
#define SPI_SEL2_Pin GPIO_PIN_15
#define SPI_SEL2_GPIO_Port GPIOA
#define ChipSelect() HAL_GPIO_WritePin(SPI_SEL2_GPIO_Port,SPI_SEL2_Pin,GPIO_PIN_RESET)
#define Chipdeselect() HAL_GPIO_WritePin(SPI_SEL2_GPIO_Port,GPIO_PIN_SET)
#define COMMAND_IDENTIFICATION 0x90
uint8_t buffer_tx[4];
uint8_t buffer_rx[2];
uint16_t GetIdentification()
{
buffer_tx[0] = COMMAND_IDENTIFICATION;
buffer_tx[1] = 0x0;
buffer_tx[2] = 0x0;
buffer_tx[3] = 0x0;
ChipSelect();
HAL_SPI_Transmit(&hspi1,buffer_tx,4,1000); // send 0x90,0x0,0x0
HAL_SPI_Receive(&hspi1,buffer_rx,2,1000); // receive 0xFF,0xFF
Chipdeselect();
return ((uint8_t)buffer_rx[0] << 8) | (uint8_t)buffer_rx[1];
}
int main(void)
{
MX_SPI1_Init();
HAL_Delay(1000);
uint16_t id = GetIdentification();
printf("Manufacturer ID: 0x%.4X\r\n",id);
while
{
}
}
我该如何解决?
解决方法
如果不访问硬件设置,很难知道出了什么问题。似乎您处在正确的轨道上,只是挂在那儿,检查所有可能的错误源,然后将问题缩小到简单且可测试的范围。并首先找出是否可能是硬件问题,然后尝试修复软件。
一些提示可能会有所帮助:
-
SPI数据线上的
-
0xFF(在这种情况下为DO / MISO)可能意味着其始终处于HIGH状态。因此,闪存IC无法回答您的命令。在驱动程序上工作时,可以使用逻辑分析仪,示波器或类似工具进行验证。如果可能,还检查期望的数据位在MOSI引脚上是否可见。 CLK可见吗?
-
用万用表检查所有引脚电压。测量值看起来是否像预期的那样? (/保持引脚?)
-
用于芯片选择(PA15)的GPIO引脚是否配置为输出引脚? (也可以在STM32CubeMX配置器软件中完成。)
/* configure /CS pin to be an output pin */
GPIO_InitTypeDef init = {0};
init.Pin = SPI_SEL2_Pin;
init.Speed = GPIO_SPEED_FREQ_LOW;
init.Mode = GPIO_MODE_OUTPUT_PP;
init.Pull = GPIO_NOPULL;
if (HAL_GPIO_Init(SPI_SEL2_GPIO_Port,&init) != HAL_OK) {
Error_Handler();
}
ChipDeselect();
HAL_Delay(100);
-
SPI外设是否正确配置? c检查SPI时钟极性和相位(即hspi1.Init.CLKPolarity和hspi1.Init.CLKPhase),位顺序(即hspi1.Init.FirstBit(MSB / LSB在前))和时钟速度(即hspi1.Init。 BaudRatePrescaler(从低处开始,然后根据需要将其增加)。检查datasheet,有时反复试验也可以解决问题。
-
闪存芯片可能处于掉电状态。 -快速浏览数据表后,似乎还有
0x90
的另一条命令,指令号为0xAB
,称为“释放掉电/设备ID”(请参见9.3节)。数据表中的7)。 -请注意,此指令仅在以“简写形式”使用时,会唤醒芯片,并引用数据表:
要使器件从掉电状态释放,可通过将/ CS引脚驱动为低电平,将指令代码“ ABh”移位并将/ CS驱动为高电平来发出指令,如图38a所示。
-
搜索网络,然后尝试查找类似的驱动程序作为参考。这些Winbond闪存IC非常普遍。一旦确定了如何从闪存中读取制造商ID,就可以按照数据表中的说明逐步将其他命令/功能集成到驱动程序中。 (还请阅读状态和配置寄存器(第7章)中的部分,稍后可能需要在写操作期间对状态位进行轮询。)将这些内容拆分为一些小功能,可以单独进行测试... >
-
HAL_SPI_TransmitReceive()一次就能发送和接收数据。