diff options
| author | Miklos Szeredi <mszeredi@suse.cz> | 2008-04-30 03:54:37 -0400 |
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2008-04-30 11:29:50 -0400 |
| commit | e4ad08fe64afca4ef79ecc4c624e6e871688da0d (patch) | |
| tree | 5b8b390b874700041dc0c095e8ba9ac3ed42ea77 | |
| parent | 76f1418b485da2707531178e517bbb5cf06b3c76 (diff) | |
mm: bdi: add separate writeback accounting capability
Add a new BDI capability flag: BDI_CAP_NO_ACCT_WB. If this flag is
set, then don't update the per-bdi writeback stats from
test_set_page_writeback() and test_clear_page_writeback().
Misc cleanups:
- convert bdi_cap_writeback_dirty() and friends to static inline functions
- create a flag that includes all three dirty/writeback related flags,
since almst all users will want to have them toghether
Signed-off-by: Miklos Szeredi <mszeredi@suse.cz>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
| -rw-r--r-- | fs/configfs/inode.c | 2 | ||||
| -rw-r--r-- | fs/hugetlbfs/inode.c | 2 | ||||
| -rw-r--r-- | fs/ocfs2/dlm/dlmfs.c | 2 | ||||
| -rw-r--r-- | fs/ramfs/inode.c | 2 | ||||
| -rw-r--r-- | fs/sysfs/inode.c | 2 | ||||
| -rw-r--r-- | include/linux/backing-dev.h | 77 | ||||
| -rw-r--r-- | kernel/cgroup.c | 2 | ||||
| -rw-r--r-- | mm/page-writeback.c | 4 | ||||
| -rw-r--r-- | mm/shmem.c | 2 | ||||
| -rw-r--r-- | mm/swap_state.c | 2 |
10 files changed, 67 insertions, 30 deletions
diff --git a/fs/configfs/inode.c b/fs/configfs/inode.c index 4c1ebff778ee..b9a1d810346d 100644 --- a/fs/configfs/inode.c +++ b/fs/configfs/inode.c | |||
| @@ -47,7 +47,7 @@ static const struct address_space_operations configfs_aops = { | |||
| 47 | 47 | ||
| 48 | static struct backing_dev_info configfs_backing_dev_info = { | 48 | static struct backing_dev_info configfs_backing_dev_info = { |
| 49 | .ra_pages = 0, /* No readahead */ | 49 | .ra_pages = 0, /* No readahead */ |
| 50 | .capabilities = BDI_CAP_NO_ACCT_DIRTY | BDI_CAP_NO_WRITEBACK, | 50 | .capabilities = BDI_CAP_NO_ACCT_AND_WRITEBACK, |
| 51 | }; | 51 | }; |
| 52 | 52 | ||
| 53 | static const struct inode_operations configfs_inode_operations ={ | 53 | static const struct inode_operations configfs_inode_operations ={ |
diff --git a/fs/hugetlbfs/inode.c b/fs/hugetlbfs/inode.c index 9783723e8ffe..aeabf80f81a5 100644 --- a/fs/hugetlbfs/inode.c +++ b/fs/hugetlbfs/inode.c | |||
| @@ -45,7 +45,7 @@ static const struct inode_operations hugetlbfs_inode_operations; | |||
| 45 | 45 | ||
| 46 | static struct backing_dev_info hugetlbfs_backing_dev_info = { | 46 | static struct backing_dev_info hugetlbfs_backing_dev_info = { |
| 47 | .ra_pages = 0, /* No readahead */ | 47 | .ra_pages = 0, /* No readahead */ |
| 48 | .capabilities = BDI_CAP_NO_ACCT_DIRTY | BDI_CAP_NO_WRITEBACK, | 48 | .capabilities = BDI_CAP_NO_ACCT_AND_WRITEBACK, |
| 49 | }; | 49 | }; |
| 50 | 50 | ||
| 51 | int sysctl_hugetlb_shm_group; | 51 | int sysctl_hugetlb_shm_group; |
diff --git a/fs/ocfs2/dlm/dlmfs.c b/fs/ocfs2/dlm/dlmfs.c index 61a000f8524c..e48aba698b77 100644 --- a/fs/ocfs2/dlm/dlmfs.c +++ b/fs/ocfs2/dlm/dlmfs.c | |||
| @@ -327,7 +327,7 @@ clear_fields: | |||
| 327 | 327 | ||
| 328 | static struct backing_dev_info dlmfs_backing_dev_info = { | 328 | static struct backing_dev_info dlmfs_backing_dev_info = { |
| 329 | .ra_pages = 0, /* No readahead */ | 329 | .ra_pages = 0, /* No readahead */ |
| 330 | .capabilities = BDI_CAP_NO_ACCT_DIRTY | BDI_CAP_NO_WRITEBACK, | 330 | .capabilities = BDI_CAP_NO_ACCT_AND_WRITEBACK, |
| 331 | }; | 331 | }; |
| 332 | 332 | ||
| 333 | static struct inode *dlmfs_get_root_inode(struct super_block *sb) | 333 | static struct inode *dlmfs_get_root_inode(struct super_block *sb) |
diff --git a/fs/ramfs/inode.c b/fs/ramfs/inode.c index 8428d5b2711d..b13123424e49 100644 --- a/fs/ramfs/inode.c +++ b/fs/ramfs/inode.c | |||
| @@ -44,7 +44,7 @@ static const struct inode_operations ramfs_dir_inode_operations; | |||
| 44 | 44 | ||
| 45 | static struct backing_dev_info ramfs_backing_dev_info = { | 45 | static struct backing_dev_info ramfs_backing_dev_info = { |
| 46 | .ra_pages = 0, /* No readahead */ | 46 | .ra_pages = 0, /* No readahead */ |
| 47 | .capabilities = BDI_CAP_NO_ACCT_DIRTY | BDI_CAP_NO_WRITEBACK | | 47 | .capabilities = BDI_CAP_NO_ACCT_AND_WRITEBACK | |
| 48 | BDI_CAP_MAP_DIRECT | BDI_CAP_MAP_COPY | | 48 | BDI_CAP_MAP_DIRECT | BDI_CAP_MAP_COPY | |
| 49 | BDI_CAP_READ_MAP | BDI_CAP_WRITE_MAP | BDI_CAP_EXEC_MAP, | 49 | BDI_CAP_READ_MAP | BDI_CAP_WRITE_MAP | BDI_CAP_EXEC_MAP, |
| 50 | }; | 50 | }; |
diff --git a/fs/sysfs/inode.c b/fs/sysfs/inode.c index d9262f74f94e..f8b82e73b3bf 100644 --- a/fs/sysfs/inode.c +++ b/fs/sysfs/inode.c | |||
| @@ -30,7 +30,7 @@ static const struct address_space_operations sysfs_aops = { | |||
| 30 | 30 | ||
| 31 | static struct backing_dev_info sysfs_backing_dev_info = { | 31 | static struct backing_dev_info sysfs_backing_dev_info = { |
| 32 | .ra_pages = 0, /* No readahead */ | 32 | .ra_pages = 0, /* No readahead */ |
| 33 | .capabilities = BDI_CAP_NO_ACCT_DIRTY | BDI_CAP_NO_WRITEBACK, | 33 | .capabilities = BDI_CAP_NO_ACCT_AND_WRITEBACK, |
| 34 | }; | 34 | }; |
| 35 | 35 | ||
| 36 | static const struct inode_operations sysfs_inode_operations ={ | 36 | static const struct inode_operations sysfs_inode_operations ={ |
diff --git a/include/linux/backing-dev.h b/include/linux/backing-dev.h index c49a2d045e11..13ab79d99268 100644 --- a/include/linux/backing-dev.h +++ b/include/linux/backing-dev.h | |||
| @@ -12,6 +12,7 @@ | |||
| 12 | #include <linux/log2.h> | 12 | #include <linux/log2.h> |
| 13 | #include <linux/proportions.h> | 13 | #include <linux/proportions.h> |
| 14 | #include <linux/kernel.h> | 14 | #include <linux/kernel.h> |
| 15 | #include <linux/fs.h> | ||
| 15 | #include <asm/atomic.h> | 16 | #include <asm/atomic.h> |
| 16 | 17 | ||
| 17 | struct page; | 18 | struct page; |
| @@ -151,22 +152,43 @@ int bdi_set_max_ratio(struct backing_dev_info *bdi, unsigned int max_ratio); | |||
| 151 | 152 | ||
| 152 | /* | 153 | /* |
| 153 | * Flags in backing_dev_info::capability | 154 | * Flags in backing_dev_info::capability |
| 154 | * - The first two flags control whether dirty pages will contribute to the | 155 | * |
| 155 | * VM's accounting and whether writepages() should be called for dirty pages | 156 | * The first three flags control whether dirty pages will contribute to the |
| 156 | * (something that would not, for example, be appropriate for ramfs) | 157 | * VM's accounting and whether writepages() should be called for dirty pages |
| 157 | * - These flags let !MMU mmap() govern direct device mapping vs immediate | 158 | * (something that would not, for example, be appropriate for ramfs) |
| 158 | * copying more easily for MAP_PRIVATE, especially for ROM filesystems | 159 | * |
| 160 | * WARNING: these flags are closely related and should not normally be | ||
| 161 | * used separately. The BDI_CAP_NO_ACCT_AND_WRITEBACK combines these | ||
| 162 | * three flags into a single convenience macro. | ||
| 163 | * | ||
| 164 | * BDI_CAP_NO_ACCT_DIRTY: Dirty pages shouldn't contribute to accounting | ||
| 165 | * BDI_CAP_NO_WRITEBACK: Don't write pages back | ||
| 166 | * BDI_CAP_NO_ACCT_WB: Don't automatically account writeback pages | ||
| 167 | * | ||
| 168 | * These flags let !MMU mmap() govern direct device mapping vs immediate | ||
| 169 | * copying more easily for MAP_PRIVATE, especially for ROM filesystems. | ||
| 170 | * | ||
| 171 | * BDI_CAP_MAP_COPY: Copy can be mapped (MAP_PRIVATE) | ||
| 172 | * BDI_CAP_MAP_DIRECT: Can be mapped directly (MAP_SHARED) | ||
| 173 | * BDI_CAP_READ_MAP: Can be mapped for reading | ||
| 174 | * BDI_CAP_WRITE_MAP: Can be mapped for writing | ||
| 175 | * BDI_CAP_EXEC_MAP: Can be mapped for execution | ||
| 159 | */ | 176 | */ |
| 160 | #define BDI_CAP_NO_ACCT_DIRTY 0x00000001 /* Dirty pages shouldn't contribute to accounting */ | 177 | #define BDI_CAP_NO_ACCT_DIRTY 0x00000001 |
| 161 | #define BDI_CAP_NO_WRITEBACK 0x00000002 /* Don't write pages back */ | 178 | #define BDI_CAP_NO_WRITEBACK 0x00000002 |
| 162 | #define BDI_CAP_MAP_COPY 0x00000004 /* Copy can be mapped (MAP_PRIVATE) */ | 179 | #define BDI_CAP_MAP_COPY 0x00000004 |
| 163 | #define BDI_CAP_MAP_DIRECT 0x00000008 /* Can be mapped directly (MAP_SHARED) */ | 180 | #define BDI_CAP_MAP_DIRECT 0x00000008 |
| 164 | #define BDI_CAP_READ_MAP 0x00000010 /* Can be mapped for reading */ | 181 | #define BDI_CAP_READ_MAP 0x00000010 |
| 165 | #define BDI_CAP_WRITE_MAP 0x00000020 /* Can be mapped for writing */ | 182 | #define BDI_CAP_WRITE_MAP 0x00000020 |
| 166 | #define BDI_CAP_EXEC_MAP 0x00000040 /* Can be mapped for execution */ | 183 | #define BDI_CAP_EXEC_MAP 0x00000040 |
| 184 | #define BDI_CAP_NO_ACCT_WB 0x00000080 | ||
| 185 | |||
| 167 | #define BDI_CAP_VMFLAGS \ | 186 | #define BDI_CAP_VMFLAGS \ |
| 168 | (BDI_CAP_READ_MAP | BDI_CAP_WRITE_MAP | BDI_CAP_EXEC_MAP) | 187 | (BDI_CAP_READ_MAP | BDI_CAP_WRITE_MAP | BDI_CAP_EXEC_MAP) |
| 169 | 188 | ||
| 189 | #define BDI_CAP_NO_ACCT_AND_WRITEBACK \ | ||
| 190 | (BDI_CAP_NO_WRITEBACK | BDI_CAP_NO_ACCT_DIRTY | BDI_CAP_NO_ACCT_WB) | ||
| 191 | |||
| 170 | #if defined(VM_MAYREAD) && \ | 192 | #if defined(VM_MAYREAD) && \ |
| 171 | (BDI_CAP_READ_MAP != VM_MAYREAD || \ | 193 | (BDI_CAP_READ_MAP != VM_MAYREAD || \ |
| 172 | BDI_CAP_WRITE_MAP != VM_MAYWRITE || \ | 194 | BDI_CAP_WRITE_MAP != VM_MAYWRITE || \ |
| @@ -206,17 +228,32 @@ void clear_bdi_congested(struct backing_dev_info *bdi, int rw); | |||
| 206 | void set_bdi_congested(struct backing_dev_info *bdi, int rw); | 228 | void set_bdi_congested(struct backing_dev_info *bdi, int rw); |
| 207 | long congestion_wait(int rw, long timeout); | 229 | long congestion_wait(int rw, long timeout); |
| 208 | 230 | ||
| 209 | #define bdi_cap_writeback_dirty(bdi) \ | ||
| 210 | (!((bdi)->capabilities & BDI_CAP_NO_WRITEBACK)) | ||
| 211 | 231 | ||
| 212 | #define bdi_cap_account_dirty(bdi) \ | 232 | static inline bool bdi_cap_writeback_dirty(struct backing_dev_info *bdi) |
| 213 | (!((bdi)->capabilities & BDI_CAP_NO_ACCT_DIRTY)) | 233 | { |
| 234 | return !(bdi->capabilities & BDI_CAP_NO_WRITEBACK); | ||
| 235 | } | ||
| 236 | |||
| 237 | static inline bool bdi_cap_account_dirty(struct backing_dev_info *bdi) | ||
| 238 | { | ||
| 239 | return !(bdi->capabilities & BDI_CAP_NO_ACCT_DIRTY); | ||
| 240 | } | ||
| 214 | 241 | ||
| 215 | #define mapping_cap_writeback_dirty(mapping) \ | 242 | static inline bool bdi_cap_account_writeback(struct backing_dev_info *bdi) |
| 216 | bdi_cap_writeback_dirty((mapping)->backing_dev_info) | 243 | { |
| 244 | /* Paranoia: BDI_CAP_NO_WRITEBACK implies BDI_CAP_NO_ACCT_WB */ | ||
| 245 | return !(bdi->capabilities & (BDI_CAP_NO_ACCT_WB | | ||
| 246 | BDI_CAP_NO_WRITEBACK)); | ||
| 247 | } | ||
| 217 | 248 | ||
| 218 | #define mapping_cap_account_dirty(mapping) \ | 249 | static inline bool mapping_cap_writeback_dirty(struct address_space *mapping) |
| 219 | bdi_cap_account_dirty((mapping)->backing_dev_info) | 250 | { |
| 251 | return bdi_cap_writeback_dirty(mapping->backing_dev_info); | ||
| 252 | } | ||
| 220 | 253 | ||
| 254 | static inline bool mapping_cap_account_dirty(struct address_space *mapping) | ||
| 255 | { | ||
| 256 | return bdi_cap_account_dirty(mapping->backing_dev_info); | ||
| 257 | } | ||
| 221 | 258 | ||
| 222 | #endif /* _LINUX_BACKING_DEV_H */ | 259 | #endif /* _LINUX_BACKING_DEV_H */ |
diff --git a/kernel/cgroup.c b/kernel/cgroup.c index b9d467d83fc1..fbc6fc8949b4 100644 --- a/kernel/cgroup.c +++ b/kernel/cgroup.c | |||
| @@ -575,7 +575,7 @@ static struct inode_operations cgroup_dir_inode_operations; | |||
| 575 | static struct file_operations proc_cgroupstats_operations; | 575 | static struct file_operations proc_cgroupstats_operations; |
| 576 | 576 | ||
| 577 | static struct backing_dev_info cgroup_backing_dev_info = { | 577 | static struct backing_dev_info cgroup_backing_dev_info = { |
| 578 | .capabilities = BDI_CAP_NO_ACCT_DIRTY | BDI_CAP_NO_WRITEBACK, | 578 | .capabilities = BDI_CAP_NO_ACCT_AND_WRITEBACK, |
| 579 | }; | 579 | }; |
| 580 | 580 | ||
| 581 | static struct inode *cgroup_new_inode(mode_t mode, struct super_block *sb) | 581 | static struct inode *cgroup_new_inode(mode_t mode, struct super_block *sb) |
diff --git a/mm/page-writeback.c b/mm/page-writeback.c index 2a9942f5387c..bbcb916190c9 100644 --- a/mm/page-writeback.c +++ b/mm/page-writeback.c | |||
| @@ -1246,7 +1246,7 @@ int test_clear_page_writeback(struct page *page) | |||
| 1246 | radix_tree_tag_clear(&mapping->page_tree, | 1246 | radix_tree_tag_clear(&mapping->page_tree, |
| 1247 | page_index(page), | 1247 | page_index(page), |
| 1248 | PAGECACHE_TAG_WRITEBACK); | 1248 | PAGECACHE_TAG_WRITEBACK); |
| 1249 | if (bdi_cap_writeback_dirty(bdi)) { | 1249 | if (bdi_cap_account_writeback(bdi)) { |
| 1250 | __dec_bdi_stat(bdi, BDI_WRITEBACK); | 1250 | __dec_bdi_stat(bdi, BDI_WRITEBACK); |
| 1251 | __bdi_writeout_inc(bdi); | 1251 | __bdi_writeout_inc(bdi); |
| 1252 | } | 1252 | } |
| @@ -1275,7 +1275,7 @@ int test_set_page_writeback(struct page *page) | |||
| 1275 | radix_tree_tag_set(&mapping->page_tree, | 1275 | radix_tree_tag_set(&mapping->page_tree, |
| 1276 | page_index(page), | 1276 | page_index(page), |
| 1277 | PAGECACHE_TAG_WRITEBACK); | 1277 | PAGECACHE_TAG_WRITEBACK); |
| 1278 | if (bdi_cap_writeback_dirty(bdi)) | 1278 | if (bdi_cap_account_writeback(bdi)) |
| 1279 | __inc_bdi_stat(bdi, BDI_WRITEBACK); | 1279 | __inc_bdi_stat(bdi, BDI_WRITEBACK); |
| 1280 | } | 1280 | } |
| 1281 | if (!PageDirty(page)) | 1281 | if (!PageDirty(page)) |
diff --git a/mm/shmem.c b/mm/shmem.c index e6d9298aa22a..e2a6ae1a44e9 100644 --- a/mm/shmem.c +++ b/mm/shmem.c | |||
| @@ -201,7 +201,7 @@ static struct vm_operations_struct shmem_vm_ops; | |||
| 201 | 201 | ||
| 202 | static struct backing_dev_info shmem_backing_dev_info __read_mostly = { | 202 | static struct backing_dev_info shmem_backing_dev_info __read_mostly = { |
| 203 | .ra_pages = 0, /* No readahead */ | 203 | .ra_pages = 0, /* No readahead */ |
| 204 | .capabilities = BDI_CAP_NO_ACCT_DIRTY | BDI_CAP_NO_WRITEBACK, | 204 | .capabilities = BDI_CAP_NO_ACCT_AND_WRITEBACK, |
| 205 | .unplug_io_fn = default_unplug_io_fn, | 205 | .unplug_io_fn = default_unplug_io_fn, |
| 206 | }; | 206 | }; |
| 207 | 207 | ||
diff --git a/mm/swap_state.c b/mm/swap_state.c index 50757ee3f9f3..d8aadaf2a0ba 100644 --- a/mm/swap_state.c +++ b/mm/swap_state.c | |||
| @@ -33,7 +33,7 @@ static const struct address_space_operations swap_aops = { | |||
| 33 | }; | 33 | }; |
| 34 | 34 | ||
| 35 | static struct backing_dev_info swap_backing_dev_info = { | 35 | static struct backing_dev_info swap_backing_dev_info = { |
| 36 | .capabilities = BDI_CAP_NO_ACCT_DIRTY | BDI_CAP_NO_WRITEBACK, | 36 | .capabilities = BDI_CAP_NO_ACCT_AND_WRITEBACK, |
| 37 | .unplug_io_fn = swap_unplug_io_fn, | 37 | .unplug_io_fn = swap_unplug_io_fn, |
| 38 | }; | 38 | }; |
| 39 | 39 | ||
