hostfs_kern.c

来自「Linux Kernel 2.6.9 for OMAP1710」· C语言 代码 · 共 1,031 行 · 第 1/2 页

C
1,031
字号
	.writepage 	= hostfs_writepage,	.readpage	= hostfs_readpage,/* 	.set_page_dirty = __set_page_dirty_nobuffers, */	.prepare_write	= hostfs_prepare_write,	.commit_write	= hostfs_commit_write};static int init_inode(struct inode *inode, struct dentry *dentry){	char *name;	int type, err = -ENOMEM;	int maj, min;	dev_t rdev = 0;	if(dentry){		name = dentry_name(dentry, 0);		if(name == NULL)			goto out;		type = file_type(name, &maj, &min);		/*Reencode maj and min with the kernel encoding.*/		rdev = MKDEV(maj, min);		kfree(name);	}	else type = OS_TYPE_DIR;	err = 0;	if(type == OS_TYPE_SYMLINK)		inode->i_op = &page_symlink_inode_operations;	else if(type == OS_TYPE_DIR)		inode->i_op = &hostfs_dir_iops;	else inode->i_op = &hostfs_iops;	if(type == OS_TYPE_DIR) inode->i_fop = &hostfs_dir_fops;	else inode->i_fop = &hostfs_file_fops;	if(type == OS_TYPE_SYMLINK)		inode->i_mapping->a_ops = &hostfs_link_aops;	else inode->i_mapping->a_ops = &hostfs_aops;	switch (type) {	case OS_TYPE_CHARDEV:		init_special_inode(inode, S_IFCHR, rdev);		break;	case OS_TYPE_BLOCKDEV:		init_special_inode(inode, S_IFBLK, rdev);		break;	case OS_TYPE_FIFO:		init_special_inode(inode, S_IFIFO, 0);		break;	case OS_TYPE_SOCK:		init_special_inode(inode, S_IFSOCK, 0);		break;	} out:	return(err);}int hostfs_create(struct inode *dir, struct dentry *dentry, int mode,                 struct nameidata *nd){	struct inode *inode;	char *name;	int error, fd;	error = -ENOMEM;	inode = iget(dir->i_sb, 0);	if(inode == NULL) goto out;	error = init_inode(inode, dentry);	if(error)		goto out_put;	error = -ENOMEM;	name = dentry_name(dentry, 0);	if(name == NULL)		goto out_put;	fd = file_create(name,			 mode & S_IRUSR, mode & S_IWUSR, mode & S_IXUSR,			 mode & S_IRGRP, mode & S_IWGRP, mode & S_IXGRP,			 mode & S_IROTH, mode & S_IWOTH, mode & S_IXOTH);	if(fd < 0)		error = fd;	else error = read_name(inode, name);	kfree(name);	if(error)		goto out_put;	HOSTFS_I(inode)->fd = fd;	HOSTFS_I(inode)->mode = FMODE_READ | FMODE_WRITE;	d_instantiate(dentry, inode);	return(0); out_put:	iput(inode); out:	return(error);}struct dentry *hostfs_lookup(struct inode *ino, struct dentry *dentry,                            struct nameidata *nd){	struct inode *inode;	char *name;	int err;	err = -ENOMEM;	inode = iget(ino->i_sb, 0);	if(inode == NULL)		goto out;	err = init_inode(inode, dentry);	if(err)		goto out_put;	err = -ENOMEM;	name = dentry_name(dentry, 0);	if(name == NULL)		goto out_put;	err = read_name(inode, name);	kfree(name);	if(err == -ENOENT){		iput(inode);		inode = NULL;	}	else if(err)		goto out_put;	d_add(dentry, inode);	dentry->d_op = &hostfs_dentry_ops;	return(NULL); out_put:	iput(inode); out:	return(ERR_PTR(err));}static char *inode_dentry_name(struct inode *ino, struct dentry *dentry){        char *file;	int len;	file = inode_name(ino, dentry->d_name.len + 1);	if(file == NULL) return(NULL);        strcat(file, "/");	len = strlen(file);        strncat(file, dentry->d_name.name, dentry->d_name.len);	file[len + dentry->d_name.len] = '\0';        return(file);}int hostfs_link(struct dentry *to, struct inode *ino, struct dentry *from){        char *from_name, *to_name;        int err;        if((from_name = inode_dentry_name(ino, from)) == NULL)                return(-ENOMEM);        to_name = dentry_name(to, 0);	if(to_name == NULL){		kfree(from_name);		return(-ENOMEM);	}        err = link_file(to_name, from_name);        kfree(from_name);        kfree(to_name);        return(err);}int hostfs_unlink(struct inode *ino, struct dentry *dentry){	char *file;	int err;	if((file = inode_dentry_name(ino, dentry)) == NULL) return(-ENOMEM);	if(append)		return(-EPERM);	err = unlink_file(file);	kfree(file);	return(err);}int hostfs_symlink(struct inode *ino, struct dentry *dentry, const char *to){	char *file;	int err;	if((file = inode_dentry_name(ino, dentry)) == NULL) return(-ENOMEM);	err = make_symlink(file, to);	kfree(file);	return(err);}int hostfs_mkdir(struct inode *ino, struct dentry *dentry, int mode){	char *file;	int err;	if((file = inode_dentry_name(ino, dentry)) == NULL) return(-ENOMEM);	err = do_mkdir(file, mode);	kfree(file);	return(err);}int hostfs_rmdir(struct inode *ino, struct dentry *dentry){	char *file;	int err;	if((file = inode_dentry_name(ino, dentry)) == NULL) return(-ENOMEM);	err = do_rmdir(file);	kfree(file);	return(err);}int hostfs_mknod(struct inode *dir, struct dentry *dentry, int mode, dev_t dev){	struct inode *inode;	char *name;	int err = -ENOMEM;	inode = iget(dir->i_sb, 0);	if(inode == NULL)		goto out;	err = init_inode(inode, dentry);	if(err)		goto out_put;	err = -ENOMEM;	name = dentry_name(dentry, 0);	if(name == NULL)		goto out_put;	init_special_inode(inode, mode, dev);	err = do_mknod(name, mode, dev);	if(err)		goto out_free;	err = read_name(inode, name);	kfree(name);	if(err)		goto out_put;	d_instantiate(dentry, inode);	return(0); out_free:	kfree(name); out_put:	iput(inode); out:	return(err);}int hostfs_rename(struct inode *from_ino, struct dentry *from,		  struct inode *to_ino, struct dentry *to){	char *from_name, *to_name;	int err;	if((from_name = inode_dentry_name(from_ino, from)) == NULL)		return(-ENOMEM);	if((to_name = inode_dentry_name(to_ino, to)) == NULL){		kfree(from_name);		return(-ENOMEM);	}	err = rename_file(from_name, to_name);	kfree(from_name);	kfree(to_name);	return(err);}void hostfs_truncate(struct inode *ino){	not_implemented();}int hostfs_permission(struct inode *ino, int desired, struct nameidata *nd){	char *name;	int r = 0, w = 0, x = 0, err;	if(desired & MAY_READ) r = 1;	if(desired & MAY_WRITE) w = 1;	if(desired & MAY_EXEC) x = 1;	name = inode_name(ino, 0);	if(name == NULL) return(-ENOMEM);	err = access_file(name, r, w, x);	kfree(name);	if(!err) err = vfs_permission(ino, desired);	return(err);}int hostfs_setattr(struct dentry *dentry, struct iattr *attr){	struct hostfs_iattr attrs;	char *name;	int err;	if(append)		attr->ia_valid &= ~ATTR_SIZE;	attrs.ia_valid = 0;	if(attr->ia_valid & ATTR_MODE){		attrs.ia_valid |= HOSTFS_ATTR_MODE;		attrs.ia_mode = attr->ia_mode;	}	if(attr->ia_valid & ATTR_UID){		if((dentry->d_inode->i_sb->s_dev == ROOT_DEV) &&		   (attr->ia_uid == 0))			attr->ia_uid = getuid();		attrs.ia_valid |= HOSTFS_ATTR_UID;		attrs.ia_uid = attr->ia_uid;	}	if(attr->ia_valid & ATTR_GID){		if((dentry->d_inode->i_sb->s_dev == ROOT_DEV) &&		   (attr->ia_gid == 0))			attr->ia_gid = getuid();		attrs.ia_valid |= HOSTFS_ATTR_GID;		attrs.ia_gid = attr->ia_gid;	}	if(attr->ia_valid & ATTR_SIZE){		attrs.ia_valid |= HOSTFS_ATTR_SIZE;		attrs.ia_size = attr->ia_size;	}	if(attr->ia_valid & ATTR_ATIME){		attrs.ia_valid |= HOSTFS_ATTR_ATIME;		attrs.ia_atime = attr->ia_atime;	}	if(attr->ia_valid & ATTR_MTIME){		attrs.ia_valid |= HOSTFS_ATTR_MTIME;		attrs.ia_mtime = attr->ia_mtime;	}	if(attr->ia_valid & ATTR_CTIME){		attrs.ia_valid |= HOSTFS_ATTR_CTIME;		attrs.ia_ctime = attr->ia_ctime;	}	if(attr->ia_valid & ATTR_ATIME_SET){		attrs.ia_valid |= HOSTFS_ATTR_ATIME_SET;	}	if(attr->ia_valid & ATTR_MTIME_SET){		attrs.ia_valid |= HOSTFS_ATTR_MTIME_SET;	}	name = dentry_name(dentry, 0);	if(name == NULL) return(-ENOMEM);	err = set_attr(name, &attrs);	kfree(name);	if(err)		return(err);	return(inode_setattr(dentry->d_inode, attr));}int hostfs_getattr(struct vfsmount *mnt, struct dentry *dentry,	   struct kstat *stat){	generic_fillattr(dentry->d_inode, stat);	return(0);}static struct inode_operations hostfs_iops = {	.create		= hostfs_create,	.link		= hostfs_link,	.unlink		= hostfs_unlink,	.symlink	= hostfs_symlink,	.mkdir		= hostfs_mkdir,	.rmdir		= hostfs_rmdir,	.mknod		= hostfs_mknod,	.rename		= hostfs_rename,	.truncate	= hostfs_truncate,	.permission	= hostfs_permission,	.setattr	= hostfs_setattr,	.getattr	= hostfs_getattr,};static struct inode_operations hostfs_dir_iops = {	.create		= hostfs_create,	.lookup		= hostfs_lookup,	.link		= hostfs_link,	.unlink		= hostfs_unlink,	.symlink	= hostfs_symlink,	.mkdir		= hostfs_mkdir,	.rmdir		= hostfs_rmdir,	.mknod		= hostfs_mknod,	.rename		= hostfs_rename,	.truncate	= hostfs_truncate,	.permission	= hostfs_permission,	.setattr	= hostfs_setattr,	.getattr	= hostfs_getattr,};int hostfs_link_readpage(struct file *file, struct page *page){	char *buffer, *name;	long long start;	int err;	start = page->index << PAGE_CACHE_SHIFT;	buffer = kmap(page);	name = inode_name(page->mapping->host, 0);	if(name == NULL) return(-ENOMEM);	err = do_readlink(name, buffer, PAGE_CACHE_SIZE);	kfree(name);	if(err == PAGE_CACHE_SIZE)		err = -E2BIG;	else if(err > 0){		flush_dcache_page(page);		SetPageUptodate(page);		if (PageError(page)) ClearPageError(page);		err = 0;	}	kunmap(page);	unlock_page(page);	return(err);}static struct address_space_operations hostfs_link_aops = {	.readpage	= hostfs_link_readpage,};static int hostfs_fill_sb_common(struct super_block *sb, void *d, int silent){	struct inode *root_inode;	char *name, *data = d;	int err;	sb->s_blocksize = 1024;	sb->s_blocksize_bits = 10;	sb->s_magic = HOSTFS_SUPER_MAGIC;	sb->s_op = &hostfs_sbops;	if((data == NULL) || (*data == '\0'))		data = root_ino;	err = -ENOMEM;	name = kmalloc(strlen(data) + 1, GFP_KERNEL);	if(name == NULL)		goto out;	strcpy(name, data);	root_inode = iget(sb, 0);	if(root_inode == NULL)		goto out_free;	err = init_inode(root_inode, NULL);	if(err)		goto out_put;	HOSTFS_I(root_inode)->host_filename = name;	err = -ENOMEM;	sb->s_root = d_alloc_root(root_inode);	if(sb->s_root == NULL)		goto out_put;	err = read_inode(root_inode);	if(err)		goto out_put;	return(0); out_put:	iput(root_inode); out_free:	kfree(name); out:	return(err);}static struct super_block *hostfs_read_sb(struct file_system_type *type,					     int flags, const char *dev_name,					     void *data){	return(get_sb_nodev(type, flags, data, hostfs_fill_sb_common));}static struct file_system_type hostfs_type = {	.owner 		= THIS_MODULE,	.name 		= "hostfs",	.get_sb 	= hostfs_read_sb,	.kill_sb	= kill_anon_super,	.fs_flags 	= 0,};static int __init init_hostfs(void){	return(register_filesystem(&hostfs_type));}static void __exit exit_hostfs(void){	unregister_filesystem(&hostfs_type);}module_init(init_hostfs)module_exit(exit_hostfs)MODULE_LICENSE("GPL");/* * Overrides for Emacs so that we follow Linus's tabbing style. * Emacs will notice this stuff at the end of the file and automatically * adjust the settings for this buffer only.  This must remain at the end * of the file. * --------------------------------------------------------------------------- * Local variables: * c-file-style: "linux" * End: */

⌨️ 快捷键说明

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