目前需要对一颗SPIFLASH(型号固定)进行离线烧录,因为对上位机还不是太熟练,所以还是采用串口命令控制的方式进行设计
整体思路构思为
上位机串口——》AT32的串口——》数据缓存区——》AT32的SPI口——》SPIFLASH
数据缓存区的数据是提前烧录进AT32的,修改需要人工
现在想到两个思路
第一种为串口通过Y-MODEM协议接收数据
上电后从串口获得数据缓存在单片机内部
需要烧录时再按最开始的离线方式进行
优点:方便一根线就搞定
缺点:受制于单片的内部存储空间
第二种为添加USB从U盘获取数据
通过串口命令从U盘烧录指定文件
优点:文件烧录灵活
缺点:需要U盘配合
现在完成第一步离线烧录方式
主要分成三步
第一步 外设初始化
串口、SPI外设、IO
/* config usart1 nvic interrupt */
nvic_priority_group_config(NVIC_PRIORITY_GROUP_4);
nvic_irq_enable(USART1_IRQn, 0 ,0);
/* configure usart1 param */
usart_init(USART1, 115200, USART_DATA_8BITS, USART_STOP_1_BIT);
usart_receiver_enable(USART1, TRUE);
usart_interrupt_enable(USART1, USART_RDBF_INT, TRUE);
usart_wakeup_id_set(USART1, 0x01);
usart_wakeup_mode_set(USART1, USART_WAKEUP_BY_MATCHING_ID);
usart_receiver_mute_enable(USART1, TRUE);
usart_enable(USART1, TRUE);
FLASH_CS_HIGH();
crm_periph_clock_enable(CRM_SPI2_PERIPH_CLOCK, TRUE);
spi_default_para_init(&spi_init_struct);
spi_init_struct.transmission_mode = SPI_TRANSMIT_FULL_DUPLEX;
spi_init_struct.master_slave_mode = SPI_MODE_MASTER;
spi_init_struct.mclk_freq_division = SPI_MCLK_DIV_64;//SPI_MCLK_DIV_8;
spi_init_struct.first_bit_transmission = SPI_FIRST_BIT_MSB;
spi_init_struct.frame_bit_num = SPI_FRAME_8BIT;
spi_init_struct.clock_polarity = SPI_CLOCK_POLARITY_HIGH;
spi_init_struct.clock_phase = SPI_CLOCK_PHASE_2EDGE;
spi_init_struct.cs_mode_selection = SPI_CS_SOFTWARE_MODE;
spi_init(SPI2, &spi_init_struct);
spi_enable(SPI2, TRUE);
void at32_board_init()
{
/* initialize delay function */
delay_init();
/* configure led in at_start_board */
at32_led_init(LED2);
at32_led_init(LED3);
at32_led_init(LED4);
at32_led_off(LED2);
at32_led_off(LED3);
at32_led_off(LED4);
/* configure button in at_start board */
at32_button_init();
}
第二部 ,针对SPIFLASH编写驱动
#define SPIF_BLOCKERASE 0xD8
#define SPIF_SECTORERASE 0x20
#define SPIF_CHIPERASE 0xC7
#define SPIF_POWERDOWN 0xB9
#define SPIF_RELEASEPOWERDOWN 0xAB
#define SPIF_DEVICEID 0xAB
#define SPIF_MANUFACTDEVICEID 0x90
#define SPIF_JEDECDEVICEID 0x9F
#define FLASH_SPI_DUMMY_BYTE 0xA5
写操作
uint8_t spi_byte_write(uint8_t data)
{
uint8_t brxbuff;
spi_i2s_dma_transmitter_enable(SPI2, FALSE);
spi_i2s_dma_receiver_enable(SPI2, FALSE);
spi_i2s_data_transmit(SPI2, data);
while(spi_i2s_flag_get(SPI2, SPI_I2S_RDBF_FLAG) == RESET);
brxbuff = spi_i2s_data_receive(SPI2);
while(spi_i2s_flag_get(SPI2, SPI_I2S_BF_FLAG) != RESET);
return brxbuff;
}
读操作
uint8_t spi_byte_read(void)
{
return (spi_byte_write(FLASH_SPI_DUMMY_BYTE));
}
第三步,应用代码,根据串口命令执行不同的操作
if((usart1_rx_buffer[0] == MATCH_ID_VAL) && (usart1_rx_buffer[1] == DATA2_VAL) && \
(usart1_rx_buffer[2] == MATCH_ID_VAL) && (usart1_rx_buffer[3] == DATA4_VAL))
{
usart1_rx_buffer[3]=0;
usart_receiver_mute_enable(USART1, TRUE);
at32_led_off(LED2);
at32_led_off(LED3);
at32_led_off(LED4);
/* read data */
for(i=0;i<32;i++)
{
spi_bytes_read(i*BUF_SIZE,rx_buffer, BUF_SIZE);
/* printf read data */
// printf("Read Data: ");
for(index = 0; index < BUF_SIZE; index++)
{
printf("0x%02x,", rx_buffer[index]);
}
for(index = 0; index < BUF_SIZE; index++)
{
rx_buffer[index]=0;
}
}
}
基本操作就完成了
|