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 11b2e8fb6eSPeter Zijlstra #include <linux/percpu_counter.h> 12b2e8fb6eSPeter Zijlstra #include <linux/log2.h> 1304fbfdc1SPeter Zijlstra #include <linux/proportions.h> 14*cf0ca9feSPeter Zijlstra #include <linux/kernel.h> 151da177e4SLinus Torvalds #include <asm/atomic.h> 161da177e4SLinus Torvalds 173fcfab16SAndrew Morton struct page; 18*cf0ca9feSPeter Zijlstra struct device; 193fcfab16SAndrew Morton 201da177e4SLinus Torvalds /* 211da177e4SLinus Torvalds * Bits in backing_dev_info.state 221da177e4SLinus Torvalds */ 231da177e4SLinus Torvalds enum bdi_state { 241da177e4SLinus Torvalds BDI_pdflush, /* A pdflush thread is working this device */ 251da177e4SLinus Torvalds BDI_write_congested, /* The write queue is getting full */ 261da177e4SLinus Torvalds BDI_read_congested, /* The read queue is getting full */ 271da177e4SLinus Torvalds BDI_unused, /* Available bits start here */ 281da177e4SLinus Torvalds }; 291da177e4SLinus Torvalds 301da177e4SLinus Torvalds typedef int (congested_fn)(void *, int); 311da177e4SLinus Torvalds 32b2e8fb6eSPeter Zijlstra enum bdi_stat_item { 33c9e51e41SPeter Zijlstra BDI_RECLAIMABLE, 3469cb51d1SPeter Zijlstra BDI_WRITEBACK, 35b2e8fb6eSPeter Zijlstra NR_BDI_STAT_ITEMS 36b2e8fb6eSPeter Zijlstra }; 37b2e8fb6eSPeter Zijlstra 38b2e8fb6eSPeter Zijlstra #define BDI_STAT_BATCH (8*(1+ilog2(nr_cpu_ids))) 39b2e8fb6eSPeter Zijlstra 401da177e4SLinus Torvalds struct backing_dev_info { 411da177e4SLinus Torvalds unsigned long ra_pages; /* max readahead in PAGE_CACHE_SIZE units */ 421da177e4SLinus Torvalds unsigned long state; /* Always use atomic bitops on this */ 431da177e4SLinus Torvalds unsigned int capabilities; /* Device capabilities */ 441da177e4SLinus Torvalds congested_fn *congested_fn; /* Function pointer if device is md/dm */ 451da177e4SLinus Torvalds void *congested_data; /* Pointer to aux data for congested func */ 461da177e4SLinus Torvalds void (*unplug_io_fn)(struct backing_dev_info *, struct page *); 471da177e4SLinus Torvalds void *unplug_io_data; 48b2e8fb6eSPeter Zijlstra 49b2e8fb6eSPeter Zijlstra struct percpu_counter bdi_stat[NR_BDI_STAT_ITEMS]; 5004fbfdc1SPeter Zijlstra 5104fbfdc1SPeter Zijlstra struct prop_local_percpu completions; 5204fbfdc1SPeter Zijlstra int dirty_exceeded; 53*cf0ca9feSPeter Zijlstra 54*cf0ca9feSPeter Zijlstra struct device *dev; 551da177e4SLinus Torvalds }; 561da177e4SLinus Torvalds 57b2e8fb6eSPeter Zijlstra int bdi_init(struct backing_dev_info *bdi); 58b2e8fb6eSPeter Zijlstra void bdi_destroy(struct backing_dev_info *bdi); 59b2e8fb6eSPeter Zijlstra 60*cf0ca9feSPeter Zijlstra int bdi_register(struct backing_dev_info *bdi, struct device *parent, 61*cf0ca9feSPeter Zijlstra const char *fmt, ...); 62*cf0ca9feSPeter Zijlstra int bdi_register_dev(struct backing_dev_info *bdi, dev_t dev); 63*cf0ca9feSPeter Zijlstra void bdi_unregister(struct backing_dev_info *bdi); 64*cf0ca9feSPeter Zijlstra 65b2e8fb6eSPeter Zijlstra static inline void __add_bdi_stat(struct backing_dev_info *bdi, 66b2e8fb6eSPeter Zijlstra enum bdi_stat_item item, s64 amount) 67e0bf68ddSPeter Zijlstra { 68b2e8fb6eSPeter Zijlstra __percpu_counter_add(&bdi->bdi_stat[item], amount, BDI_STAT_BATCH); 69e0bf68ddSPeter Zijlstra } 70e0bf68ddSPeter Zijlstra 71b2e8fb6eSPeter Zijlstra static inline void __inc_bdi_stat(struct backing_dev_info *bdi, 72b2e8fb6eSPeter Zijlstra enum bdi_stat_item item) 73e0bf68ddSPeter Zijlstra { 74b2e8fb6eSPeter Zijlstra __add_bdi_stat(bdi, item, 1); 75b2e8fb6eSPeter Zijlstra } 76b2e8fb6eSPeter Zijlstra 77b2e8fb6eSPeter Zijlstra static inline void inc_bdi_stat(struct backing_dev_info *bdi, 78b2e8fb6eSPeter Zijlstra enum bdi_stat_item item) 79b2e8fb6eSPeter Zijlstra { 80b2e8fb6eSPeter Zijlstra unsigned long flags; 81b2e8fb6eSPeter Zijlstra 82b2e8fb6eSPeter Zijlstra local_irq_save(flags); 83b2e8fb6eSPeter Zijlstra __inc_bdi_stat(bdi, item); 84b2e8fb6eSPeter Zijlstra local_irq_restore(flags); 85b2e8fb6eSPeter Zijlstra } 86b2e8fb6eSPeter Zijlstra 87b2e8fb6eSPeter Zijlstra static inline void __dec_bdi_stat(struct backing_dev_info *bdi, 88b2e8fb6eSPeter Zijlstra enum bdi_stat_item item) 89b2e8fb6eSPeter Zijlstra { 90b2e8fb6eSPeter Zijlstra __add_bdi_stat(bdi, item, -1); 91b2e8fb6eSPeter Zijlstra } 92b2e8fb6eSPeter Zijlstra 93b2e8fb6eSPeter Zijlstra static inline void dec_bdi_stat(struct backing_dev_info *bdi, 94b2e8fb6eSPeter Zijlstra enum bdi_stat_item item) 95b2e8fb6eSPeter Zijlstra { 96b2e8fb6eSPeter Zijlstra unsigned long flags; 97b2e8fb6eSPeter Zijlstra 98b2e8fb6eSPeter Zijlstra local_irq_save(flags); 99b2e8fb6eSPeter Zijlstra __dec_bdi_stat(bdi, item); 100b2e8fb6eSPeter Zijlstra local_irq_restore(flags); 101b2e8fb6eSPeter Zijlstra } 102b2e8fb6eSPeter Zijlstra 103b2e8fb6eSPeter Zijlstra static inline s64 bdi_stat(struct backing_dev_info *bdi, 104b2e8fb6eSPeter Zijlstra enum bdi_stat_item item) 105b2e8fb6eSPeter Zijlstra { 106b2e8fb6eSPeter Zijlstra return percpu_counter_read_positive(&bdi->bdi_stat[item]); 107b2e8fb6eSPeter Zijlstra } 108b2e8fb6eSPeter Zijlstra 109b2e8fb6eSPeter Zijlstra static inline s64 __bdi_stat_sum(struct backing_dev_info *bdi, 110b2e8fb6eSPeter Zijlstra enum bdi_stat_item item) 111b2e8fb6eSPeter Zijlstra { 112b2e8fb6eSPeter Zijlstra return percpu_counter_sum_positive(&bdi->bdi_stat[item]); 113b2e8fb6eSPeter Zijlstra } 114b2e8fb6eSPeter Zijlstra 115b2e8fb6eSPeter Zijlstra static inline s64 bdi_stat_sum(struct backing_dev_info *bdi, 116b2e8fb6eSPeter Zijlstra enum bdi_stat_item item) 117b2e8fb6eSPeter Zijlstra { 118b2e8fb6eSPeter Zijlstra s64 sum; 119b2e8fb6eSPeter Zijlstra unsigned long flags; 120b2e8fb6eSPeter Zijlstra 121b2e8fb6eSPeter Zijlstra local_irq_save(flags); 122b2e8fb6eSPeter Zijlstra sum = __bdi_stat_sum(bdi, item); 123b2e8fb6eSPeter Zijlstra local_irq_restore(flags); 124b2e8fb6eSPeter Zijlstra 125b2e8fb6eSPeter Zijlstra return sum; 126b2e8fb6eSPeter Zijlstra } 127b2e8fb6eSPeter Zijlstra 128b2e8fb6eSPeter Zijlstra /* 129b2e8fb6eSPeter Zijlstra * maximal error of a stat counter. 130b2e8fb6eSPeter Zijlstra */ 131b2e8fb6eSPeter Zijlstra static inline unsigned long bdi_stat_error(struct backing_dev_info *bdi) 132b2e8fb6eSPeter Zijlstra { 133b2e8fb6eSPeter Zijlstra #ifdef CONFIG_SMP 134b2e8fb6eSPeter Zijlstra return nr_cpu_ids * BDI_STAT_BATCH; 135b2e8fb6eSPeter Zijlstra #else 136b2e8fb6eSPeter Zijlstra return 1; 137b2e8fb6eSPeter Zijlstra #endif 138e0bf68ddSPeter Zijlstra } 1391da177e4SLinus Torvalds 1401da177e4SLinus Torvalds /* 1411da177e4SLinus Torvalds * Flags in backing_dev_info::capability 1421da177e4SLinus Torvalds * - The first two flags control whether dirty pages will contribute to the 1431da177e4SLinus Torvalds * VM's accounting and whether writepages() should be called for dirty pages 1441da177e4SLinus Torvalds * (something that would not, for example, be appropriate for ramfs) 1451da177e4SLinus Torvalds * - These flags let !MMU mmap() govern direct device mapping vs immediate 1461da177e4SLinus Torvalds * copying more easily for MAP_PRIVATE, especially for ROM filesystems 1471da177e4SLinus Torvalds */ 1481da177e4SLinus Torvalds #define BDI_CAP_NO_ACCT_DIRTY 0x00000001 /* Dirty pages shouldn't contribute to accounting */ 1491da177e4SLinus Torvalds #define BDI_CAP_NO_WRITEBACK 0x00000002 /* Don't write pages back */ 1501da177e4SLinus Torvalds #define BDI_CAP_MAP_COPY 0x00000004 /* Copy can be mapped (MAP_PRIVATE) */ 1511da177e4SLinus Torvalds #define BDI_CAP_MAP_DIRECT 0x00000008 /* Can be mapped directly (MAP_SHARED) */ 1521da177e4SLinus Torvalds #define BDI_CAP_READ_MAP 0x00000010 /* Can be mapped for reading */ 1531da177e4SLinus Torvalds #define BDI_CAP_WRITE_MAP 0x00000020 /* Can be mapped for writing */ 1541da177e4SLinus Torvalds #define BDI_CAP_EXEC_MAP 0x00000040 /* Can be mapped for execution */ 1551da177e4SLinus Torvalds #define BDI_CAP_VMFLAGS \ 1561da177e4SLinus Torvalds (BDI_CAP_READ_MAP | BDI_CAP_WRITE_MAP | BDI_CAP_EXEC_MAP) 1571da177e4SLinus Torvalds 1581da177e4SLinus Torvalds #if defined(VM_MAYREAD) && \ 1591da177e4SLinus Torvalds (BDI_CAP_READ_MAP != VM_MAYREAD || \ 1601da177e4SLinus Torvalds BDI_CAP_WRITE_MAP != VM_MAYWRITE || \ 1611da177e4SLinus Torvalds BDI_CAP_EXEC_MAP != VM_MAYEXEC) 1621da177e4SLinus Torvalds #error please change backing_dev_info::capabilities flags 1631da177e4SLinus Torvalds #endif 1641da177e4SLinus Torvalds 1651da177e4SLinus Torvalds extern struct backing_dev_info default_backing_dev_info; 1661da177e4SLinus Torvalds void default_unplug_io_fn(struct backing_dev_info *bdi, struct page *page); 1671da177e4SLinus Torvalds 1681da177e4SLinus Torvalds int writeback_in_progress(struct backing_dev_info *bdi); 1691da177e4SLinus Torvalds 1701da177e4SLinus Torvalds static inline int bdi_congested(struct backing_dev_info *bdi, int bdi_bits) 1711da177e4SLinus Torvalds { 1721da177e4SLinus Torvalds if (bdi->congested_fn) 1731da177e4SLinus Torvalds return bdi->congested_fn(bdi->congested_data, bdi_bits); 1741da177e4SLinus Torvalds return (bdi->state & bdi_bits); 1751da177e4SLinus Torvalds } 1761da177e4SLinus Torvalds 1771da177e4SLinus Torvalds static inline int bdi_read_congested(struct backing_dev_info *bdi) 1781da177e4SLinus Torvalds { 1791da177e4SLinus Torvalds return bdi_congested(bdi, 1 << BDI_read_congested); 1801da177e4SLinus Torvalds } 1811da177e4SLinus Torvalds 1821da177e4SLinus Torvalds static inline int bdi_write_congested(struct backing_dev_info *bdi) 1831da177e4SLinus Torvalds { 1841da177e4SLinus Torvalds return bdi_congested(bdi, 1 << BDI_write_congested); 1851da177e4SLinus Torvalds } 1861da177e4SLinus Torvalds 1871da177e4SLinus Torvalds static inline int bdi_rw_congested(struct backing_dev_info *bdi) 1881da177e4SLinus Torvalds { 1891da177e4SLinus Torvalds return bdi_congested(bdi, (1 << BDI_read_congested)| 1901da177e4SLinus Torvalds (1 << BDI_write_congested)); 1911da177e4SLinus Torvalds } 1921da177e4SLinus Torvalds 1933fcfab16SAndrew Morton void clear_bdi_congested(struct backing_dev_info *bdi, int rw); 1943fcfab16SAndrew Morton void set_bdi_congested(struct backing_dev_info *bdi, int rw); 1953fcfab16SAndrew Morton long congestion_wait(int rw, long timeout); 1963fcfab16SAndrew Morton 1971da177e4SLinus Torvalds #define bdi_cap_writeback_dirty(bdi) \ 1981da177e4SLinus Torvalds (!((bdi)->capabilities & BDI_CAP_NO_WRITEBACK)) 1991da177e4SLinus Torvalds 2001da177e4SLinus Torvalds #define bdi_cap_account_dirty(bdi) \ 2011da177e4SLinus Torvalds (!((bdi)->capabilities & BDI_CAP_NO_ACCT_DIRTY)) 2021da177e4SLinus Torvalds 2031da177e4SLinus Torvalds #define mapping_cap_writeback_dirty(mapping) \ 2041da177e4SLinus Torvalds bdi_cap_writeback_dirty((mapping)->backing_dev_info) 2051da177e4SLinus Torvalds 2061da177e4SLinus Torvalds #define mapping_cap_account_dirty(mapping) \ 2071da177e4SLinus Torvalds bdi_cap_account_dirty((mapping)->backing_dev_info) 2081da177e4SLinus Torvalds 2091da177e4SLinus Torvalds 2101da177e4SLinus Torvalds #endif /* _LINUX_BACKING_DEV_H */ 211