sys_link是创建硬链接的函数,从这个函数的代码中我们看到,硬链接的原理。文件查找和操作函数见之前的文章。这里就不贴了。
代码语言:javascript复制// 创建硬链接
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;
}
结构如下。
在这里插入图片描述