diff options
Diffstat (limited to 'include/linux/fs.h')
| -rw-r--r-- | include/linux/fs.h | 152 | 
1 files changed, 47 insertions, 105 deletions
| diff --git a/include/linux/fs.h b/include/linux/fs.h index bbf812ce89a8..c8510da6cc6d 100644 --- a/include/linux/fs.h +++ b/include/linux/fs.h @@ -41,6 +41,7 @@  #include <linux/stddef.h>  #include <linux/mount.h>  #include <linux/cred.h> +#include <linux/mnt_idmapping.h>  #include <asm/byteorder.h>  #include <uapi/linux/fs.h> @@ -1220,13 +1221,13 @@ static inline int fcntl_setlk(unsigned int fd, struct file *file,  #if BITS_PER_LONG == 32  static inline int fcntl_getlk64(struct file *file, unsigned int cmd, -				struct flock64 __user *user) +				struct flock64 *user)  {  	return -EINVAL;  }  static inline int fcntl_setlk64(unsigned int fd, struct file *file, -				unsigned int cmd, struct flock64 __user *user) +				unsigned int cmd, struct flock64 *user)  {  	return -EACCES;  } @@ -1599,6 +1600,11 @@ struct super_block {  	struct list_head	s_inodes_wb;	/* writeback inodes */  } __randomize_layout; +static inline struct user_namespace *i_user_ns(const struct inode *inode) +{ +	return inode->i_sb->s_user_ns; +} +  /* Helper functions so that in most cases filesystems will   * not need to deal directly with kuid_t and kgid_t and can   * instead deal with the raw numeric values that are stored @@ -1606,50 +1612,22 @@ struct super_block {   */  static inline uid_t i_uid_read(const struct inode *inode)  { -	return from_kuid(inode->i_sb->s_user_ns, inode->i_uid); +	return from_kuid(i_user_ns(inode), inode->i_uid);  }  static inline gid_t i_gid_read(const struct inode *inode)  { -	return from_kgid(inode->i_sb->s_user_ns, inode->i_gid); +	return from_kgid(i_user_ns(inode), inode->i_gid);  }  static inline void i_uid_write(struct inode *inode, uid_t uid)  { -	inode->i_uid = make_kuid(inode->i_sb->s_user_ns, uid); +	inode->i_uid = make_kuid(i_user_ns(inode), uid);  }  static inline void i_gid_write(struct inode *inode, gid_t gid)  { -	inode->i_gid = make_kgid(inode->i_sb->s_user_ns, gid); -} - -/** - * kuid_into_mnt - map a kuid down into a mnt_userns - * @mnt_userns: user namespace of the relevant mount - * @kuid: kuid to be mapped - * - * Return: @kuid mapped according to @mnt_userns. - * If @kuid has no mapping INVALID_UID is returned. - */ -static inline kuid_t kuid_into_mnt(struct user_namespace *mnt_userns, -				   kuid_t kuid) -{ -	return make_kuid(mnt_userns, __kuid_val(kuid)); -} - -/** - * kgid_into_mnt - map a kgid down into a mnt_userns - * @mnt_userns: user namespace of the relevant mount - * @kgid: kgid to be mapped - * - * Return: @kgid mapped according to @mnt_userns. - * If @kgid has no mapping INVALID_GID is returned. - */ -static inline kgid_t kgid_into_mnt(struct user_namespace *mnt_userns, -				   kgid_t kgid) -{ -	return make_kgid(mnt_userns, __kgid_val(kgid)); +	inode->i_gid = make_kgid(i_user_ns(inode), gid);  }  /** @@ -1663,7 +1641,7 @@ static inline kgid_t kgid_into_mnt(struct user_namespace *mnt_userns,  static inline kuid_t i_uid_into_mnt(struct user_namespace *mnt_userns,  				    const struct inode *inode)  { -	return kuid_into_mnt(mnt_userns, inode->i_uid); +	return mapped_kuid_fs(mnt_userns, i_user_ns(inode), inode->i_uid);  }  /** @@ -1677,69 +1655,7 @@ static inline kuid_t i_uid_into_mnt(struct user_namespace *mnt_userns,  static inline kgid_t i_gid_into_mnt(struct user_namespace *mnt_userns,  				    const struct inode *inode)  { -	return kgid_into_mnt(mnt_userns, inode->i_gid); -} - -/** - * kuid_from_mnt - map a kuid up into a mnt_userns - * @mnt_userns: user namespace of the relevant mount - * @kuid: kuid to be mapped - * - * Return: @kuid mapped up according to @mnt_userns. - * If @kuid has no mapping INVALID_UID is returned. - */ -static inline kuid_t kuid_from_mnt(struct user_namespace *mnt_userns, -				   kuid_t kuid) -{ -	return KUIDT_INIT(from_kuid(mnt_userns, kuid)); -} - -/** - * kgid_from_mnt - map a kgid up into a mnt_userns - * @mnt_userns: user namespace of the relevant mount - * @kgid: kgid to be mapped - * - * Return: @kgid mapped up according to @mnt_userns. - * If @kgid has no mapping INVALID_GID is returned. - */ -static inline kgid_t kgid_from_mnt(struct user_namespace *mnt_userns, -				   kgid_t kgid) -{ -	return KGIDT_INIT(from_kgid(mnt_userns, kgid)); -} - -/** - * mapped_fsuid - return caller's fsuid mapped up into a mnt_userns - * @mnt_userns: user namespace of the relevant mount - * - * Use this helper to initialize a new vfs or filesystem object based on - * the caller's fsuid. A common example is initializing the i_uid field of - * a newly allocated inode triggered by a creation event such as mkdir or - * O_CREAT. Other examples include the allocation of quotas for a specific - * user. - * - * Return: the caller's current fsuid mapped up according to @mnt_userns. - */ -static inline kuid_t mapped_fsuid(struct user_namespace *mnt_userns) -{ -	return kuid_from_mnt(mnt_userns, current_fsuid()); -} - -/** - * mapped_fsgid - return caller's fsgid mapped up into a mnt_userns - * @mnt_userns: user namespace of the relevant mount - * - * Use this helper to initialize a new vfs or filesystem object based on - * the caller's fsgid. A common example is initializing the i_gid field of - * a newly allocated inode triggered by a creation event such as mkdir or - * O_CREAT. Other examples include the allocation of quotas for a specific - * user. - * - * Return: the caller's current fsgid mapped up according to @mnt_userns. - */ -static inline kgid_t mapped_fsgid(struct user_namespace *mnt_userns) -{ -	return kgid_from_mnt(mnt_userns, current_fsgid()); +	return mapped_kgid_fs(mnt_userns, i_user_ns(inode), inode->i_gid);  }  /** @@ -1753,7 +1669,7 @@ static inline kgid_t mapped_fsgid(struct user_namespace *mnt_userns)  static inline void inode_fsuid_set(struct inode *inode,  				   struct user_namespace *mnt_userns)  { -	inode->i_uid = mapped_fsuid(mnt_userns); +	inode->i_uid = mapped_fsuid(mnt_userns, i_user_ns(inode));  }  /** @@ -1767,7 +1683,7 @@ static inline void inode_fsuid_set(struct inode *inode,  static inline void inode_fsgid_set(struct inode *inode,  				   struct user_namespace *mnt_userns)  { -	inode->i_gid = mapped_fsgid(mnt_userns); +	inode->i_gid = mapped_fsgid(mnt_userns, i_user_ns(inode));  }  /** @@ -1784,10 +1700,18 @@ static inline void inode_fsgid_set(struct inode *inode,  static inline bool fsuidgid_has_mapping(struct super_block *sb,  					struct user_namespace *mnt_userns)  { -	struct user_namespace *s_user_ns = sb->s_user_ns; +	struct user_namespace *fs_userns = sb->s_user_ns; +	kuid_t kuid; +	kgid_t kgid; -	return kuid_has_mapping(s_user_ns, mapped_fsuid(mnt_userns)) && -	       kgid_has_mapping(s_user_ns, mapped_fsgid(mnt_userns)); +	kuid = mapped_fsuid(mnt_userns, fs_userns); +	if (!uid_valid(kuid)) +		return false; +	kgid = mapped_fsgid(mnt_userns, fs_userns); +	if (!gid_valid(kgid)) +		return false; +	return kuid_has_mapping(fs_userns, kuid) && +	       kgid_has_mapping(fs_userns, kgid);  }  extern struct timespec64 current_time(struct inode *inode); @@ -2249,6 +2173,7 @@ struct super_operations {  #define S_ENCRYPTED	(1 << 14) /* Encrypted file (using fs/crypto/) */  #define S_CASEFOLD	(1 << 15) /* Casefolded file */  #define S_VERITY	(1 << 16) /* Verity file (using fs/verity/) */ +#define S_KERNEL_FILE	(1 << 17) /* File is in use by the kernel (eg. fs/cachefiles) */  /*   * Note that nosuid etc flags are inode-specific: setting some file-system @@ -2418,6 +2343,8 @@ static inline void kiocb_clone(struct kiocb *kiocb, struct kiocb *kiocb_src,   *			Used to detect that mark_inode_dirty() should not move   * 			inode between dirty lists.   * + * I_PINNING_FSCACHE_WB	Inode is pinning an fscache object for writeback. + *   * Q: What is the difference between I_WILL_FREE and I_FREEING?   */  #define I_DIRTY_SYNC		(1 << 0) @@ -2440,6 +2367,7 @@ static inline void kiocb_clone(struct kiocb *kiocb, struct kiocb *kiocb_src,  #define I_CREATING		(1 << 15)  #define I_DONTCACHE		(1 << 16)  #define I_SYNC_QUEUED		(1 << 17) +#define I_PINNING_FSCACHE_WB	(1 << 18)  #define I_DIRTY_INODE (I_DIRTY_SYNC | I_DIRTY_DATASYNC)  #define I_DIRTY (I_DIRTY_INODE | I_DIRTY_PAGES) @@ -2724,6 +2652,21 @@ static inline struct user_namespace *file_mnt_user_ns(struct file *file)  {  	return mnt_user_ns(file->f_path.mnt);  } + +/** + * is_idmapped_mnt - check whether a mount is mapped + * @mnt: the mount to check + * + * If @mnt has an idmapping attached different from the + * filesystem's idmapping then @mnt is mapped. + * + * Return: true if mount is mapped, false if not. + */ +static inline bool is_idmapped_mnt(const struct vfsmount *mnt) +{ +	return mnt_user_ns(mnt) != mnt->mnt_sb->s_user_ns; +} +  extern long vfs_truncate(const struct path *, loff_t);  int do_truncate(struct user_namespace *, struct dentry *, loff_t start,  		unsigned int time_attrs, struct file *filp); @@ -2847,8 +2790,6 @@ static inline int filemap_fdatawait(struct address_space *mapping)  extern bool filemap_range_has_page(struct address_space *, loff_t lstart,  				  loff_t lend); -extern bool filemap_range_needs_writeback(struct address_space *, -					  loff_t lstart, loff_t lend);  extern int filemap_write_and_wait_range(struct address_space *mapping,  				        loff_t lstart, loff_t lend);  extern int __filemap_fdatawrite_range(struct address_space *mapping, @@ -3152,6 +3093,7 @@ extern void unlock_new_inode(struct inode *);  extern void discard_new_inode(struct inode *);  extern unsigned int get_next_ino(void);  extern void evict_inodes(struct super_block *sb); +void dump_mapping(const struct address_space *);  /*   * Userspace may rely on the the inode number being non-zero. For example, glibc |