[APM32F4] APM32F4XX USB OTA升级含源码和上位机升级工具

[复制链接]
 楼主| 18270291392 发表于 2023-9-1 16:15 | 显示全部楼层 |阅读模式
本帖最后由 18270291392 于 2023-9-21 08:50 编辑

#申请原创# 近期在研究USB CDC协议,使用USB Virtual Port Com功能与上位机通讯做了OTA功能。
开发平台:MDK529
开发硬件:APM32F411
首先介绍一下程序执行逻辑:
程序由两部分组成:Boot+APP. Flash由三部分组成:Boot+Updata_Flag+APP.
Boot程序逻辑:Boot程序会在上电的时候读取升级标志位:如果Flash中有已升级的程序则进行跳转APP区,否则就一直跑升级协议。
APP程序逻辑:APP程上电后首先会设置中断向量偏移地址,然后一直跑升级协议,如果有检测到升级指令,则擦除升级标志位,然后重启进入Boot程序。
帖子末尾有源码和升级工具。
升级程序固件适用于APM32F4系列MCU,升级工具适用于使用此协议定义升级的任何mcu。
首先看下手册Flash分布,Flash总共8个扇区。
QQ截图20230901150502.png
接下来进行Flash分区。
QQ截图20230901152151.png
扇区 0 和 扇区 1做Boot区。
QQ截图20230901152329.png
扇区 2做APP跳转判断区。
QQ截图20230901152521.png
扇区 3到扇区 7做APP程序区。
既然分区已经分好,接下来就进行Boot代码编写。使用Virtual Port Com SDK。
微信图片_20230901153222.png
打开MDK工程,进行Flash大小设置,Flash空间占用32k。勾选使用Use Micro LIB。
对main.c文件进行修改
  1. #include "main.h"
  2. #include "usb_device.h"
  3. #include "gpio.h"
  4. #include "Update.h"

  5. void SystemClock_Config(void);
  6. typedef void (*pFunction)(void);
  7. pFunction Jump_To_Application;
  8. uint32_t JumpAddress;

  9. void Jump_To_App(uint32_t address)
  10. {
  11.     if (((*(__IO uint32_t*)address) & 0x2FFE0000) == 0x20000000)
  12.     {
  13.         JumpAddress = *(__IO uint32_t*) (address + 4);
  14.         
  15.                 Jump_To_Application = (pFunction) JumpAddress;
  16.                         
  17.         __set_MSP(*(__IO uint32_t*) address);
  18.                
  19.         Jump_To_Application();
  20.     }
  21. }

  22. void USB_Disconnected(void)
  23. {
  24.         __HAL_RCC_USB_OTG_FS_FORCE_RESET();
  25.         HAL_Delay(200);
  26.         __HAL_RCC_USB_OTG_FS_RELEASE_RESET();
  27.         
  28.         GPIO_InitTypeDef GPIO_Initure;
  29.         __HAL_RCC_GPIOA_CLK_ENABLE();

  30.         GPIO_Initure.Pin = GPIO_PIN_11 | GPIO_PIN_12;
  31.         GPIO_Initure.Mode = GPIO_MODE_OUTPUT_PP;
  32.         GPIO_Initure.Pull = GPIO_PULLDOWN;
  33.         GPIO_Initure.Speed = GPIO_SPEED_FREQ_HIGH;
  34.         HAL_GPIO_Init(GPIOA, &GPIO_Initure);

  35.         HAL_GPIO_WritePin(GPIOA, GPIO_PIN_11, GPIO_PIN_RESET);
  36.         HAL_GPIO_WritePin(GPIOA, GPIO_PIN_12, GPIO_PIN_RESET);
  37.         HAL_Delay(300);
  38. }
  39. int main(void)
  40. {
  41.         uint8_t R_Buff[1] = {0};
  42.         HAL_Init();
  43.         SystemClock_Config();

  44.         Read_Flash_Data(R_Buff, 1, FLASH_Updata_Flag_ADDR);
  45.         
  46.         if(R_Buff[0] != 0x55)
  47.         {
  48.                 USB_Disconnected();
  49.                 MX_GPIO_Init();
  50.                 MX_USB_DEVICE_Init();
  51.                 printf("Boot Code Start\r\n");
  52.                 printf("%s\r\n", Device_Version_Info_Str);
  53.         }
  54.         else
  55.         {

  56.         }
  57.         
  58.         while (1)
  59.         {        
  60.                 if(R_Buff[0] == 0x55)
  61.                 {
  62.                         Jump_To_App(FLASH_APP_ADDR);
  63.                 }
  64.                 else
  65.                 {
  66.                         Data_Handler();
  67.                 }
  68.         }
  69. }

  70. /**
  71.   * [url=home.php?mod=space&uid=247401]@brief[/url] System Clock Configuration
  72.   * @retval None
  73.   */
  74. void SystemClock_Config(void)
  75. {
  76.   RCC_OscInitTypeDef RCC_OscInitStruct = {0};
  77.   RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};

  78.   /** Configure the main internal regulator output voltage
  79.   */
  80.   __HAL_RCC_PWR_CLK_ENABLE();
  81.   __HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE1);

  82.   /** Initializes the RCC Oscillators according to the specified parameters
  83.   * in the RCC_OscInitTypeDef structure.
  84.   */
  85.   RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
  86.   RCC_OscInitStruct.HSEState = RCC_HSE_ON;
  87.   RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
  88.   RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
  89.   RCC_OscInitStruct.PLL.PLLM = 4;
  90.   RCC_OscInitStruct.PLL.PLLN = 96;
  91.   RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV2;
  92.   RCC_OscInitStruct.PLL.PLLQ = 4;
  93.   if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
  94.   {
  95.     Error_Handler();
  96.   }

  97.   /** Initializes the CPU, AHB and APB buses clocks
  98.   */
  99.   RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
  100.                               |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
  101.   RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
  102.   RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
  103.   RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;
  104.   RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;

  105.   if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_3) != HAL_OK)
  106.   {
  107.     Error_Handler();
  108.   }
  109. }

  110. /* USER CODE BEGIN 4 */

  111. /* USER CODE END 4 */

  112. /**
  113.   * [url=home.php?mod=space&uid=247401]@brief[/url]  This function is executed in case of error occurrence.
  114.   * @retval None
  115.   */
  116. void Error_Handler(void)
  117. {
  118.   /* USER CODE BEGIN Error_Handler_Debug */
  119.   /* User can add his own implementation to report the HAL error return state */
  120.   __disable_irq();
  121.   while (1)
  122.   {
  123.   }
  124.   /* USER CODE END Error_Handler_Debug */
  125. }

  126. #ifdef  USE_FULL_ASSERT
  127. /**
  128.   * [url=home.php?mod=space&uid=247401]@brief[/url]  Reports the name of the source file and the source line number
  129.   *         where the assert_param error has occurred.
  130.   * @param  file: pointer to the source file name
  131.   * @param  line: assert_param error line source number
  132.   * @retval None
  133.   */
  134. void assert_failed(uint8_t *file, uint32_t line)
  135. {
  136.   /* USER CODE BEGIN 6 */
  137.   /* User can add his own implementation to report the file name and line number,
  138.      ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
  139.   /* USER CODE END 6 */
  140. }
  141. #endif /* USE_FULL_ASSERT */
增加Update.c文件
  1. #include "Update.h"


  2. int fputc(int ch, FILE *f)
  3. {
  4.         while(CDC_Transmit_FS((uint8_t *)&ch, 1) == USBD_BUSY);
  5.         return ch;
  6. }

  7. void Set_Device_Info(void)
  8. {
  9.         uint32_t deviceserial2;
  10.         deviceserial2 = *(uint32_t *) DEVICE_ID3;
  11.         
  12.         sprintf((char *)&Device_SSID_Info, "%d", deviceserial2);
  13.         strcpy((char *)Device_Name_Info, Device_Name_Info_Str);
  14.         strcpy((char *)Device_Version_Info, Device_Version_Info_Str);
  15. }


  16. void Write_Flash(uint8_t *data, uint16_t DataLen, uint32_t Addr)
  17. {
  18.         uint16_t i = 0;
  19.         uint32_t Data = 0;
  20.         uint32_t temp = 0;
  21.    
  22.         HAL_FLASH_Unlock();
  23.         for(i = 0; i < DataLen; i += 4)
  24.         {
  25.         Data = 0;        
  26.         for(uint8_t j = 0; j < 4; j++)
  27.         {
  28.             temp = data[i + j];
  29.             Data |= temp << 8 * j;
  30.         }
  31.         
  32.                 if (HAL_FLASH_Program(FLASH_TYPEPROGRAM_WORD, FLASH_APP_ADDR + Addr + i, Data) == HAL_OK)
  33.                 {
  34.             
  35.                 }
  36.         }
  37.         HAL_FLASH_Lock();
  38. }


  39. void Write_Updata_Flag_Flash(void)
  40. {
  41.     HAL_FLASH_Unlock();
  42.     HAL_FLASH_Program(FLASH_TYPEPROGRAM_WORD, FLASH_Updata_Flag_ADDR, 0x55555555);
  43.     HAL_FLASH_Lock();
  44. }


  45. void Erase_Updata_Flag_Flash(void)
  46. {
  47.         FLASH_EraseInitTypeDef EraseInitStruct;
  48.         HAL_FLASH_Unlock();
  49.         EraseInitStruct.TypeErase = FLASH_TYPEERASE_SECTORS;
  50.         EraseInitStruct.VoltageRange = FLASH_VOLTAGE_RANGE_3;
  51.         EraseInitStruct.Sector = UPDATA_FLAG_SECTOR;
  52.         EraseInitStruct.NbSectors = UPDATA_FLAG_SECTOR_NUM;
  53.         if (HAL_FLASHEx_Erase(&EraseInitStruct, &PageError) != HAL_OK)
  54.         {

  55.     }
  56.     HAL_FLASH_Lock();
  57. }

  58. void Erase_APP_Flash(void)
  59. {
  60.         FLASH_EraseInitTypeDef EraseInitStruct;
  61.         HAL_FLASH_Unlock();
  62.         EraseInitStruct.TypeErase = FLASH_TYPEERASE_SECTORS;
  63.         EraseInitStruct.VoltageRange = FLASH_VOLTAGE_RANGE_3;
  64.         EraseInitStruct.Sector = UPDATA_APP_SECTOR;
  65.         EraseInitStruct.NbSectors = UPDATA_APP_SECTOR_NUM;
  66.         if (HAL_FLASHEx_Erase(&EraseInitStruct, &PageError) != HAL_OK)
  67.         {

  68.     }
  69.     HAL_FLASH_Lock();
  70. }

  71. void Read_Flash_Data(uint8_t* pBuffer, uint32_t NumToRead, uint32_t ReadAddr)
  72. {
  73.     uint32_t i;
  74.     for (i = 0; i < NumToRead; i++)
  75.     {
  76.         *((uint8_t*)pBuffer + i) = *((uint8_t*)ReadAddr + i);
  77.     }
  78. }
