11da177e4SLinus Torvalds /* 21da177e4SLinus Torvalds * include/linux/backing-dev.h 31da177e4SLinus Torvalds * 41da177e4SLinus Torvalds * low-level device information and state which is propagated up through 51da177e4SLinus Torvalds * to high-level code. 61da177e4SLinus Torvalds */ 71da177e4SLinus Torvalds 81da177e4SLinus Torvalds #ifndef _LINUX_BACKING_DEV_H 91da177e4SLinus Torvalds #define _LINUX_BACKING_DEV_H 101da177e4SLinus Torvalds 11cf0ca9feSPeter Zijlstra #include <linux/kernel.h> 12e4ad08feSMiklos Szeredi #include <linux/fs.h> 1303ba3782SJens Axboe #include <linux/sched.h> 14a212b105STejun Heo #include <linux/blkdev.h> 1503ba3782SJens Axboe #include <linux/writeback.h> 1652ebea74STejun Heo #include <linux/blk-cgroup.h> 1766114cadSTejun Heo #include <linux/backing-dev-defs.h> 181da177e4SLinus Torvalds 198077c0d9SMikulas Patocka int __must_check bdi_init(struct backing_dev_info *bdi); 20b2e8fb6eSPeter Zijlstra void bdi_destroy(struct backing_dev_info *bdi); 21b2e8fb6eSPeter Zijlstra 22d2cc4ddeSJoe Perches __printf(3, 4) 23cf0ca9feSPeter Zijlstra int bdi_register(struct backing_dev_info *bdi, struct device *parent, 24cf0ca9feSPeter Zijlstra const char *fmt, ...); 25cf0ca9feSPeter Zijlstra int bdi_register_dev(struct backing_dev_info *bdi, dev_t dev); 26cf0ca9feSPeter Zijlstra void bdi_unregister(struct backing_dev_info *bdi); 27b4caecd4SChristoph Hellwig int __must_check bdi_setup_and_register(struct backing_dev_info *, char *); 280e175a18SCurt Wohlgemuth void bdi_start_writeback(struct backing_dev_info *bdi, long nr_pages, 290e175a18SCurt Wohlgemuth enum wb_reason reason); 30c5444198SChristoph Hellwig void bdi_start_background_writeback(struct backing_dev_info *bdi); 31f0054bb1STejun Heo void wb_workfn(struct work_struct *work); 32f0054bb1STejun Heo void wb_wakeup_delayed(struct bdi_writeback *wb); 33cf0ca9feSPeter Zijlstra 3403ba3782SJens Axboe extern spinlock_t bdi_lock; 3566f3b8e2SJens Axboe extern struct list_head bdi_list; 3666f3b8e2SJens Axboe 37839a8e86STejun Heo extern struct workqueue_struct *bdi_wq; 38839a8e86STejun Heo 39d6c10f1fSTejun Heo static inline bool wb_has_dirty_io(struct bdi_writeback *wb) 4003ba3782SJens Axboe { 41d6c10f1fSTejun Heo return test_bit(WB_has_dirty_io, &wb->state); 4203ba3782SJens Axboe } 4303ba3782SJens Axboe 4495a46c65STejun Heo static inline bool bdi_has_dirty_io(struct backing_dev_info *bdi) 4595a46c65STejun Heo { 4695a46c65STejun Heo /* 4795a46c65STejun Heo * @bdi->tot_write_bandwidth is guaranteed to be > 0 if there are 4895a46c65STejun Heo * any dirty wbs. See wb_update_write_bandwidth(). 4995a46c65STejun Heo */ 5095a46c65STejun Heo return atomic_long_read(&bdi->tot_write_bandwidth); 5195a46c65STejun Heo } 5295a46c65STejun Heo 5393f78d88STejun Heo static inline void __add_wb_stat(struct bdi_writeback *wb, 5493f78d88STejun Heo enum wb_stat_item item, s64 amount) 55e0bf68ddSPeter Zijlstra { 5693f78d88STejun Heo __percpu_counter_add(&wb->stat[item], amount, WB_STAT_BATCH); 57e0bf68ddSPeter Zijlstra } 58e0bf68ddSPeter Zijlstra 5993f78d88STejun Heo static inline void __inc_wb_stat(struct bdi_writeback *wb, 6093f78d88STejun Heo enum wb_stat_item item) 61e0bf68ddSPeter Zijlstra { 6293f78d88STejun Heo __add_wb_stat(wb, item, 1); 63b2e8fb6eSPeter Zijlstra } 64b2e8fb6eSPeter Zijlstra 6593f78d88STejun Heo static inline void inc_wb_stat(struct bdi_writeback *wb, enum wb_stat_item item) 66b2e8fb6eSPeter Zijlstra { 67b2e8fb6eSPeter Zijlstra unsigned long flags; 68b2e8fb6eSPeter Zijlstra 69b2e8fb6eSPeter Zijlstra local_irq_save(flags); 7093f78d88STejun Heo __inc_wb_stat(wb, item); 71b2e8fb6eSPeter Zijlstra local_irq_restore(flags); 72b2e8fb6eSPeter Zijlstra } 73b2e8fb6eSPeter Zijlstra 7493f78d88STejun Heo static inline void __dec_wb_stat(struct bdi_writeback *wb, 7593f78d88STejun Heo enum wb_stat_item item) 76b2e8fb6eSPeter Zijlstra { 7793f78d88STejun Heo __add_wb_stat(wb, item, -1); 78b2e8fb6eSPeter Zijlstra } 79b2e8fb6eSPeter Zijlstra 8093f78d88STejun Heo static inline void dec_wb_stat(struct bdi_writeback *wb, enum wb_stat_item item) 81b2e8fb6eSPeter Zijlstra { 82b2e8fb6eSPeter Zijlstra unsigned long flags; 83b2e8fb6eSPeter Zijlstra 84b2e8fb6eSPeter Zijlstra local_irq_save(flags); 8593f78d88STejun Heo __dec_wb_stat(wb, item); 86b2e8fb6eSPeter Zijlstra local_irq_restore(flags); 87b2e8fb6eSPeter Zijlstra } 88b2e8fb6eSPeter Zijlstra 8993f78d88STejun Heo static inline s64 wb_stat(struct bdi_writeback *wb, enum wb_stat_item item) 90b2e8fb6eSPeter Zijlstra { 9193f78d88STejun Heo return percpu_counter_read_positive(&wb->stat[item]); 92b2e8fb6eSPeter Zijlstra } 93b2e8fb6eSPeter Zijlstra 9493f78d88STejun Heo static inline s64 __wb_stat_sum(struct bdi_writeback *wb, 9593f78d88STejun Heo enum wb_stat_item item) 96b2e8fb6eSPeter Zijlstra { 9793f78d88STejun Heo return percpu_counter_sum_positive(&wb->stat[item]); 98b2e8fb6eSPeter Zijlstra } 99b2e8fb6eSPeter Zijlstra 10093f78d88STejun Heo static inline s64 wb_stat_sum(struct bdi_writeback *wb, enum wb_stat_item item) 101b2e8fb6eSPeter Zijlstra { 102b2e8fb6eSPeter Zijlstra s64 sum; 103b2e8fb6eSPeter Zijlstra unsigned long flags; 104b2e8fb6eSPeter Zijlstra 105b2e8fb6eSPeter Zijlstra local_irq_save(flags); 10693f78d88STejun Heo sum = __wb_stat_sum(wb, item); 107b2e8fb6eSPeter Zijlstra local_irq_restore(flags); 108b2e8fb6eSPeter Zijlstra 109b2e8fb6eSPeter Zijlstra return sum; 110b2e8fb6eSPeter Zijlstra } 111b2e8fb6eSPeter Zijlstra 11293f78d88STejun Heo extern void wb_writeout_inc(struct bdi_writeback *wb); 113dd5656e5SMiklos Szeredi 114b2e8fb6eSPeter Zijlstra /* 115b2e8fb6eSPeter Zijlstra * maximal error of a stat counter. 116b2e8fb6eSPeter Zijlstra */ 11793f78d88STejun Heo static inline unsigned long wb_stat_error(struct bdi_writeback *wb) 118b2e8fb6eSPeter Zijlstra { 119b2e8fb6eSPeter Zijlstra #ifdef CONFIG_SMP 12093f78d88STejun Heo return nr_cpu_ids * WB_STAT_BATCH; 121b2e8fb6eSPeter Zijlstra #else 122b2e8fb6eSPeter Zijlstra return 1; 123b2e8fb6eSPeter Zijlstra #endif 124e0bf68ddSPeter Zijlstra } 1251da177e4SLinus Torvalds 126189d3c4aSPeter Zijlstra int bdi_set_min_ratio(struct backing_dev_info *bdi, unsigned int min_ratio); 127a42dde04SPeter Zijlstra int bdi_set_max_ratio(struct backing_dev_info *bdi, unsigned int max_ratio); 128189d3c4aSPeter Zijlstra 1291da177e4SLinus Torvalds /* 1301da177e4SLinus Torvalds * Flags in backing_dev_info::capability 131e4ad08feSMiklos Szeredi * 132e4ad08feSMiklos Szeredi * The first three flags control whether dirty pages will contribute to the 1331da177e4SLinus Torvalds * VM's accounting and whether writepages() should be called for dirty pages 1341da177e4SLinus Torvalds * (something that would not, for example, be appropriate for ramfs) 135e4ad08feSMiklos Szeredi * 136e4ad08feSMiklos Szeredi * WARNING: these flags are closely related and should not normally be 137e4ad08feSMiklos Szeredi * used separately. The BDI_CAP_NO_ACCT_AND_WRITEBACK combines these 138e4ad08feSMiklos Szeredi * three flags into a single convenience macro. 139e4ad08feSMiklos Szeredi * 140e4ad08feSMiklos Szeredi * BDI_CAP_NO_ACCT_DIRTY: Dirty pages shouldn't contribute to accounting 141e4ad08feSMiklos Szeredi * BDI_CAP_NO_WRITEBACK: Don't write pages back 142e4ad08feSMiklos Szeredi * BDI_CAP_NO_ACCT_WB: Don't automatically account writeback pages 1435a537485SMaxim Patlasov * BDI_CAP_STRICTLIMIT: Keep number of dirty pages below bdi threshold. 14489e9b9e0STejun Heo * 14589e9b9e0STejun Heo * BDI_CAP_CGROUP_WRITEBACK: Supports cgroup-aware writeback. 1461da177e4SLinus Torvalds */ 147e4ad08feSMiklos Szeredi #define BDI_CAP_NO_ACCT_DIRTY 0x00000001 148e4ad08feSMiklos Szeredi #define BDI_CAP_NO_WRITEBACK 0x00000002 149b4caecd4SChristoph Hellwig #define BDI_CAP_NO_ACCT_WB 0x00000004 150b4caecd4SChristoph Hellwig #define BDI_CAP_STABLE_WRITES 0x00000008 151b4caecd4SChristoph Hellwig #define BDI_CAP_STRICTLIMIT 0x00000010 15289e9b9e0STejun Heo #define BDI_CAP_CGROUP_WRITEBACK 0x00000020 1531da177e4SLinus Torvalds 154e4ad08feSMiklos Szeredi #define BDI_CAP_NO_ACCT_AND_WRITEBACK \ 155e4ad08feSMiklos Szeredi (BDI_CAP_NO_WRITEBACK | BDI_CAP_NO_ACCT_DIRTY | BDI_CAP_NO_ACCT_WB) 156e4ad08feSMiklos Szeredi 1575129a469SJörn Engel extern struct backing_dev_info noop_backing_dev_info; 1581da177e4SLinus Torvalds 1591da177e4SLinus Torvalds int writeback_in_progress(struct backing_dev_info *bdi); 1601da177e4SLinus Torvalds 161a212b105STejun Heo static inline struct backing_dev_info *inode_to_bdi(struct inode *inode) 162a212b105STejun Heo { 163a212b105STejun Heo struct super_block *sb; 164a212b105STejun Heo 165a212b105STejun Heo if (!inode) 166a212b105STejun Heo return &noop_backing_dev_info; 167a212b105STejun Heo 168a212b105STejun Heo sb = inode->i_sb; 169a212b105STejun Heo #ifdef CONFIG_BLOCK 170a212b105STejun Heo if (sb_is_blkdev_sb(sb)) 171a212b105STejun Heo return blk_get_backing_dev_info(I_BDEV(inode)); 172a212b105STejun Heo #endif 173a212b105STejun Heo return sb->s_bdi; 174a212b105STejun Heo } 175a212b105STejun Heo 176ec8a6f26STejun Heo static inline int wb_congested(struct bdi_writeback *wb, int cong_bits) 1771da177e4SLinus Torvalds { 178ec8a6f26STejun Heo struct backing_dev_info *bdi = wb->bdi; 179ec8a6f26STejun Heo 1801da177e4SLinus Torvalds if (bdi->congested_fn) 181ec8a6f26STejun Heo return bdi->congested_fn(bdi->congested_data, cong_bits); 182ec8a6f26STejun Heo return wb->congested->state & cong_bits; 1831da177e4SLinus Torvalds } 1841da177e4SLinus Torvalds 1858aa7e847SJens Axboe long congestion_wait(int sync, long timeout); 1860e093d99SMel Gorman long wait_iff_congested(struct zone *zone, int sync, long timeout); 1873965c9aeSWanpeng Li int pdflush_proc_obsolete(struct ctl_table *table, int write, 1883965c9aeSWanpeng Li void __user *buffer, size_t *lenp, loff_t *ppos); 1891da177e4SLinus Torvalds 1907d311cdaSDarrick J. Wong static inline bool bdi_cap_stable_pages_required(struct backing_dev_info *bdi) 1917d311cdaSDarrick J. Wong { 1927d311cdaSDarrick J. Wong return bdi->capabilities & BDI_CAP_STABLE_WRITES; 1937d311cdaSDarrick J. Wong } 1947d311cdaSDarrick J. Wong 195e4ad08feSMiklos Szeredi static inline bool bdi_cap_writeback_dirty(struct backing_dev_info *bdi) 196e4ad08feSMiklos Szeredi { 197e4ad08feSMiklos Szeredi return !(bdi->capabilities & BDI_CAP_NO_WRITEBACK); 198e4ad08feSMiklos Szeredi } 1991da177e4SLinus Torvalds 200e4ad08feSMiklos Szeredi static inline bool bdi_cap_account_dirty(struct backing_dev_info *bdi) 201e4ad08feSMiklos Szeredi { 202e4ad08feSMiklos Szeredi return !(bdi->capabilities & BDI_CAP_NO_ACCT_DIRTY); 203e4ad08feSMiklos Szeredi } 2041da177e4SLinus Torvalds 205e4ad08feSMiklos Szeredi static inline bool bdi_cap_account_writeback(struct backing_dev_info *bdi) 206e4ad08feSMiklos Szeredi { 207e4ad08feSMiklos Szeredi /* Paranoia: BDI_CAP_NO_WRITEBACK implies BDI_CAP_NO_ACCT_WB */ 208e4ad08feSMiklos Szeredi return !(bdi->capabilities & (BDI_CAP_NO_ACCT_WB | 209e4ad08feSMiklos Szeredi BDI_CAP_NO_WRITEBACK)); 210e4ad08feSMiklos Szeredi } 2111da177e4SLinus Torvalds 212e4ad08feSMiklos Szeredi static inline bool mapping_cap_writeback_dirty(struct address_space *mapping) 213e4ad08feSMiklos Szeredi { 214de1414a6SChristoph Hellwig return bdi_cap_writeback_dirty(inode_to_bdi(mapping->host)); 215e4ad08feSMiklos Szeredi } 216e4ad08feSMiklos Szeredi 217e4ad08feSMiklos Szeredi static inline bool mapping_cap_account_dirty(struct address_space *mapping) 218e4ad08feSMiklos Szeredi { 219de1414a6SChristoph Hellwig return bdi_cap_account_dirty(inode_to_bdi(mapping->host)); 220e4ad08feSMiklos Szeredi } 2211da177e4SLinus Torvalds 22203ba3782SJens Axboe static inline int bdi_sched_wait(void *word) 22303ba3782SJens Axboe { 22403ba3782SJens Axboe schedule(); 22503ba3782SJens Axboe return 0; 22603ba3782SJens Axboe } 22703ba3782SJens Axboe 22889e9b9e0STejun Heo #ifdef CONFIG_CGROUP_WRITEBACK 22989e9b9e0STejun Heo 23052ebea74STejun Heo struct bdi_writeback_congested * 23152ebea74STejun Heo wb_congested_get_create(struct backing_dev_info *bdi, int blkcg_id, gfp_t gfp); 23252ebea74STejun Heo void wb_congested_put(struct bdi_writeback_congested *congested); 23352ebea74STejun Heo struct bdi_writeback *wb_get_create(struct backing_dev_info *bdi, 23452ebea74STejun Heo struct cgroup_subsys_state *memcg_css, 23552ebea74STejun Heo gfp_t gfp); 23652ebea74STejun Heo void __inode_attach_wb(struct inode *inode, struct page *page); 23752ebea74STejun Heo void wb_memcg_offline(struct mem_cgroup *memcg); 23852ebea74STejun Heo void wb_blkcg_offline(struct blkcg *blkcg); 239703c2708STejun Heo int inode_congested(struct inode *inode, int cong_bits); 24052ebea74STejun Heo 24189e9b9e0STejun Heo /** 24289e9b9e0STejun Heo * inode_cgwb_enabled - test whether cgroup writeback is enabled on an inode 24389e9b9e0STejun Heo * @inode: inode of interest 24489e9b9e0STejun Heo * 24589e9b9e0STejun Heo * cgroup writeback requires support from both the bdi and filesystem. 24689e9b9e0STejun Heo * Test whether @inode has both. 24789e9b9e0STejun Heo */ 24889e9b9e0STejun Heo static inline bool inode_cgwb_enabled(struct inode *inode) 24989e9b9e0STejun Heo { 25089e9b9e0STejun Heo struct backing_dev_info *bdi = inode_to_bdi(inode); 25189e9b9e0STejun Heo 25289e9b9e0STejun Heo return bdi_cap_account_dirty(bdi) && 25389e9b9e0STejun Heo (bdi->capabilities & BDI_CAP_CGROUP_WRITEBACK) && 25489e9b9e0STejun Heo (inode->i_sb->s_type->fs_flags & FS_CGROUP_WRITEBACK); 25589e9b9e0STejun Heo } 25689e9b9e0STejun Heo 25752ebea74STejun Heo /** 25852ebea74STejun Heo * wb_tryget - try to increment a wb's refcount 25952ebea74STejun Heo * @wb: bdi_writeback to get 26052ebea74STejun Heo */ 26152ebea74STejun Heo static inline bool wb_tryget(struct bdi_writeback *wb) 26252ebea74STejun Heo { 26352ebea74STejun Heo if (wb != &wb->bdi->wb) 26452ebea74STejun Heo return percpu_ref_tryget(&wb->refcnt); 26552ebea74STejun Heo return true; 26652ebea74STejun Heo } 26752ebea74STejun Heo 26852ebea74STejun Heo /** 26952ebea74STejun Heo * wb_get - increment a wb's refcount 27052ebea74STejun Heo * @wb: bdi_writeback to get 27152ebea74STejun Heo */ 27252ebea74STejun Heo static inline void wb_get(struct bdi_writeback *wb) 27352ebea74STejun Heo { 27452ebea74STejun Heo if (wb != &wb->bdi->wb) 27552ebea74STejun Heo percpu_ref_get(&wb->refcnt); 27652ebea74STejun Heo } 27752ebea74STejun Heo 27852ebea74STejun Heo /** 27952ebea74STejun Heo * wb_put - decrement a wb's refcount 28052ebea74STejun Heo * @wb: bdi_writeback to put 28152ebea74STejun Heo */ 28252ebea74STejun Heo static inline void wb_put(struct bdi_writeback *wb) 28352ebea74STejun Heo { 28452ebea74STejun Heo if (wb != &wb->bdi->wb) 28552ebea74STejun Heo percpu_ref_put(&wb->refcnt); 28652ebea74STejun Heo } 28752ebea74STejun Heo 28852ebea74STejun Heo /** 28952ebea74STejun Heo * wb_find_current - find wb for %current on a bdi 29052ebea74STejun Heo * @bdi: bdi of interest 29152ebea74STejun Heo * 29252ebea74STejun Heo * Find the wb of @bdi which matches both the memcg and blkcg of %current. 29352ebea74STejun Heo * Must be called under rcu_read_lock() which protects the returend wb. 29452ebea74STejun Heo * NULL if not found. 29552ebea74STejun Heo */ 29652ebea74STejun Heo static inline struct bdi_writeback *wb_find_current(struct backing_dev_info *bdi) 29752ebea74STejun Heo { 29852ebea74STejun Heo struct cgroup_subsys_state *memcg_css; 29952ebea74STejun Heo struct bdi_writeback *wb; 30052ebea74STejun Heo 30152ebea74STejun Heo memcg_css = task_css(current, memory_cgrp_id); 30252ebea74STejun Heo if (!memcg_css->parent) 30352ebea74STejun Heo return &bdi->wb; 30452ebea74STejun Heo 30552ebea74STejun Heo wb = radix_tree_lookup(&bdi->cgwb_tree, memcg_css->id); 30652ebea74STejun Heo 30752ebea74STejun Heo /* 30852ebea74STejun Heo * %current's blkcg equals the effective blkcg of its memcg. No 30952ebea74STejun Heo * need to use the relatively expensive cgroup_get_e_css(). 31052ebea74STejun Heo */ 31152ebea74STejun Heo if (likely(wb && wb->blkcg_css == task_css(current, blkio_cgrp_id))) 31252ebea74STejun Heo return wb; 31352ebea74STejun Heo return NULL; 31452ebea74STejun Heo } 31552ebea74STejun Heo 31652ebea74STejun Heo /** 31752ebea74STejun Heo * wb_get_create_current - get or create wb for %current on a bdi 31852ebea74STejun Heo * @bdi: bdi of interest 31952ebea74STejun Heo * @gfp: allocation mask 32052ebea74STejun Heo * 32152ebea74STejun Heo * Equivalent to wb_get_create() on %current's memcg. This function is 32252ebea74STejun Heo * called from a relatively hot path and optimizes the common cases using 32352ebea74STejun Heo * wb_find_current(). 32452ebea74STejun Heo */ 32552ebea74STejun Heo static inline struct bdi_writeback * 32652ebea74STejun Heo wb_get_create_current(struct backing_dev_info *bdi, gfp_t gfp) 32752ebea74STejun Heo { 32852ebea74STejun Heo struct bdi_writeback *wb; 32952ebea74STejun Heo 33052ebea74STejun Heo rcu_read_lock(); 33152ebea74STejun Heo wb = wb_find_current(bdi); 33252ebea74STejun Heo if (wb && unlikely(!wb_tryget(wb))) 33352ebea74STejun Heo wb = NULL; 33452ebea74STejun Heo rcu_read_unlock(); 33552ebea74STejun Heo 33652ebea74STejun Heo if (unlikely(!wb)) { 33752ebea74STejun Heo struct cgroup_subsys_state *memcg_css; 33852ebea74STejun Heo 33952ebea74STejun Heo memcg_css = task_get_css(current, memory_cgrp_id); 34052ebea74STejun Heo wb = wb_get_create(bdi, memcg_css, gfp); 34152ebea74STejun Heo css_put(memcg_css); 34252ebea74STejun Heo } 34352ebea74STejun Heo return wb; 34452ebea74STejun Heo } 34552ebea74STejun Heo 34652ebea74STejun Heo /** 34752ebea74STejun Heo * inode_attach_wb - associate an inode with its wb 34852ebea74STejun Heo * @inode: inode of interest 34952ebea74STejun Heo * @page: page being dirtied (may be NULL) 35052ebea74STejun Heo * 35152ebea74STejun Heo * If @inode doesn't have its wb, associate it with the wb matching the 35252ebea74STejun Heo * memcg of @page or, if @page is NULL, %current. May be called w/ or w/o 35352ebea74STejun Heo * @inode->i_lock. 35452ebea74STejun Heo */ 35552ebea74STejun Heo static inline void inode_attach_wb(struct inode *inode, struct page *page) 35652ebea74STejun Heo { 35752ebea74STejun Heo if (!inode->i_wb) 35852ebea74STejun Heo __inode_attach_wb(inode, page); 35952ebea74STejun Heo } 36052ebea74STejun Heo 36152ebea74STejun Heo /** 36252ebea74STejun Heo * inode_detach_wb - disassociate an inode from its wb 36352ebea74STejun Heo * @inode: inode of interest 36452ebea74STejun Heo * 36552ebea74STejun Heo * @inode is being freed. Detach from its wb. 36652ebea74STejun Heo */ 36752ebea74STejun Heo static inline void inode_detach_wb(struct inode *inode) 36852ebea74STejun Heo { 36952ebea74STejun Heo if (inode->i_wb) { 37052ebea74STejun Heo wb_put(inode->i_wb); 37152ebea74STejun Heo inode->i_wb = NULL; 37252ebea74STejun Heo } 37352ebea74STejun Heo } 37452ebea74STejun Heo 37552ebea74STejun Heo /** 37652ebea74STejun Heo * inode_to_wb - determine the wb of an inode 37752ebea74STejun Heo * @inode: inode of interest 37852ebea74STejun Heo * 37952ebea74STejun Heo * Returns the wb @inode is currently associated with. 38052ebea74STejun Heo */ 38152ebea74STejun Heo static inline struct bdi_writeback *inode_to_wb(struct inode *inode) 38252ebea74STejun Heo { 38352ebea74STejun Heo return inode->i_wb; 38452ebea74STejun Heo } 38552ebea74STejun Heo 386*ebe41ab0STejun Heo struct wb_iter { 387*ebe41ab0STejun Heo int start_blkcg_id; 388*ebe41ab0STejun Heo struct radix_tree_iter tree_iter; 389*ebe41ab0STejun Heo void **slot; 390*ebe41ab0STejun Heo }; 391*ebe41ab0STejun Heo 392*ebe41ab0STejun Heo static inline struct bdi_writeback *__wb_iter_next(struct wb_iter *iter, 393*ebe41ab0STejun Heo struct backing_dev_info *bdi) 394*ebe41ab0STejun Heo { 395*ebe41ab0STejun Heo struct radix_tree_iter *titer = &iter->tree_iter; 396*ebe41ab0STejun Heo 397*ebe41ab0STejun Heo WARN_ON_ONCE(!rcu_read_lock_held()); 398*ebe41ab0STejun Heo 399*ebe41ab0STejun Heo if (iter->start_blkcg_id >= 0) { 400*ebe41ab0STejun Heo iter->slot = radix_tree_iter_init(titer, iter->start_blkcg_id); 401*ebe41ab0STejun Heo iter->start_blkcg_id = -1; 402*ebe41ab0STejun Heo } else { 403*ebe41ab0STejun Heo iter->slot = radix_tree_next_slot(iter->slot, titer, 0); 404*ebe41ab0STejun Heo } 405*ebe41ab0STejun Heo 406*ebe41ab0STejun Heo if (!iter->slot) 407*ebe41ab0STejun Heo iter->slot = radix_tree_next_chunk(&bdi->cgwb_tree, titer, 0); 408*ebe41ab0STejun Heo if (iter->slot) 409*ebe41ab0STejun Heo return *iter->slot; 410*ebe41ab0STejun Heo return NULL; 411*ebe41ab0STejun Heo } 412*ebe41ab0STejun Heo 413*ebe41ab0STejun Heo static inline struct bdi_writeback *__wb_iter_init(struct wb_iter *iter, 414*ebe41ab0STejun Heo struct backing_dev_info *bdi, 415*ebe41ab0STejun Heo int start_blkcg_id) 416*ebe41ab0STejun Heo { 417*ebe41ab0STejun Heo iter->start_blkcg_id = start_blkcg_id; 418*ebe41ab0STejun Heo 419*ebe41ab0STejun Heo if (start_blkcg_id) 420*ebe41ab0STejun Heo return __wb_iter_next(iter, bdi); 421*ebe41ab0STejun Heo else 422*ebe41ab0STejun Heo return &bdi->wb; 423*ebe41ab0STejun Heo } 424*ebe41ab0STejun Heo 425*ebe41ab0STejun Heo /** 426*ebe41ab0STejun Heo * bdi_for_each_wb - walk all wb's of a bdi in ascending blkcg ID order 427*ebe41ab0STejun Heo * @wb_cur: cursor struct bdi_writeback pointer 428*ebe41ab0STejun Heo * @bdi: bdi to walk wb's of 429*ebe41ab0STejun Heo * @iter: pointer to struct wb_iter to be used as iteration buffer 430*ebe41ab0STejun Heo * @start_blkcg_id: blkcg ID to start iteration from 431*ebe41ab0STejun Heo * 432*ebe41ab0STejun Heo * Iterate @wb_cur through the wb's (bdi_writeback's) of @bdi in ascending 433*ebe41ab0STejun Heo * blkcg ID order starting from @start_blkcg_id. @iter is struct wb_iter 434*ebe41ab0STejun Heo * to be used as temp storage during iteration. rcu_read_lock() must be 435*ebe41ab0STejun Heo * held throughout iteration. 436*ebe41ab0STejun Heo */ 437*ebe41ab0STejun Heo #define bdi_for_each_wb(wb_cur, bdi, iter, start_blkcg_id) \ 438*ebe41ab0STejun Heo for ((wb_cur) = __wb_iter_init(iter, bdi, start_blkcg_id); \ 439*ebe41ab0STejun Heo (wb_cur); (wb_cur) = __wb_iter_next(iter, bdi)) 440*ebe41ab0STejun Heo 44189e9b9e0STejun Heo #else /* CONFIG_CGROUP_WRITEBACK */ 44289e9b9e0STejun Heo 44389e9b9e0STejun Heo static inline bool inode_cgwb_enabled(struct inode *inode) 44489e9b9e0STejun Heo { 44589e9b9e0STejun Heo return false; 44689e9b9e0STejun Heo } 44789e9b9e0STejun Heo 44852ebea74STejun Heo static inline struct bdi_writeback_congested * 44952ebea74STejun Heo wb_congested_get_create(struct backing_dev_info *bdi, int blkcg_id, gfp_t gfp) 45052ebea74STejun Heo { 45152ebea74STejun Heo return bdi->wb.congested; 45252ebea74STejun Heo } 45352ebea74STejun Heo 45452ebea74STejun Heo static inline void wb_congested_put(struct bdi_writeback_congested *congested) 45552ebea74STejun Heo { 45652ebea74STejun Heo } 45752ebea74STejun Heo 45852ebea74STejun Heo static inline bool wb_tryget(struct bdi_writeback *wb) 45952ebea74STejun Heo { 46052ebea74STejun Heo return true; 46152ebea74STejun Heo } 46252ebea74STejun Heo 46352ebea74STejun Heo static inline void wb_get(struct bdi_writeback *wb) 46452ebea74STejun Heo { 46552ebea74STejun Heo } 46652ebea74STejun Heo 46752ebea74STejun Heo static inline void wb_put(struct bdi_writeback *wb) 46852ebea74STejun Heo { 46952ebea74STejun Heo } 47052ebea74STejun Heo 47152ebea74STejun Heo static inline struct bdi_writeback *wb_find_current(struct backing_dev_info *bdi) 47252ebea74STejun Heo { 47352ebea74STejun Heo return &bdi->wb; 47452ebea74STejun Heo } 47552ebea74STejun Heo 47652ebea74STejun Heo static inline struct bdi_writeback * 47752ebea74STejun Heo wb_get_create_current(struct backing_dev_info *bdi, gfp_t gfp) 47852ebea74STejun Heo { 47952ebea74STejun Heo return &bdi->wb; 48052ebea74STejun Heo } 48152ebea74STejun Heo 48252ebea74STejun Heo static inline void inode_attach_wb(struct inode *inode, struct page *page) 48352ebea74STejun Heo { 48452ebea74STejun Heo } 48552ebea74STejun Heo 48652ebea74STejun Heo static inline void inode_detach_wb(struct inode *inode) 48752ebea74STejun Heo { 48852ebea74STejun Heo } 48952ebea74STejun Heo 49052ebea74STejun Heo static inline struct bdi_writeback *inode_to_wb(struct inode *inode) 49152ebea74STejun Heo { 49252ebea74STejun Heo return &inode_to_bdi(inode)->wb; 49352ebea74STejun Heo } 49452ebea74STejun Heo 49552ebea74STejun Heo static inline void wb_memcg_offline(struct mem_cgroup *memcg) 49652ebea74STejun Heo { 49752ebea74STejun Heo } 49852ebea74STejun Heo 49952ebea74STejun Heo static inline void wb_blkcg_offline(struct blkcg *blkcg) 50052ebea74STejun Heo { 50152ebea74STejun Heo } 50252ebea74STejun Heo 503*ebe41ab0STejun Heo struct wb_iter { 504*ebe41ab0STejun Heo int next_id; 505*ebe41ab0STejun Heo }; 506*ebe41ab0STejun Heo 507*ebe41ab0STejun Heo #define bdi_for_each_wb(wb_cur, bdi, iter, start_blkcg_id) \ 508*ebe41ab0STejun Heo for ((iter)->next_id = (start_blkcg_id); \ 509*ebe41ab0STejun Heo ({ (wb_cur) = !(iter)->next_id++ ? &(bdi)->wb : NULL; }); ) 510*ebe41ab0STejun Heo 511703c2708STejun Heo static inline int inode_congested(struct inode *inode, int cong_bits) 512703c2708STejun Heo { 513703c2708STejun Heo return wb_congested(&inode_to_bdi(inode)->wb, cong_bits); 514703c2708STejun Heo } 515703c2708STejun Heo 51689e9b9e0STejun Heo #endif /* CONFIG_CGROUP_WRITEBACK */ 51789e9b9e0STejun Heo 518703c2708STejun Heo static inline int inode_read_congested(struct inode *inode) 519703c2708STejun Heo { 520703c2708STejun Heo return inode_congested(inode, 1 << WB_sync_congested); 521703c2708STejun Heo } 522703c2708STejun Heo 523703c2708STejun Heo static inline int inode_write_congested(struct inode *inode) 524703c2708STejun Heo { 525703c2708STejun Heo return inode_congested(inode, 1 << WB_async_congested); 526703c2708STejun Heo } 527703c2708STejun Heo 528703c2708STejun Heo static inline int inode_rw_congested(struct inode *inode) 529703c2708STejun Heo { 530703c2708STejun Heo return inode_congested(inode, (1 << WB_sync_congested) | 531703c2708STejun Heo (1 << WB_async_congested)); 532703c2708STejun Heo } 533703c2708STejun Heo 534ec8a6f26STejun Heo static inline int bdi_congested(struct backing_dev_info *bdi, int cong_bits) 535ec8a6f26STejun Heo { 536ec8a6f26STejun Heo return wb_congested(&bdi->wb, cong_bits); 537ec8a6f26STejun Heo } 538ec8a6f26STejun Heo 539ec8a6f26STejun Heo static inline int bdi_read_congested(struct backing_dev_info *bdi) 540ec8a6f26STejun Heo { 541ec8a6f26STejun Heo return bdi_congested(bdi, 1 << WB_sync_congested); 542ec8a6f26STejun Heo } 543ec8a6f26STejun Heo 544ec8a6f26STejun Heo static inline int bdi_write_congested(struct backing_dev_info *bdi) 545ec8a6f26STejun Heo { 546ec8a6f26STejun Heo return bdi_congested(bdi, 1 << WB_async_congested); 547ec8a6f26STejun Heo } 548ec8a6f26STejun Heo 549ec8a6f26STejun Heo static inline int bdi_rw_congested(struct backing_dev_info *bdi) 550ec8a6f26STejun Heo { 551ec8a6f26STejun Heo return bdi_congested(bdi, (1 << WB_sync_congested) | 552ec8a6f26STejun Heo (1 << WB_async_congested)); 553ec8a6f26STejun Heo } 554ec8a6f26STejun Heo 5551da177e4SLinus Torvalds #endif /* _LINUX_BACKING_DEV_H */ 556