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

📄 ops_file.c

📁 linux 内核源代码
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * 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 + -