增加Update.h文件
  1. #ifndef __UPDATE_H
  2. #define __UPDATE_H

  3. #include "main.h"
  4. #include <string.h>
  5. #include <stdbool.h>
  6. #include <stdio.h>
  7. #include "usbd_cdc_if.h"

  8. #define ADDR_FLASH_SECTOR_0     ((uint32_t)0x08000000) /* Base [url=home.php?mod=space&uid=72445]@[/url] of Sector 0, 16 Kbytes */
  9. #define ADDR_FLASH_SECTOR_1     ((uint32_t)0x08004000) /* Base [url=home.php?mod=space&uid=72445]@[/url] of Sector 1, 16 Kbytes */
  10. #define ADDR_FLASH_SECTOR_2     ((uint32_t)0x08008000) /* Base [url=home.php?mod=space&uid=72445]@[/url] of Sector 2, 16 Kbytes */
  11. #define ADDR_FLASH_SECTOR_3     ((uint32_t)0x0800C000) /* Base [url=home.php?mod=space&uid=72445]@[/url] of Sector 3, 16 Kbytes */
  12. #define ADDR_FLASH_SECTOR_4     ((uint32_t)0x08010000) /* Base [url=home.php?mod=space&uid=72445]@[/url] of Sector 4, 64 Kbytes */
  13. #define ADDR_FLASH_SECTOR_5     ((uint32_t)0x08020000) /* Base [url=home.php?mod=space&uid=72445]@[/url] of Sector 5, 128 Kbytes */
  14. #define ADDR_FLASH_SECTOR_6     ((uint32_t)0x08040000) /* Base [url=home.php?mod=space&uid=72445]@[/url] of Sector 6, 128 Kbytes */
  15. #define ADDR_FLASH_SECTOR_7     ((uint32_t)0x08060000) /* Base [url=home.php?mod=space&uid=72445]@[/url] of Sector 7, 128 Kbytes */

  16. #define FLASH_Updata_Flag_ADDR        ADDR_FLASH_SECTOR_2
  17. #define FLASH_APP_ADDR        ADDR_FLASH_SECTOR_3

  18. #define UPDATA_FLAG_SECTOR                 FLASH_SECTOR_2
  19. #define UPDATA_FLAG_SECTOR_NUM         1

  20. #define UPDATA_APP_SECTOR                 FLASH_SECTOR_3
  21. #define UPDATA_APP_SECTOR_NUM         3


  22. #define         DEVICE_ID3      (UID_BASE + 0x8)
  23. #define Device_Name_Info_Str         "APM32F411"
  24. #define Device_SSID_Info_Str          "1234"
  25. #define Device_Version_Info_Str "Boot V 1.0"

  26. extern uint8_t Device_Name_Info[12];
  27. extern uint8_t Device_SSID_Info[12];
  28. extern uint8_t Device_Version_Info[12];
  29. extern uint32_t PageError;


  30. void Erase_APP_Flash(void);
  31. void Erase_Updata_Flag_Flash(void);
  32. void Write_Updata_Flag_Flash(void);
  33. void Write_Flash(uint8_t *data, uint16_t DataLen, uint32_t Addr);
  34. void Read_Flash_Data(uint8_t* pBuffer, uint32_t NumToRead, uint32_t ReadAddr);


  35. void ClearRxQueue(void);
  36. void OnDataReceived(uint8_t val);
  37. void Data_Handler(void);

  38. #endif

