从简单的文件偏移、读写进一步研究MCU程序的下载和软件升级(上)

      关于MCU固件更新和下载,在上大学的时候老师并没有详细的去讲解,只是知道程序xxx.c编译后生成xxx.hex或者xxx.bin,然后将对应的xxx.hex和xxx.bin下载到MCU上,然后五花八门的程序就开始运行了,还有就是程序在正常运行中,通过远程获取更新包,然后更新程序,而程序只有一个部分更新,而不影响其它的部分。这就是所谓的软件升级。

      工作后,对于学习技术知识,不要仅停留在表面上,而是要深入的去分析实现原理,程序是怎么下载到MCU的?我们的软件又是怎么更新而又不会影响其它部分的,这一节,我们将借助一个简单的文件操作例子,来阐述最基本的原理。

       首先,我们编写一段程序,来创建一个1.bin的文件,并在这个文件中写入数据1,2,3,4,5,6,7,8,9,10,例程如下:

test.c

#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>


int main(void)
{
	int fd = -1 ; 
	char buffer[] = {1,2,3,4,5,6,7,8,9,10};
	fd = open("1.bin",O_RDWR | O_CREAT);
	if(fd < -1)
	{
		printf("open file fail\n");
		return -1 ;
	}
	write(fd,buffer,10);
	close(fd);
	return 0 ;	
}

运行结果,在当前路径下会生成一个1.bin的文件,用WinHex这个软件打开这个文件可以看到我们刚刚用程序写入的数据:

     现在,我们希望,改变1.bin里面某个地址的值,而不影响其它的数据,在下面的程序中,我们改变offset=5这个地址,将该地址的值修改为16,例程如下:

test1.c

#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>

int main(void)
{
	int i  ;
	int fd = -1 ; 
	char buffer[11] = {0};
	int data = 16 ;
	//1、打开1.bin这个文件 
	fd = open("1.bin",O_RDWR);
	if(fd < -1)
	{
		printf("open file fail\n");
		return -1 ;
	}
	//读出1.bin中的内容 
	read(fd,buffer,11);
	for(i = 0 ; i < 10 ; i++)
	{
		printf("buffer[%d]:%d\n",i,buffer[i]);
	}
	printf("xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx读原始数据完毕xxxxxxxxxxxxxxxxxxxxxxxxxxxx\n");
	//2、将1.bin从0地址开始偏移到地址5 
	lseek(fd,5,SEEK_SET);
	//3、将data=16这个值写到1.bin这个文件的偏移地址5 
	write(fd,&data,1); 
	//4、清buffer 
	memset(buffer,0,11);
	//5、将地址偏移重新改到地址0
	lseek(fd,0,SEEK_SET); 
	//6、读出修改后1.bin中的内容 
	read(fd,buffer,11);
	for(i = 0 ; i < 10 ; i++)
	{
		printf("buffer[%d]:%d\n",i,buffer[i]);
	} 
	printf("xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx读改写后的数据完毕xxxxxxxxxxxxxxxxxxxxxxxxx\n");
	//7、关闭文件描述符 
	close(fd);
	return 0 ;	
}

      在这个程序中,我们先读取原先1.bin中的数据,接着通过lseek函数将文件偏移到offset=5的地址,然后使用write,将data=16这个数据写入到offset=5这个地址,改写这个地址的数据,接下来调用lseek将偏移地址改写回从0开始,再读出改写后的所有数据,效果如下:

     很明显,第二个写入改变offset=5这个地址的数据的程序并没有影响其它数据,而是以覆盖的形式直接改写了偏移地址的数据。

      举这个简单的例子能说明什么呢?这跟我们程序最终的更新原理其实是一样的,我们再写一个程序:

      test2.c:

#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>


int main(void)
{
	int i ;
	int fd = -1 ; 
	char buffer[] = {2,1,4,2,3,1,4,8,9,6};
	char buffer1[11];
	fd = open("1.bin",O_RDWR | O_CREAT);
	if(fd < -1)
	{
		printf("open file fail\n");
		return -1 ;
	}
	//偏移到地址0 
	lseek(fd,0,SEEK_SET);
	//写入10个数据 
	write(fd,buffer,10);
	//偏移到地址0 
	lseek(fd,0,SEEK_SET);
	//读出1.bin中的内容 
	read(fd,buffer1,11);
	for(i = 0 ; i < 10 ; i++)
	{
		printf("buffer1[%d]:%d\n",i,buffer1[i]);
	}
	close(fd);
	return 0 ;	
}

运行结果:

     从这里我们可以得知,数据从偏移地址0到偏移地址9都被修改了,这也就是我们MCU固件更新的原理。

     无论是固件更新还是软件升级原理都是差不多的,最后就是简单的将二进制数据覆盖对应的地址区域,在这里,我举的这个例子仅仅只是为了阐述最基本原理,然而MCU固件下载,程序更新其中不乏还是有很多复杂的流程,比如先将要更新的数据拷贝到一个和主程序不相干的区域,在这里我们简单叫做备份区吧。不管是备份区还是数据区还是其它区域,这些区域需要类似Window操作系统一样有一个分区表,然后来指定对应区域存储的数据,有些MCU内存比较小,还需要外挂一些存储芯片,常见的有NAND FLASH,SPI FLASH,NOR FLASH等等,然后在备份区将数据覆盖到原始区域,最后再删除备份区域的数据,我们在下一节中,将整理一个MCU的固件下载以及软件升级的原理。

在实际工程应用中,我们时常会遇到为解决某个老产品的BUG,需要在工程现场更新设备的FPGA代码,或者参加电信测试时需要现场升级设备FPGA程序以便于调试。公司现阶段所用的Altera FPGA程序代码一般存放于芯片配套的FLASH存储器中,而常见的对印制板上FLASH编程有几种方法,原始的方法是使用编程器,这种方法需要要将芯片取下,十分不便,或者通过JTAG接口连接到PC机上,但需要专用下载软件(一般由芯片生产厂商提供)。在测试现场或调测机房现场,要找到FPGA的专用下载线是比较困难的,且Altera FPGA的专用下载软件并不是每个PC设备上都有的。有时仅为了更新一个FPGA的程序就需要研发或客服人员亲自到现场去烧写程序,这既不便捷,也使得设备维护成本大大增加。  经过可行性与成本的考虑,我们找到一种既方便实用又低成本的方法来实现FPGA程序的在线现升级。即在MCU中(单片机或ARM均可)用软件来模拟XModem协议,将程序文件传输到FPGA的FLASH中。这种方法使用WINDOWS自带的超级终端软件来传送文件,无需安装专用软件,硬件支持仅需要一根通用串口线,只要在目标板MCU上增加一段实现XModem协议传输的代码,就可以方便的实现FPGA程序下载了。这种特点不仅方便了客服人员,也给研发和生产人员在现场调试和软件升级、修改中带来极大方便。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Engineer-Bruce_Yang

谢谢您

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值