通過(guò)對(duì)stm32內(nèi)部的flash的讀寫可以實(shí)現(xiàn)對(duì)stm32的編程操作。
stm32 的內(nèi)置可編程Flash在許多場(chǎng)合具有十分重要的意義。如其支持ICP特性使得開(kāi)發(fā)人員對(duì)stm32可以警醒調(diào)試開(kāi)發(fā),可以通過(guò)JTAG和SWD接口對(duì)stm32進(jìn)行程序燒寫;支持IAP特性使得開(kāi)發(fā)人員可以在stm32運(yùn)行程序的時(shí)候?qū)ζ鋬?nèi)部程序進(jìn)行更新操作。對(duì)一些對(duì)數(shù)據(jù)安全有要求的場(chǎng)合,可編程FLASH可以結(jié)合stm32內(nèi)部唯一的身份標(biāo)識(shí)實(shí)現(xiàn)各種各樣的防破解方案。并且stm32的FLASH在一些輕量級(jí)的防掉電存儲(chǔ)方案中也有立足之地。
stm32的FLASH分為主存儲(chǔ)塊和信息塊。主存儲(chǔ)塊用于保存具體的程序代碼和用戶數(shù)據(jù),信息塊用于負(fù)責(zé)由stm32出廠是放置2KB的啟動(dòng)程序(Bootloader)和512B的用戶配置信息區(qū)。
主存儲(chǔ)塊是以頁(yè)為單位劃分的,一頁(yè)大小為1KB。范圍為從地址0x08000000開(kāi)始的128KB內(nèi)。
對(duì)Flash 的寫入操作要 “先擦除后寫入”的原則;
stm32的內(nèi)置flash 編程操作都是以頁(yè)為單位寫入的,而寫入的操作必須要以16位半字寬度數(shù)據(jù)位單位,允許跨頁(yè)寫,寫入非16位數(shù)據(jù)時(shí)將導(dǎo)致stm32內(nèi)部總線錯(cuò)誤。
進(jìn)行內(nèi)置flash讀寫時(shí),必須要打開(kāi)內(nèi)部Rc振蕩器。
main.c:
001 | #include "stm32f10x.h" |
006 | void RCC_Configuration(void); |
007 | void GPIO_Configuration(void); |
008 | void USART_Configuration(void); |
012 | u16 data[5]={0x0001,0x0002,0x0003,0x0004,0x0005}; |
018 | USART_Configuration(); |
024 | FLASH_ClearFlag(FLASH_FLAG_EOP|FLASH_FLAG_PGERR|FLASH_FLAG_WRPRTERR); |
026 | FLASH_ErasePage(0x8002000); |
030 | FLASH_ProgramHalfWord((0x8002000 +count*2),data[count]); //flash 為一個(gè)字節(jié)存儲(chǔ),16位數(shù)據(jù)必須地址加2 |
040 | printf("The Five Data Is : "); |
045 | printf("%d ",*(u8 *)(0x8002000 + count*2)); //讀取方法 |
056 | void GPIO_Configuration(void) |
058 | GPIO_InitTypeDef GPIO_InitStructure; |
060 | GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; |
062 | GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; |
063 | GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; |
064 | GPIO_Init(GPIOA , &GPIO_InitStructure); |
066 | GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; |
067 | GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; |
068 | GPIO_Init(GPIOA , &GPIO_InitStructure); |
071 | void RCC_Configuration(void) |
074 | ErrorStatus HSEStartUpStatus; |
079 | RCC_HSEConfig(RCC_HSE_ON); |
081 | HSEStartUpStatus = RCC_WaitForHSEStartUp(); |
083 | if(HSEStartUpStatus == SUCCESS) |
086 | RCC_HCLKConfig(RCC_SYSCLK_Div1); |
088 | RCC_PCLK2Config(RCC_HCLK_Div1); |
090 | RCC_PCLK1Config(RCC_HCLK_Div2); |
092 | FLASH_SetLatency(FLASH_Latency_2); |
094 | FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable); |
096 | RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9); |
100 | while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET); |
102 | RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK); |
104 | while(RCC_GetSYSCLKSource() != 0x08); |
107 | RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_USART1, ENABLE); |
109 | //RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); |
111 | //RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR,ENABLE); |
112 | //RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR|RCC_APB1Periph_BKP|RCC_APB1Periph_WWDG, ENABLE); |
117 | void USART_Configuration(void) |
119 | USART_InitTypeDef USART_InitStructure; |
120 | USART_ClockInitTypeDef USART_ClockInitStructure; |
122 | USART_ClockInitStructure.USART_Clock = USART_Clock_Disable; |
123 | USART_ClockInitStructure.USART_CPOL = USART_CPOL_Low; |
124 | USART_ClockInitStructure.USART_CPHA = USART_CPHA_2Edge; |
125 | USART_ClockInitStructure.USART_LastBit = USART_LastBit_Disable; |
126 | USART_ClockInit(USART1 , &USART_ClockInitStructure); |
128 | USART_InitStructure.USART_BaudRate = 9600; |
129 | USART_InitStructure.USART_WordLength = USART_WordLength_8b; |
130 | USART_InitStructure.USART_StopBits = USART_StopBits_1; |
131 | USART_InitStructure.USART_Parity = USART_Parity_No; |
132 | USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; |
133 | USART_InitStructure.USART_Mode = USART_Mode_Rx|USART_Mode_Tx; |
134 | USART_Init(USART1,&USART_InitStructure); |
136 | USART_Cmd(USART1,ENABLE); |
141 | int fputc(int ch,FILE *f) |
143 | USART_SendData(USART1,(u8) ch); |
144 | while(USART_GetFlagStatus(USART1,USART_FLAG_TC) == RESET); |
評(píng)論