修改usbd_cdc_if.c文件,修改如下函数即可。
  1. #include "Update.h"

  2. static int8_t CDC_Receive_FS(uint8_t* Buf, uint32_t *Len)
  3. {
  4.         /* USER CODE BEGIN 6 */
  5.         USBD_CDC_SetRxBuffer(&hUsbDeviceFS, &Buf[0]);
  6.         USBD_CDC_ReceivePacket(&hUsbDeviceFS);
  7.         for(int i = 0; i < *Len; i++)
  8.         {
  9.                 OnDataReceived(Buf[i]);
  10.         }
  11.         return (USBD_OK);
  12.         /* USER CODE END 6 */
  13. }
接下来进行APP代码修改。可以复制Boot代码进行修改。对Flash起始地址和大小进行修改。
微信图片_20230901155613.png
修改main.c文件
  1. #include "main.h"
  2. #include "usb_device.h"
  3. #include "gpio.h"
  4. #include "Update.h"

  5. void SystemClock_Config(void);

  6. void USB_Disconnected(void)
  7. {
  8.         __HAL_RCC_USB_OTG_FS_FORCE_RESET();
  9.         HAL_Delay(200);
  10.         __HAL_RCC_USB_OTG_FS_RELEASE_RESET();
  11.         
  12.         GPIO_InitTypeDef GPIO_Initure;
  13.         __HAL_RCC_GPIOA_CLK_ENABLE();

  14.         GPIO_Initure.Pin = GPIO_PIN_11 | GPIO_PIN_12;
  15.         GPIO_Initure.Mode = GPIO_MODE_OUTPUT_PP;
  16.         GPIO_Initure.Pull = GPIO_PULLDOWN;
  17.         GPIO_Initure.Speed = GPIO_SPEED_FREQ_HIGH;
  18.         HAL_GPIO_Init(GPIOA, &GPIO_Initure);

  19.         HAL_GPIO_WritePin(GPIOA, GPIO_PIN_11, GPIO_PIN_RESET);
  20.         HAL_GPIO_WritePin(GPIOA, GPIO_PIN_12, GPIO_PIN_RESET);
  21.         HAL_Delay(300);
  22. }
  23. int main(void)
  24. {
  25.         SCB->VTOR = FLASH_APP_ADDR;
  26.         HAL_Init();
  27.         SystemClock_Config();
  28.         USB_Disconnected();

  29.         MX_GPIO_Init();
  30.         MX_USB_DEVICE_Init();
  31.         printf("APP Code Start\r\n");
  32.         printf("%s\r\n", Device_Version_Info_Str);
  33.         while (1)
  34.         {
  35.                 Data_Handler();
  36.         }
  37. }

  38. /**
  39.   * [url=home.php?mod=space&uid=247401]@brief[/url] System Clock Configuration
  40.   * @retval None
  41.   */
  42. void SystemClock_Config(void)
  43. {
  44.   RCC_OscInitTypeDef RCC_OscInitStruct = {0};
  45.   RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};

  46.   /** Configure the main internal regulator output voltage
  47.   */
  48.   __HAL_RCC_PWR_CLK_ENABLE();
  49.   __HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE1);

  50.   /** Initializes the RCC Oscillators according to the specified parameters
  51.   * in the RCC_OscInitTypeDef structure.
  52.   */
  53.   RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
  54.   RCC_OscInitStruct.HSEState = RCC_HSE_ON;
  55.   RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
  56.   RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
  57.   RCC_OscInitStruct.PLL.PLLM = 4;
  58.   RCC_OscInitStruct.PLL.PLLN = 96;
  59.   RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV2;
  60.   RCC_OscInitStruct.PLL.PLLQ = 4;
  61.   if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
  62.   {
  63.     Error_Handler();
  64.   }

  65.   /** Initializes the CPU, AHB and APB buses clocks
  66.   */
  67.   RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
  68.                               |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
  69.   RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
  70.   RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
  71.   RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;
  72.   RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;

  73.   if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_3) != HAL_OK)
  74.   {
  75.     Error_Handler();
  76.   }
  77. }

  78. /* USER CODE BEGIN 4 */

  79. /* USER CODE END 4 */

  80. /**
  81.   * [url=home.php?mod=space&uid=247401]@brief[/url]  This function is executed in case of error occurrence.
  82.   * @retval None
  83.   */
  84. void Error_Handler(void)
  85. {
  86.   /* USER CODE BEGIN Error_Handler_Debug */
  87.   /* User can add his own implementation to report the HAL error return state */
  88.   __disable_irq();
  89.   while (1)
  90.   {
  91.   }
  92.   /* USER CODE END Error_Handler_Debug */
  93. }

  94. #ifdef  USE_FULL_ASSERT
  95. /**
  96.   * [url=home.php?mod=space&uid=247401]@brief[/url]  Reports the name of the source file and the source line number
  97.   *         where the assert_param error has occurred.
  98.   * @param  file: pointer to the source file name
  99.   * @param  line: assert_param error line source number
  100.   * @retval None
  101.   */
  102. void assert_failed(uint8_t *file, uint32_t line)
  103. {
  104.   /* USER CODE BEGIN 6 */
  105.   /* User can add his own implementation to report the file name and line number,
  106.      ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
  107.   /* USER CODE END 6 */
  108. }
  109. #endif /* USE_FULL_ASSERT */
