在文件系统中,当需要执行写操作时,肯定是需要查找需要写入的块。那么如何查找块哪?

在Ext4系统中,有两个函数是可能执行查找操作的:ext4_getblk()、ext4_get_block()。

其中ext4_getblk(),原型如下:

structbuffer_head *ext4_getblk(handle_t *handle, struct inode *inode,

ext4_lblk_tblock, int create, int *errp)

该函数的主要功能:基于给定的inode,查找或创建block,并返回与其映射的buffer;create标识用于表示当查找不到时,是否执行分配块操作。

本节我们不关注块分配操作;因此create为0。

ext4_get_block()函数原型如下:

intext4_get_block(struct inode *inode, sector_t iblock,

struct buffer_head *bh, int create)

该函数的主要功能:基于给定的inode,查找文件的逻辑块iblock,并将其与bh进行映射,如果没有查找到,若create为1,则执行块分配操作为文件申请新的block,然后再与bh建立映射。这个函数用于适配VFS层接口。

无论ext4_getblk()还是ext4_get_block(),都是先初始化structext4_map_blocks map,然后调用函数ext4_map_blocks(),实现块查找功能。

先来看一下ext4_map_blocks这个结构体:

struct ext4_map_blocks{

ext4_fsblk_t m_pblk;

ext4_lblk_t m_lblk;

unsigned int m_len;

unsigned int m_flags;

};

在分析ext4_map_blocks结构体之前,我们先来熟悉两个概念:物理块号、逻辑块号。对于一个存储设备(这个设备是可以是存储分区)格式化为某个文件系统的过程中,一般,都会给予给定的block值对整个存储分区划分很多个大小为block size的单元,每个单元,我们都称之为块,每一个块都有一个块号,是以存储分区起始位置为偏移计算的,我们一般称之为物理块号。当某一个块被分配给某一个文件用来存放Data后,那么这个块给予文件起始位置又被赋予了一个块号,这个块号只对于该文件有效,这个块号我们一般称之为逻辑块号。

ext4_map_blocks结构体主要用来描述需要映射的空间信息,包含映射空间的物理块号,逻辑块号,空间大小,属性标识。在查找之前,一般都是知道逻辑块号和空间大小的。

ext4_map_blocks()的原型如下:

intext4_map_blocks(handle_t *handle, struct inode *inode,

struct ext4_map_blocks *map, int flags)

1.首先调用ext4_es_lookup_extent()查找extent status tree,extent status tree是描述extent在内存中信息与状态的树,可以理解为extent
tree在内存的中组织。在查找extent status tree的过程中,先检查cache extents是否满足条件,否则遍历整个extent status tree。

2.如果在缓存中没有查找到,那么我们就需要从磁盘上读出extent tree数据,然后查找。实现函数为ext4_ext_find_extent()。那么是如何实现的哪?

说起来,也十分的简单。

由前述[磁盘布局extent tree]可知,extent tree在磁盘上都是以extent block为单位存储的。

首先从inode中找出根节点,在根节点中的四个entry中通过二分法查找包含指定block的entry,然后通过该entry信息读取下一级extent
block到buffer中。该buffer中包含很多entry,继续通过二分法查找包含指定block的enrty,然后再读取下一级entry,依次类推;直至到叶子节点或无下级的节点。

从而我们得到与制定block最相近的extent(若查找不到,则返回小于block的最大的extent,即左侧的extent).

假如寻找到合适的包含block块号的extent,则计算出对应的物理block和在这个extent中以block为起始点到extent终点的长度(如果这个长度大约所要求的长度,则取所要求的长度)。

out:

if (allocated > map->m_len)

allocated = map->m_len;

ext4_ext_show_leaf(inode, path);

map->m_flags |= EXT4_MAP_MAPPED;

map->m_pblk = newblock;

map->m_len = allocated;

如果没有找到块,因为之前的Create标识已经说明即使没有找到,也不需要创建,所以直接返回0.

OK,块查找操作分析结束。

作者:Younger Liu,

本作品采用知识共享署名-非商业性使用-相同方式共享 3.0 未本地化版本许可协议进行许可。

