📄 ops_file.c
字号:
/* * Copyright (C) Sistina Software, Inc. 1997-2003 All rights reserved. * Copyright (C) 2004-2006 Red Hat, Inc. All rights reserved. * * This copyrighted material is made available to anyone wishing to use, * modify, copy, or redistribute it subject to the terms and conditions * of the GNU General Public License version 2. */#include <linux/slab.h>#include <linux/spinlock.h>#include <linux/completion.h>#include <linux/buffer_head.h>#include <linux/pagemap.h>#include <linux/uio.h>#include <linux/blkdev.h>#include <linux/mm.h>#include <linux/fs.h>#include <linux/gfs2_ondisk.h>#include <linux/ext2_fs.h>#include <linux/crc32.h>#include <linux/lm_interface.h>#include <linux/writeback.h>#include <asm/uaccess.h>#include "gfs2.h"#include "incore.h"#include "bmap.h"#include "dir.h"#include "glock.h"#include "glops.h"#include "inode.h"#include "lm.h"#include "log.h"#include "meta_io.h"#include "ops_file.h"#include "ops_vm.h"#include "quota.h"#include "rgrp.h"#include "trans.h"#include "util.h"#include "eaops.h"/* * Most fields left uninitialised to catch anybody who tries to * use them. f_flags set to prevent file_accessed() from touching * any other part of this. Its use is purely as a flag so that we * know (in readpage()) whether or not do to locking. */struct file gfs2_internal_file_sentinel = { .f_flags = O_NOATIME|O_RDONLY,};static int gfs2_read_actor(read_descriptor_t *desc, struct page *page, unsigned long offset, unsigned long size){ char *kaddr; unsigned long count = desc->count; if (size > count) size = count; kaddr = kmap(page); memcpy(desc->arg.data, kaddr + offset, size); kunmap(page); desc->count = count - size; desc->written += size; desc->arg.buf += size; return size;}int gfs2_internal_read(struct gfs2_inode *ip, struct file_ra_state *ra_state, char *buf, loff_t *pos, unsigned size){ struct inode *inode = &ip->i_inode; read_descriptor_t desc; desc.written = 0; desc.arg.data = buf; desc.count = size; desc.error = 0; do_generic_mapping_read(inode->i_mapping, ra_state, &gfs2_internal_file_sentinel, pos, &desc, gfs2_read_actor); return desc.written ? desc.written : desc.error;}/** * gfs2_llseek - seek to a location in a file * @file: the file * @offset: the offset * @origin: Where to seek from (SEEK_SET, SEEK_CUR, or SEEK_END) * * SEEK_END requires the glock for the file because it references the * file's size. * * Returns: The new offset, or errno */static loff_t gfs2_llseek(struct file *file, loff_t offset, int origin){ struct gfs2_inode *ip = GFS2_I(file->f_mapping->host); struct gfs2_holder i_gh; loff_t error; if (origin == 2) { error = gfs2_glock_nq_init(ip->i_gl, LM_ST_SHARED, LM_FLAG_ANY, &i_gh); if (!error) { error = remote_llseek(file, offset, origin); gfs2_glock_dq_uninit(&i_gh); } } else error = remote_llseek(file, offset, origin); return error;}/** * gfs2_readdir - Read directory entries from a directory * @file: The directory to read from * @dirent: Buffer for dirents * @filldir: Function used to do the copying * * Returns: errno */static int gfs2_readdir(struct file *file, void *dirent, filldir_t filldir){ struct inode *dir = file->f_mapping->host; struct gfs2_inode *dip = GFS2_I(dir); struct gfs2_holder d_gh; u64 offset = file->f_pos; int error; gfs2_holder_init(dip->i_gl, LM_ST_SHARED, GL_ATIME, &d_gh); error = gfs2_glock_nq_atime(&d_gh); if (error) { gfs2_holder_uninit(&d_gh); return error; } error = gfs2_dir_read(dir, &offset, dirent, filldir); gfs2_glock_dq_uninit(&d_gh); file->f_pos = offset; return error;}/** * fsflags_cvt * @table: A table of 32 u32 flags * @val: a 32 bit value to convert * * This function can be used to convert between fsflags values and * GFS2's own flags values. * * Returns: the converted flags */static u32 fsflags_cvt(const u32 *table, u32 val){ u32 res = 0; while(val) { if (val & 1) res |= *table; table++; val >>= 1; } return res;}static const u32 fsflags_to_gfs2[32] = { [3] = GFS2_DIF_SYNC, [4] = GFS2_DIF_IMMUTABLE, [5] = GFS2_DIF_APPENDONLY, [7] = GFS2_DIF_NOATIME, [12] = GFS2_DIF_EXHASH, [14] = GFS2_DIF_INHERIT_JDATA, [20] = GFS2_DIF_INHERIT_DIRECTIO,};static const u32 gfs2_to_fsflags[32] = { [gfs2fl_Sync] = FS_SYNC_FL, [gfs2fl_Immutable] = FS_IMMUTABLE_FL, [gfs2fl_AppendOnly] = FS_APPEND_FL, [gfs2fl_NoAtime] = FS_NOATIME_FL, [gfs2fl_ExHash] = FS_INDEX_FL, [gfs2fl_InheritDirectio] = FS_DIRECTIO_FL, [gfs2fl_InheritJdata] = FS_JOURNAL_DATA_FL,};static int gfs2_get_flags(struct file *filp, u32 __user *ptr){ struct inode *inode = filp->f_path.dentry->d_inode; struct gfs2_inode *ip = GFS2_I(inode); struct gfs2_holder gh; int error; u32 fsflags; gfs2_holder_init(ip->i_gl, LM_ST_SHARED, GL_ATIME, &gh); error = gfs2_glock_nq_atime(&gh); if (error) return error; fsflags = fsflags_cvt(gfs2_to_fsflags, ip->i_di.di_flags); if (!S_ISDIR(inode->i_mode)) { if (ip->i_di.di_flags & GFS2_DIF_JDATA) fsflags |= FS_JOURNAL_DATA_FL; if (ip->i_di.di_flags & GFS2_DIF_DIRECTIO) fsflags |= FS_DIRECTIO_FL; } if (put_user(fsflags, ptr)) error = -EFAULT; gfs2_glock_dq_m(1, &gh); gfs2_holder_uninit(&gh); return error;}void gfs2_set_inode_flags(struct inode *inode){ struct gfs2_inode *ip = GFS2_I(inode); struct gfs2_dinode_host *di = &ip->i_di; unsigned int flags = inode->i_flags; flags &= ~(S_SYNC|S_APPEND|S_IMMUTABLE|S_NOATIME|S_DIRSYNC); if (di->di_flags & GFS2_DIF_IMMUTABLE) flags |= S_IMMUTABLE; if (di->di_flags & GFS2_DIF_APPENDONLY) flags |= S_APPEND; if (di->di_flags & GFS2_DIF_NOATIME) flags |= S_NOATIME; if (di->di_flags & GFS2_DIF_SYNC) flags |= S_SYNC; inode->i_flags = flags;}/* Flags that can be set by user space */#define GFS2_FLAGS_USER_SET (GFS2_DIF_JDATA| \ GFS2_DIF_DIRECTIO| \ GFS2_DIF_IMMUTABLE| \ GFS2_DIF_APPENDONLY| \ GFS2_DIF_NOATIME| \ GFS2_DIF_SYNC| \ GFS2_DIF_SYSTEM| \ GFS2_DIF_INHERIT_DIRECTIO| \ GFS2_DIF_INHERIT_JDATA)/** * gfs2_set_flags - set flags on an inode * @inode: The inode * @flags: The flags to set * @mask: Indicates which flags are valid * */static int do_gfs2_set_flags(struct file *filp, u32 reqflags, u32 mask){ struct inode *inode = filp->f_path.dentry->d_inode; struct gfs2_inode *ip = GFS2_I(inode); struct gfs2_sbd *sdp = GFS2_SB(inode); struct buffer_head *bh; struct gfs2_holder gh; int error; u32 new_flags, flags; error = gfs2_glock_nq_init(ip->i_gl, LM_ST_EXCLUSIVE, 0, &gh); if (error) return error; flags = ip->i_di.di_flags; new_flags = (flags & ~mask) | (reqflags & mask); if ((new_flags ^ flags) == 0) goto out; error = -EINVAL; if ((new_flags ^ flags) & ~GFS2_FLAGS_USER_SET) goto out; error = -EPERM; if (IS_IMMUTABLE(inode) && (new_flags & GFS2_DIF_IMMUTABLE)) goto out; if (IS_APPEND(inode) && (new_flags & GFS2_DIF_APPENDONLY)) goto out; if (((new_flags ^ flags) & GFS2_DIF_IMMUTABLE) && !capable(CAP_LINUX_IMMUTABLE)) goto out; if (!IS_IMMUTABLE(inode)) { error = permission(inode, MAY_WRITE, NULL); if (error) goto out; } error = gfs2_trans_begin(sdp, RES_DINODE, 0); if (error) goto out; error = gfs2_meta_inode_buffer(ip, &bh); if (error) goto out_trans_end; gfs2_trans_add_bh(ip->i_gl, bh, 1); ip->i_di.di_flags = new_flags; gfs2_dinode_out(ip, bh->b_data); brelse(bh); gfs2_set_inode_flags(inode);out_trans_end: gfs2_trans_end(sdp);out: gfs2_glock_dq_uninit(&gh); return error;}static int gfs2_set_flags(struct file *filp, u32 __user *ptr){ struct inode *inode = filp->f_path.dentry->d_inode; u32 fsflags, gfsflags; if (get_user(fsflags, ptr)) return -EFAULT; gfsflags = fsflags_cvt(fsflags_to_gfs2, fsflags); if (!S_ISDIR(inode->i_mode)) { if (gfsflags & GFS2_DIF_INHERIT_JDATA) gfsflags ^= (GFS2_DIF_JDATA | GFS2_DIF_INHERIT_JDATA); if (gfsflags & GFS2_DIF_INHERIT_DIRECTIO) gfsflags ^= (GFS2_DIF_DIRECTIO | GFS2_DIF_INHERIT_DIRECTIO); return do_gfs2_set_flags(filp, gfsflags, ~0); } return do_gfs2_set_flags(filp, gfsflags, ~GFS2_DIF_JDATA);}static long gfs2_ioctl(struct file *filp, unsigned int cmd, unsigned long arg){ switch(cmd) { case FS_IOC_GETFLAGS: return gfs2_get_flags(filp, (u32 __user *)arg); case FS_IOC_SETFLAGS: return gfs2_set_flags(filp, (u32 __user *)arg); } return -ENOTTY;}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -