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;
}

结构如下。
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值