[ext4]空间管理 - 查找块的更多相关文章

  1. [ext4]空间管理 - 与分配相关的关键数据结构

     在块分配机制中,涉及到几个主要的数据结构. 通过ext4_allocation_request描述块请求,然后基于块查找结果即上层需求来决定是否执行块分配操作. 在分配过程中,为了更好执行分配, ...

  2. [ext4]空间管理 - 分配机制

     在Ext4系统中,存在很多分配策略,比如预分配.多块分配.延迟分配等   Prealloc预分配 在ext4系统中,对于小文件和大文件的空间申请请求,都有不同的分配策略.对用小文件的空间请求,e ...

  3. [ext4]13 空间管理 - Prealloc分配机制

     作者:Younger Liu, 本作品采用知识共享署名-非商业性使用-相同方式共享 3.0 未本地化版本许可协议进行许可. 在ext4系统中,对于小文件和大文件的空间申请请求,都有不同的分配策略 ...

  4. Oracle表空间管理

    oracle表空间相关常用命令小结: 1.ALTER DATABASE SET DEFAULT BIGFILE TABLESPACE;              //修改表空间数据文件类型 2.ALT ...

  5. 本地管理表空间(LMT)与自动段空间管理(ASSM)概念

    创建表空间时,extent management local 定义本地管理表空间(LMT),segment space management auto 定义自动段空间管理(ASSM). extent ...

  6. ORACLE表空间管理维护

    1:表空间概念 在ORACLE数据库中,所有数据从逻辑结构上看都是存放在表空间当中,当然表空间下还有段.区.块等逻辑结构.从物理结构上看是放在数据文件中.一个表空间可由多个数据文件组成. 如下图所示, ...

  7. ORACLE表空间管理方式segment和extent

    A permanent tablespace contains persistent schema objects. Objects in permanent tablespaces are stor ...

  8. 【Oracle】表空间管理

    --表空间管理为主.附带 权限管理.数据字典 /* 表空间是逻辑结构,数据文件是物理结构 一个表空间对应多个段segment 段可以对应多个数据文件.跨磁盘 一个段对应多个盘区 extent 一个盘区 ...

  9. Oracle数据库体系结构(7) 表空间管理1

    表空间是Oracle数据库最大的逻辑存储结构,有一系列段构成.Oracle数据库对象存储结构的管理主要是通过表空间的管理实现的. 1.表空间的分类 表空间根据存储类型不同分为系统表空间和非系统表空间 ...

随机推荐

  1. SQL中PIVOT和UNPIVOT行列转换

    DECLARE @sql_col VARCHAR(8000); DECLARE @sql_str VARCHAR(8000); DECLARE @sql_ VARCHAR(MAX); SELECT @ ...

  2. 学习ASP.NET MVC(十一)——分页

    在这一篇文章中,我们将学习如何在MVC页面中实现分页的方法.分页功能是一个非常实用,常用的功能,当数据量过多的时候,必然要使用分页.在今天这篇文章中,我们学习如果在MVC页面中使用PagedList. ...

  3. UWP--数据绑定的几种方式

    1.后台代码:   2.后台定义属性,前台XAML 中绑定:     3.XAML 中定义资源并应用(资源中自定义对象):       4.用元素值绑定:

  4. android 透明状态栏方法及其适配键盘上推(二)

    在上一篇文章中介绍了一种设置透明状态栏及其适配键盘上推得方法.但是上一篇介绍的方法中有个缺点,就是不能消除掉statusbar的阴影.很多手机如(三星,Nexus都带有阴影).即使我用了: <a ...

  5. 纪念一下java学习之第一个get请求。

    环境,变量及工具: 1.java jdk1.8.X 2.tomcat 8.5.11 3. 环境变量: CATALINA_HOME: tomcat安装地址   JAVA_HOME:  JDK安装地址   ...

  6. 【VB超简单入门】五、基本输出输入

    之前讲了VB IDE的基本操作和概念,接下来要开始将VB语言的编程了. 程序最重要的部分是输出和输入,输入数据,经过计算机处理,再输出结果.本文将介绍两种最基本的输出输入方法,分别是Print.Msg ...

  7. Linux中的grep命令

    grep - print lines matching a pattern 参数: -a 将binary文件以text文件的方式查找 -i 忽略大小写 --color=zuto 加颜色匹配字符串 -v ...

  8. iOS开发寻找最近公共view

    新技能 #pragma mark --寻找最近公共view + (NSArray *)superViews:(UIView *)view{ if (view==nil) { return @[]; } ...

  9. centOS下调整swap

    [root@localhost /]# mkdir swap [root@localhost /]# cd swap [root@localhost swap]# dd if=/dev/zero of ...

  10. [原]node.js使用感想

    最近尝试了使用node.js,但因为不是太深入(小项目,还没做完),所以不能谈心得谈经验,就来谈谈使用感想. node.js和以往的cgi接口的服务器+cgi程序(如apache+phpmod)中的单 ...