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 + -
显示快捷键?