[ Linux Busybox ] nandwrite 命令解析

news2025/1/12 12:29:57

文章目录

    • 相关结构体
    • 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/mtd2
        opt_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/mtd3
    xioctl(fd, MEMGETINFO, &meminfo);    // 获取内存信息

    mtdoffset = xstrtou(opt_s, 0);            // 获取mtd偏移量,默认为0
    if (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);    // 从获取标准输入中获取数据,大小为内存写入的大小,数据保存在filebuf
        if (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 实现流程图

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Proteus仿真】【STM32单片机】水质监测报警系统设计

文章目录 一、功能简介二、软件设计三、实验现象联系作者 一、功能简介 本项目使用Proteus8仿真STM32单片机控制器&#xff0c;使用按键、LED、蜂鸣器、LCD1602、PCF8591 ADC、PH传感器、浑浊度传感器、DS18B20温度传感器、继电器模块等。 主要功能&#xff1a; 系统运行后&a…

汽车标定技术(七)--基于模型开发如何生成完整的A2L文件(2)

目录 1. 自定义ASAP2文件 2. asap2userlib.tlc需要修改的部分 3. 标定量观测量地址替换 3.1 由elf文件替换 3.2 由map文件替换 3.3 正则表达式&#xff08;含asap2post.m修改方法&#xff09; 4.小结 书接上文汽车标定技术(五)--基于模型开发如何生成完整的A2L文件(1)-C…

考研408-计算机网络 第二章-物理层学习笔记及习题

第二章 物理层 一 通信基础 1.1 物理层基本概念 1.1.1 认识物理层 物理层目的&#xff1a;解决如何在连接各种计算机的传输媒体上传输数据比特流&#xff0c;而不是具体的传输媒体。 物理层主要任务&#xff1a;确认与传输媒体接口有关的一些特性&#xff0c;需要进行定义标…

Kafka(消息队列)--简介

1、kafka&#xff1a; 是一个高吞吐的分布式消息系统&#xff0c;与Hdfs比较相似&#xff0c;但是与hdfs的区别是在于hdfs是存储的是历史的、海量的数据&#xff0c;然而kafka存储的是实时的、最新的数据。 2、消息队列&#xff1a; 指的是在Kafka中的数据队列。可以存放数据在…

华为数通方向HCIP-DataCom H12-831题库(多选题:141-160)

第141题 项目实施方案主要包含哪些项? A、风险控制措施 B、人员安排 C、时间规划 D、割接流程 答案:ABCD 解析: 项目实施方案主要包括:时间规划、人员安排、割接流程、风险控制措施。 第142题 以下哪些广域网链路属于报文交换网? A、PSIN B、ISDN C、X.25 D、ATM E、MSTP…

大语言模型(LLM)综述(六):大型语言模型的基准和评估

A Survey of Large Language Models 前言7 CAPACITY AND EVALUATION7.1 基本能力7.1.1 语言生成7.1.2 知识利用7.1.3 复杂推理 7.2 高级能力7.2.1 人类对齐7.2.2 与外部环境的交互7.2.3 工具操作 7.3 基准和评估方法7.3.1 综合评价基准7.3.2 评估方法 7.4 实证评估7.4.1 实验设…

数字化工厂的实施挑战与解决方法

随着科技的快速发展&#xff0c;数字化工厂成为现代制造业的重要趋势之一。数字化工厂通过整合信息技术和制造技术&#xff0c;可以提高生产效率、降低成本、优化资源利用&#xff0c;同时也给企业带来了更多的挑战。 1. 技术挑战 数字化工厂实施过程中最大的挑战之一是技术方…

运动耳机怎么选?精选五款优秀的运动耳机分享

​对于喜欢运动的我来说&#xff0c;不管是在室内健身&#xff0c;还是在户外跑步&#xff0c;我都会边运动边听歌&#xff0c;运动耳机里放着自己喜欢的音乐会很放松。然而&#xff0c;市面上的运动耳机品牌和型号众多&#xff0c;质量参差不齐。今天&#xff0c;我们就为大家…

ZKP8.2 FRI (Univariate) Polynomial Commitment

ZKP学习笔记 ZK-Learning MOOC课程笔记 Lecture 8: FRI-based Polynomial Commitments and Fiat-Shamir (Justin Thaler) 8.2 FRI (Univariate) Polynomial Commitment Recall: Univariate Polynomial Commitments Initial Attempt from Lecture 4 (Merkle Tree) Fixing…

AI智能助理源码系统+智能创作 带完整搭建教程

大家好啊&#xff0c;罗峰今天来给大家介绍一款AI智能助理源码系统。 以下是部分核心功能实现代码模块 系统特色功能一览&#xff1a; 高效便捷&#xff1a;AI智能助理能够帮助用户完成各种任务&#xff0c;无论是写作、论文、代码、小说、创意策划&#xff0c;还是Excel表格…

【Node.js入门】1.1Node.js 简介

Node.js入门之—1.1Node.js 简介 文章目录 Node.js入门之—1.1Node.js 简介什么是 Node.js错误说法 Node.js 的特点跨平台三方类库自带http服务器非阻塞I/O事件驱动单线程 Node.js 的应用场合适合用Node.js的场合不适合用Node.js的场合弥补Node.js不足的解决方案 什么是 Node.j…

如何利用ChannelPipeline在Netty中搭建无懈可击的数据处理流水线?

在上篇文章&#xff08;Netty 入门 — ChannelHandler&#xff0c; Netty 的数据加工厂&#xff09;提到 ChannelHandler 虽然是一个好的打工人&#xff0c;但是在我们实际业务线中&#xff0c;他不可能一个人干所有的活啊&#xff0c;毕竟都 21 世纪了&#xff0c;我们是要讲究…

通过对多个思维链进行元推理来回答问题11.8

通过对多个思维链进行元推理来回答问题 摘要1 引言2 背景3 方法3.1 生成推理链3.2 对推理链进行元推理 4 实验4.1 实验设置4.1.1 数据集4.1.2 方法 摘要 现代的多跳问题回答&#xff08;QA&#xff09;系统通常将问题分解为一系列推理步骤&#xff0c;称为思维链&#xff08;C…

基于SSM的旅游管理系统的设计与实现

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

el-select下拉框默认显示全部选项

这里一直不默认显示全部,但这部分代码是正确的, 经过查询发现: 所以找到了这段: 发现之前role_id和status是 修改成0后就可以默认选中全部选项了

程序设计:控制台输出二叉树 二叉树的形象显示

本文指导你编写一个输出到字符控制台的形象的二叉树展示。 目录 一般的Tree显示方式 理想的显示方式 实现方法 计算显示位置 输出数据 计算显示位置的代码 输出数据的代码 一般的Tree显示方式 编写二叉树算法时调试是很头疼的&#xff0c;如何显示成一目了然的树结构呢…

C/C++轻量级并发TCP服务器框架Zinx-游戏服务器开发003:架构搭建-需求分析及TCP通信方式的实现

文章目录 1 项目总体架构2 项目需求2.1 服务器职责2.2 消息的格式和定义 3 基于Tcp连接的通信方式3.1 通道层实现GameChannel类3.1.1 TcpChannel类3.1.2 Tcp工厂类3.1.3 创建主函数&#xff0c;添加Tcp的监听套接字3.1.4 代码测试 3.2 协议层与消息类3.2.1 消息的定义3.2.2 消息…

算法训练 第六周

一、用栈实现队列 本题要求我们使用栈这个数据结构来模拟实现队列的各种操作&#xff0c;我们的具体思路是使用两个栈&#xff0c;将一个栈当作输入栈&#xff0c;用于压入 push传入的数据&#xff1b;另一个栈当作输出栈&#xff0c;用于 pop和 peek 操作。每次 pop 或 peek 时…

CHOME、EDGE无法打开网页问题处理方法

最近有多位同时反馈CHOME/EDGE无法打开网页&#xff0c;如下提示 右键点CHOME/EDGE图标&#xff0c;在属性-兼容性&#xff0c;勾上以兼容模式运行这个程序&#xff0c;确定即可。

【Verilog 教程】7.4Verilog CIC 滤波器设计

积分梳状滤波器&#xff08;CIC&#xff0c;Cascaded Integrator Comb&#xff09;&#xff0c;一般用于数字下变频&#xff08;DDC&#xff09;和数字上变频&#xff08;DUC&#xff09;系统。CIC 滤波器结构简单&#xff0c;没有乘法器&#xff0c;只有加法器、积分器和寄存器…