[ Linux Busybox ] nandwrite 命令解析

2023-11-08 12:20

本文主要是介绍[ Linux Busybox ] nandwrite 命令解析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

    • 相关结构体
    • nandwrite 函数实现
    • nandwrite 实现流程图


文件路径:busybox-1.20.2/miscutils/nandwrite.c

相关结构体

MTD 相关信息结构体

struct mtd_info_user {__u8 type;              // MTD 设备类型__u32 flags;            // MTD设备属性标志__u32 size;             // mtd设备的大小__u32 erasesize;        // MTD设备的擦除单元大小,对于 NandFlash来说就是 Block的大小__u32 writesize;        // MTD设备的读写单元大小,对于 NandFlash来说就是page 的大小__u32 oobsize;          // oob区域大小__u64 padding;          // 有效的oob区域大小
};

nandwrite 函数实现

假如要将 mtd2 拷贝到 mtd3 分区中,使用的命令是:nandwrite /dev/mtd3 /dev/mtd2

int nandwrite_main(int argc UNUSED_PARAM, char **argv)
{/* Buffer for OOB data */unsigned char *oobbuf;unsigned opts;int fd;ssize_t cnt;unsigned mtdoffset, meminfo_writesize, blockstart, limit;unsigned end_addr = ~0;struct mtd_info_user meminfo;struct mtd_oob_buf oob;unsigned char *filebuf;const char *opt_s = "0", *opt_f = "-", *opt_l;if (IS_NANDDUMP) {                            // 从命令行获取参数opt_complementary = "=1";opts = getopt32(argv, "os:bf:l:", &opt_s, &opt_f, &opt_l);} else { /* nandwrite */opt_complementary = "-1:?2";opts = getopt32(argv, "ps:", &opt_s);}argv += optind;if (IS_NANDWRITE && argv[1])                    // argv[1]为  /dev/mtd2opt_f = argv[1];if (!LONE_DASH(opt_f)) {                        // 判断输入的参数是否为文件,根据命令为文件添加权限int tmp_fd = xopen(opt_f,IS_NANDDUMP ? O_WRONLY | O_TRUNC | O_CREAT : O_RDONLY);xmove_fd(tmp_fd, IS_NANDDUMP ? STDOUT_FILENO : STDIN_FILENO);   // 将文件内容放在标准输出或者标准输入中。}fd = xopen(argv[0], O_RDWR);        // 打开文件,argv[0]为/dev/mtd3xioctl(fd, MEMGETINFO, &meminfo);    // 获取内存信息mtdoffset = xstrtou(opt_s, 0);            // 获取mtd偏移量,默认为0if (IS_NANDDUMP && (opts & OPT_l)) {unsigned length = xstrtou(opt_l, 0);if (length < meminfo.size - mtdoffset)end_addr = mtdoffset + length;}/* Pull it into a CPU register (hopefully) - smaller code that way */meminfo_writesize = meminfo.writesize;        // 获取每次写入内存的大小(一般为页大小)if (mtdoffset & (meminfo_writesize - 1))        // 判断写入的地址是否页对齐bb_error_msg_and_die("start address is not page aligned");filebuf = xmalloc(meminfo_writesize);            // 根据每次写入的大小分配buf和oob内存oobbuf = xmalloc(meminfo.oobsize);oob.start  = 0;                    // 开始地址oob.length = meminfo.oobsize;        // oob大小oob.ptr    = oobbuf;                // oob值blockstart = mtdoffset & ~(meminfo.erasesize - 1);        // 获得块起始地址if (blockstart != mtdoffset) {unsigned tmp;/* mtdoffset is in the middle of an erase block, verify that* this block is OK. Advance mtdoffset only if this block is* bad.*/tmp = next_good_eraseblock(fd, &meminfo, blockstart);if (tmp != blockstart) {/* bad block(s), advance mtdoffset */if (IS_NANDDUMP & !(opts & OPT_b)) {int bad_len = MIN(tmp, end_addr) - mtdoffset;dump_bad(&meminfo, bad_len, !(opts & OPT_o));}mtdoffset = tmp;}}cnt = -1;limit = MIN(meminfo.size, end_addr);                // 获取写入总空间,meminfo.size为mtd总空间大小while (mtdoffset < limit) {                            // 循环往mtd写入数值,直到超出mtd总空间大小int input_fd = IS_NANDWRITE ? STDIN_FILENO : fd;    // 若为IS_NANDWRITE指令,将输入input_fd指向标准输入,输出output_fd指向文件int output_fd = IS_NANDWRITE ? fd : STDOUT_FILENO;blockstart = mtdoffset & ~(meminfo.erasesize - 1);   // 获得块起始地址if (blockstart == mtdoffset) {                // 若是对齐的,开始检测坏块/* starting a new eraseblock */mtdoffset = next_good_eraseblock(fd, &meminfo, blockstart);         // 检测坏块,若检测到跳过,具体实现见下面一个函数if (IS_NANDWRITE)printf("Writing at 0x%08x\n", mtdoffset);else if (mtdoffset > blockstart) {int bad_len = MIN(mtdoffset, limit) - blockstart;dump_bad(&meminfo, bad_len, !(opts & OPT_o));}if (mtdoffset >= limit)        // 偏移量超出mtd总大小跳出break;}xlseek(fd, mtdoffset, SEEK_SET);    // 将读写位置移到文件开头/* get some more data from input */cnt = full_read(input_fd, filebuf, meminfo_writesize);    // 从获取标准输入中获取数据,大小为内存写入的大小,数据保存在filebufif (cnt == 0) {/* even with -p, we do not pad past the end of input* (-p only zero-pads last incomplete page)*/break;}if (cnt < meminfo_writesize) {            // 从标准输出中获取到数据的大小 若小于 写入的标准大小if (IS_NANDDUMP)bb_error_msg_and_die("short read");if (!(opts & OPT_p))bb_error_msg_and_die("input size is not rounded up to page size, ""use -p to zero pad");/* zero pad to end of write block */memset(filebuf + cnt, 0, meminfo_writesize - cnt);    // 在数据后面填充0}xwrite(output_fd, filebuf, meminfo_writesize);        // 将filebuf数据拷贝到 mtd中if (IS_NANDDUMP && !(opts & OPT_o)) {/* Dump OOB data */oob.start = mtdoffset;xioctl(fd, MEMREADOOB, &oob);xwrite(output_fd, oobbuf, meminfo.oobsize);}mtdoffset += meminfo_writesize;            // 指向下个要写入的地址if (cnt < meminfo_writesize)                // 若本次获取数据的量小于写入的大小,则跳出break;}if (IS_NANDWRITE && cnt != 0) {         // 填满了整个MTD,但是我们在输入时达到EOF了吗/* We filled entire MTD, but did we reach EOF on input? */if (full_read(STDIN_FILENO, filebuf, meminfo_writesize) != 0) {/* no */bb_error_msg_and_die("not enough space in MTD device");}}if (ENABLE_FEATURE_CLEAN_UP) {free(filebuf);close(fd);}return EXIT_SUCCESS;
}

标准输出输出定义

#define  STDIN_FILENO   0  /* Standard input.  */
#define  STDOUT_FILENO  1  /* Standard output.  */
#define  STDERR_FILENO  2  /* Standard error output.  */

检测坏块实现

static unsigned next_good_eraseblock(int fd, struct mtd_info_user *meminfo,unsigned block_offset)
{while (1) {            // 循环检测,便于跳过坏块loff_t offs;if (block_offset >= meminfo->size) {                // 1、传入的块偏移量大于等于mtd总大小if (IS_NANDWRITE)bb_error_msg_and_die("not enough space in MTD device");return block_offset; /* let the caller exit */    // 返回}offs = block_offset;if (xioctl(fd, MEMGETBADBLOCK, &offs) == 0)        // 2、判断是否为坏块return block_offset;                           // 若不是返回地址/* ioctl returned 1 => "bad block" */if (IS_NANDWRITE)                                    // 若是坏块跳过,并指向下一块的地址检测printf("Skipping bad block at 0x%08x\n", block_offset);block_offset += meminfo->erasesize;}
}

nandwrite 实现流程图

在这里插入图片描述

这篇关于[ Linux Busybox ] nandwrite 命令解析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/369829

相关文章

在Linux中改变echo输出颜色的实现方法

《在Linux中改变echo输出颜色的实现方法》在Linux系统的命令行环境下,为了使输出信息更加清晰、突出,便于用户快速识别和区分不同类型的信息,常常需要改变echo命令的输出颜色,所以本文给大家介... 目python录在linux中改变echo输出颜色的方法技术背景实现步骤使用ANSI转义码使用tpu

linux hostname设置全过程

《linuxhostname设置全过程》:本文主要介绍linuxhostname设置全过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录查询hostname设置步骤其它相关点hostid/etc/hostsEDChina编程A工具license破解注意事项总结以RHE

PostgreSQL的扩展dict_int应用案例解析

《PostgreSQL的扩展dict_int应用案例解析》dict_int扩展为PostgreSQL提供了专业的整数文本处理能力,特别适合需要精确处理数字内容的搜索场景,本文给大家介绍PostgreS... 目录PostgreSQL的扩展dict_int一、扩展概述二、核心功能三、安装与启用四、字典配置方法

Linux中压缩、网络传输与系统监控工具的使用完整指南

《Linux中压缩、网络传输与系统监控工具的使用完整指南》在Linux系统管理中,压缩与传输工具是数据备份和远程协作的桥梁,而系统监控工具则是保障服务器稳定运行的眼睛,下面小编就来和大家详细介绍一下它... 目录引言一、压缩与解压:数据存储与传输的优化核心1. zip/unzip:通用压缩格式的便捷操作2.

深度解析Java DTO(最新推荐)

《深度解析JavaDTO(最新推荐)》DTO(DataTransferObject)是一种用于在不同层(如Controller层、Service层)之间传输数据的对象设计模式,其核心目的是封装数据,... 目录一、什么是DTO?DTO的核心特点:二、为什么需要DTO?(对比Entity)三、实际应用场景解析

深度解析Java项目中包和包之间的联系

《深度解析Java项目中包和包之间的联系》文章浏览阅读850次,点赞13次,收藏8次。本文详细介绍了Java分层架构中的几个关键包:DTO、Controller、Service和Mapper。_jav... 目录前言一、各大包1.DTO1.1、DTO的核心用途1.2. DTO与实体类(Entity)的区别1

Java中的雪花算法Snowflake解析与实践技巧

《Java中的雪花算法Snowflake解析与实践技巧》本文解析了雪花算法的原理、Java实现及生产实践,涵盖ID结构、位运算技巧、时钟回拨处理、WorkerId分配等关键点,并探讨了百度UidGen... 目录一、雪花算法核心原理1.1 算法起源1.2 ID结构详解1.3 核心特性二、Java实现解析2.

Linux中SSH服务配置的全面指南

《Linux中SSH服务配置的全面指南》作为网络安全工程师,SSH(SecureShell)服务的安全配置是我们日常工作中不可忽视的重要环节,本文将从基础配置到高级安全加固,全面解析SSH服务的各项参... 目录概述基础配置详解端口与监听设置主机密钥配置认证机制强化禁用密码认证禁止root直接登录实现双因素

使用Python绘制3D堆叠条形图全解析

《使用Python绘制3D堆叠条形图全解析》在数据可视化的工具箱里,3D图表总能带来眼前一亮的效果,本文就来和大家聊聊如何使用Python实现绘制3D堆叠条形图,感兴趣的小伙伴可以了解下... 目录为什么选择 3D 堆叠条形图代码实现:从数据到 3D 世界的搭建核心代码逐行解析细节优化应用场景:3D 堆叠图

深度解析Python装饰器常见用法与进阶技巧

《深度解析Python装饰器常见用法与进阶技巧》Python装饰器(Decorator)是提升代码可读性与复用性的强大工具,本文将深入解析Python装饰器的原理,常见用法,进阶技巧与最佳实践,希望可... 目录装饰器的基本原理函数装饰器的常见用法带参数的装饰器类装饰器与方法装饰器装饰器的嵌套与组合进阶技巧