以上就是OTA升级的全部配置步骤。Update.h有很多宏定义可以修改。
下面介绍一下升级工具的使用。
1.将Boot程序烧录到板子中。
2.APP工程编译出的bin文件可以通过升级工具烧写到板子中。
3.打开升级工具,下图是只有Boot程序的板子。
QQ截图20230901161230.png
4.打开升级文件升级后截图,烧写成功,板子会自动运行APP代码。
微信图片_20230901161349.png
升级过程如下
d1013223d554c6bd6bf8e8aa6609f4a0.gif
喜欢朋友可以下载下面的升级工具和程序固件进行测试。链接不能直接跳转,需要手动复制粘贴网页地址才能跳转。
APP和Boot固件: 升级固件.zip (35.66 KB, 下载次数: 30)
APM32F4XX升级工具:CSDN:链接:https://download.csdn.net/download/dailin2012/88288374
百度网盘:链接:https://pan.baidu.com/s/1iYWV7D9ABpPxeaiuQe3DTw
提取码:mwjs
下面有源码下载。回复帖子可见。
APM32F411Boot源码:
游客,如果您要查看本帖隐藏内容请回复

APM32F411APP源码:
游客,如果您要查看本帖隐藏内容请回复

下载源码后,可在Update.h文件对一些参数的宏定义进行修。如修改设备名称,版本等等信息,Updata_Flag地址和占用空间,APP地址和占用Flash空间等。此协议也可以移植到其他系列的MCU上如APM32F103.只需要将FLASH读写函数修改成APM32F103的接口就可以与上位机通讯进行升级。

