⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 inode.c

📁 linux 内核源代码
💻 C
📖 第 1 页 / 共 2 页
字号:
/* *  inode.c * *  Copyright (C) 1995, 1996 by Volker Lendecke *  Modified for big endian by J.F. Chadima and David S. Miller *  Modified 1997 Peter Waltenberg, Bill Hawes, David Woodhouse for 2.1 dcache *  Modified 1998 Wolfram Pienkoss for NLS *  Modified 2000 Ben Harris, University of Cambridge for NFS NS meta-info * */#include <linux/module.h>#include <asm/system.h>#include <asm/uaccess.h>#include <asm/byteorder.h>#include <linux/time.h>#include <linux/kernel.h>#include <linux/mm.h>#include <linux/string.h>#include <linux/stat.h>#include <linux/errno.h>#include <linux/file.h>#include <linux/fcntl.h>#include <linux/slab.h>#include <linux/vmalloc.h>#include <linux/init.h>#include <linux/smp_lock.h>#include <linux/vfs.h>#include <linux/ncp_fs.h>#include <net/sock.h>#include "ncplib_kernel.h"#include "getopt.h"static void ncp_delete_inode(struct inode *);static void ncp_put_super(struct super_block *);static int  ncp_statfs(struct dentry *, struct kstatfs *);static struct kmem_cache * ncp_inode_cachep;static struct inode *ncp_alloc_inode(struct super_block *sb){	struct ncp_inode_info *ei;	ei = (struct ncp_inode_info *)kmem_cache_alloc(ncp_inode_cachep, GFP_KERNEL);	if (!ei)		return NULL;	return &ei->vfs_inode;}static void ncp_destroy_inode(struct inode *inode){	kmem_cache_free(ncp_inode_cachep, NCP_FINFO(inode));}static void init_once(struct kmem_cache *cachep, void *foo){	struct ncp_inode_info *ei = (struct ncp_inode_info *) foo;	mutex_init(&ei->open_mutex);	inode_init_once(&ei->vfs_inode);}static int init_inodecache(void){	ncp_inode_cachep = kmem_cache_create("ncp_inode_cache",					     sizeof(struct ncp_inode_info),					     0, (SLAB_RECLAIM_ACCOUNT|						SLAB_MEM_SPREAD),					     init_once);	if (ncp_inode_cachep == NULL)		return -ENOMEM;	return 0;}static void destroy_inodecache(void){	kmem_cache_destroy(ncp_inode_cachep);}static int ncp_remount(struct super_block *sb, int *flags, char* data){	*flags |= MS_NODIRATIME;	return 0;}static const struct super_operations ncp_sops ={	.alloc_inode	= ncp_alloc_inode,	.destroy_inode	= ncp_destroy_inode,	.drop_inode	= generic_delete_inode,	.delete_inode	= ncp_delete_inode,	.put_super	= ncp_put_super,	.statfs		= ncp_statfs,	.remount_fs	= ncp_remount,};extern struct dentry_operations ncp_root_dentry_operations;#if defined(CONFIG_NCPFS_EXTRAS) || defined(CONFIG_NCPFS_NFS_NS)extern const struct address_space_operations ncp_symlink_aops;extern int ncp_symlink(struct inode*, struct dentry*, const char*);#endif/* * Fill in the ncpfs-specific information in the inode. */static void ncp_update_dirent(struct inode *inode, struct ncp_entry_info *nwinfo){	NCP_FINFO(inode)->DosDirNum = nwinfo->i.DosDirNum;	NCP_FINFO(inode)->dirEntNum = nwinfo->i.dirEntNum;	NCP_FINFO(inode)->volNumber = nwinfo->volume;}void ncp_update_inode(struct inode *inode, struct ncp_entry_info *nwinfo){	ncp_update_dirent(inode, nwinfo);	NCP_FINFO(inode)->nwattr = nwinfo->i.attributes;	NCP_FINFO(inode)->access = nwinfo->access;	memcpy(NCP_FINFO(inode)->file_handle, nwinfo->file_handle,			sizeof(nwinfo->file_handle));	DPRINTK("ncp_update_inode: updated %s, volnum=%d, dirent=%u\n",		nwinfo->i.entryName, NCP_FINFO(inode)->volNumber,		NCP_FINFO(inode)->dirEntNum);}static void ncp_update_dates(struct inode *inode, struct nw_info_struct *nwi){	/* NFS namespace mode overrides others if it's set. */	DPRINTK(KERN_DEBUG "ncp_update_dates_and_mode: (%s) nfs.mode=0%o\n",		nwi->entryName, nwi->nfs.mode);	if (nwi->nfs.mode) {		/* XXX Security? */		inode->i_mode = nwi->nfs.mode;	}	inode->i_blocks = (inode->i_size + NCP_BLOCK_SIZE - 1) >> NCP_BLOCK_SHIFT;	inode->i_mtime.tv_sec = ncp_date_dos2unix(nwi->modifyTime, nwi->modifyDate);	inode->i_ctime.tv_sec = ncp_date_dos2unix(nwi->creationTime, nwi->creationDate);	inode->i_atime.tv_sec = ncp_date_dos2unix(0, nwi->lastAccessDate);	inode->i_atime.tv_nsec = 0;	inode->i_mtime.tv_nsec = 0;	inode->i_ctime.tv_nsec = 0;}static void ncp_update_attrs(struct inode *inode, struct ncp_entry_info *nwinfo){	struct nw_info_struct *nwi = &nwinfo->i;	struct ncp_server *server = NCP_SERVER(inode);	if (nwi->attributes & aDIR) {		inode->i_mode = server->m.dir_mode;		/* for directories dataStreamSize seems to be some		   Object ID ??? */		inode->i_size = NCP_BLOCK_SIZE;	} else {		inode->i_mode = server->m.file_mode;		inode->i_size = le32_to_cpu(nwi->dataStreamSize);#ifdef CONFIG_NCPFS_EXTRAS		if ((server->m.flags & (NCP_MOUNT_EXTRAS|NCP_MOUNT_SYMLINKS)) 		 && (nwi->attributes & aSHARED)) {			switch (nwi->attributes & (aHIDDEN|aSYSTEM)) {				case aHIDDEN:					if (server->m.flags & NCP_MOUNT_SYMLINKS) {						if (/* (inode->i_size >= NCP_MIN_SYMLINK_SIZE)						 && */ (inode->i_size <= NCP_MAX_SYMLINK_SIZE)) {							inode->i_mode = (inode->i_mode & ~S_IFMT) | S_IFLNK;							NCP_FINFO(inode)->flags |= NCPI_KLUDGE_SYMLINK;							break;						}					}					/* FALLTHROUGH */				case 0:					if (server->m.flags & NCP_MOUNT_EXTRAS)						inode->i_mode |= S_IRUGO;					break;				case aSYSTEM:					if (server->m.flags & NCP_MOUNT_EXTRAS)						inode->i_mode |= (inode->i_mode >> 2) & S_IXUGO;					break;				/* case aSYSTEM|aHIDDEN: */				default:					/* reserved combination */					break;			}		}#endif	}	if (nwi->attributes & aRONLY) inode->i_mode &= ~S_IWUGO;}void ncp_update_inode2(struct inode* inode, struct ncp_entry_info *nwinfo){	NCP_FINFO(inode)->flags = 0;	if (!atomic_read(&NCP_FINFO(inode)->opened)) {		NCP_FINFO(inode)->nwattr = nwinfo->i.attributes;		ncp_update_attrs(inode, nwinfo);	}	ncp_update_dates(inode, &nwinfo->i);	ncp_update_dirent(inode, nwinfo);}/* * Fill in the inode based on the ncp_entry_info structure. */static void ncp_set_attr(struct inode *inode, struct ncp_entry_info *nwinfo){	struct ncp_server *server = NCP_SERVER(inode);	NCP_FINFO(inode)->flags = 0;		ncp_update_attrs(inode, nwinfo);	DDPRINTK("ncp_read_inode: inode->i_mode = %u\n", inode->i_mode);	inode->i_nlink = 1;	inode->i_uid = server->m.uid;	inode->i_gid = server->m.gid;	ncp_update_dates(inode, &nwinfo->i);	ncp_update_inode(inode, nwinfo);}#if defined(CONFIG_NCPFS_EXTRAS) || defined(CONFIG_NCPFS_NFS_NS)static const struct inode_operations ncp_symlink_inode_operations = {	.readlink	= generic_readlink,	.follow_link	= page_follow_link_light,	.put_link	= page_put_link,	.setattr	= ncp_notify_change,};#endif/* * Get a new inode. */struct inode * ncp_iget(struct super_block *sb, struct ncp_entry_info *info){	struct inode *inode;	if (info == NULL) {		printk(KERN_ERR "ncp_iget: info is NULL\n");		return NULL;	}	inode = new_inode(sb);	if (inode) {		atomic_set(&NCP_FINFO(inode)->opened, info->opened);		inode->i_ino = info->ino;		ncp_set_attr(inode, info);		if (S_ISREG(inode->i_mode)) {			inode->i_op = &ncp_file_inode_operations;			inode->i_fop = &ncp_file_operations;		} else if (S_ISDIR(inode->i_mode)) {			inode->i_op = &ncp_dir_inode_operations;			inode->i_fop = &ncp_dir_operations;#ifdef CONFIG_NCPFS_NFS_NS		} else if (S_ISCHR(inode->i_mode) || S_ISBLK(inode->i_mode) || S_ISFIFO(inode->i_mode) || S_ISSOCK(inode->i_mode)) {			init_special_inode(inode, inode->i_mode,				new_decode_dev(info->i.nfs.rdev));#endif#if defined(CONFIG_NCPFS_EXTRAS) || defined(CONFIG_NCPFS_NFS_NS)		} else if (S_ISLNK(inode->i_mode)) {			inode->i_op = &ncp_symlink_inode_operations;			inode->i_data.a_ops = &ncp_symlink_aops;#endif		} else {			make_bad_inode(inode);		}		insert_inode_hash(inode);	} else		printk(KERN_ERR "ncp_iget: iget failed!\n");	return inode;}static voidncp_delete_inode(struct inode *inode){	truncate_inode_pages(&inode->i_data, 0);	if (S_ISDIR(inode->i_mode)) {		DDPRINTK("ncp_delete_inode: put directory %ld\n", inode->i_ino);	}	if (ncp_make_closed(inode) != 0) {		/* We can't do anything but complain. */		printk(KERN_ERR "ncp_delete_inode: could not close\n");	}	clear_inode(inode);}static void ncp_stop_tasks(struct ncp_server *server) {	struct sock* sk = server->ncp_sock->sk;			sk->sk_error_report = server->error_report;	sk->sk_data_ready   = server->data_ready;	sk->sk_write_space  = server->write_space;	del_timer_sync(&server->timeout_tm);	flush_scheduled_work();}static const struct ncp_option ncp_opts[] = {	{ "uid",	OPT_INT,	'u' },	{ "gid",	OPT_INT,	'g' },	{ "owner",	OPT_INT,	'o' },	{ "mode",	OPT_INT,	'm' },	{ "dirmode",	OPT_INT,	'd' },	{ "timeout",	OPT_INT,	't' },	{ "retry",	OPT_INT,	'r' },	{ "flags",	OPT_INT,	'f' },	{ "wdogpid",	OPT_INT,	'w' },	{ "ncpfd",	OPT_INT,	'n' },	{ "infofd",	OPT_INT,	'i' },	/* v5 */	{ "version",	OPT_INT,	'v' },	{ NULL,		0,		0 } };static int ncp_parse_options(struct ncp_mount_data_kernel *data, char *options) {	int optval;	char *optarg;	unsigned long optint;	int version = 0;	int ret;	data->flags = 0;	data->int_flags = 0;	data->mounted_uid = 0;	data->wdog_pid = NULL;	data->ncp_fd = ~0;	data->time_out = 10;	data->retry_count = 20;	data->uid = 0;	data->gid = 0;	data->file_mode = 0600;	data->dir_mode = 0700;	data->info_fd = -1;	data->mounted_vol[0] = 0;		while ((optval = ncp_getopt("ncpfs", &options, ncp_opts, NULL, &optarg, &optint)) != 0) {		ret = optval;		if (ret < 0)			goto err;		switch (optval) {			case 'u':				data->uid = optint;				break;			case 'g':				data->gid = optint;				break;			case 'o':				data->mounted_uid = optint;				break;			case 'm':				data->file_mode = optint;				break;			case 'd':				data->dir_mode = optint;				break;			case 't':				data->time_out = optint;				break;			case 'r':				data->retry_count = optint;				break;			case 'f':				data->flags = optint;				break;			case 'w':				data->wdog_pid = find_get_pid(optint);				break;			case 'n':				data->ncp_fd = optint;				break;			case 'i':				data->info_fd = optint;				break;			case 'v':				ret = -ECHRNG;				if (optint < NCP_MOUNT_VERSION_V4)					goto err;				if (optint > NCP_MOUNT_VERSION_V5)					goto err;				version = optint;				break;					}	}	return 0;err:	put_pid(data->wdog_pid);	data->wdog_pid = NULL;	return ret;}static int ncp_fill_super(struct super_block *sb, void *raw_data, int silent){	struct ncp_mount_data_kernel data;	struct ncp_server *server;	struct file *ncp_filp;	struct inode *root_inode;	struct inode *sock_inode;	struct socket *sock;	int error;	int default_bufsize;#ifdef CONFIG_NCPFS_PACKET_SIGNING	int options;#endif	struct ncp_entry_info finfo;	data.wdog_pid = NULL;	server = kzalloc(sizeof(struct ncp_server), GFP_KERNEL);	if (!server)		return -ENOMEM;	sb->s_fs_info = server;	error = -EFAULT;	if (raw_data == NULL)		goto out;	switch (*(int*)raw_data) {		case NCP_MOUNT_VERSION:			{				struct ncp_mount_data* md = (struct ncp_mount_data*)raw_data;				data.flags = md->flags;				data.int_flags = NCP_IMOUNT_LOGGEDIN_POSSIBLE;				data.mounted_uid = md->mounted_uid;				data.wdog_pid = find_get_pid(md->wdog_pid);				data.ncp_fd = md->ncp_fd;				data.time_out = md->time_out;				data.retry_count = md->retry_count;				data.uid = md->uid;				data.gid = md->gid;				data.file_mode = md->file_mode;				data.dir_mode = md->dir_mode;				data.info_fd = -1;				memcpy(data.mounted_vol, md->mounted_vol,					NCP_VOLNAME_LEN+1);			}			break;		case NCP_MOUNT_VERSION_V4:			{				struct ncp_mount_data_v4* md = (struct ncp_mount_data_v4*)raw_data;				data.flags = md->flags;				data.int_flags = 0;				data.mounted_uid = md->mounted_uid;				data.wdog_pid = find_get_pid(md->wdog_pid);				data.ncp_fd = md->ncp_fd;				data.time_out = md->time_out;				data.retry_count = md->retry_count;				data.uid = md->uid;				data.gid = md->gid;				data.file_mode = md->file_mode;				data.dir_mode = md->dir_mode;				data.info_fd = -1;				data.mounted_vol[0] = 0;			}			break;		default:			error = -ECHRNG;			if (memcmp(raw_data, "vers", 4) == 0) {				error = ncp_parse_options(&data, raw_data);			}			if (error)				goto out;			break;	}	error = -EBADF;	ncp_filp = fget(data.ncp_fd);	if (!ncp_filp)		goto out;	error = -ENOTSOCK;	sock_inode = ncp_filp->f_path.dentry->d_inode;	if (!S_ISSOCK(sock_inode->i_mode))		goto out_fput;	sock = SOCKET_I(sock_inode);	if (!sock)		goto out_fput;			if (sock->type == SOCK_STREAM)		default_bufsize = 0xF000;	else		default_bufsize = 1024;	sb->s_flags |= MS_NODIRATIME;	/* probably even noatime */	sb->s_maxbytes = 0xFFFFFFFFU;	sb->s_blocksize = 1024;	/* Eh...  Is this correct? */	sb->s_blocksize_bits = 10;	sb->s_magic = NCP_SUPER_MAGIC;	sb->s_op = &ncp_sops;	server = NCP_SBP(sb);	memset(server, 0, sizeof(*server));	server->ncp_filp = ncp_filp;	server->ncp_sock = sock;		if (data.info_fd != -1) {		struct socket *info_sock;		error = -EBADF;		server->info_filp = fget(data.info_fd);		if (!server->info_filp)			goto out_fput;

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -