dataHandling.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382
  1. #include "main.h"
  2. #include "dataHandling.h"
  3. #include "md5c.h"
  4. void Flash_ReadBytes(uint16_t* sorBuf,uint32_t FlashAddr,uint16_t len)
  5. {
  6. uint16_t* p = sorBuf;
  7. uint8_t i = 0,j = 0;
  8. uint32_t addr = FlashAddr;
  9. while(len--)
  10. {
  11. i = *(uint32_t*)addr++;
  12. j = *(uint32_t*)addr++;
  13. *p++ = j<<8|i;
  14. }
  15. }
  16. uint16_t iapbuf[1024];
  17. uint16_t iapversion[5];
  18. uint16_t iapmd5[10];
  19. // 定义一个长度为16的字符数组,用于存储MD5摘要
  20. unsigned char digest[16];
  21. int App2_MD5_Check(uint32_t addr,unsigned int all_len)
  22. {
  23. // 定义变量i和update_len
  24. unsigned int i,update_len;
  25. // 定义一个MD5上下文结构体
  26. MD5_CTX md5c;
  27. // 减去摘要的长度,得到实际数据长度
  28. // all_len -= 16;
  29. // 初始化MD5上下文结构体
  30. MD5Init(&md5c);
  31. // 循环处理数据
  32. for(i=0;i<all_len;)
  33. {
  34. // 判断剩余数据长度是否大于512,如果是则取512,否则取剩余长度
  35. if(all_len>(i+512))
  36. update_len = 512;
  37. else
  38. update_len = all_len-i;
  39. // 将数据从指定地址拷贝到iapbuf缓冲区
  40. memcpy(iapbuf,(const void *)(addr+i),update_len);
  41. // 更新MD5上下文
  42. MD5Update (&md5c, (unsigned char *)iapbuf, update_len);
  43. // 更新i的值
  44. i += update_len;
  45. }
  46. // 计算最后的MD5摘要
  47. MD5Final(digest,&md5c);
  48. // 验证摘要是否与给定地址的数据一致
  49. for(i=0;i<16;++i)
  50. {
  51. // if(digest[i]!=*(unsigned char *)(addr+all_len+i))
  52. if(digest[i]!=*(unsigned char *)(MD5_ADDR+i))
  53. break;
  54. }
  55. // 如果验证通过,则返回1,否则返回0
  56. if(i>=16)
  57. return 1;
  58. else
  59. return 0;
  60. }
  61. void delay_sys_us(uint32_t Delay)//1个delay,大概1.5us
  62. {
  63. uint32_t cnt = Delay * 8;
  64. uint32_t i = 0;
  65. for(i = 0; i < cnt; i++)__NOP();
  66. }
  67. uint16_t STMFLASH_ReadHalfWord(uint32_t faddr)
  68. {
  69. return *(uint16_t*)faddr;
  70. }
  71. void STMFLASH_Read(uint32_t ReadAddr,uint16_t *pBuffer,uint16_t NumToRead)
  72. {
  73. uint16_t i;
  74. for(i=0;i<NumToRead;i++)
  75. {
  76. pBuffer[i]=STMFLASH_ReadHalfWord(ReadAddr);//读取2个字节.
  77. ReadAddr+=2;//偏移2个字节.
  78. }
  79. }
  80. void STMFLASH_Write_NoCheck(uint32_t WriteAddr,uint16_t *pBuffer,uint16_t NumToWrite)
  81. {
  82. uint16_t i;
  83. for(i=0;i<NumToWrite;i++)
  84. {
  85. HAL_FLASH_Program(FLASH_TYPEPROGRAM_HALFWORD,WriteAddr,pBuffer[i]);
  86. WriteAddr+=2;//地址增加2.
  87. }
  88. }
  89. // 定义Flash缓冲区
  90. uint16_t Flashbuf[2048] __attribute__((at(0X20001000)));
  91. // Flash写入函数
  92. void Flash_WriteBytes(uint16_t* sorBuf, uint32_t FlashAddr, uint16_t len)
  93. {
  94. uint32_t Offset_ADDR = 0, Page_StartAddr = 0, i = 0;
  95. // 计算偏移地址和起始地址
  96. Offset_ADDR = FlashAddr % FLASH_PAGE_SIZE;
  97. Page_StartAddr = FlashAddr - Offset_ADDR;
  98. // 设置PageError
  99. uint32_t PageError = 0;
  100. // 初始化擦除参数
  101. FLASH_EraseInitTypeDef f;
  102. f.TypeErase = FLASH_TYPEERASE_PAGES;
  103. f.PageAddress = Page_StartAddr;
  104. f.NbPages = 1;
  105. // 读取Flash缓冲区数据
  106. Flash_ReadBytes(Flashbuf, Page_StartAddr, 0x400);//写一半先1024字节
  107. // 将源缓冲区数据写入Flash缓冲区
  108. for (i = 0; i < len; i++)
  109. Flashbuf[Offset_ADDR / 2 + i] = sorBuf[i]; //后一半也存入缓存中
  110. // 解锁FLASH
  111. HAL_FLASH_Unlock();
  112. // 擦除FLASH
  113. HAL_FLASHEx_Erase(&f, &PageError);
  114. // 写入Flash缓冲区数据到FLASH
  115. for (uint16_t i = 0; i < 0x400; i++)
  116. {
  117. HAL_FLASH_Program(FLASH_TYPEPROGRAM_HALFWORD, Page_StartAddr + (i * 2), Flashbuf[i]);
  118. }
  119. // 锁住FLASH
  120. HAL_FLASH_Lock();
  121. }
  122. void Set_App2_Flag(void)
  123. {
  124. uint16_t update_flag1[2];
  125. update_flag = Startup_APP2;
  126. update_flag1[1] = (uint16_t)update_flag;
  127. update_flag1[0] = (uint16_t)(update_flag >> 16);
  128. Flash_WriteBytes(update_flag1, StartMode_Addr, 2);
  129. NVIC_SystemReset();
  130. }
  131. void Set_jump_Flag(void)
  132. {
  133. uint16_t update_flag1[2];
  134. update_flag = Jump_app;
  135. update_flag1[1] = (uint16_t)update_flag;
  136. update_flag1[0] = (uint16_t)(update_flag >> 16);
  137. Flash_WriteBytes(update_flag1, StartMode_Addr, 2);
  138. }
  139. void Set_normal_Flag(void)
  140. {
  141. uint16_t update_flag1[2];
  142. update_flag = Startup_Normal;
  143. update_flag1[1] = (uint16_t)update_flag;
  144. update_flag1[0] = (uint16_t)(update_flag >> 16);
  145. Flash_WriteBytes(update_flag1, StartMode_Addr, 2);
  146. }
  147. void convertHexToAscii(uint16_t version[4], uint32_t* System_version) {
  148. *System_version = (((((version[0] >> 8 & 0xF) << 4) | (version[0] & 0xF)) & 0xFF) << 24) |
  149. (((((version[1] >> 8 & 0xF) << 4) | (version[1] & 0xF)) & 0xFF) << 16) |
  150. (((((version[2] >> 8 & 0xF) << 4) | (version[2] & 0xF)) & 0xFF) << 8) |
  151. ((((version[3] >> 8 & 0xF) << 4) | (version[3] & 0xF)) & 0xFF);
  152. }
  153. /*
  154. uint8_t hexData[] = {0x34, 0x34, 0x30, 0x30, 0x00};
  155. size_t length = sizeof(hexData) / sizeof(hexData[0]);
  156. uint16_t intValue = convertHexToInt(hexData, length);
  157. */
  158. void convertToUint16(uint8_t* uint8Data, size_t length, uint16_t* uint16Data)
  159. {
  160. for (size_t i = 0; i < length; i += 2)
  161. {
  162. uint16Data[i / 2] = (uint16_t)uint8Data[i] << 8 | uint8Data[i + 1];
  163. }
  164. }
  165. void search_seat(const uint8_t* hexData)
  166. {
  167. int i, j, k;
  168. for (i = 0; i < sizeof(hexData); i++) {
  169. if (hexData[i] == 0x00) { // 找到第一个占位符
  170. break; // 结束循环
  171. }
  172. binName[i] = hexData[i]; // 将第一个占位符之前的数据放入binName中
  173. }
  174. for (j = i + 1; j < sizeof(hexData); j++) {
  175. if (hexData[j] == 0x00) { // 找到第二个占位符
  176. break; // 结束循环
  177. }
  178. binLength[j - i - 1] = hexData[j]; // 将第一个占位符至第二个占位符之间的数据放入binLength中
  179. }
  180. for (k = j + 1; k < sizeof(hexData); k++) {
  181. if (hexData[k] == 0x00) { // 找到第三个占位符
  182. break; // 结束循环
  183. }
  184. binMd5[k - j - 1] = hexData[k]; // 将第三个占位符至第四个占位符之间的数据放入binMd5中
  185. }
  186. }
  187. uint16_t crc16_xmodem(uint8_t *data, uint16_t length) {
  188. uint16_t crc = 0;
  189. for (uint16_t i = 0; i < length; i++) {
  190. crc ^= (uint16_t)data[i] << 8;
  191. for (uint8_t j = 0; j < 8; j++) {
  192. if (crc & 0x8000) {
  193. crc = (crc << 1) ^ 0x1021;
  194. } else {
  195. crc <<= 1;
  196. }
  197. }
  198. }
  199. return crc;
  200. }
  201. void Set_Update_Down(void)
  202. {
  203. // uint16_t update_flag1[2];
  204. // update_flag = Startup_Update;
  205. // update_flag1[0] = (uint16_t)update_flag;
  206. // update_flag1[1] = (uint16_t)(update_flag >> 16);
  207. // Flash_WriteBytes(update_flag1, StartMode_Addr, 2);
  208. uint16_t update_flag1[2];
  209. update_flag = Startup_Update;
  210. update_flag1[0] = (uint16_t)update_flag;
  211. update_flag1[1] = update_flag & 0xFFFF;
  212. memcpy(&update_flag1, &update_flag, sizeof(uint16_t) * 2);
  213. Flash_WriteBytes(update_flag1, StartMode_Addr, sizeof(update_flag1));
  214. int txLen = sprintf((char *)USART_IAP_RX, "reboot\r\n");
  215. HAL_UART_Transmit_IT(&huart1,USART_IAP_RX,txLen);
  216. while (huart1.gState == HAL_UART_STATE_BUSY_TX)
  217. {
  218. HAL_Delay(1);
  219. }
  220. __NOP();__NOP();
  221. // __disable_irq();
  222. NVIC_SystemReset();
  223. }
  224. void readFlashToArr(uint8_t *arr, uint32_t addr) {
  225. if(lenindex!=0)
  226. {
  227. for (int i = 0; i < 5; i++) {
  228. arr[i] = *(uint8_t*)(addr + i);
  229. }
  230. }
  231. else
  232. {
  233. for (int i = 0; i < 4; i++) {
  234. arr[i] = *(uint8_t*)(addr + i);
  235. }
  236. }
  237. }
  238. #if STM32_FLASH_SIZE<256
  239. #define STM_SECTOR_SIZE 1024 //字节
  240. #else
  241. #define STM_SECTOR_SIZE 2048
  242. #endif
  243. uint16_t STMFLASH_BUF[STM_SECTOR_SIZE/2];//最多是2K字节
  244. void STMFLASH_Write(uint32_t WriteAddr,uint16_t *pBuffer,uint16_t NumToWrite)
  245. {
  246. uint32_t secpos; //扇区地址
  247. uint16_t secoff; //扇区内偏移地址(16位字计算)
  248. uint16_t secremain; //扇区内剩余地址(16位字计算)
  249. uint16_t i;
  250. uint32_t offaddr; //去掉0X08000000后的地址
  251. if(WriteAddr<STM32_FLASH_BASE||(WriteAddr>=(STM32_FLASH_BASE+1024*STM32_FLASH_SIZE)))return;//非法地址
  252. HAL_FLASH_Unlock(); //解锁
  253. offaddr=WriteAddr-STM32_FLASH_BASE; //实际偏移地址.
  254. secpos=offaddr/STM_SECTOR_SIZE; //扇区地址 0~127 for STM32F103RBT6
  255. secoff=(offaddr%STM_SECTOR_SIZE)/2; //在扇区内的偏移(2个字节为基本单位.)
  256. secremain=STM_SECTOR_SIZE/2-secoff; //扇区剩余空间大小
  257. if(NumToWrite<=secremain)secremain=NumToWrite;//不大于该扇区范围
  258. while(1)
  259. {
  260. STMFLASH_Read(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE,STMFLASH_BUF,STM_SECTOR_SIZE/2);//读出整个扇区的内容
  261. for(i=0;i<secremain;i++) //校验数据
  262. {
  263. if(STMFLASH_BUF[secoff+i]!=0XFFFF)break;//需要擦除
  264. }
  265. if(i<secremain) //需要擦除
  266. {
  267. FLASH_PageErase(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE); //擦除这个扇区
  268. FLASH_WaitForLastOperation(FLASH_WAITETIME); //等待上次操作完成
  269. CLEAR_BIT(FLASH->CR, FLASH_CR_PER); //清除CR寄存器的PER位,此操作应该在FLASH_PageErase()中完成!
  270. //但是HAL库里面并没有做,应该是HAL库bug!
  271. for(i=0;i<secremain;i++)//复制
  272. {
  273. STMFLASH_BUF[i+secoff]=pBuffer[i];
  274. }
  275. STMFLASH_Write_NoCheck(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE,STMFLASH_BUF,STM_SECTOR_SIZE/2);//写入整个扇区
  276. }else
  277. {
  278. FLASH_WaitForLastOperation(FLASH_WAITETIME); //等待上次操作完成
  279. STMFLASH_Write_NoCheck(WriteAddr,pBuffer,secremain);//写已经擦除了的,直接写入扇区剩余区间.
  280. }
  281. if(NumToWrite==secremain)break;//写入结束了
  282. else//写入未结束
  283. {
  284. secpos++; //扇区地址增1
  285. secoff=0; //偏移位置为0
  286. pBuffer+=secremain; //指针偏移
  287. WriteAddr+=secremain*2; //写地址偏移(16位数据地址,需要*2)
  288. NumToWrite-=secremain; //字节(16位)数递减
  289. if(NumToWrite>(STM_SECTOR_SIZE/2))secremain=STM_SECTOR_SIZE/2;//下一个扇区还是写不完
  290. else secremain=NumToWrite;//下一个扇区可以写完了
  291. }
  292. };
  293. HAL_FLASH_Lock(); //上锁
  294. }
  295. uint16_t pagesize;
  296. uint32_t addrtowrite;
  297. int z;
  298. void iap_write_appbin(uint32_t appxaddr, uint32_t appbuf, uint32_t appsize)
  299. {
  300. uint16_t size = appsize;
  301. uint32_t fwaddr = appbuf; // 当前写入的地址
  302. addrtowrite = ADD_UPDATE_PROG;
  303. pagesize =(size + 2048) / 2048;
  304. memset(iapbuf, 0, sizeof(iapbuf));
  305. uint8_t ACK_Arr[5] = "ack=";
  306. for( z =1; z<=pagesize ;z++)
  307. {
  308. HAL_GPIO_WritePin(GPIOB, GPIO_PIN_0, GPIO_PIN_SET);
  309. HAL_Delay(1);
  310. HAL_GPIO_WritePin(GPIOB, GPIO_PIN_0, GPIO_PIN_RESET);
  311. Flash_ReadBytes(&iapbuf[0], fwaddr, 2048);//2048
  312. Flash_WriteBytes(&iapbuf[0],addrtowrite,1024);
  313. delay_sys_us(80);
  314. memset(iapbuf, 0, sizeof(iapbuf));
  315. int txLen;
  316. txLen = sprintf((char *)USART_IAP_RX, "%s%d\r\n", ACK_Arr,z);
  317. HAL_UART_Transmit_IT(&huart1,USART_IAP_RX,txLen);
  318. fwaddr = fwaddr + 0x800;
  319. addrtowrite = addrtowrite + 0x800;
  320. }
  321. }