eMMC读写的数据流图

简介: eMMC读写的数据流图

前言

之前在《深入浅出MMC子系统》介绍了MMC子系统,从中可以窥探出eMMC读写的全流程,对于其中的数据流,了解它也非常重要。将其中的数据流抽出来分析,深入理解里面每一层使用的数据结构,会让我们对MMC的理解更为深刻。

640.png

存储基本知识

   先了解一些存储的基本概念,以我的开发板为例,存储介质为容量8G的eMMC。系统里输入fdisk -l可查看磁盘信息,红框的为8G eMMC的信息。

有两个用户分区,分别为p1和p2。

   对于p1分区,起始逻辑块为20480,终止逻辑块为282623。一共有262144(282623 - 20480 + 1)个逻辑块。一般eMMC每个块大小为512或1024字节,这里我的是512字节。所以p1分区大小为262144 * 512 = 134217728字节。134217728/1024/1024=128M。

640.png

   对于磁盘来说,存储的最小单位是扇区,但是在文件系统层进行I/O操作时以是块为单位的。

640.png

int _submit_bh(int rw, struct buffer_head *bh, unsigned long bio_flags)

   _submit_bh是fs/buffer.c的一个函数,第一个参数rw代表读写,对于buffer_head,用官方的话来解释。

在历史上,buffer_head被用来映射页面中的单个块,当然也作为I/O的单位通过文件系统和块层。现在,基本的I/O单元是bio, buffer_heads用于提取块映射(通过get_block_t调用),用于跟踪页面中的状态(通过page_mapping),以及出于向后兼容性的原因包装bio提交(例如submit_bh)。

struct buffer_head {
  unsigned long b_state; /* 缓冲区状态位图 */
  struct buffer_head *b_this_page;/* 页缓存的圆形链表 */
  struct page *b_page;    /* buffer映射到的页面 */
  sector_t b_blocknr; /* 起始块号 */
  size_t b_size; /* 映射大小 */
  char *b_data; /* 指向page里的数据 */
  struct block_device *b_bdev;
  bh_end_io_t *b_end_io; /* I/O completion */
  void *b_private; /* reserved for b_end_io */
  struct list_head b_assoc_buffers; /* associated with another mapping */
  struct address_space *b_assoc_map;  /* mapping this buffer is
               associated with */
  atomic_t b_count; /* 用户正在使用的buffer_head */};

   buffer_head描述的东西大概如下图

640.png

   前面提到,磁盘用扇区描述最小单元,文件系统用block描述最小单元。

bio->bi_iter.bi_sector = bh->b_blocknr * (bh->b_size >> 9);

这里将block乘以512转换成字节大小赋值给扇区描述,不过前面看到我板子上的eMMCsector大小也是512字节,但并不是所有磁盘的扇区大小都是512字节,所以这里是为了兼容,后面对磁盘进行传输时还需要再处理一步,也就是将sector再除以扇区大小单位。

bio->bi_iter.bi_sector = bh->b_blocknr * (bh->b_size >> 9);
bio->bi_io_vec[0].bv_page = bh->b_page;
bio->bi_io_vec[0].bv_len = bh->b_size;
bio->bi_io_vec[0].bv_offset = bh_offset(bh);
bio->bi_iter.bi_size = bh->b_size;
bio->bi_private = bh;

数据流经过

640.png

号主:一枚机械专业本科生,经历了转行,从外包逆袭到芯片原厂的Linux驱动开发工程师,深入操作系统的世界,贯彻终身学习、终身成长的理念。平时喜欢折腾,寒冬之下,抱团取暖,期待你来一起探讨技术、搞自媒体副业,程序员接单和投资理财。【对了,不定期送闲置开发板、书籍、键盘等等】。

如果你想了解我的转行经验,欢迎找我交流~gongzhong号【哆哆jarvis】

一起不断探索自我、走出迷茫、找到热爱,希望和你成为朋友,一起成长~

相关文章
|
小程序
uniapp如何分包 & 分包配置后无法读取static文件夹
uniapp如何分包 & 分包配置后无法读取static文件夹
889 0
uniapp如何分包 & 分包配置后无法读取static文件夹
|
Shell Linux
uboot启动流程源码分析(二)
uboot启动流程源码分析(二)
341 0
|
NoSQL Linux 编译器
Linux下交叉编译gdb,gdbserver+gdb的使用以及通过gdb调试core文件
交叉编译gdb和gdbserver 1、下载gdb:下载地址为:http://ftp.gnu.org/gnu/gdb/按照一般的想法,最新版本越好,因此下载7.2这个版本。当然,凡事无绝对。我们以gdb-7.2.tar.bz2 这个文件为例。
8401 1
|
数据可视化 数据挖掘 UED
Plotly交互式数据可视化
【10月更文挑战第12天】本文介绍了如何使用 Plotly 实现交互式数据可视化,涵盖从安装 Plotly 到数据准备、图表创建、添加交互功能、导出图表及自定义图表样式的全过程。通过具体示例,展示了如何创建和优化交互式折线图,提升数据分析的效率和趣味性。
639 60
|
存储 缓存 网络协议
深入理解Linux网络——内核是如何发送网络包的
一、相关实际问题 1. 查看内核发送数据消耗的CPU时应该看sy还是si 2. 在服务器上查看/proc/softirqs,为什么NET_RX要比NET_TX大得多 3. 发送网络数据的时候都涉及那些内存拷贝操作 4. 零拷贝到底是怎么回事 5. 为什么Kafka的网络性能很突出
|
Linux API 云计算
探索Linux操作系统的模块化设计
【6月更文挑战第12天】本文深入探讨了Linux操作系统的模块化设计,揭示了其背后的技术原理与实践应用。通过分析Linux内核的模块化特性,我们了解到这一设计如何促进了系统的灵活性、可维护性以及安全性。文章将详细阐述模块加载机制、模块间的通信方式以及模块化给系统开发和维护带来的影响。
|
缓存 算法 程序员
深入浅出MMC子系统
深入浅出MMC子系统
1643 0
深入浅出MMC子系统
|
Linux 调度 数据中心
Linux cgroup资源隔离各个击破之 - io隔离
Linux Cgroup blkio子系统的用法. blkio子系统支持两种IO隔离策略 .1. cfq io调度器,支持按权重分配IO处理的时间片,从而达到IO调度和限制的目的,权重取值范围100-1000。通过以下两个文件进行配置。 blkio.weight
11649 0
|
存储 Java 关系型数据库
分布式定时任务框架Quartz总结和实践(2)—持久化到Mysql数据库
本文主要介绍分布式定时任务框架Quartz集成SpringBoot持久化数据到Mysql数据库的操作,上一篇文章使用Quartz创建定时任务都是保存在内存中,如果服务重启定时任务就会失效,所以Quartz官方也提供将定时任务等信息持久化到Mysql数据库的功能,本文主要实现这种Quartz的这种使用方式。
2213 0
分布式定时任务框架Quartz总结和实践(2)—持久化到Mysql数据库