10.Lab Nine —— file system-下

news2025/1/16 11:11:15

 Symbolic links

添加符号链接

1.添加有关symlink系统调用的定义声明,包括kernel/syscall.h, kernel/syscall.c, user/usys.pluser/user.h.

2.添加新的文件类型T_SYMLINK到kernel/stat.h中,添加新的文件标识位O_NOFOLLOW到kernel/fcntl.h中

3.在kenel/sysfile.c中实现sys_symlink()函数

该函数用于生成符号链接,该符号链接相当于一个特殊的独立文件,里面存储的是目标文件的路径

4.修改kernel/sysfile中的sys_open()函数

该函数可以用来打开文件,对于符号链接一般情况下需要打开的是其目标文件,因此需要额外处理

(此处需要注意避免成环问题的出现)

/* 系统调用open的实现 */
uint64
sys_open(void)
{
  /* 路径名缓冲区 */
  char path[MAXPATH];
  /* 文件描述符和打开模式 */
  int fd, omode;
  /* 文件结构指针 */
  struct file *f;
  /* i节点指针 */
  struct inode *ip;
  /* 临时变量,用于存储函数调用返回值 */
  int n;

  /* 获取路径名和打开模式,如果出错则返回-1 */
  if((n = argstr(0, path, MAXPATH)) < 0 || argint(1, &omode) < 0)
    return -1;

  /* 开始一个文件操作 */
  begin_op();

  /* 如果需要创建文件 */
  if(omode & O_CREATE){
    /* 调用create函数创建文件,如果失败则返回-1 */
    ip = create(path, T_FILE, 0, 0);
    if(ip == 0){
      end_op();
      return -1;
    }
  } else {
    int symlink_depth = 0;
    /* 查找已存在的文件,如果失败则返回-1 */
    while(1){
      if((ip = namei(path)) == 0){
        end_op();
        return -1;
      }
      /* 加锁i节点 */
      ilock(ip);

      if(ip->type == T_SYMLINK && (omode & O_NOFOLLOW) == 0){
        //在跟踪符号链接时需要额外考虑到符号链接的目标可能还是符号链接, 此时需要递归的去跟踪目标链接直至得到真正的文件. 而这其中需要解决两个问题: 一是符号链接可能成环, 这样会一直递归地跟踪下去, 因此需要进行成环的检测; 另一方面是需要对链接的深度进行限制, 以减轻系统负担,这里把最大深度设成10
        if(++symlink_depth > 10){
          // too many layer of symlinks, might be a loop
          iunlockput(ip);
          end_op();
          return -1;
        }
        if(readi(ip, 0, (uint64)path, 0, MAXPATH) < 0){
          iunlockput(ip);
          end_op();
          return -1;
        }
        iunlockput(ip);
        
      }else{
        break;
      }
    }

    /* 如果是目录文件且不是只读打开,返回-1 */
    if(ip->type == T_DIR && omode != O_RDONLY){
      // // 如果是目录,并且打开模式不是 O_RDONLY(只读),则不能打开
      iunlockput(ip);
      end_op();
      return -1;
    }
  }

  /* 如果是设备文件,检查设备号是否有效,否则返回-1 */
  if(ip->type == T_DEVICE && (ip->major < 0 || ip->major >= NDEV)){
    iunlockput(ip);
    end_op();
    return -1;
  }

  /* 分配文件结构,如果失败则返回-1 */
  if((f = filealloc()) == 0 || (fd = fdalloc(f)) < 0){
    if(f)
      fileclose(f);
    iunlockput(ip);
    end_op();
    return -1;
  }

  // //处理符号链接,即软连接,如果
  // if(ip->type == T_SYMLINK && !(omode & O_NOFOLLOW)){ //如果不是不可追踪
  //   //若符号链接指向的仍旧是符号链接,则递归跟随它,直到找到真正的指向文件
  //   //深度不能超过MAX_SYMLINK_DEPTH
  //   for(int i = 0; i < MAX_SYMLINK_DEPTH; i++){
  //     //读出符号链接指的路径
  //   }
  // }

  /* 根据文件类型设置文件结构的类型和主要设备号 */
  if(ip->type == T_DEVICE){
    f->type = FD_DEVICE;
    f->major = ip->major;
  } else {
    f->type = FD_INODE;
    f->off = 0;
  }
  /* 设置文件结构的i节点指针和读写权限 */
  f->ip = ip;
  f->readable = !(omode & O_WRONLY);
  f->writable = (omode & O_WRONLY) || (omode & O_RDWR);

  /* 如果要求截断文件,且文件类型是普通文件,则截断文件 */
  if((omode & O_TRUNC) && ip->type == T_FILE){
    itrunc(ip);
  }

  /* 解锁i节点 */
  iunlock(ip);
  /* 结束文件操作 */
  end_op();

  /* 返回文件描述符 */
  return fd;
}

这里也可以看一下硬链接的实现方式(kernel/sysfile)

// Create the path new as a link to the same inode as old.
//xv6硬链接实现方式
uint64 sys_link(void) {
  char name[DIRSIZ], new[MAXPATH], old[MAXPATH];
  struct inode *dp, *ip;

  // 从用户态获取参数 old 和 new,分别表示旧路径和新路径
  if(argstr(0, old, MAXPATH) < 0 || argstr(1, new, MAXPATH) < 0)
    return -1;

  // 开始文件系统操作事务
  begin_op();

  // 根据旧路径名找到 inode
  if((ip = namei(old)) == 0){
    end_op();
    return -1;
  }

  // 对 inode 进行加锁
  ilock(ip);

  // 检查旧路径对应的 inode 是否为目录,如果是目录则不能创建硬链接
  if(ip->type == T_DIR){
    iunlockput(ip); // 解锁 inode 并释放引用
    end_op();
    return -1;
  }

  // 增加 inode 的链接数,因为即将创建一个新的硬链接指向该 inode
  ip->nlink++;
  iupdate(ip); // 更新 inode 的磁盘信息
  iunlock(ip); // 解锁 inode

  // 解析新路径中的目录名和文件名
  if((dp = nameiparent(new, name)) == 0)
    goto bad;

  // 对目录的 inode 进行加锁
  ilock(dp);

  // 检查新路径所在的目录的设备号与旧路径所在的设备号是否相同
  // 如果不相同,或者在新路径所在的目录中无法创建新的目录项,说明创建硬链接失败
  if(dp->dev != ip->dev || dirlink(dp, name, ip->inum) < 0){
    iunlockput(dp); // 解锁目录的 inode 并释放引用
    goto bad;
  }

  // 解锁目录的 inode 并释放引用
  iunlockput(dp);

  // 释放旧路径对应的 inode 的引用
  iput(ip);

  // 结束文件系统操作事务
  end_op();

  return 0;

bad:
  // 创建硬链接失败时,需要回滚之前对旧路径 inode 的修改,即减少链接数
  ilock(ip);
  ip->nlink--;
  iupdate(ip);
  iunlockput(ip);
  end_op();
  return -1;
}

5.最后在Makefile中添加对测试文件symlinktest.c 的编译.

测试

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2165965.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Rust和Go谁会更胜一筹

在国内&#xff0c;我认为Go语言会成为未来的主流&#xff0c;因为国内程序员号称码农&#xff0c;比较适合搬砖&#xff0c;而Rust对心智要求太高了&#xff0c;不适合搬砖。 就个人经验来看&#xff0c;Go语言简单&#xff0c;下限低&#xff0c;没有什么心智成本&#xff0c…

828华为云征文|华为云Flexus X实例Windows Server 2019安装护卫神防火墙——为企业运维安全发挥重要作用!!!

前言 公司最近需要选购一台华为云Windows服务器部署产品应用&#xff0c;但是考虑到Windows的安全性至关重要。护卫神防火墙无疑是守护Windows系统安全的得力助手。 华为云以其强大的性能和稳定的服务&#xff0c;为众多企业和开发者提供了可靠的云端基础设施。在网络环境日益复…

微信小程序-分包加载

文章目录 微信小程序-分包加载概述基本使用打包和引用原则独立分包分包预下载 微信小程序-分包加载 概述 小程序的代码通常是由许多页面、组件以及资源等组成&#xff0c;随着小程序功能的增加&#xff0c;代码量也会逐渐增加&#xff0c;体积过大就会导致用户打开速度变慢&a…

python --qt5(webview)/防多开/套壳网页/多次点击激活旧窗口

pyqtwebengine=5.12 PyQt5==5.12class MyWindow(QMainWindow):def __init__(self):super(MyWindow, self).__init__()self.browser = QWebEngineView(self) # 如果不写self则新生成一个窗口self.browser.setWindowTitle(技术领域占比分析)self.browser.setWindowIcon(QIcon(LO…

嵌入式linux裸机调试之windows、linux联合gdb

系列文章目录 提示:这里可以添加系列文章的所有文章的目录,目录需要自己手动添加 提示:写完文章后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 系列文章目录前言一、gdb-multiarch安装二、windows、linux联合gdb1.windows下启动JLinkGDBServer.exe2.Linux下…

Axure原型系统:药企内部管理平台原型设计

在当今竞争激烈的医药行业中&#xff0c;高效的内部管理系统是企业成功的关键之一。为了满足药企对市场部与销售部精细化管理的需求&#xff0c;我们精心设计了一款基于Axure RP的药企内部管理平台原型。这款原型以管理员角色为核心&#xff0c;旨在通过直观、高效的界面和强大…

Linux系统重启后MySQL数据丢失问题的解决(磁盘挂载)

今天分享一个在Linux系统中经常遇到的问题:系统重启后发现MySQL无法启动,而且数据似乎丢失了。这个问题可能会让人惊慌失措,但别担心,通常情况下这只是因为数据盘没有正确挂载导致的。现在我们将深入探讨这个问题的原因、解决方法以及如何预防它的再次发生。 1 问题描述 想象一…

HTTP Status 404 - /brand-demo/selectAllServlet错误解决原因-Servlet/JavaWeb/IDEA

