diff options
-rw-r--r-- | fs/Makefile | 3 | ||||
-rw-r--r-- | fs/stack.c | 40 | ||||
-rw-r--r-- | include/linux/fs_stack.h | 31 |
3 files changed, 73 insertions, 1 deletions
diff --git a/fs/Makefile b/fs/Makefile index 9a5ce9323bfd..b9ffa63f77fc 100644 --- a/fs/Makefile +++ b/fs/Makefile | |||
@@ -10,7 +10,8 @@ obj-y := open.o read_write.o file_table.o super.o \ | |||
10 | ioctl.o readdir.o select.o fifo.o locks.o dcache.o inode.o \ | 10 | ioctl.o readdir.o select.o fifo.o locks.o dcache.o inode.o \ |
11 | attr.o bad_inode.o file.o filesystems.o namespace.o aio.o \ | 11 | attr.o bad_inode.o file.o filesystems.o namespace.o aio.o \ |
12 | seq_file.o xattr.o libfs.o fs-writeback.o \ | 12 | seq_file.o xattr.o libfs.o fs-writeback.o \ |
13 | pnode.o drop_caches.o splice.o sync.o utimes.o | 13 | pnode.o drop_caches.o splice.o sync.o utimes.o \ |
14 | stack.o | ||
14 | 15 | ||
15 | ifeq ($(CONFIG_BLOCK),y) | 16 | ifeq ($(CONFIG_BLOCK),y) |
16 | obj-y += buffer.o bio.o block_dev.o direct-io.o mpage.o ioprio.o | 17 | obj-y += buffer.o bio.o block_dev.o direct-io.o mpage.o ioprio.o |
diff --git a/fs/stack.c b/fs/stack.c new file mode 100644 index 000000000000..5ddbc34535f9 --- /dev/null +++ b/fs/stack.c | |||
@@ -0,0 +1,40 @@ | |||
1 | #include <linux/module.h> | ||
2 | #include <linux/fs.h> | ||
3 | #include <linux/fs_stack.h> | ||
4 | |||
5 | /* does _NOT_ require i_mutex to be held. | ||
6 | * | ||
7 | * This function cannot be inlined since i_size_{read,write} is rather | ||
8 | * heavy-weight on 32-bit systems | ||
9 | */ | ||
10 | void fsstack_copy_inode_size(struct inode *dst, const struct inode *src) | ||
11 | { | ||
12 | i_size_write(dst, i_size_read((struct inode *)src)); | ||
13 | dst->i_blocks = src->i_blocks; | ||
14 | } | ||
15 | EXPORT_SYMBOL_GPL(fsstack_copy_inode_size); | ||
16 | |||
17 | /* copy all attributes; get_nlinks is optional way to override the i_nlink | ||
18 | * copying | ||
19 | */ | ||
20 | void fsstack_copy_attr_all(struct inode *dest, const struct inode *src, | ||
21 | int (*get_nlinks)(struct inode *)) | ||
22 | { | ||
23 | if (!get_nlinks) | ||
24 | dest->i_nlink = src->i_nlink; | ||
25 | else | ||
26 | dest->i_nlink = (*get_nlinks)(dest); | ||
27 | |||
28 | dest->i_mode = src->i_mode; | ||
29 | dest->i_uid = src->i_uid; | ||
30 | dest->i_gid = src->i_gid; | ||
31 | dest->i_rdev = src->i_rdev; | ||
32 | dest->i_atime = src->i_atime; | ||
33 | dest->i_mtime = src->i_mtime; | ||
34 | dest->i_ctime = src->i_ctime; | ||
35 | dest->i_blkbits = src->i_blkbits; | ||
36 | dest->i_flags = src->i_flags; | ||
37 | |||
38 | fsstack_copy_inode_size(dest, src); | ||
39 | } | ||
40 | EXPORT_SYMBOL_GPL(fsstack_copy_attr_all); | ||
diff --git a/include/linux/fs_stack.h b/include/linux/fs_stack.h new file mode 100644 index 000000000000..bb516ceeefc9 --- /dev/null +++ b/include/linux/fs_stack.h | |||
@@ -0,0 +1,31 @@ | |||
1 | #ifndef _LINUX_FS_STACK_H | ||
2 | #define _LINUX_FS_STACK_H | ||
3 | |||
4 | /* This file defines generic functions used primarily by stackable | ||
5 | * filesystems; none of these functions require i_mutex to be held. | ||
6 | */ | ||
7 | |||
8 | #include <linux/fs.h> | ||
9 | |||
10 | /* externs for fs/stack.c */ | ||
11 | extern void fsstack_copy_attr_all(struct inode *dest, const struct inode *src, | ||
12 | int (*get_nlinks)(struct inode *)); | ||
13 | |||
14 | extern void fsstack_copy_inode_size(struct inode *dst, const struct inode *src); | ||
15 | |||
16 | /* inlines */ | ||
17 | static inline void fsstack_copy_attr_atime(struct inode *dest, | ||
18 | const struct inode *src) | ||
19 | { | ||
20 | dest->i_atime = src->i_atime; | ||
21 | } | ||
22 | |||
23 | static inline void fsstack_copy_attr_times(struct inode *dest, | ||
24 | const struct inode *src) | ||
25 | { | ||
26 | dest->i_atime = src->i_atime; | ||
27 | dest->i_mtime = src->i_mtime; | ||
28 | dest->i_ctime = src->i_ctime; | ||
29 | } | ||
30 | |||
31 | #endif /* _LINUX_FS_STACK_H */ | ||