Commit f6498b77 authored by Johannes Weiner's avatar Johannes Weiner Committed by akpm

mm: zswap: add basic meminfo and vmstat coverage

Currently it requires poking at debugfs to figure out the size and
population of the zswap cache on a host.  There are no counters for reads
and writes against the cache.  As a result, it's difficult to understand
zswap behavior on production systems.

Print zswap memory consumption and how many pages are zswapped out in
/proc/meminfo.  Count zswapouts and zswapins in /proc/vmstat.

Link: https://lkml.kernel.org/r/20220510152847.230957-6-hannes@cmpxchg.orgSigned-off-by: default avatarJohannes Weiner <hannes@cmpxchg.org>
Acked-by: default avatarDavid Hildenbrand <david@redhat.com>
Cc: Dan Streetman <ddstreet@ieee.org>
Cc: Michal Hocko <mhocko@suse.com>
Cc: Minchan Kim <minchan@kernel.org>
Cc: Roman Gushchin <guro@fb.com>
Cc: Seth Jennings <sjenning@redhat.com>
Cc: Shakeel Butt <shakeelb@google.com>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
parent b3fbd58f
...@@ -964,6 +964,8 @@ Example output. You may not have all of these fields. ...@@ -964,6 +964,8 @@ Example output. You may not have all of these fields.
Mlocked: 0 kB Mlocked: 0 kB
SwapTotal: 0 kB SwapTotal: 0 kB
SwapFree: 0 kB SwapFree: 0 kB
Zswap: 1904 kB
Zswapped: 7792 kB
Dirty: 12 kB Dirty: 12 kB
Writeback: 0 kB Writeback: 0 kB
AnonPages: 4654780 kB AnonPages: 4654780 kB
...@@ -1055,6 +1057,10 @@ SwapTotal ...@@ -1055,6 +1057,10 @@ SwapTotal
SwapFree SwapFree
Memory which has been evicted from RAM, and is temporarily Memory which has been evicted from RAM, and is temporarily
on the disk on the disk
Zswap
Memory consumed by the zswap backend (compressed size)
Zswapped
Amount of anonymous memory stored in zswap (original size)
Dirty Dirty
Memory which is waiting to get written back to the disk Memory which is waiting to get written back to the disk
Writeback Writeback
......
...@@ -86,6 +86,13 @@ static int meminfo_proc_show(struct seq_file *m, void *v) ...@@ -86,6 +86,13 @@ static int meminfo_proc_show(struct seq_file *m, void *v)
show_val_kb(m, "SwapTotal: ", i.totalswap); show_val_kb(m, "SwapTotal: ", i.totalswap);
show_val_kb(m, "SwapFree: ", i.freeswap); show_val_kb(m, "SwapFree: ", i.freeswap);
#ifdef CONFIG_ZSWAP
seq_printf(m, "Zswap: %8lu kB\n",
(unsigned long)(zswap_pool_total_size >> 10));
seq_printf(m, "Zswapped: %8lu kB\n",
(unsigned long)atomic_read(&zswap_stored_pages) <<
(PAGE_SHIFT - 10));
#endif
show_val_kb(m, "Dirty: ", show_val_kb(m, "Dirty: ",
global_node_page_state(NR_FILE_DIRTY)); global_node_page_state(NR_FILE_DIRTY));
show_val_kb(m, "Writeback: ", show_val_kb(m, "Writeback: ",
......
...@@ -620,6 +620,11 @@ static inline int mem_cgroup_swappiness(struct mem_cgroup *mem) ...@@ -620,6 +620,11 @@ static inline int mem_cgroup_swappiness(struct mem_cgroup *mem)
} }
#endif #endif
#ifdef CONFIG_ZSWAP
extern u64 zswap_pool_total_size;
extern atomic_t zswap_stored_pages;
#endif
#if defined(CONFIG_SWAP) && defined(CONFIG_MEMCG) && defined(CONFIG_BLK_CGROUP) #if defined(CONFIG_SWAP) && defined(CONFIG_MEMCG) && defined(CONFIG_BLK_CGROUP)
extern void __cgroup_throttle_swaprate(struct page *page, gfp_t gfp_mask); extern void __cgroup_throttle_swaprate(struct page *page, gfp_t gfp_mask);
static inline void cgroup_throttle_swaprate(struct page *page, gfp_t gfp_mask) static inline void cgroup_throttle_swaprate(struct page *page, gfp_t gfp_mask)
......
...@@ -136,6 +136,10 @@ enum vm_event_item { PGPGIN, PGPGOUT, PSWPIN, PSWPOUT, ...@@ -136,6 +136,10 @@ enum vm_event_item { PGPGIN, PGPGOUT, PSWPIN, PSWPOUT,
#ifdef CONFIG_KSM #ifdef CONFIG_KSM
COW_KSM, COW_KSM,
#endif #endif
#ifdef CONFIG_ZSWAP
ZSWPIN,
ZSWPOUT,
#endif
#ifdef CONFIG_X86 #ifdef CONFIG_X86
DIRECT_MAP_LEVEL2_SPLIT, DIRECT_MAP_LEVEL2_SPLIT,
DIRECT_MAP_LEVEL3_SPLIT, DIRECT_MAP_LEVEL3_SPLIT,
......
...@@ -1396,6 +1396,10 @@ const char * const vmstat_text[] = { ...@@ -1396,6 +1396,10 @@ const char * const vmstat_text[] = {
#ifdef CONFIG_KSM #ifdef CONFIG_KSM
"cow_ksm", "cow_ksm",
#endif #endif
#ifdef CONFIG_ZSWAP
"zswpin",
"zswpout",
#endif
#ifdef CONFIG_X86 #ifdef CONFIG_X86
"direct_map_level2_splits", "direct_map_level2_splits",
"direct_map_level3_splits", "direct_map_level3_splits",
......
...@@ -42,9 +42,9 @@ ...@@ -42,9 +42,9 @@
* statistics * statistics
**********************************/ **********************************/
/* Total bytes used by the compressed storage */ /* Total bytes used by the compressed storage */
static u64 zswap_pool_total_size; u64 zswap_pool_total_size;
/* The number of compressed pages currently stored in zswap */ /* The number of compressed pages currently stored in zswap */
static atomic_t zswap_stored_pages = ATOMIC_INIT(0); atomic_t zswap_stored_pages = ATOMIC_INIT(0);
/* The number of same-value filled pages currently stored in zswap */ /* The number of same-value filled pages currently stored in zswap */
static atomic_t zswap_same_filled_pages = ATOMIC_INIT(0); static atomic_t zswap_same_filled_pages = ATOMIC_INIT(0);
...@@ -1243,6 +1243,7 @@ static int zswap_frontswap_store(unsigned type, pgoff_t offset, ...@@ -1243,6 +1243,7 @@ static int zswap_frontswap_store(unsigned type, pgoff_t offset,
/* update stats */ /* update stats */
atomic_inc(&zswap_stored_pages); atomic_inc(&zswap_stored_pages);
zswap_update_total_size(); zswap_update_total_size();
count_vm_event(ZSWPOUT);
return 0; return 0;
...@@ -1285,11 +1286,10 @@ static int zswap_frontswap_load(unsigned type, pgoff_t offset, ...@@ -1285,11 +1286,10 @@ static int zswap_frontswap_load(unsigned type, pgoff_t offset,
zswap_fill_page(dst, entry->value); zswap_fill_page(dst, entry->value);
kunmap_atomic(dst); kunmap_atomic(dst);
ret = 0; ret = 0;
goto freeentry; goto stats;
} }
if (!zpool_can_sleep_mapped(entry->pool->zpool)) { if (!zpool_can_sleep_mapped(entry->pool->zpool)) {
tmp = kmalloc(entry->length, GFP_ATOMIC); tmp = kmalloc(entry->length, GFP_ATOMIC);
if (!tmp) { if (!tmp) {
ret = -ENOMEM; ret = -ENOMEM;
...@@ -1304,10 +1304,8 @@ static int zswap_frontswap_load(unsigned type, pgoff_t offset, ...@@ -1304,10 +1304,8 @@ static int zswap_frontswap_load(unsigned type, pgoff_t offset,
src += sizeof(struct zswap_header); src += sizeof(struct zswap_header);
if (!zpool_can_sleep_mapped(entry->pool->zpool)) { if (!zpool_can_sleep_mapped(entry->pool->zpool)) {
memcpy(tmp, src, entry->length); memcpy(tmp, src, entry->length);
src = tmp; src = tmp;
zpool_unmap_handle(entry->pool->zpool, entry->handle); zpool_unmap_handle(entry->pool->zpool, entry->handle);
} }
...@@ -1326,7 +1324,8 @@ static int zswap_frontswap_load(unsigned type, pgoff_t offset, ...@@ -1326,7 +1324,8 @@ static int zswap_frontswap_load(unsigned type, pgoff_t offset,
kfree(tmp); kfree(tmp);
BUG_ON(ret); BUG_ON(ret);
stats:
count_vm_event(ZSWPIN);
freeentry: freeentry:
spin_lock(&tree->lock); spin_lock(&tree->lock);
zswap_entry_put(tree, entry); zswap_entry_put(tree, entry);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment