《Linux内核设计的艺术:图解Linux操作系统架构设计与实现原理》——2.10 初始化缓冲区管理结构

简介: 本节书摘来自华章计算机《Linux内核设计的艺术:图解Linux操作系统架构设计与实现原理》一书中的第2章,第2.10节,作者:新设计团队著, 更多章节内容可以访问云栖社区“华章计算机”公众号查看。

2.10 初始化缓冲区管理结构

缓冲区是内存与外设(如硬盘,以后以硬盘为例)进行数据交互的媒介。内存与硬盘最大的区别在于,硬盘的作用仅仅是对数据信息以很低的成本做大量数据的断电保存,并不参与运算(因为CPU无法到硬盘上进行寻址),而内存除了需要对数据进行保存以外,更重要的是要与CPU、总线配合进行数据运算。缓冲区则介于两者之间,它既对数据信息进行保存,也能够参与一些像查找、组织之类的间接、辅助性运算。有了缓冲区这个媒介以后,对外设而言,它仅需要考虑与缓冲区进行数据交互是否符合要求,而不需要考虑内存如何使用这些交互的数据;对内存而言,它也仅需要考虑与缓冲区交互的条件是否成熟,而不需要关心此时外设对缓冲区的交互情况。两者的组织、管理和协调将由操作系统统一操作。
操作系统通过hash_table[NR_HASH]、buffer_head双向环链表组成的复杂的哈希表管理缓冲区。
操作系统通过调用buffer_init()函数对缓冲区进行设置,执行代码如下:

//代码路径:init/main.c:
void main(void)
{
    …
    buffer_init(buffer_memory_end);
    …
}

在buffer_init()函数里,从内核的末端及缓冲区的末端同时开始,方向相对增长、配对地做出buffer_head、缓冲块,直到不足一对buffer_head、缓冲块。在第2章开始时设定的内存格局下,有3000多对buffer_head、缓冲块,buffer_head在低地址端,缓冲块在高地址端。
将buffer_head的成员设备号b_dev、引用次数b_count、“更新”标志b_uptodate、“脏”标志b_dirt、“锁定”标志b_lock设置为0。如图2-24所示,将b_data指针指向对应的缓冲块。利用buffer_head的b_prev_free、b_next_free,将所有的buffer_head形成双向链表。使free_list指向第一个buffer_head,并利用free_list将buffer_head形成双向链表链接成双向环链表,如图2-25所示。
注意图2-26顶部所示的内存的变化。在紧靠系统内核的部分,多出了一块用黑色表示的内存区域,那里面存储的就是缓冲区管理结构。由于它管理着3000多个缓冲块,因此它占用的内存空间的大小,与内核几乎差不多。图2-26中也对空闲表的双向链表结构给出了形象的说明。

image

最后,对hash_table[307]进行设置,将hash_table[307]的所有项全部设置为NULL,如图2-26第二步所示。

image

对应的代码如下:

//代码路径:fs/buffer.c:
    …
struct buffer_head * start_buffer= (struct buffer_head *) &end;
struct buffer_head * hash_table[NR_HASH];
static struct buffer_head * free_list;
    …
void buffer_init(long buffer_end)
{
    struct buffer_head * h= start_buffer;
    void * b;
    int i;

    if (buffer_end== 1<<20)
         b= (void *) (640*1024);
    else
         b= (void *) buffer_end;

//h、b分别从缓冲区的低地址端和高地址端开始,每次对进buffer_head、缓冲块各一个
//忽略剩余不足一对buffer_head、缓冲块的空间
    while ( (b -= BLOCK_SIZE) >= ((void *) (h + 1)) ) {
         h->b_dev= 0;
         h->b_dirt= 0;
         h->b_count= 0;
         h->b_lock= 0;
         h->b_uptodate= 0;
         h->b_wait= NULL;
         h->b_next= NULL;        //这两项初始化为空,后续的使用将与hash_table挂接
         h->b_prev= NULL;
         h->b_data= (char *) b;    //每个buffer_head关联一个缓冲块
         h->b_prev_free= h-1;    //这两项使buffer_head分别与前、
         h->b_next_free= h + 1;    // 后buffer_head挂接,形成双向链表
         h++;
         NR_BUFFERS++;
         if (b== (void *) 0x100000)    //避开ROMBIOS&VGA
               b= (void *) 0xA0000;
    }
    h--;
    free_list= start_buffer;        // free_list指向第一个buffer_head
    free_list->b_prev_free= h;        //使buffer_head双向链表
    h->b_next_free= free_list;        //形成双向环链表
    for (i=0;i<NR_HASH;i++)        //清空hash_table[307]
         hash_table[i]=NULL;
}

