在USB启动原文件上添加上下面代码即可完成校验头的添加,不过一般的做法是把此程序成单独文件在Makefile中进行编译,一般形式为:
gcc check_head.c -o check_head
./check_head no_check_head.bin have_check_head.bin
注:编译器要使用gcc,平台(Linux)编译器,不能使用交叉编译工具链中的gcc,因为此程序的编译工作只是在平台编译就完成的和要使用的开发板没有关系。check_head.c为添加校验头的程序,no_check_head.bin为不带校验头的bin文件,have_check_head.bin为带校验头的bin文件,上述名字均可以自行更改。
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#define BUFSIZE (16*1024) //16k个字节
#define IMG_SIZE (16*1024)
#define SPL_HEADER_SIZE 16 //16个字节
#define SPL_HEADER "Any character " //只要为16个字节就行
int main (int argc, char *argv[])
{
FILE *fp;
char *Buf, *a;
int BufLen;
int nbytes, fileLen;
unsigned int checksum, count;
int i;
// 1. 检查是否为3个参数,因为执行时为./mkx210 xxx.bin yyy.bin;xxx.bin为不带校验头
if (argc != 3)
{
printf("Usage: %s <source file> <destination file>\n", argv[0]);
return -1;
}
// 2. 分配16K的buffer
BufLen = BUFSIZE; //16k个字节
Buf = (char *)malloc(BufLen); //因为malloc失败会返回0
if (!Buf)
{
printf("Alloc buffer failed!\n");
return -1;
}
memset(Buf, 0x00, BufLen); //初始化为0
// 3. 读源bin文件到buffer
// 3.1 打开源bin
fp = fopen(argv[1], "rb");
if( fp == NULL)
{
printf("source file open error\n");
free(Buf);
return -1;
}
// 3.2 获取源bin文件长度 //此时获取的是不带校验头的bin文件
fseek(fp, 0L, SEEK_END); // 定位到文件尾
fileLen = ftell(fp); // 得到文件长度
fseek(fp, 0L, SEEK_SET); // 再次定位到文件头
// 3.3 源bin长度不得超过16K-16byte //因为不带校验头的bin文件没有16字节的校验头
count = (fileLen < (IMG_SIZE - SPL_HEADER_SIZE)) //IMG_SIZE-SPL_HEADER_SIZE=16K-16byte
? fileLen : (IMG_SIZE - SPL_HEADER_SIZE); //判断是否超过16K-16byte,如果超过就强制截取
// 3.4 buffer[0~15]存放"Any character " //[0~15]为16个字节
memcpy(&Buf[0], SPL_HEADER, SPL_HEADER_SIZE); //把前16个字节内容存储为SPL_HEADER宏中的内容
// 3.5 读源bin到buffer[16] //此时不带校验头的bin文件中的真正内容已经读到
//了buffer中从buffer[16]开始的内存中
nbytes = fread(Buf + SPL_HEADER_SIZE, 1, count, fp);
if ( nbytes != count )
{
printf("source file read error\n");
free(Buf);
fclose(fp);
return -1;
}
fclose(fp);
// 4. 计算校验和
// 4.1 从第16byte开始计算,把buffer中所有的字节数据加和起来得到的结果,可以参照程序2理解
a = Buf + SPL_HEADER_SIZE; //a为一个指针,指向Buf中第16byte开始的地方
for(i = 0, checksum = 0; i < IMG_SIZE - SPL_HEADER_SIZE; i++)
checksum += (0x000000FF) & *a++; //0xFF刚好代表一个字节,具体理解可以参照程序2 //checksum=checksum+(0x000000FF)& *a++
// 4.2 将校验和保存在buffer[8~15] //保存完之后此时Buf中存储的就是带校验和的有效文件
a = Buf + 8; // Buf是210.bin的起始地址,+8表示向后位移2个字,
//也就是说写入到第3个字
*( (unsigned int *)a ) = checksum;
// 5. 拷贝buffer中的内容到目的bin //也就是210.bin
// 5.1 打开目的bin
fp = fopen(argv[2], "wb");
if (fp == NULL)
{
printf("destination file open error\n");
free(Buf);
return -1;
}
// 5.2 将16k的buffer拷贝到目的bin中
a = Buf;
nbytes = fwrite( a, 1, BufLen, fp);
if ( nbytes != BufLen )
{
printf("destination file write error\n");
free(Buf);
fclose(fp);
return -1;
}
free(Buf);
fclose(fp);
return 0;
}
程序2::
#include<stdio.h>
main()
{
int a=0;
printf("%d\n",a+0xf);
}
结果为:15