@harpsword
2015-11-14T15:18:23.000000Z
字数 1748
阅读 1940
文件系统
这篇文章承接之一,主要介绍 truncate.c文件。
/*
* linux/fs/truncate.c
*
* (C) 1991 Linus Torvalds
*/
#include <linux/sched.h>
#include <sys/stat.h>
//// 释放所有一次间接块
// 参数dev是文件系统所在设备的设备号;block是逻辑块号
static void free_ind(int dev,int block)
{
struct buffer_head * bh;
unsigned short * p;
int i;
// 首先判断参数的有效性
if (!block)
return;
// 然后读取一次间接块,并释放其上表明使用的所有逻辑块,然后释放该一次间接块
// 的缓冲块,函数free_block用于释放设备上指定逻辑块号的磁盘块。
if ((bh=bread(dev,block))) {
p = (unsigned short *) bh->b_data; // 指向缓冲块数据区
for (i=0;i<512;i++,p++) // 每个逻辑块上可有512个块号
if (*p)
free_block(dev,*p); // 释放指定的设备逻辑块
brelse(bh); // 然后释放间接块占用的缓冲块
}
// 最后释放设备上的一次间接块
free_block(dev,block);
}
//// 释放所有第二次间接块
// 参数同free_ind函数
static void free_dind(int dev,int block)
{
struct buffer_head * bh;
unsigned short * p;
int i;
if (!block)
return;
// 读取二次间接块的一级块,并释放其上表明使用的所有的逻辑块,然后释放该一级块的缓冲区
if ((bh=bread(dev,block))) {
p = (unsigned short *) bh->b_data;
for (i=0;i<512;i++,p++)
// 假如使用 if (p) ,会成功
// 但是 p == (unsigned short *) 0 可能有效
if (*p)
free_ind(dev,*p); // 释放所有一次间接块
// 释放二次间接块占用的缓冲块
brelse(bh);
}
// 最后释放设备上的二次间接块
free_block(dev,block);
}
//// 截断文件数据函数
// 将节点对应的文件长度截为0,并释放所占用的设备空间
void truncate(struct m_inode * inode)
{
int i;
// 首先判断指定i节点的有效性,如果不是常规文件或者是目录文件,则返回
if (!(S_ISREG(inode->i_mode) || S_ISDIR(inode->i_mode)))
return;
// 然后释放i节点的7个直接逻辑块,并将这7个逻辑块项全置零。
for (i=0;i<7;i++)
if (inode->i_zone[i]) { // 如果块号不为0,则释放
free_block(inode->i_dev,inode->i_zone[i]);
inode->i_zone[i]=0;
}
free_ind(inode->i_dev,inode->i_zone[7]); // 释放所有一次间接块
free_dind(inode->i_dev,inode->i_zone[8]); // 释放所有二次间接块
inode->i_zone[7] = inode->i_zone[8] = 0; // 逻辑块项7、8置零
inode->i_size = 0; // 文件大小置零
inode->i_dirt = 1; // 置节点已修改标志
// 最后重置文件修改时间和i节点改变时间为当前时间。宏CURRENT_TIME定义在
// include/linux/sched.h中,用于取得从1970:0:0:0开始到现在为止经过的秒数。
inode->i_mtime = inode->i_ctime = CURRENT_TIME;
}