国民技术Cortex-M0系列单片机IAP升级

考虑到设备部署到现场后有可能需要进行软件升级,之前做过PIC系列单片机的升级,现在想做个国民技术N32G031系列Cortex-M0内核的单片机IAP方案。

因为国民技术系列单片机在很多大程度上都模仿了STM32,所以我想其升级方案极有可能差不多。于是在网上下载了STM32官方使用YMODEM协议实现的IAP,下载地址:STSW-STM32008 - STM32F10xxx in-application programming using the USART (AN2557) - STMicroelectronics,使用野火的STM32开发实测过是没有问题的,于是在它的基础上进行修改,移植到N32G031系列单片机中来,经过一番折腾还是搞定了,现在把相关内容分享下,另外资源可以在以下链接中下载,无需积分。

【免费】国民技术N32G031使用YMODEM协议实现IAP资源-CSDN文库

另外,关于YMODEM协议及secureCRT的使用可以参考以下链接:

stm32 Bootloader设计(YModem协议)-CSDN博客

关于secureCRT的破解方法请参考以下链接:

尝试SecureCRT_securecrt issue date-CSDN博客

主要介绍IAP程序的main.c文件,代码如下:

/*** @file main.c* @author Power* @version V1.0.1** @copyright Copyright (c) 2023, DS.*/
/*MIPS的全称是Million Instructions Per Second,每秒百万指令(西方或者国际上的计量体系中1M(兆)=100万=1000000);Mhz,是指单片机CPU的主频兆赫兹。单条指令执行时间:STM32F10X单片机在主频为72MHz下,C语言程序执行一条指令需要的时间可认为10ns~100ns。国民技术系列N32G031 MCU,以主频48MHz为例,这里估算的C语言执行一条指令的时间约为20ns-200ns
*/#include "main.h"
#include "common.h"
#include "ymodem.h"#define 	COMn                2
#define     RS485_GPIOx_CLK     RCC_APB2_PERIPH_GPIOF   //peripheral adress
#define     RS485_GPIO_PIN      GPIO_PIN_6              //pin address
#define     RS485_GPIOx         GPIOF                   //port address
#define     RS485_L()           GPIO_ResetBits(RS485_GPIOx, RS485_GPIO_PIN)//RECEIVE_ mode
#define     RS485_H()           GPIO_SetBits(RS485_GPIOx, RS485_GPIO_PIN)//TRANSMIT modetypedef enum 
{COM1 = 0,COM2 = 1
} COM_TypeDef; /* Private typedef -----------------------------------------------------------*/
/* Private define ------------------------------------------------------------*/
/* Private macro -------------------------------------------------------------*/
/* Private variables ---------------------------------------------------------*/
extern pFunction Jump_To_Application;
extern uint32_t JumpAddress;
uint16_t    led_cnt = 0;
volatile uint8_t Flag_1ms;      //1 milli-second timeout flag
uint8_t cTemp = 10;				//for RS485 mode switching delay/* Private function prototypes -----------------------------------------------*/
static void IAP_Init(void);
void RCC_Configuration(void);
void GPIO_Configuration(void);const uint16_t COM_TX_PIN[COMn] = {EVAL_COM1_TX_PIN, EVAL_COM2_TX_PIN};
const uint16_t COM_RX_PIN[COMn] = {EVAL_COM1_RX_PIN, EVAL_COM2_RX_PIN};
const uint32_t COM_TX_PORT_CLK[COMn] = {EVAL_COM1_TX_GPIO_CLK, EVAL_COM2_TX_GPIO_CLK}; 
const uint32_t COM_RX_PORT_CLK[COMn] = {EVAL_COM1_RX_GPIO_CLK, EVAL_COM2_RX_GPIO_CLK};
const uint32_t COM_USART_CLK[COMn] = {EVAL_COM1_CLK, EVAL_COM2_CLK};
GPIO_Module* COM_TX_PORT[COMn] = {EVAL_COM1_TX_GPIO_PORT, EVAL_COM2_TX_GPIO_PORT}; 
GPIO_Module* COM_RX_PORT[COMn] = {EVAL_COM1_RX_GPIO_PORT, EVAL_COM2_RX_GPIO_PORT};
USART_Module* COM_USART[COMn] = {EVAL_COM1, EVAL_COM2};/* Private functions ---------------------------------------------------------*/
/**
*@name: RS485_Configuration
*@description: RS485 GPIO initialization
*@params: none
*@return: none
*/
void RS485_Configuration(void)
{GPIO_InitType GPIO_InitStructure;RCC_EnableAPB2PeriphClk(RS485_GPIOx_CLK, ENABLE);/* -2- Configure GPIOx_PIN in output push-pull mode */GPIO_InitStruct(&GPIO_InitStructure);GPIO_InitStructure.Pin = RS485_GPIO_PIN;GPIO_InitStructure.GPIO_Mode = GPIO_MODE_OUTPUT_PP;GPIO_InitStructure.GPIO_Speed = GPIO_SPEED_HIGH;GPIO_InitPeripheral(RS485_GPIOx, &GPIO_InitStructure);GPIO_ResetBits(RS485_GPIOx, RS485_GPIO_PIN);
}/**
*@name: Process_1MS
*@description: 1 milli-second timeout process
*@params: none
*@return: none
*/
void Process_1MS(void)
{if (Flag_1ms == 0){return;}Flag_1ms = 0;if (led_cnt % 500 == 0) //LED toggle to see whether the timer works correctly.{GPIO_WriteBit(GPIOF, GPIO_PIN_7, (Bit_OperateType)(1 - GPIO_ReadOutputDataBit(GPIOF, GPIO_PIN_7)));}led_cnt++;if (led_cnt >= 60000){led_cnt = 0;}
}/**
*@name: RS485_Mode_Switching
*@description: switch the RS485 work mode
*@params: mode: 1: receive, 0: transmit
*@return: none
*/
void RS485_Mode_Switching(uint8_t mode)
{cTemp = 10;if(mode){RS485_L();}else{RS485_H();}	while (cTemp-- != 0);
}/*** @brief  Main program.* @param  None* @retval None*/
int main(void)
{/* Flash unlock */FLASH_Unlock();/* Initialize Key Button mounted on N32G031-EVAL board *///N32_EVAL_PBInit(BUTTON_KEY, BUTTON_MODE_GPIO);/* Test if Key push-button on N32G031 Board is pressed *///if (N32_EVAL_PBGetState(BUTTON_KEY)  == 0x00)    if (1){/* If Key is pressed *//* Execute the IAP driver in order to re-program the Flash */RCC_Configuration();GPIO_Configuration();RS485_Configuration();TIM3_Configuration();IAP_Init();        RS485_Mode_Switching(0);SerialPutString("\r\n===============================================================");SerialPutString("\r\n=              (C) COPYRIGHT 2023 DS Power Supply             =");SerialPutString("\r\n=                                                             =");SerialPutString("\r\n=     In-Application Programming  (Version 1.0.0)             =");SerialPutString("\r\n=                                                             =");SerialPutString("\r\n=                           By POWER          		      =");SerialPutString("\r\n===============================================================");SerialPutString("\r\n\r\n");Main_Menu();}/* Keep the user application running */else{/* Test if user code is programmed starting from address "ApplicationAddress" */if (((*(__IO uint32_t*)ApplicationAddress) & 0x2FFE0000) == 0x20000000){/* Jump to user application */JumpAddress = *(__IO uint32_t*)(ApplicationAddress + 4);Jump_To_Application = (pFunction) JumpAddress;/* Initialize user application's Stack Pointer */__set_MSP(*(__IO uint32_t*) ApplicationAddress);Jump_To_Application();}}while (1){}
}/*** @brief  Configures COM port.* @param  COM: Specifies the COM port to be configured.*   This parameter can be one of following parameters:    *     @arg COM1*     @arg COM2  * @param  USART_InitStruct: pointer to a USART_InitTypeDef structure that*   contains the configuration information for the specified USART peripheral.* @retval None*/
void IAP_COMInit(COM_TypeDef COM, USART_InitType* USART_InitStruct)
{GPIO_InitType GPIO_InitStructure;/* Enable GPIO clock */	RCC_EnableAPB2PeriphClk(COM_TX_PORT_CLK[COM] | RCC_APB2_PERIPH_AFIO, ENABLE);/* Enable UART clock */if (COM == COM1){RCC_EnableAPB2PeriphClk(COM_USART_CLK[COM], ENABLE); }else{RCC_EnableAPB1PeriphClk(COM_USART_CLK[COM], ENABLE);}GPIO_InitStruct(&GPIO_InitStructure);/* Configure USART Tx as alternate function push-pull */GPIO_InitStructure.GPIO_Mode = GPIO_MODE_AF_PP;GPIO_InitStructure.Pin = COM_TX_PIN[COM];GPIO_InitStructure.GPIO_Alternate = GPIO_AF4_USART1;//alternate functionGPIO_InitPeripheral(COM_TX_PORT[COM], &GPIO_InitStructure);    /* Configure USART Rx as input floating */  GPIO_InitStructure.Pin = GPIO_PIN_10;GPIO_InitStructure.GPIO_Alternate = GPIO_AF4_USART1;//alternate functionGPIO_InitPeripheral(COM_RX_PORT[COM], &GPIO_InitStructure);/* USART configuration */USART_Init(COM_USART[COM], USART_InitStruct);/* Enable USART */USART_Enable(COM_USART[COM], ENABLE);
}/*** @brief  Initialize the IAP: Configure RCC, USART and GPIOs.* @param  None* @retval None*/
void IAP_Init(void)
{USART_InitType USART_InitStructure;/* USART resources configuration (Clock, GPIO pins and USART registers) ----*//* USART configured as follow:- BaudRate = 115200 baud- Word Length = 8 Bits- One Stop Bit- No parity- Hardware flow control disabled (RTS and CTS signals)- Receive and transmit enabled*/USART_InitStructure.BaudRate = 115200;USART_InitStructure.WordLength = USART_WL_8B;USART_InitStructure.StopBits = USART_STPB_1;USART_InitStructure.Parity = USART_PE_NO;USART_InitStructure.HardwareFlowControl = USART_HFCTRL_NONE;USART_InitStructure.Mode = USART_MODE_RX | USART_MODE_TX;IAP_COMInit(COM1, &USART_InitStructure);
}/*** @brief  Configures the different system clocks.*/
static void RCC_Configuration(void)
{//PCLK1 = HCLK/4, set the prescaler of the APB1 clock and timer3 uses APB1 clockRCC_ConfigPclk1(RCC_HCLK_DIV4);//Enable GPIO clocksRCC_EnableAPB2PeriphClk(RCC_APB2_PERIPH_GPIOA | RCC_APB2_PERIPH_AFIO | RCC_APB2_PERIPH_GPIOF, ENABLE);//TIM3 clock enableRCC_EnableAPB1PeriphClk(RCC_APB1_PERIPH_TIM3, ENABLE);
}/**
*@name: GPIO_Configuration
*@description: IO initialization
*@params: none
*@return: none
*/
static void GPIO_Configuration(void)
{GPIO_InitType GPIO_InitStructure;GPIO_InitStruct(&GPIO_InitStructure);//PF7: LEDGPIO_InitStructure.Pin       = GPIO_PIN_7;GPIO_InitStructure.GPIO_Mode = GPIO_MODE_OUTPUT_PP;GPIO_InitPeripheral(GPIOF, &GPIO_InitStructure);//set the default IO levelGPIO_SetBits(GPIOF, GPIO_PIN_7);
}#ifdef USE_FULL_ASSERT/*** @brief  Reports the name of the source file and the source line number*         where the assert_param error has occurred.* @param file pointer to the source file name* @param line assert_param error line source number*/
void assert_failed(const uint8_t* expr, const uint8_t* file, uint32_t line)
{/* User can add his own implementation to report the file name and line number,ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) *//* Infinite loop */while (1){}
}#endif/*
=============关于补码的简洁概括==============正数,本身就是补码。负数,就用它的正数,减一取反,即可得到补码。如,已知:+9 的二进制是:0000 1001。下面求-9 补码:先减一:0000 1001 - 1 = 0000 1000;再取反:1111 0111。所以有:-9 补码 = 1111 0111。这不就完了吗!简不简单? 意不意外?如果把一个值赋给一个有符号类型,如果补码的最高位是1,则是负数,还原成实际的负数值的步骤是:1、先按位取反2、再加1例如(int表示32位有符号):int a = -552305;(按正数552305的补码先减一再按位取反得到-552305二进制补码是:1111 1001 0010 1000 1111)(short表示16位无符号)short c = (short)a;由于short最大只有16位,因此高4位被忽略,剩下1001 0010 1000 1111又由于赋值给short,所以最高位表示符号位,这里是1表示负数,则将1001 0010 1000 1111除符号位外按位取反再加1得到1110 1101 0111 0001,最终的结果就是除符号位外的数据:110 1101 0111 0001=-28017
=============================================*//*** @}*//*** @}*/

