linux系统调用之sys_link(基于linux0.11)

sys_link是创建硬链接的函数,从这个函数的代码中我们看到,硬链接的原理。文件查找和操作函数见之前的文章。这里就不贴了。

// 创建硬链接
int sys_link(const char * oldname, const char * newname)
{
    struct dir_entry * de;
    struct m_inode * oldinode, * dir;
    struct buffer_head * bh;
    const char * basename;
    int namelen;
    // 根据路径名找到文件的inode节点
    oldinode=namei(oldname);
    if (!oldinode)
        return -ENOENT;
    // 不能给目录创建硬链接
    if (S_ISDIR(oldinode->i_mode)) {
        // 不需要使用inode了,解除引用
        iput(oldinode);
        return -EPERM;
    }
    // 找出newname最后一级目录的inode和newname中的文件名
    dir = dir_namei(newname,&namelen,&basename);
    // 路径不存在
    if (!dir) {
        iput(oldinode);
        return -EACCES;
    }
    // 路径是一个目录,所以文件名是空
    if (!namelen) {
        iput(oldinode);
        iput(dir);
        return -EPERM;
    }
    // 不能跨文件系统创建硬链接
    if (dir->i_dev != oldinode->i_dev) {
        iput(dir);
        iput(oldinode);
        return -EXDEV;
    }
    // 权限检验
    if (!permission(dir,MAY_WRITE)) {
        iput(dir);
        iput(oldinode);
        return -EACCES;
    }
    // 在目录下找文件名等于basename的项
    bh = find_entry(&dir,basename,namelen,&de);
    // 找到的话说明文件名已经存在,则不能再创建
    if (bh) {
        brelse(bh);
        iput(dir);
        iput(oldinode);
        return -EEXIST;
    }
    // 没有则新增一个目录项,de保存找到的目录项
    bh = add_entry(dir,basename,namelen,&de);
    // 新增是否成功
    if (!bh) {
        iput(dir);
        iput(oldinode);
        return -ENOSPC;
    }
    // 硬链接的inode和旧文件的inode号一样
    de->inode = oldinode->i_num;
    // 新增了一项,需要回写硬盘
    bh->b_dirt = 1; 
    brelse(bh);
    iput(dir);
    // 引用数加1,创建硬链接即多了一个索引指向inode节点,所以inode引用数加一即可,为0才能删除文件
    oldinode->i_nlinks++;
    oldinode->i_ctime = CURRENT_TIME;
    // inode信息有更新,需要回写硬盘
    oldinode->i_dirt = 1;
    iput(oldinode);
    return 0;
}

结构如下。

在这里插入图片描述

原文发布于微信公众号 - 编程杂技(theanarkh)

原文发表时间:2019-05-05

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

扫码关注云+社区

领取腾讯云代金券