yaffs2文件系统的工作流程和checkpoint机制

本文详细介绍了YAFFS2文件系统的挂载过程及关键步骤,包括初始化superblock和yaffs_param、扫描分区检查checkpoint机制、启动后台程序完成垃圾回收等。此外,还深入探讨了checkpoint机制的工作原理。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

      本文主要讲解一下一个具体的文件系统,yaffs2文件系统,当然针对需要深入研究的人或者说对文件系统完全不了解的读者可以先看一下以下两篇推荐文章。

http://www.ibm.com/developerworks/cn/linux/l-vfs/    解析 Linux 中的 VFS 文件系统机制

http://www.cnblogs.com/yuyijq/archive/2013/02/24/2923855.html       Linux虚拟文件系统(VFS)

      像所有的其它文件系统一样,yaffs2文件系统也会在mount的过程中挂载到VFS上并且初始化与设备相关的结构体变量,这样上可承VFS下可接MTD设备驱动程序。
      在yaffs2文件系统的mount过程中,主要完成了yaffs_internal_read_super_mtd函数中的功能,从函数名称我们可以看出函数完成了两个方面的功能,一个是读取了super的数据,另一个是读取了mtd的数据。现在具体介绍这个函数完成的功能,函数主要任务分为以下几个步骤:

  •       填充superblock和yaffs_param。
  •       yaffs_guts_initialise(dev);//扫描分区,检查是否需要checkpoint机制
  •       yaffs_bg_start(dev);//启动后台程序,完成垃圾回收

 1 填充superblock和yaffs_param  

         总体上来讲linux下的文件系统主要分为三大块:一是上层的文件系统调用,二是虚拟文件系统VFS,三是挂载到VFS中的各个实际文件系统。linux以一组通用对象的角度看待所有的文件系统。这些对象是superblock、inode、dentry和文件。超级块在每个文件系统的根上,超级块描述和维护文件系统的状态。文件系统中管理的每个对象(文件和目录)在linux中称为inode。dentry用来实现名称和inode之间的映射。dentry还维护目录和文件之间的关系,从而支持在文件系统中的移动。
      1)初始化与VFS的接口 
             sb->s_magic = YAFFS_MAGIC;
             sb->s_op = &yaffs_super_ops;   
             yaffs_dev_to_lc(dev)->super = sb;
             inode->i_op = &yaffs_dir_inode_operations;
             inode->i_fop = &yaffs_dir_operations;
       在yaffs2文件系统中魔数的值定义如下:
/* Give us a  Y=0x59,
 * Give us an A=0x41,
 * Give us an FF=0xff
 * Give us an S=0x53
 * And what have we got...
 */
#define YAFFS_MAGIC 0x5941ff53

      在函数中完成了以上的赋值过程,从而yaffs2文件系统能够成功的与VFS文件系统进行对接。

     2)初始化与MTD的接口

                param->total_bytes_per_chunk = mtd->writesize;
                param->chunks_per_block = mtd->erasesize / mtd->writesize;

                n_blocks = YCALCBLOCKS(mtd->size, mtd->erasesize);
                param->start_block = 0;
                param->end_block = n_blocks - 1;

                 yaffs_mtd_drv_install(dev);
                 param->sb_dirty_fn = yaffs_set_super_dirty;
                 param->gc_control_fn = yaffs_gc_control_callback;
                 param->skip_checkpt_rd = options.skip_checkpoint_read;
                 param->skip_checkpt_wr = options.skip_checkpoint_write;

2  yaffs_guts_initialise(dev);//扫描分区,检查是否需要checkpoint机制

yaffs_init_nand,初始化nand
yaffs_init_tmp_buffers,初始化临时buffer和cache
yaffs_init_blocks,初始化擦除block信息
yaffs_init_tnodes_and_objs初始化tnodes和objs
yaffs_create_initial_dir初始化跟目录
yaffs_summary_init初始化summary
yaffs2_checkpt_restore
         如果上面checkpoint的数据正确则直接进行load操作,调用
                {
                       yaffs_check_obj_details_loaded
基础设施搭建完毕,后面就是扫描flash在内存中建立完整的文件视图,YAFFS2首先尝试从checkpoint中恢复文件系统的信息, 
◇yaffs2_checkpt_open为读出checkpoint信息做一些初始化工作,首先创建一个checkpt_ buffer,初始化checkpt_block_list数组; 
◇yaffs2_rd_checkpt_validity_marker读取checkpoint头部信息,并判断是否正确; 
◇yaffs2_rd_checkpt_dev读取struct yaffs_checkpt_dev结构的信息,用来填充yaffs_dev结构相应的字段,以及flash上各个擦除块yaffs_block_info信息,还有chunk_bits表示flash上各个chunk位图; 
◇yaffs2_rd_checkpt_objs读取各个文件的信息,如果是目录就建立目录结构,如果是文件就建立Tnodes树; 
◇yaffs2_rd_checkpt_validity_marker读出checkpoint尾部信息,并判断是否正确; 
◇yaffs2_rd_checkpt_sum读出校验和,并和计算的比较是否一致。 

                   }
         否则,或者第一次启动则
         {
                     yaffs_deinit_blocks(dev);
                     yaffs_deinit_tnodes_and_objs(dev)
                      yaffs_init_blocks(dev))
                      yaffs_init_tnodes_and_objs(dev);
                      yaffs_create_initial_dir(dev)
                       yaffs2_scan_backwards(dev)
