www.pudn.com > linux011.rar > fcntl.c


/* passed
 *  linux/fs/fcntl.c
 *
 *  (C) 1991  Linus Torvalds
 */
#include  

#include // 字符串头文件。主要定义了一些有关字符串操作的嵌入函数。
#include // 错误号头文件。包含系统中各种出错号。(Linus 从minix 中引进的)。
#include // 调度程序头文件,定义了任务结构task_struct、初始任务0 的数据, 
									// 还有一些有关描述符参数设置和获取的嵌入式汇编函数宏语句。
#include // 内核头文件。含有一些内核常用函数的原形定义。
#include // 段操作头文件。定义了有关段寄存器操作的嵌入式汇编函数。

#include // 文件控制头文件。用于文件及其描述符的操作控制常数符号的定义。
#include // 文件状态头文件。含有文件或文件系统状态结构stat{}和常量。

extern int sys_close(int fd);// 关闭文件系统调用。(fs/open.c, 192)
 
//// 复制文件句柄(描述符)。 
// 参数fd 是欲复制的文件句柄,arg 指定新文件句柄的最小数值。 
// 返回新文件句柄或出错码。 
static int dupfd(unsigned int fd, unsigned int arg)
{ 
// 如果文件句柄值大于一个程序最多打开文件数NR_OPEN,或者该句柄的文件结构不存在,则出错, 
// 返回出错码并退出。
	if (fd >= NR_OPEN || !current->filp[fd])
		return -EBADF; 
// 如果指定的新句柄值arg 大于最多打开文件数,则出错,返回出错码并退出。
	if (arg >= NR_OPEN)
		return -EINVAL; 
// 在当前进程的文件结构指针数组中寻找索引号大于等于arg 但还没有使用的项。
	while (arg < NR_OPEN)
		if (current->filp[arg])
			arg++;
		else
			break; 
// 如果找到的新句柄值arg 大于最多打开文件数,则出错,返回出错码并退出。
	if (arg >= NR_OPEN)
		return -EMFILE; 
// 在执行时关闭标志位图中复位该句柄位。也即在运行exec()类函数时不关闭该句柄。
	current->close_on_exec &= ~(1<filp[arg] = current->filp[fd])->f_count++;
	return arg;		// 返回新的文件句柄。
}
 
//// 复制文件句柄系统调用函数。 
// 复制指定文件句柄oldfd,新句柄值等于newfd。如果newfd 已经打开,则首先关闭之。
int sys_dup2(unsigned int oldfd, unsigned int newfd)
{
	sys_close(newfd);			// 若句柄newfd 已经打开,则首先关闭之。
	return dupfd(oldfd,newfd);	// 复制并返回新句柄。
}
 
//// 复制文件句柄系统调用函数。 
// 复制指定文件句柄oldfd,新句柄的值是当前最小的未用句柄。
int sys_dup(unsigned int fildes)
{
	return dupfd(fildes,0);
}
 
//// 文件控制系统调用函数。 
// 参数fd 是文件句柄,cmd 是操作命令(参见include/fcntl.h,23-30 行)。
int sys_fcntl(unsigned int fd, unsigned int cmd, unsigned long arg)
{	
	struct file * filp;
 
// 如果文件句柄值大于一个进程最多打开文件数NR_OPEN,或者该句柄的文件结构指针为空,则出错, 
// 返回出错码并退出。
	if (fd >= NR_OPEN || !(filp = current->filp[fd]))
		return -EBADF; 
// 根据不同命令cmd 进行分别处理。
	switch (cmd) {
		case F_DUPFD:	// 复制文件句柄。
			return dupfd(fd,arg);
		case F_GETFD:	// 取文件句柄的执行时关闭标志。
			return (current->close_on_exec>>fd)&1;
		case F_SETFD:	// 设置句柄执行时关闭标志。arg 位0 置位是设置,否则关闭。
			if (arg&1)
				current->close_on_exec |= (1<close_on_exec &= ~(1<f_flags;
		case F_SETFL:	// 设置文件状态和访问模式(根据arg 设置添加、非阻塞标志)。
			filp->f_flags &= ~(O_APPEND | O_NONBLOCK);
			filp->f_flags |= arg & (O_APPEND | O_NONBLOCK);
			return 0;
		case F_GETLK:	case F_SETLK:	case F_SETLKW:	// 未实现。
			return -1;
		default:
			return -1;
	}
}