这个系列文章将讲述文件系统的安装过程,以函数f2fs_fill_super为起点。
f2fs_fill_super:通过调用sb_set_blocksize来检查和设置super_block的块大小字段和快设备的块大小。然后调用read_raw_super_block来对物理设备上的f2fs_super_block进行读取。接下来就可以对f2fs_sb_info的某些字段进行初始化了。init_sb_info函数用刚才读取的f2fs_super_block对f2fs_sb_info进行初始化。init_percpu_info对sbi的几个统计的数据进行了per cpu的初始化。然后是对f2fs_sb_info的meta_inode进行赋值,这个字段是f2fs用来管理元数据page cache的inode,所以它在磁盘上面起始时没有具体的存在形式的,所以这里的初始化通过调用函数f2fs_iget仅仅inode->i_mapping->a_ops =&f2fs_meta_aops对相应的aops赋值了。接下来就是通过get_valid_checkpoint读取最新版本的f2fs_checkpoint。接着利用读取的f2fs_checkpoint对f2fs_sb_info再次进行某些字段的初始化。然后对管理文件系统脏的inode的链表inode_list和相关锁inode_lock进行初始化。这里的链表的个数是NR_INODE_TYPE,分别对应DIR_INODE/FILE_INODE/DIRTY_META三种类型。接下来调用函数init_extent_cache_info来完成管理extent信息的结构。然后函数init_ino_entry_info初始化ino_management的信息和最大孤儿节点个数的信息。接着调用build_segment_manager构造文件系统在内存中的segment的管理结构f2fs_sm_info。然后调用build_node_manager构造文件系统在内存中的node的管理结构f2fs_nm_info。接着调用build_gc_manager来初始化gc_manager,主要是确定select_victim的函数。然后是对f2fs_sb_info的node_inode进行赋值,同meta_inode,磁盘上没有具体的存在形式,所以只是对aops的赋值。然后调用f2fs_join_shrinker将新安装的文件系统的实例加入到f2fs这种文件系统的链表中。接着调用recover_orphan_inodes来将orphan inode中的所有的inode全部删除掉。接着调用函数f2fs_iget来读取root inode,然后d_make_root来构造root inode的目录项。f2fs_build_stats 来构造统计信息f2fs_stat_info。接着检查手否设置标志DISABLE_ROLL_FORWARD来决定是否进行前滚操作,这个都是通过函数recover_fsync_data来完成的。如果设置了BG_GC并且文件系统不是只读的,那么就调用start_gc_thread来启动gc线程。最后根据前面读取f2fs_super_block是否有问题来决定是否调用函数f2fs_commit_super来修复f2fs_super_block。最后更新f2fs_sb_info的CP_TIME和REQ_TIME时间。
static int f2fs_fill_super(struct super_block *sb, void *data, int silent)
{
struct f2fs_sb_info *sbi;
struct f2fs_super_block *raw_super;
struct inode *root;
int err;
bool retry = true, need_fsck = false;
char *options = NULL;
int recovery, i, valid_super_block;
struct curseg_info *seg_i;
try_on