注意看代码中struct buffer_head * h = start_buffer这一行。这一行中的start_buffer确定了缓冲区的起始位置,这也就回答了2.2节中关于缓冲区起始点位置的这个问题。它是在buffer.c中定义的:

struct buffer_head * start_buffer= (s truct buffer_head *) &end;

这个end就是内核代码末端的地址。在代码编写阶段,设计者事先较难准确估算这个地址,于是就在内核模块链接期间设置end这个值,然后在这里使用。

相关文章
|
7月前
|
安全 网络协议 Linux
深入理解Linux内核模块:加载机制、参数传递与实战开发
本文深入解析了Linux内核模块的加载机制、参数传递方式及实战开发技巧。内容涵盖模块基础概念、加载与卸载流程、生命周期管理、参数配置方法,并通过“Hello World”模块和字符设备驱动实例,带领读者逐步掌握模块开发技能。同时,介绍了调试手段、常见问题排查、开发规范及高级特性,如内核线程、模块间通信与性能优化策略。适合希望深入理解Linux内核机制、提升系统编程能力的技术人员阅读与实践。
682 1
|
7月前
|
Ubuntu Linux
Ubuntu 23.04 用上 Linux 6.2 内核,预计下放到 22.04 LTS 版本
Linux 6.2 带来了多项内容更新,修复了 AMD 锐龙处理器设备在启用 fTPM 后的运行卡顿问题,还增强了文件系统。
|
7月前
|
Ubuntu Linux
Ubuntu 23.10 现在由Linux内核6.3提供支持
如果你想在你的个人电脑上测试一下Ubuntu 23.10的最新开发快照,你可以从官方下载服务器下载最新的每日构建ISO。然而,请记住,这是一个预发布版本,所以不要在生产机器上使用或安装它。
|
7月前
|
监控 Ubuntu Linux
什么Linux,Linux内核及Linux操作系统
上面只是简单的介绍了一下Linux操作系统的几个核心组件,其实Linux的整体架构要复杂的多。单纯从Linux内核的角度,它要管理CPU、内存、网卡、硬盘和输入输出等设备,因此内核本身分为进程调度,内存管理,虚拟文件系统,网络接口等4个核心子系统。
464 0
|
7月前
|
Web App开发 缓存 Rust
|
7月前
|
Ubuntu 安全 Linux
Ubuntu 发行版更新 Linux 内核,修复 17 个安全漏洞
本地攻击者可以利用上述漏洞,攻击 Ubuntu 22.10、Ubuntu 22.04、Ubuntu 20.04 LTS 发行版,导致拒绝服务(系统崩溃)或执行任意代码。
|
6月前
|
Linux 应用服务中间件 Shell
二、Linux文本处理与文件操作核心命令
熟悉了Linux的基本“行走”后,就该拿起真正的“工具”干活了。用grep这个“放大镜”在文件里搜索内容,用find这个“探测器”在系统中寻找文件,再用tar把东西打包带走。最关键的是要学会使用管道符|,它像一条流水线,能把这些命令串联起来,让简单工具组合出强大的功能,比如 ps -ef | grep 'nginx' 就能快速找出nginx进程。
735 1
二、Linux文本处理与文件操作核心命令
|
6月前
|
Linux
linux命令—stat
`stat` 是 Linux 系统中用于查看文件或文件系统详细状态信息的命令。相比 `ls -l`,它提供更全面的信息,包括文件大小、权限、所有者、时间戳(最后访问、修改、状态变更时间)、inode 号、设备信息等。其常用选项包括 `-f` 查看文件系统状态、`-t` 以简洁格式输出、`-L` 跟踪符号链接,以及 `-c` 或 `--format` 自定义输出格式。通过这些选项,用户可以灵活获取所需信息,适用于系统调试、权限检查、磁盘管理等场景。
435 137
|
6月前
|
安全 Ubuntu Unix
一、初识 Linux 与基本命令
玩转Linux命令行,就像探索一座新城市。首先要熟悉它的“地图”,也就是/根目录下/etc(放配置)、/home(住家)这些核心区域。然后掌握几个“生存口令”:用ls看周围,cd去别处,mkdir建新房,cp/mv搬东西,再用cat或tail看文件内容。最后,别忘了随时按Tab键,它能帮你自动补全命令和路径,是提高效率的第一神器。
1141 59