yaffs2_scan_backwards流程: 
如果上述5个步骤都没有错误,则从checkpoint挂载成功。否则就扫描整个flash,  ◇扫描整个flash OOB信息,按照bi->seq_number顺序对block进行排序; 
◇其次从seq_number最大的block开始往seq_number减小的方向扫描; 
◇最后从physical chunk index大的chunk开始,并向减小的方向扫描; 扫描的功能主要由yaffs2_scan_chunk函数完成: 
◇读取chunk的tags信息,根据tags.chunk_used判断该chunk是空闲的还是被使用的;根据tags.chunk_id判断是data chunk,还是object header 
◇如果是data chunk,分几种情况: a)根据tags.obj_id查找散列表,找到object对象,则把tags.chunk_id转换成文件内的偏移,拿这个偏移和文件对象的shrink_size比较,如果在文件的大小范围内则调用yaffs_put _chunk_in_file把该chunk加入到对应的tnodes tree中,否则删除该chunk;这一般发生在正常关闭文件的情况下,也即先扫描到object header,后扫描到文件的data chunk,这种情况文件的实际大小应该由扫描碰到的第一个object header所记录的大小决定; b)根据tags.obj_id查找散列表,不存在则创建新的object对象,并插入到散列表中,则把tags.chunk_id转换成文件内的偏移,拿这个偏移和文件对象的shrink_size比较,如果在文件的大小范围内则调用yaffs_put_chunk_in_file把该chunk加入到对应的tnodes tree中,否则删除该chunk;这一般发生在非正常关闭文件的情况下,也即先扫描到data chunk,后扫描到文件的object header,这种情况文件大小由扫描到的第一个data chunk决定。 ◇如果是object header,分几种情况: 
a)object header被扫描到,obiect散列表中并无对应的文件,那么根据tags.obj_id、tags. extra_obj_type或者tags.obj_id、oh->type创建新的object对象,并使用object header的信息初始化object对象,并根据oh->parent_obj_id创建父亲的object对象,把两者联系起来; b)object header被扫描到,obiect散列表中已经存在对应的object对象,且对象已经有关联的object header,说明这个object header是过时的,所以就直接删除该chunk;
          }

3  yaffs_bg_start(dev);//启动后台程序,完成垃圾回收


4 全局变量yaffs_auto_checkpoint

至于在文件系统的yaffs_auto_checkpoint,需要特别说明:
If 0 only checkpoint on unmount.
If 1 checkpoint on sync.
If 2 checkpoint on write_super and sync.
Default is 1.
也就是在默认值的情况下,只有在使用sync命令或者umount命令之后,文件系统才会将checkpoint信息写回。否则在每一次的启动过程中,设备都会重新扫描整个partition信息。
yaffs_sync_fs和yaffs_write_super函数都会调用到yaffs_do_sync_fs(sb, request_checkpoint)函数,而函数的第二个参数的值就是根据yaffs_auto_checkpoint是1还是 2决定。
在VFS机制中,yaffs_sync_fs函数是在fs/super.c文件中调用的,这个文件中的系统调用只有在sync和umount的过程中才会被调用到。而yaffs_write_super函数是在/mm/backing-dev.c文件中通过创建一个内核的守护线程完成的,当超级块中有脏数据时则进行sync_super的操作,最终将checkpoint信息进行restore。