其实也很简单,主要就是串口和RS485相关外设的初始化,串口这里没有使用中断的方法。

外设初始化完成后就会进入主菜单,根据用户输入的值进行不同的操作,剩下就是YMODEM协议的理解。后面有空准备自己做一个上位机来实现YMODEM协议,因为自己有逻辑分析仪,可以抓取到实际传输的数据。

我刚开始调试的时候一直出现在secureCRT中输入菜单编号后单片机没有反应的情况,仔细检测了代码发现和STM32的也差不多,没有什么问题。于是使能串口的接收中断,我测试在secureCRT中选择菜单后串口收到的是什么,后来查看确实有问题,无论secureCRT中输入什么内容,打印出来的都是两个字节,有时候是0xFD 0xFF,有时又是0xF9 0xFF的,感觉莫名其妙的。后来想起来之前使用N32G031串口发送有时少一字节的问题,特别是在RS485通信中最容易出现,再查看本项目的下面这个函数:

/*** @brief  Print a character on the HyperTerminal* @param  c: The character to be printed* @retval None*/
void SerialPutChar(uint8_t c)
{USART_SendData(EVAL_COM1, c);while (USART_GetFlagStatus(EVAL_COM1, USART_FLAG_TXDE) == RESET);   //while (USART_GetFlagStatus(EVAL_COM1, USART_FLAG_TXDE) == RESET)//{//}
}

