操作系统实验九 proc文件系统的实现(哈工大李治军)(二)

简介: 操作系统实验九 proc文件系统的实现(哈工大李治军)(二)

让 proc 文件可读


文件位置:fs/read_write.c


添加extern,表示proc_read函数是从外部调用的

/*新增proc_read函数外部调用*/
extern int proc_read(int dev,char* buf,int count,unsigned long *pos);


然后在sys_read函数中仿照其他if语句,加上 S_IFPROC() 的分支,添加proc文件的proc_read()调用:

if (inode->i_pipe)
  return (file->f_mode&1)?read_pipe(inode,buf,count):-EIO;
  /*新增proc_read调用*/
  if (S_ISPROC(inode->i_mode))
  return proc_read(inode->i_zone[0],&file->f_pos,buf,count);
  if (S_ISCHR(inode->i_mode))
  return rw_char(READ,inode->i_zone[0],buf,count,&file->f_pos);


需要传给处理函数的参数包括:


inode->i_zone[0],这就是 mknod() 时指定的 dev ——设备编号

buf,指向用户空间,就是 read() 的第二个参数,用来接收数据

count,就是 read() 的第三个参数,说明 buf 指向的缓冲区大小

&file->f_pos,f_pos 是上一次读文件结束时“文件位置指针”的指向。这里必须传指针,因为处理函数需要根据传给 buf 的数据量修改 f_pos 的值。


实现 proc_read 函数


proc 文件的处理函数的功能是根据设备编号,把不同的内容写入到用户空间的 buf。写入的数据要从 f_pos 指向的位置开始,每次最多写 count 个字节,并根据实际写入的字节数调整 f_pos 的值,最后返回实际写入的字节数。当设备编号表明要读的是 psinfo 的内容时,就要按照 psinfo 的形式组织数据。


实现此函数可能要用到如下几个函数:


malloc() 函数

free() 函数

包含 linux/kernel.h 头文件后,就可以使用 malloc() 和 free() 函数。它们是可以被核心态代码调用的,唯一的限制是一次申请的内存大小不能超过一个页面。


文件位置:fs/proc.c


代码来自一位大佬的博客————参考代码