评论

感谢  发表于 2024-4-17 09:02
感謝分享  发表于 2023-11-10 11:12
 楼主| 18270291392 发表于 2023-9-2 11:42 | 显示全部楼层
本帖最后由 18270291392 于 2023-9-21 08:50 编辑

测试
D戴琳 发表于 2023-9-2 11:57 | 显示全部楼层
要代码
 楼主| 18270291392 发表于 2023-9-3 20:38 | 显示全部楼层
本帖最后由 18270291392 于 2023-9-21 08:52 编辑

printf("测试");
knk 发表于 2023-9-3 23:49 | 显示全部楼层
CDC没有winusb好用吧
 楼主| 18270291392 发表于 2023-9-4 08:27 | 显示全部楼层
knk 发表于 2023-9-3 23:49
CDC没有winusb好用吧

做个OTA升级,我认为使用CDC Virtual Port Com是最合适,毕竟有现成 Virtual Port Com SDK

评论

knk
[url=home.php?mod=space&uid=2787538]@18270291392[/url] :是个好办法,一直以为COM设备没有VID和PID  发表于 2023-9-6 22:40
[url=home.php?mod=space&uid=2477484]@knk[/url] :你可以把百度网盘链接下载上位机试试,我的上位机会自动打开我的usb设备  发表于 2023-9-5 09:30
[url=home.php?mod=space&uid=2477484]@knk[/url] :我的上位机工具是靠检查uid和pid进行区分的。  发表于 2023-9-5 09:29
knk
如果电脑上有多个串口,怎么找到自己想要的那个?用WinUSB和HID就不会有这个问题  发表于 2023-9-4 22:44
yjwpm 发表于 2023-9-4 11:23 | 显示全部楼层
chenjun89 发表于 2023-9-5 07:39 来自手机 | 显示全部楼层
感谢,下载用用看。
stly 发表于 2023-9-5 22:40 | 显示全部楼层
18270291392 发表于 2023-9-3 20:38
怎么没人问问题啊,都看得懂吗?

这么嚣张
forgot 发表于 2023-9-7 08:06 | 显示全部楼层
非常好的技术分享,源码回复可见,感谢楼主
chenqianqian 发表于 2023-9-7 08:17 来自手机 | 显示全部楼层
感谢分享,正好需要学习。
WinHuiDong 发表于 2023-9-7 20:22 | 显示全部楼层
感谢分享,正好需要学习。
caizhiwei 发表于 2023-9-8 09:53 | 显示全部楼层
用USB host U盘升级,或者用USB Device模拟优盘升级不香吗? 还需要配合一个不通用的上位机;

如果用Virtual Port Com ,那直接使用ymodem协议不香吗?

升级的宗旨是 用户使用方便,售后维护人员都能零成本学习使用,才是好的方案哦

仅从USB 方案讨论,欢迎探讨
cmyldd 发表于 2023-9-12 22:37 | 显示全部楼层
上位机的源码可以开放出来吗? 在学习上位机的
CW32TechSupport 发表于 2023-9-15 14:10 | 显示全部楼层
Thanks
rongwinzip 发表于 2023-9-16 10:02 | 显示全部楼层
回复看看
 楼主| 18270291392 发表于 2023-9-21 08:53 | 显示全部楼层
caizhiwei 发表于 2023-9-8 09:53
用USB host U盘升级,或者用USB Device模拟优盘升级不香吗? 还需要配合一个不通用的上位机;

如果用Virtu ...

ymodem难道不需要上位机吗?
hfx 发表于 2023-9-21 13:08 | 显示全部楼层
学习一下,谢谢分享~~~~~
caizhiwei 发表于 2023-10-2 09:34 | 显示全部楼层
18270291392 发表于 2023-9-21 08:53
ymodem难道不需要上位机吗?

需要,任何一个终端都行,甚至windows自带的都行,Linux平台就更通用了,所以不需要单独开发呢
南河壹号 发表于 2023-10-26 09:18 | 显示全部楼层
谢谢分享
您需要登录后才可以回帖 登录 | 注册

本版积分规则

2

主题

11

帖子

1

粉丝
快速回复 在线客服 返回列表 返回顶部

2

主题

11

帖子

1

粉丝
快速回复 在线客服 返回列表 返回顶部