<think>好的,用户问的是Yaffs的开机自检处理在哪里。首先,我需要回忆一下Yaffs文件系统的结构。Yaffs是专门为NAND闪存设计的文件系统,所以它的启动过程应该包括对NAND的检查。 用户之前问过关于`yaffs_check_chunk_erased`函数的解析,现在这个问题是关于开机自检的,可能是在启动时进行的检查。我需要确定Yaffs在挂载过程中会执行哪些自检步骤。 Yaffs的挂载过程通常在`yaffs_mount`或类似的函数中处理。这时候文件系统会扫描NAND设备,检查各个块的状态,可能包括坏块检测、元数据校验等。比如,在挂载时,Yaffs会遍历所有块,确认它们的元数据是否正确,并建立内部的数据结构。 另外,开机自检可能涉及检查超级块(superblock)是否存在,或者是否有有效的文件系统结构。Yaffs可能会在初始化时调用`yaffs_check_dev_fns`来验证设备函数的有效性,确保底层驱动正常工作。 还有可能在`yaffs_guts_initialise`函数中进行初始化操作,这里可能包括对NAND设备的检查,比如读取块的状态,确认擦除次数等。如果发现坏块,Yaffs会进行标记,避免后续使用。 用户可能想知道具体的代码位置,比如函数名或文件路径。需要指出主要的函数,比如`yaffs_mount`中的扫描过程,或者`yaffs_scan_backwards`函数,该函数在挂载时扫描所有块,检查它们的元数据数据状态。 另外,Yaffs在处理挂载时会调用`yaffs_checkpt_mount`来加载检查点,如果存在检查点,可能会进行恢复,这也算是一种自检机制。如果没有检查点,会进行完整的扫描,这可能包括对每个块的ECC检查,确保数据完整性。 需要确认Yaffs在挂载过程中是否调用了类似`yaffs_verify_blk`这样的函数来验证块的有效性。同时,`yaffs_check_alloc_available`可能会检查是否有足够的可用块,这也是启动时的一部分自检。 总结来说,Yaffs的开机自检主要集中在挂载阶段,通过扫描NAND设备,验证块的状态元数据,确保文件系统的一致性完整性。具体的代码实现可能在`yaffs_mount`、`yaffs_scan_backwards`以及相关的初始化函数中。</think>--- ### **YAFFS开机自检的核心逻辑位置** YAFFS文件系统的开机自检(初始化检查)主要在**挂载过程(mount)**中完成,具体代码逻辑集中在以下几个关键函数: --- #### **1. 挂载入口函数:`yaffs_mount()`** - **位置**:`yaffs_guts.c`(或类似文件,取决于具体实现版本) - **作用**:初始化NAND设备、扫描物理块结构、重建文件系统元数据。 - **核心操作**: ```c if (!yaffs_check_dev_fns(dev)) { // 检查设备驱动函数是否有效 /* 错误处理 */ } yaffs_init_nand_blocks(dev); // 初始化NAND块管理结构 yaffs_scan_backwards(dev); // 逆向扫描块以重建元数据 ``` --- #### **2. 块扫描与元数据重建:`yaffs_scan_backwards()`** - **位置**:`yaffs_guts.c` - **关键自检逻辑**: - **物理块状态检查**:调用`yaffs_check_chunk_erased`(即用户之前分析的函数)验证块是否已擦除。 - **ECC校验**:通过`yaffs_rd_chunk_tags_nand`读取数据时自动执行ECC校验。 - **元数据一致性检查**:解析每个块的`yaffs_ext_tags`,验证对象ID、序列号等是否合法。 - **坏块标记**:发现不可恢复错误时调用`yaffs_mark_bad`标记坏块。 --- #### **3. 检查点恢复(可选):`yaffs_checkpt_mount()`** - **位置**:`yaffs_checkptrw.c` - **作用**:若启用检查点(checkpoint)功能,尝试从检查点恢复文件系统状态。 - **自检行为**: ```c if (yaffs_checkpt_check_validity(dev)) { // 验证检查点有效性 yaffs_checkpt_restore(dev); // 恢复元数据 } else { yaffs_scan_backwards(dev); // 回退到完整扫描 } ``` --- ### **自检流程总结** | 步骤 | 操作 | 目的 | |------|------|------| | 1. **设备驱动验证** | 检查NAND驱动函数是否有效 | 确保底层操作(读/写/擦除)可用 | | 2. **块状态扫描** | 遍历所有NAND块,检查擦除状态、ECC错误 | 识别坏块、残留数据 | | 3. **元数据重建** | 解析块标签中的对象/文件信息 | 恢复目录树、文件索引 | | 4. **检查点恢复** | 加载最近保存的检查点(若存在) | 加速挂载过程,确保一致性 | --- ### **典型自检触发场景** 1. **系统启动时挂载YAFFS分区** ```c // 内核挂载调用示例 mount("/dev/mtdblock0", "/flash", "yaffs", 0, NULL); ``` 2. **热插拔后重新挂载设备** 3. **文件系统异常卸载后的恢复** --- ### **调试与日志** - **跟踪日志**:通过`yaffs_trace()`输出调试信息(需启用`YAFFS_TRACE_MOUNT`等标志)。 - **错误统计**:`dev->n_erase_failures`(擦除失败计数)、`dev->n_bad_chunks`(坏块计数)等计数器反映自检结果。 --- ### **设计特点** - **逆向扫描优化**:从最后写入的块向前扫描,快速定位最新有效元数据。 - **惰性擦除检查**:仅在挂载时验证关键块,运行时动态检查其他块。 - **软/硬容错结合**:通过ECC纠正位错误,坏块隔离处理硬件缺陷。 --- ### **性能权衡** - **时间开销**:全盘扫描可能较慢,检查点机制可显著优化。 - **可靠性优先**:即使延长挂载时间,也要确保元数据完整。 此设计体现了YAFFS对NAND闪存**不稳定存储介质特性**的深度适配,其自检机制是保障数据可靠性的核心环节。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值