我忽然想起来了,还有一个非常重要的语句没有调用,那就是:

 while (USART_GetFlagStatus(EVAL_COM1, USART_FLAG_TXC) == RESET);

解释如下:

/*==========关于串口直接发送正常,而485出现少发1字节的问题==============

            我看我的代码发现我的检测标志位是USART_FLAG_TXDE(发送寄存器空)。

            以前一直用这个写也没啥问题,但是现在出现了丢失字节,我就怀疑可能是寄存器空,

            但是数据还没有完全发出(注意:当你用USART_FLAG_TXDE标志位去认为已经发完,并去控制RTS引脚的时候,485芯片会从发送转接收,

            导致如果有字节没有发完就不发了)。

            因此,我们只需要再加一个标志位检测就好。========================================================================

USART_FLAG_TXC表示数据发送完成,只有当它置1才表示最后一个数据发送结束了。加了这个标志位检测后RS485的收发就正常工作了。

另外,关于用户应用程序的中断向量表偏移的问题,除了在MDK中设置ROM的偏移值外,还要设置中断向量表。N32G031系列没有像STM32系列那样有类似SCB->VTOR的成员设置中断向量表的偏移,IAP接收完bin文件后用户程序总是不执行。我在网上搜索,终于找到了一篇文章专门介绍这个的:

【精选】国民技术N32G030F6S7使用ymodem协议更新固件_n32g452 ymodem远程升级-CSDN博客

感谢这位博主,要不然我不知道还要折腾多久。这个芯片厂家应该要把资料准备得齐全一些,尤其是这种关键的内容。

下面是在secureCRT中主菜单和发送完成bin后的界面:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/195769.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

基于SSM的智能仓储系统研究与设计

末尾获取源码 开发语言:Java Java开发工具:JDK1.8 后端框架:SSM 前端:采用JSP技术开发 数据库:MySQL5.7和Navicat管理工具结合 服务器:Tomcat8.5 开发软件:IDEA / Eclipse 是否Maven项目&#x…

开源与闭源:大模型时代的技术交融与商业平衡

一、开源和闭源的优劣势比较 1.1 开源 优势: 1.技术共享与吸引人才: 开源促进了技术共享,吸引了全球范围内的人才参与大模型的发展,形成了庞大的开发者社区。 2.推动创新: 开源模式鼓励开发者共同参与,推动…

【STM32】RTC(实时时钟)

1.RTC简介 本质:计数器 RTC中断是外部中断(EXTI) 当VDD掉电的时候,Vbat可以通过电源--->实时计时 STM32的RTC外设(Real Time Clock),实质是一个 掉电 后还继续运行的定时器。从定时器的角度…

腾讯云4核8G服务器性能如何多少钱一年?

腾讯云服务器4核8G配置优惠价格表,轻量应用服务器和CVM云服务器均有活动,云服务器CVM标准型S5实例4核8G配置价格15个月1437.3元,5年6490.44元,轻量应用服务器4核8G12M带宽一年446元、529元15个月,腾讯云百科txybk.com分…

牛客——OR36 链表的回文结构(C语言,配图,快慢指针)

本题是没有对C的支持的,但因为Cpp支持C,所以这里就用C写了,可以面向更多用户 链表的回文结构_牛客题霸_牛客网 (nowcoder.com) 思路一:链表翻转 简单的想想整形我们怎么比较,就是将整形A 依次取尾,放到整形…

航天联志Aisino-AISINO26081R服务器通过调BIOS用U盘重新做系统(windows系统通用)

产品名称:航天联志Aisino系列服务器 产品型号:AISINO26081R CPU架构:Intel 的CPU,所以支持Windows Server all 和Linux系统(重装完系统可以用某60驱动管家更新所有硬件驱动) 操作系统:本次我安装的服务器系统为Serv…

python趣味编程-5分钟实现一个Tic Tac Toe游戏(含源码、步骤讲解)

The Tic Tac Toe In Python是用 Python 编程语言编写的,这个Tic Tac Toe Game In Python是一个简单的基于 GUI 的策略游戏板,非常容易理解和使用。 所有的游戏规则都是一样的,就像我们玩实时井字棋一样,这是一个简单的多人游戏。 Python 中的 Tic Tac Toe 游戏:项目信息 …

物理驱动深度学习方法总结

一、物理驱动深度学习方法总结 现有博主更新物理驱动深度学方法总体介绍 二、 PINN介绍 PINN综述Blog介绍:内嵌物理知识神经网络 (Physics Informed Neural Network,简称PINN) 是一种科学机器在传统数值领域的应用方法&…

ping命令使用示例解析

【一】ping命令简介 ping (Packet Internet Groper)是一种因特网包探索器,用于测试网络连接量的程序。ping的一般用途有: ①【测试网络物理链路是否正常】:通过将ICMP(Internet控制消息协议)回显数据包发送到网络终端&…

[PHP]关联和操作MySQL数据库然后将数据库部署到ECS

在Mac电脑上使用VS Code进行PHP开发并关联操作MySQL数据库,然后将数据库部署到ECS。 1.安装PHP和MySQL 确保你的Mac上已经安装了PHP和MySQL。你可以使用Homebrew来安装它们: $ brew install php $ brew install mysql 安装mysql完成后记住这一句: …

蓝桥杯第三周算法竞赛D题E题

发现更多计算机知识,欢迎访问Cr不是铬的个人网站 D迷宫逃脱 拿到题目一眼应该就能看出是可以用动态规划来解决。但是怎么定义dp呢? 这个题增加难度的点就在当所在位置与下一个要去的位置互质的时候,会消耗一把钥匙。当没有钥匙的时候就不能移动了。想…

cocos----刚体

刚体(Rigidbody) 刚体(Rigidbody)是运动学(Kinematic)中的一个概念,指在运动中和受力作用后,形状和大小不变,而且内部各点的相对位置不变的物体。在 Unity3D 中&#xff…

使用 React Flow 构建一个思维导图应用

思维导图是围绕共同主题或问题将思想、概念、信息或任务分组的视觉表示。思维导图应用是一种软件应用,允许您创建、可视化和组织您的思想、想法和信息作为思维导图。本文将向您展示如何实现自己的思维导图应用程序。 在我们开始之前,我想向您展示一下我们…

ke11..--2其他界面也要提取我的locatStarage

获取浏览器里面的本地缓存 localStorage就是我们的浏览器缓存在哪都可以用 下面代码是获取打印到我们的页面上 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Title</title> </head> …

Hessian协议详解

前言 Hessian协议是一种基于二进制的轻量级远程调用协议&#xff0c;用于在分布式系统中进行跨语言的通信。它使用简单的二进制格式来序列化和反序列化数据&#xff0c;并支持多种编程语言&#xff0c;如Java、C#、Python等。Hessian协议相对于其他协议的优势在于其简单性和高…

微服务实战系列之Token

前言 什么是“Token”&#xff1f; 它是服务端生成的一串字符串&#xff0c;以作客户端进行请求的一个令牌&#xff0c;当第一次登录后&#xff0c;服务器生成一个Token便返回给客户端&#xff1b;以后客户端只携带此Token请求数据即可。 简言之&#xff0c;Token其实就是用户身…

【2022改良版】学法减分助手PRO小程序源码

【2022改良版】学法减分助手PRO小程序源码 &#xff0c;交管推出个学法减分&#xff0c;每个驾驶员可以把被扣的6分&#xff0c;以看视频答题的形式学习回来&#xff0c;然后答题这个一共二十道题每道题60秒&#xff0c; 有好多人不会&#xff0c;用咱们的小程序就可以模拟练习…

北京君正客户应用案例:掌静脉3D人脸猫眼视屏智能锁

凯迪仕在今年4月发布了智能锁旗舰新品K70 Pro Max掌静脉3D人脸猫眼视屏智能锁&#xff0c;随即这款新品也成了行业热议的焦点。凯迪仕每次新品都力求突破精益求精&#xff0c;不仅追求科技感、高级感与品质感&#xff0c;而且赋予科技温度&#xff0c;带来人文化的关怀。K70 Pr…

OpenCV快速入门:像素操作和图像变换

文章目录 前言1. 像素操作1.1 像素统计1.2 两个图像之间的操作1.2.1 图像加法操作1.2.3 图像加权混合 1.3 二值化1.4 LUT&#xff08;查找表&#xff09;1.4.1 查找表原理1.4.2 代码演示 2 图像变换2.1 旋转操作2.1.1 旋转的基本原理2.1.2 代码实现 2.2 缩放操作2.3 平移操作2.…

使用GPT-4训练数据微调GPT-3.5 RAG管道

原文&#xff1a;使用GPT-4训练数据微调GPT-3.5 RAG管道 - 知乎 OpenAI在2023年8月22日宣布&#xff0c;现在可以对GPT-3.5 Turbo进行微调了。也就是说&#xff0c;我们可以自定义自己的模型了。然后LlamaIndex就发布了0.8.7版本&#xff0c;集成了微调OpenAI gpt-3.5 turbo的…