检查xml文件的包名有无错误检查html文件的url有无写错&#xff0c;是否与Servlet的urlPatterns一致检查Servlet的urlpattern有没有写错(如写成name),检查doPost、doGet是否正常运行 注&#xff1a;IDEA新建Servlet时&#xff0c;默认的WebServlet注解中name需要改urlPatterns&…

创意实现!在uni-app小程序商品详情页轮播中嵌入视频播放功能

背景介绍 通过uni-app框架实现商城小程序商品详情页的视频与图片轮播功能&#xff0c;以提升用户体验和增加商品吸引力。通过展示商品视频和图片&#xff0c;用户可以更全面地了解商品细节&#xff0c;从而提高购买决策的便利性和满意度。这种功能适用于各类商品&#xff0c;如…

【SQL】产品分组销售

目录 语法 需求 示例 分析 代码 语法 GROUP_CONCAT(DISTINCT expression1 ORDER BY expression2 SEPARATOR sep&#xff09; GROUP_CONCAT 是 SQL 中的一个聚合函数&#xff0c;主要用于将分组后的多个行的值连接成一个字符串。这个函数在 MySQL 和一些其他数据库管理系统…

VBA技术资料MF203:添加带图标的右键菜单

我给VBA的定义&#xff1a;VBA是个人小型自动化处理的有效工具。利用好了&#xff0c;可以大大提高自己的工作效率&#xff0c;而且可以提高数据的准确度。“VBA语言専攻”提供的教程一共九套&#xff0c;分为初级、中级、高级三大部分&#xff0c;教程是对VBA的系统讲解&#…

ctf.show---->re2

做题笔记。 下载 查壳 32 ida打开。 WSL先运行一下&#xff1a; &#xff1f; 创建呗。 函数如下&#xff1a; 逻辑很清晰&#xff0c;写脚本咯 &#xff1a; #include <stdio.h> #include <string.h> #include <stdlib.h>int main() {char encode[] &qu…

计算机毕业设计 养老院管理系统的设计与实现 Java实战项目 附源码+文档+视频讲解

博主介绍&#xff1a;✌从事软件开发10年之余&#xff0c;专注于Java技术领域、Python人工智能及数据挖掘、小程序项目开发和Android项目开发等。CSDN、掘金、华为云、InfoQ、阿里云等平台优质作者✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精…

活动策划灵感TOP10分享-华媒舍

活动策划对于推广品牌、提升知名度和增加销量具有重要的作用。在当今竞争激烈的市场中&#xff0c;如何设计出与众不同、令人难以忘怀的活动策划方案是每个市场人员关心的问题。本文将介绍10个非凡的创意灵感&#xff0c;帮助您在活动策划中取得成功。 1. 主题游戏夜 通过组织…

阿里云云效多个ssh密钥对配置

实现功能 windows本地多个ssh密钥对,分别对应不同的阿里云账号的云效 实现办法 1.生成ssh密钥对 ssh-keygen -t rsa -f C:\xxx\id_rsa_customname(我这里C:\Users\admin\.ssh\id_rsa_customname) 2.配置.ssh目录的config文件 # ruiyi Host customnameHostName codeup.al…

IPSec隧道协议学习(一)

前情回顾 前面介绍的GRE隧道协议&#xff0c;可以字LAN之间通过Internet建立隧道&#xff0c;实现网络间资源共享&#xff0c;但是GRE隧道协议不能实现加密功能&#xff0c;传输的数据不受加密保护&#xff0c;为了实现在隧道间传输数据包收到加密保护&#xff0c;需要使用IPS…

贷款并非只看利息低,还有很多你知不道的地方

贷款这事儿&#xff0c;听起来像是天上掉馅饼&#xff0c;只要付点利息钱就能轻松拿到钱花&#xff0c;但实际上&#xff0c;它可是个需要精打细算的大工程。咱们得明白&#xff0c;贷款不只是利息那么简单&#xff0c;本金加利息&#xff0c;一个子儿都不能少还。所以&#xf…

B-树(不是B减树)原理剖析(1)

目录 B树的主要特性&#xff1a; B树的操作&#xff1a; B树的优点&#xff1a; 为什么要发明出B-树&#xff1f; B树的概念和原理剖析 原理图讲解(部分讲解在图中) 初始化结点&#xff1a; 处理数据数量计算(了解) 底层代码实现(加深理解) 前些日子我们学了AVl树&…

等保测评新趋势:企业如何领跑网络安全赛道

在当今数字化转型浪潮中&#xff0c;信息安全等级保护&#xff08;简称“等保”&#xff09;测评作为衡量企业网络安全管理水平的重要标尺&#xff0c;正随着网络环境的复杂度提升和法律法规的不断完善而持续进化。本文旨在深入剖析等保测评的最新趋势&#xff0c;并为企业提供…

【论文阅读】StoryMaker | 更全面的人物一致性开源工作

文章目录 1 Motivation2 背景 相关工作 Related work3 Method 方法4 效果 1 Motivation 背景是 Tuning-free personalized image generation methods无微调的个性化图像生成方式在维持脸部一致性上取得了显著性的成功。这里我不是很了解 然而&#xff0c;在多个场景中缺乏整…