#include 
#include 
#include 
#include 
#include 
#include 
#define set_bit(bitnr,addr) ({ \
register int __res ; \
__asm__("bt %2,%3;setb %%al":"=a" (__res):"a" (0),"r" (bitnr),"m" (*(addr))); \
__res; })
char proc_buf[4096] ={'\0'};
extern int vsprintf(char * buf, const char * fmt, va_list args);
//Linux0.11没有sprintf(),该函数是用于输出结果到字符串中的,所以就实现一个,这里是通过vsprintf()实现的。
int sprintf(char *buf, const char *fmt, ...)
{
  va_list args; int i;
  va_start(args, fmt);
  i=vsprintf(buf, fmt, args);
  va_end(args);
  return i;
}
int get_psinfo()
{
  int read = 0;
  read += sprintf(proc_buf+read,"%s","pid\tstate\tfather\tcounter\tstart_time\n");
  struct task_struct **p;
  for(p = &FIRST_TASK ; p <= &LAST_TASK ; ++p)
  if (*p != NULL)
  {
    read += sprintf(proc_buf+read,"%d\t",(*p)->pid);
    read += sprintf(proc_buf+read,"%d\t",(*p)->state);
    read += sprintf(proc_buf+read,"%d\t",(*p)->father);
    read += sprintf(proc_buf+read,"%d\t",(*p)->counter);
    read += sprintf(proc_buf+read,"%d\n",(*p)->start_time);
  }
  return read;
}
/*
*  参考fs/super.c mount_root()函数
*/
int get_hdinfo()
{
  int read = 0;
  int i,used;
  struct super_block * sb;
  sb=get_super(0x301);  /*磁盘设备号 3*256+1*/
  /*Blocks信息*/
  read += sprintf(proc_buf+read,"Total blocks:%d\n",sb->s_nzones);
  used = 0;
  i=sb->s_nzones;
  while(--i >= 0)
  {
  if(set_bit(i&8191,sb->s_zmap[i>>13]->b_data))
    used++;
  }
  read += sprintf(proc_buf+read,"Used blocks:%d\n",used);
  read += sprintf(proc_buf+read,"Free blocks:%d\n",sb->s_nzones-used);
  /*Inodes 信息*/
  read += sprintf(proc_buf+read,"Total inodes:%d\n",sb->s_ninodes);
  used = 0;
  i=sb->s_ninodes+1;
  while(--i >= 0)
  {
  if(set_bit(i&8191,sb->s_imap[i>>13]->b_data))
    used++;
  }
  read += sprintf(proc_buf+read,"Used inodes:%d\n",used);
  read += sprintf(proc_buf+read,"Free inodes:%d\n",sb->s_ninodes-used);
  return read;
}
int get_inodeinfo()
{
  int read = 0;
  int i;
  struct super_block * sb;
  struct m_inode *mi;
  sb=get_super(0x301);  /*磁盘设备号 3*256+1*/
  i=sb->s_ninodes+1;
  i=0;
  while(++i < sb->s_ninodes+1)
  {
  if(set_bit(i&8191,sb->s_imap[i>>13]->b_data))
  {
    mi = iget(0x301,i);
    read += sprintf(proc_buf+read,"inr:%d;zone[0]:%d\n",mi->i_num,mi->i_zone[0]);
    iput(mi);
  }
  if(read >= 4000) 
  {
    break;
  }
  }
  return read;
}
int proc_read(int dev, unsigned long * pos, char * buf, int count)
{
  int i;
  if(*pos % 1024 == 0)
  {
  if(dev == 0)
    get_psinfo();
  if(dev == 1)
    get_hdinfo();
  if(dev == 2)
    get_inodeinfo();
  }
  for(i=0;i
  {
    if(proc_buf[i+ *pos ] == '\0')  
          break; 
    put_fs_byte(proc_buf[i+ *pos],buf + i+ *pos);
  }
  *pos += i;
  return i;
}


同时修改fs/Makefile文件:

OBJS= open.o read_write.o inode.o file_table.o buffer.o super.o \
  block_dev.o char_dev.o file_dev.o stat.o exec.o pipe.o namei.o \
  bitmap.o fcntl.o ioctl.o truncate.o proc.o
//......
### Dependencies:
proc.o : proc.c ../include/linux/kernel.h ../include/linux/sched.h \
  ../include/linux/head.h ../include/linux/fs.h ../include/sys/types.h \
  ../include/linux/mm.h ../include/signal.h ../include/asm/segment.h



编译运行


再次重新编译内核make all,然后运行内核,查看psinfo(当前系统进程状态信息)和hdinfo(硬盘信息)的信息:


天道酬勤


byez

目录
相关文章
|
2月前
|
算法
数据结构实验之操作系统打印机管理器问题
本实验旨在通过实现操作系统中的打印机管理器问题,掌握队列的基本操作如入队、出队等,利用队列的先进先出特性解决先申请先打印的问题。实验包括队列的初始化、入队、出队、打印队列内容等功能,并通过菜单式界面进行交互。实验结果显示基本功能可正常执行,但在连续操作时存在执行失败的情况,需进一步优化。
56 4
|
4月前
|
存储 算法 安全
操作系统之文件系统的奥秘
【9月更文挑战第19天】本文将深入探索操作系统中不可或缺的组件——文件系统,揭示其工作原理与实现细节。我们将通过浅显的语言和生动的比喻,一步步解析文件系统如何组织数据、管理存储空间,并确保数据的完整性和安全性。文章不仅适合初学者构建基础概念,也能帮助有经验的开发者更深入地理解文件系统的高级特性。
|
1月前
|
安全 Linux 数据安全/隐私保护
深入Linux操作系统:文件系统和权限管理
在数字世界的海洋中,操作系统是连接用户与硬件的桥梁,而Linux作为其中的佼佼者,其文件系统和权限管理则是这座桥梁上不可或缺的结构。本文将带你探索Linux的文件系统结构,理解文件权限的重要性,并通过实际案例揭示如何有效地管理和控制这些权限。我们将一起航行在Linux的命令行海洋中,解锁文件系统的奥秘,并学习如何保护你的数据免受不必要的访问。
|
2月前
|
存储 安全 大数据
深入浅出操作系统:文件系统的秘密
【10月更文挑战第35天】本文将揭示文件系统背后的奥秘,从其基本概念到复杂的实现机制。我们将一起探索文件系统的结构和原理,并了解它如何影响我们的日常计算体验。通过简单的例子和比喻,文章旨在使读者对文件系统有一个清晰而深刻的理解,就像甘地所言:“你必须成为你希望在世界上看到的改变。”让我们一起成为理解操作系统的先行者。
|
4月前
|
存储 缓存 文件存储
探索操作系统中的文件系统管理
【9月更文挑战第25天】在数字世界的海洋中,操作系统是指引我们航行的灯塔。它不仅管理着硬件资源,还维护着软件的秩序。本文将深入探讨操作系统中一个至关重要的部分——文件系统管理。我们将从基础概念出发,逐步深入到文件系统的设计与实现,最后通过代码示例来直观展示文件系统的操作。让我们一起揭开文件系统管理的神秘面纱,理解其背后的逻辑与奥秘。
|
5月前
|
编解码 Linux 程序员
深度探索Linux操作系统 —— 构建根文件系统2
深度探索Linux操作系统 —— 构建根文件系统
61 12
|
5月前
|
Linux Shell 网络安全
深度探索Linux操作系统 —— 构建根文件系统1
深度探索Linux操作系统 —— 构建根文件系统
73 6
|
5月前
|
存储 人工智能 数据管理
深入理解Linux操作系统之文件系统管理探索人工智能:从理论到实践的旅程
【8月更文挑战第30天】在探索Linux的无限可能时,我们不可避免地会遇到文件系统管理这一核心话题。本文将深入浅出地介绍Linux文件系统的基础知识、操作命令及高级技巧,帮助你更有效地管理和维护你的系统。从基础概念到实践应用,我们将一步步揭开Linux文件系统的神秘面纱。
|
6月前
|
弹性计算 运维
阿里云操作系统智能助手OS Copilot实验测评报告
**OS Copilot 产品体验与功能反馈摘要** 运维人员发现OS Copilot易上手,文档清晰,助其高效排查故障(8/10分)。愿意推荐并参与开源开发。亮点在于知识问答,能快速筛选答案。相较于竞品,优点是新手友好、文档清晰,但功能扩展性待增强。期望增加系统错误排查与解决方案,并集成ECS等,以优化系统安装流程。
阿里云操作系统智能助手OS Copilot实验测评报告
|
5月前
|
存储 算法 Unix
OS—文件系统
OS—文件系统
80 0