mkv210_image.c详解
1.背景知识回顾
s5pv210启动后先执行内部iROM中的BL0,BL0执行完后会根据OMpin的配置选择一个外部设备来启动(有很多,我们实际使用的有2个:usb启动和SD卡启动)。
在usb启动时,内部BL0读取到BL1后不做校验,直接从BL1的实质内部0xd0020010开始执行,因此usb启动的景象led.bin不需要头信息,因此我们从usb启动时直接将镜像下载到0xd0020010去执行即可,不管头信息了。
从SD启动时,BL0会首先读取sd卡得到完整的镜像(完整指的是led.bin和16字节的头),然后BL0会自己根据你的实际镜像(指led.bin)来计算一个校验和checksum,然后和你完整镜像的头部中的checksum来比对。如果对应则执行BL1,如果不对应则启动失败(会转入执行2st启动,即SD2启动。如果这里已经是2st启动了,这里校验通不过就死定了)。(一开始计算出来的(之后被存储在头部16字节中),和BL0从sd卡读取出来后重新计算的(可能由SD卡于机械特性出现错误),进行对比)
2、mkv210_image.c的作用
保证文件小于16k,并为BL1添加校验头。编译链接时只得到led.bin,由led.bin得到210.bin的过程是三星的S5PV210所特有的,因此需要我们自己去完成,为此我们写了mkv210_image.c来完成。
3、代码详解
第1步:检验用户传参是不是3个。
第2步:分配16K Bbuffer并且填充为0.
第3步: 读源bin到buffer(过程:打开源bin,获取源bin长度,源bin长度不得超过16K-16byte,将源bin放入事先已经填充过的buffer[16]中,关闭源bin)
第4步: 计算校验和(在要校验的内存区域中,所有内存中的内容按照字节为单位来进行相加,最终相加的和与整镜像的头部中的checksum来比对)
第5步:拷贝buffer中的内容到目的bin,即完成由led.bin得到210.bin。
(1)main函数两个形参的作用
main函数接收2个形参:argc和argv。
argc是用户(通过命令行来)执行这个程序时,实际传递的参数个数。注意这个个数是包含程序执行本身的
argv是一个字符串数组,这个数组中存储的字符串就是一个个的传参。
譬如我们执行程序时使用./mkx210 led.bin 210.bin,则argc = 3,则argv[0] = “./mkx210” argv[1] = led.bin argv[2] = 210.bin
(2)glibc读写文件接口
linux中要读取一个文件,可以使用fopen打开文件,fread读取文件,读完之后fclose关闭文件。
要写文件用fwrite来写。这些函数是glibc的库函数,在linux中用man 3 可以查找。
(3)校验和的计算方法
校验和其实就是需要校验的内存区域中,所有内存中的内容按照字节为单位来进行相加,最终相加的和极为校验和。
/*
* mkv210_image.c作用:将usb启动时使用的xxx_usb.bin制作得到由sd卡启动的镜像xxx_sd.bin
* 本文件来自于友善之臂的裸机教程,据友善之臂的文档中讲述,本文件是一个热心网友提供,在此表示感谢。
*
* 在BL0阶段,Irom内固化的代码读取nandflash或SD卡前16K的内容,
* 并比对前16字节中的校验和是否正确,正确则继续,错误则停止。
*/
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#define BUFSIZE (16*1024)
#define IMG_SIZE (16*1024)
#define SPL_HEADER_SIZE 16
#define SPL_HEADER "S5PC110 HEADER " // 16字节,随便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
if (argc != 3)
{
printf("Usage: %s <source file> <destination file>\n", argv[0]);
return -1;
}
// 2.分配16K的buffer并请零
BufLen = BUFSIZE;
Buf = (char *)malloc(BufLen);
if (!Buf)
{
printf("malloc buffer failed!\n");
return -1;
}
memset(Buf, 0x00, BufLen);
// 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长度
fseek(fp, 0L, SEEK_END); // 定位到文件尾
fileLen = ftell(fp); // 得到文件长度
fseek(fp, 0L, SEEK_SET); // 再次定位到文件头
// 3.3源bin长度不得超过16K-16byte,
// 注意若裸机程序大于16KB,则裸机程序会出问题,在后续做LCD实验时需特别注意
count = (fileLen < (IMG_SIZE - SPL_HEADER_SIZE))
? fileLen : (IMG_SIZE - SPL_HEADER_SIZE);
// 3.4在buffer[0~15]中存放"S5PC110 HEADER "
memcpy(&Buf[0], SPL_HEADER, SPL_HEADER_SIZE);
// 3.5读源bin到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中所有的字节数据加和起来得到的结果
a = Buf + SPL_HEADER_SIZE;
for(i = 0, checksum = 0; i < IMG_SIZE - SPL_HEADER_SIZE; i++)
checksum += (0x000000FF) & *a++;
// 4.2将校验和保存在buffer[8~12]
a = Buf + 8; // Buf是xxx_sd.bin的起始地址,+8表示向后位移2个字,也就是说写入到第3个字
*((unsigned int *)a) = checksum;
// 5.拷贝buffer中的内容到目的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;
}
mkv210_image.c详解相关推荐
- mkv210_image.c文件详解
以下内容源于朱有鹏<物联网大讲堂>课程的学习,如有侵权,请告知删除. 1.mkv210_image.c的使用演示 裸机程序中的Makefile(实际上真正的项目的Makefile都是这样的 ...
- 从命令行到IDE,版本管理工具Git详解(远程仓库创建+命令行讲解+IDEA集成使用)
首先,Git已经并不只是GitHub,而是所有基于Git的平台,只要在你的电脑上面下载了Git,你就可以通过Git去管理"基于Git的平台"上的代码,常用的平台有GitHub.Gi ...
- JVM年轻代,老年代,永久代详解
秉承不重复造轮子的原则,查看印象笔记分享连接↓↓↓↓ 传送门:JVM年轻代,老年代,永久代详解 速读摘要 最近被问到了这个问题,解释的不是很清晰,有一些概念略微模糊,在此进行整理和记录,分享给大家.在 ...
- docker常用命令详解
docker常用命令详解 本文只记录docker命令在大部分情境下的使用,如果想了解每一个选项的细节,请参考官方文档,这里只作为自己以后的备忘记录下来. 根据自己的理解,总的来说分为以下几种: Doc ...
- 通俗易懂word2vec详解词嵌入-深度学习
https://blog.csdn.net/just_so_so_fnc/article/details/103304995 skip-gram 原理没看完 https://blog.csdn.net ...
- 深度学习优化函数详解(5)-- Nesterov accelerated gradient (NAG) 优化算法
深度学习优化函数详解系列目录 深度学习优化函数详解(0)– 线性回归问题 深度学习优化函数详解(1)– Gradient Descent 梯度下降法 深度学习优化函数详解(2)– SGD 随机梯度下降 ...
- CUDA之nvidia-smi命令详解---gpu
nvidia-smi是用来查看GPU使用情况的.我常用这个命令判断哪几块GPU空闲,但是最近的GPU使用状态让我很困惑,于是把nvidia-smi命令显示的GPU使用表中各个内容的具体含义解释一下. ...
- Bert代码详解(一)重点详细
这是bert的pytorch版本(与tensorflow一样的,这个更简单些,这个看懂了,tf也能看懂),地址:https://github.com/huggingface/pytorch-pretr ...
- CRF(条件随机场)与Viterbi(维特比)算法原理详解
摘自:https://mp.weixin.qq.com/s/GXbFxlExDtjtQe-OPwfokA https://www.cnblogs.com/zhibei/p/9391014.html C ...
最新文章
- 莱比特矿池CEO江卓尔:BCH作为货币不需要新功能,但出于货币竞争的考虑需要
- WPF/Silverlight深度解决方案:(十六)传值实现
- Python中list和set的区别
- 人工智能 | SLAM与Visual Odometry技术综述(浙江大学智能系统和控制研究所)
- 【maven插件】maven-resources-plugin
- Caused by: java
- 元宇宙行业深度研究报告:为什么元宇宙是下一代互联网?
- Java当中捕获异常
- 登录:应用程序错误通知
- 神经网络训练3次就准确率不变_1组高效徒手训练,6个动作每周3-5次,帮你在家高效率燃脂增肌!...
- java九种数据类型以及封装类
- gps天线拆解图片_飞宇稳定器拆解:握杆的手,不怕颤抖
- 2016年Google面筋记录
- 算法 Tricks(三)—— 判断序列是否为等差数列
- server.htaccess 具体解释以及 .htaccess 參数说明
- python图像量化及采样处理
- EXCEL用今天日期减去之前的日期得到结果
- 牛客网浙江大学机试--找出直系亲属
- JMokit中的@Mocked与@Injectable区别
- 年度盘点!Flink 社区全年的精华内容都在这里啦(内附福利)
热门文章
- 系统架构师学习笔记-数据库系统
- Qt奇淫技巧-使用QSharedMemory方式实现数据跨界面传输
- c#中使用mysql查询语句_遇到@符合怎么办_C# Mysql 查询 Rownum的解决方法
- java $ class_java文件编译后额外生成的$1.class是怎么一回事
- 单片机c语言三种经典程序结构,单片机C语言程序的结构和设计精选.docx
- AVB2 avbtool.py脚本常用命令
- dama数据管理知识体系指南第二版pdf_DMBOK数据管理 - CDMP认证培训
- 十四、List,Set,Collection,Collections
- 前端基础-html-列表
- MarckDown学习