1b2441318SGreg Kroah-Hartman // SPDX-License-Identifier: GPL-2.0
231c9afa6SSasha Levin /*
331c9afa6SSasha Levin * mm/debug.c
431c9afa6SSasha Levin *
531c9afa6SSasha Levin * mm/ specific debug routines.
631c9afa6SSasha Levin *
731c9afa6SSasha Levin */
831c9afa6SSasha Levin
982742a3aSSasha Levin #include <linux/kernel.h>
1082742a3aSSasha Levin #include <linux/mm.h>
11af658dcaSSteven Rostedt (Red Hat) #include <linux/trace_events.h>
1282742a3aSSasha Levin #include <linux/memcontrol.h>
13420adbe9SVlastimil Babka #include <trace/events/mmflags.h>
147cd12b4aSVlastimil Babka #include <linux/migrate.h>
154e462112SVlastimil Babka #include <linux/page_owner.h>
16f682a97aSAlexander Duyck #include <linux/ctype.h>
1782742a3aSSasha Levin
18edf14cdbSVlastimil Babka #include "internal.h"
198eb42beaSJohn Hubbard #include <trace/events/migrate.h>
208eb42beaSJohn Hubbard
218eb42beaSJohn Hubbard /*
228eb42beaSJohn Hubbard * Define EM() and EMe() so that MIGRATE_REASON from trace/events/migrate.h can
238eb42beaSJohn Hubbard * be used to populate migrate_reason_names[].
248eb42beaSJohn Hubbard */
258eb42beaSJohn Hubbard #undef EM
268eb42beaSJohn Hubbard #undef EMe
278eb42beaSJohn Hubbard #define EM(a, b) b,
288eb42beaSJohn Hubbard #define EMe(a, b) b
29edf14cdbSVlastimil Babka
309a2f45ffSAlexey Dobriyan const char *migrate_reason_names[MR_TYPES] = {
318eb42beaSJohn Hubbard MIGRATE_REASON
327cd12b4aSVlastimil Babka };
337cd12b4aSVlastimil Babka
34edf14cdbSVlastimil Babka const struct trace_print_flags pageflag_names[] = {
35edf14cdbSVlastimil Babka __def_pageflag_names,
36edf14cdbSVlastimil Babka {0, NULL}
37420adbe9SVlastimil Babka };
38420adbe9SVlastimil Babka
39edf14cdbSVlastimil Babka const struct trace_print_flags gfpflag_names[] = {
40edf14cdbSVlastimil Babka __def_gfpflag_names,
41edf14cdbSVlastimil Babka {0, NULL}
42edf14cdbSVlastimil Babka };
43edf14cdbSVlastimil Babka
44edf14cdbSVlastimil Babka const struct trace_print_flags vmaflag_names[] = {
45edf14cdbSVlastimil Babka __def_vmaflag_names,
46edf14cdbSVlastimil Babka {0, NULL}
4782742a3aSSasha Levin };
4882742a3aSSasha Levin
494ffca5a9SMatthew Wilcox (Oracle) #define DEF_PAGETYPE_NAME(_name) [PGTY_##_name - 0xf0] = __stringify(_name)
504ffca5a9SMatthew Wilcox (Oracle)
514ffca5a9SMatthew Wilcox (Oracle) static const char *page_type_names[] = {
524ffca5a9SMatthew Wilcox (Oracle) DEF_PAGETYPE_NAME(slab),
534ffca5a9SMatthew Wilcox (Oracle) DEF_PAGETYPE_NAME(hugetlb),
544ffca5a9SMatthew Wilcox (Oracle) DEF_PAGETYPE_NAME(offline),
554ffca5a9SMatthew Wilcox (Oracle) DEF_PAGETYPE_NAME(guard),
564ffca5a9SMatthew Wilcox (Oracle) DEF_PAGETYPE_NAME(table),
574ffca5a9SMatthew Wilcox (Oracle) DEF_PAGETYPE_NAME(buddy),
584ffca5a9SMatthew Wilcox (Oracle) DEF_PAGETYPE_NAME(unaccepted),
594ffca5a9SMatthew Wilcox (Oracle) };
604ffca5a9SMatthew Wilcox (Oracle)
page_type_name(unsigned int page_type)614ffca5a9SMatthew Wilcox (Oracle) static const char *page_type_name(unsigned int page_type)
624ffca5a9SMatthew Wilcox (Oracle) {
634ffca5a9SMatthew Wilcox (Oracle) unsigned i = (page_type >> 24) - 0xf0;
644ffca5a9SMatthew Wilcox (Oracle)
654ffca5a9SMatthew Wilcox (Oracle) if (i >= ARRAY_SIZE(page_type_names))
664ffca5a9SMatthew Wilcox (Oracle) return "unknown";
674ffca5a9SMatthew Wilcox (Oracle) return page_type_names[i];
684ffca5a9SMatthew Wilcox (Oracle) }
694ffca5a9SMatthew Wilcox (Oracle)
__dump_folio(struct folio * folio,struct page * page,unsigned long pfn,unsigned long idx)70fae7d834SMatthew Wilcox (Oracle) static void __dump_folio(struct folio *folio, struct page *page,
71fae7d834SMatthew Wilcox (Oracle) unsigned long pfn, unsigned long idx)
7282742a3aSSasha Levin {
73fae7d834SMatthew Wilcox (Oracle) struct address_space *mapping = folio_mapping(folio);
747441d349SDavid Hildenbrand int mapcount = atomic_read(&page->_mapcount);
755b57b8f2SVlastimil Babka char *type = "";
76fc36def9SPavel Tatashin
774ffca5a9SMatthew Wilcox (Oracle) mapcount = page_mapcount_is_type(mapcount) ? 0 : mapcount + 1;
78fae7d834SMatthew Wilcox (Oracle) pr_warn("page: refcount:%d mapcount:%d mapping:%p index:%#lx pfn:%#lx\n",
79fae7d834SMatthew Wilcox (Oracle) folio_ref_count(folio), mapcount, mapping,
80fae7d834SMatthew Wilcox (Oracle) folio->index + idx, pfn);
81fae7d834SMatthew Wilcox (Oracle) if (folio_test_large(folio)) {
8231a31da8SDavid Hildenbrand int pincount = 0;
8331a31da8SDavid Hildenbrand
8431a31da8SDavid Hildenbrand if (folio_has_pincount(folio))
8531a31da8SDavid Hildenbrand pincount = atomic_read(&folio->_pincount);
8631a31da8SDavid Hildenbrand
8705c5323bSDavid Hildenbrand pr_warn("head: order:%u mapcount:%d entire_mapcount:%d nr_pages_mapped:%d pincount:%d\n",
88fae7d834SMatthew Wilcox (Oracle) folio_order(folio),
8905c5323bSDavid Hildenbrand folio_mapcount(folio),
9091ec7f28SMatthew Wilcox (Oracle) folio_entire_mapcount(folio),
91eec20426SMatthew Wilcox (Oracle) folio_nr_pages_mapped(folio),
9231a31da8SDavid Hildenbrand pincount);
93452b557cSMatthew Wilcox (Oracle) }
9491f5345aSMatthew Wilcox (Oracle)
9591f5345aSMatthew Wilcox (Oracle) #ifdef CONFIG_MEMCG
96fae7d834SMatthew Wilcox (Oracle) if (folio->memcg_data)
97fae7d834SMatthew Wilcox (Oracle) pr_warn("memcg:%lx\n", folio->memcg_data);
9891f5345aSMatthew Wilcox (Oracle) #endif
99fae7d834SMatthew Wilcox (Oracle) if (folio_test_ksm(folio))
1005b57b8f2SVlastimil Babka type = "ksm ";
101fae7d834SMatthew Wilcox (Oracle) else if (folio_test_anon(folio))
1025b57b8f2SVlastimil Babka type = "anon ";
1033e9d80a8SMatthew Wilcox (Oracle) else if (mapping)
1043e9d80a8SMatthew Wilcox (Oracle) dump_mapping(mapping);
105edf14cdbSVlastimil Babka BUILD_BUG_ON(ARRAY_SIZE(pageflag_names) != __NR_PAGEFLAGS + 1);
106ff8e8116SVlastimil Babka
107fae7d834SMatthew Wilcox (Oracle) /*
108fae7d834SMatthew Wilcox (Oracle) * Accessing the pageblock without the zone lock. It could change to
109fae7d834SMatthew Wilcox (Oracle) * "isolate" again in the meantime, but since we are just dumping the
110fae7d834SMatthew Wilcox (Oracle) * state for debugging, it should be fine to accept a bit of
111fae7d834SMatthew Wilcox (Oracle) * inaccuracy here due to racing.
112fae7d834SMatthew Wilcox (Oracle) */
113fae7d834SMatthew Wilcox (Oracle) pr_warn("%sflags: %pGp%s\n", type, &folio->flags,
114fae7d834SMatthew Wilcox (Oracle) is_migrate_cma_folio(folio, pfn) ? " CMA" : "");
1158f790d0cSMatthew Wilcox (Oracle) if (page_has_type(&folio->page))
1164ffca5a9SMatthew Wilcox (Oracle) pr_warn("page_type: %x(%s)\n", folio->page.page_type >> 24,
1174ffca5a9SMatthew Wilcox (Oracle) page_type_name(folio->page.page_type));
118f2421a16SHyeonggon Yoo
119e0392cf7SMichal Hocko print_hex_dump(KERN_WARNING, "raw: ", DUMP_PREFIX_NONE, 32,
12046e8a3a0SVlastimil Babka sizeof(unsigned long), page,
12146e8a3a0SVlastimil Babka sizeof(struct page), false);
122fae7d834SMatthew Wilcox (Oracle) if (folio_test_large(folio))
1236197ab98SMatthew Wilcox (Oracle) print_hex_dump(KERN_WARNING, "head: ", DUMP_PREFIX_NONE, 32,
124fae7d834SMatthew Wilcox (Oracle) sizeof(unsigned long), folio,
125fae7d834SMatthew Wilcox (Oracle) 2 * sizeof(struct page), false);
126fae7d834SMatthew Wilcox (Oracle) }
127fae7d834SMatthew Wilcox (Oracle)
__dump_page(const struct page * page)128fae7d834SMatthew Wilcox (Oracle) static void __dump_page(const struct page *page)
129fae7d834SMatthew Wilcox (Oracle) {
130fae7d834SMatthew Wilcox (Oracle) struct folio *foliop, folio;
131fae7d834SMatthew Wilcox (Oracle) struct page precise;
1326a7de1bfSMatthew Wilcox (Oracle) unsigned long head;
133fae7d834SMatthew Wilcox (Oracle) unsigned long pfn = page_to_pfn(page);
134fae7d834SMatthew Wilcox (Oracle) unsigned long idx, nr_pages = 1;
135fae7d834SMatthew Wilcox (Oracle) int loops = 5;
136fae7d834SMatthew Wilcox (Oracle)
137fae7d834SMatthew Wilcox (Oracle) again:
138fae7d834SMatthew Wilcox (Oracle) memcpy(&precise, page, sizeof(*page));
1396a7de1bfSMatthew Wilcox (Oracle) head = precise.compound_head;
1406a7de1bfSMatthew Wilcox (Oracle) if ((head & 1) == 0) {
1416a7de1bfSMatthew Wilcox (Oracle) foliop = (struct folio *)&precise;
142fae7d834SMatthew Wilcox (Oracle) idx = 0;
143fae7d834SMatthew Wilcox (Oracle) if (!folio_test_large(foliop))
144fae7d834SMatthew Wilcox (Oracle) goto dump;
145fae7d834SMatthew Wilcox (Oracle) foliop = (struct folio *)page;
146fae7d834SMatthew Wilcox (Oracle) } else {
1476a7de1bfSMatthew Wilcox (Oracle) foliop = (struct folio *)(head - 1);
148fae7d834SMatthew Wilcox (Oracle) idx = folio_page_idx(foliop, page);
149fae7d834SMatthew Wilcox (Oracle) }
150fae7d834SMatthew Wilcox (Oracle)
151fae7d834SMatthew Wilcox (Oracle) if (idx < MAX_FOLIO_NR_PAGES) {
152fae7d834SMatthew Wilcox (Oracle) memcpy(&folio, foliop, 2 * sizeof(struct page));
153fae7d834SMatthew Wilcox (Oracle) nr_pages = folio_nr_pages(&folio);
15431a31da8SDavid Hildenbrand if (nr_pages > 1)
15531a31da8SDavid Hildenbrand memcpy(&folio.__page_2, &foliop->__page_2,
15631a31da8SDavid Hildenbrand sizeof(struct page));
157fae7d834SMatthew Wilcox (Oracle) foliop = &folio;
158fae7d834SMatthew Wilcox (Oracle) }
159fae7d834SMatthew Wilcox (Oracle)
160fae7d834SMatthew Wilcox (Oracle) if (idx > nr_pages) {
161fae7d834SMatthew Wilcox (Oracle) if (loops-- > 0)
162fae7d834SMatthew Wilcox (Oracle) goto again;
163fae7d834SMatthew Wilcox (Oracle) pr_warn("page does not match folio\n");
164fae7d834SMatthew Wilcox (Oracle) precise.compound_head &= ~1UL;
165fae7d834SMatthew Wilcox (Oracle) foliop = (struct folio *)&precise;
166fae7d834SMatthew Wilcox (Oracle) idx = 0;
167fae7d834SMatthew Wilcox (Oracle) }
168fae7d834SMatthew Wilcox (Oracle)
169fae7d834SMatthew Wilcox (Oracle) dump:
170fae7d834SMatthew Wilcox (Oracle) __dump_folio(foliop, &precise, pfn, idx);
17182742a3aSSasha Levin }
17282742a3aSSasha Levin
dump_page(const struct page * page,const char * reason)173b3a32033SMatthew Wilcox (Oracle) void dump_page(const struct page *page, const char *reason)
17482742a3aSSasha Levin {
175be7c701fSMatthew Wilcox (Oracle) if (PagePoisoned(page))
176*456620c5SLiu Ye pr_warn("page:%p is uninitialized and poisoned\n", page);
177be7c701fSMatthew Wilcox (Oracle) else
178be7c701fSMatthew Wilcox (Oracle) __dump_page(page);
179be7c701fSMatthew Wilcox (Oracle) if (reason)
180be7c701fSMatthew Wilcox (Oracle) pr_warn("page dumped because: %s\n", reason);
1814e462112SVlastimil Babka dump_page_owner(page);
18282742a3aSSasha Levin }
18382742a3aSSasha Levin EXPORT_SYMBOL(dump_page);
18482742a3aSSasha Levin
18582742a3aSSasha Levin #ifdef CONFIG_DEBUG_VM
18682742a3aSSasha Levin
dump_vma(const struct vm_area_struct * vma)18782742a3aSSasha Levin void dump_vma(const struct vm_area_struct *vma)
18882742a3aSSasha Levin {
189763ecb03SLiam R. Howlett pr_emerg("vma %px start %px end %px mm %px\n"
190152a2d19SMatthew Wilcox "prot %lx anon_vma %px vm_ops %px\n"
191152a2d19SMatthew Wilcox "pgoff %lx file %px private_data %px\n"
1923dd98c5cSSuren Baghdasaryan #ifdef CONFIG_PER_VMA_LOCK
1933dd98c5cSSuren Baghdasaryan "refcnt %x\n"
1943dd98c5cSSuren Baghdasaryan #endif
195b8eceeb9SVlastimil Babka "flags: %#lx(%pGv)\n",
196763ecb03SLiam R. Howlett vma, (void *)vma->vm_start, (void *)vma->vm_end, vma->vm_mm,
19782742a3aSSasha Levin (unsigned long)pgprot_val(vma->vm_page_prot),
19882742a3aSSasha Levin vma->anon_vma, vma->vm_ops, vma->vm_pgoff,
199b8eceeb9SVlastimil Babka vma->vm_file, vma->vm_private_data,
2003dd98c5cSSuren Baghdasaryan #ifdef CONFIG_PER_VMA_LOCK
2013dd98c5cSSuren Baghdasaryan refcount_read(&vma->vm_refcnt),
2023dd98c5cSSuren Baghdasaryan #endif
203b8eceeb9SVlastimil Babka vma->vm_flags, &vma->vm_flags);
20482742a3aSSasha Levin }
20582742a3aSSasha Levin EXPORT_SYMBOL(dump_vma);
20682742a3aSSasha Levin
dump_mm(const struct mm_struct * mm)20731c9afa6SSasha Levin void dump_mm(const struct mm_struct *mm)
20831c9afa6SSasha Levin {
209763ecb03SLiam R. Howlett pr_emerg("mm %px task_size %lu\n"
210763ecb03SLiam R. Howlett "mmap_base %lu mmap_legacy_base %lu\n"
211152a2d19SMatthew Wilcox "pgd %px mm_users %d mm_count %d pgtables_bytes %lu map_count %d\n"
21231c9afa6SSasha Levin "hiwater_rss %lx hiwater_vm %lx total_vm %lx locked_vm %lx\n"
21370f8a3caSDavidlohr Bueso "pinned_vm %llx data_vm %lx exec_vm %lx stack_vm %lx\n"
21431c9afa6SSasha Levin "start_code %lx end_code %lx start_data %lx end_data %lx\n"
21531c9afa6SSasha Levin "start_brk %lx brk %lx start_stack %lx\n"
21631c9afa6SSasha Levin "arg_start %lx arg_end %lx env_start %lx env_end %lx\n"
2170258b5fdSEric W. Biederman "binfmt %px flags %lx\n"
21831c9afa6SSasha Levin #ifdef CONFIG_AIO
219152a2d19SMatthew Wilcox "ioctx_table %px\n"
22031c9afa6SSasha Levin #endif
22131c9afa6SSasha Levin #ifdef CONFIG_MEMCG
222152a2d19SMatthew Wilcox "owner %px "
22331c9afa6SSasha Levin #endif
224152a2d19SMatthew Wilcox "exe_file %px\n"
22531c9afa6SSasha Levin #ifdef CONFIG_MMU_NOTIFIER
226984cfe4eSJason Gunthorpe "notifier_subscriptions %px\n"
22731c9afa6SSasha Levin #endif
22831c9afa6SSasha Levin #ifdef CONFIG_NUMA_BALANCING
22931c9afa6SSasha Levin "numa_next_scan %lu numa_scan_offset %lu numa_scan_seq %d\n"
23031c9afa6SSasha Levin #endif
23131c9afa6SSasha Levin "tlb_flush_pending %d\n"
232b8eceeb9SVlastimil Babka "def_flags: %#lx(%pGv)\n",
23331c9afa6SSasha Levin
234763ecb03SLiam R. Howlett mm, mm->task_size,
235763ecb03SLiam R. Howlett mm->mmap_base, mm->mmap_legacy_base,
23631c9afa6SSasha Levin mm->pgd, atomic_read(&mm->mm_users),
23731c9afa6SSasha Levin atomic_read(&mm->mm_count),
238af5b0f6aSKirill A. Shutemov mm_pgtables_bytes(mm),
23931c9afa6SSasha Levin mm->map_count,
24031c9afa6SSasha Levin mm->hiwater_rss, mm->hiwater_vm, mm->total_vm, mm->locked_vm,
24144dc1b1fSQian Cai (u64)atomic64_read(&mm->pinned_vm),
24270f8a3caSDavidlohr Bueso mm->data_vm, mm->exec_vm, mm->stack_vm,
24331c9afa6SSasha Levin mm->start_code, mm->end_code, mm->start_data, mm->end_data,
24431c9afa6SSasha Levin mm->start_brk, mm->brk, mm->start_stack,
24531c9afa6SSasha Levin mm->arg_start, mm->arg_end, mm->env_start, mm->env_end,
2460258b5fdSEric W. Biederman mm->binfmt, mm->flags,
24731c9afa6SSasha Levin #ifdef CONFIG_AIO
24831c9afa6SSasha Levin mm->ioctx_table,
24931c9afa6SSasha Levin #endif
25031c9afa6SSasha Levin #ifdef CONFIG_MEMCG
25131c9afa6SSasha Levin mm->owner,
25231c9afa6SSasha Levin #endif
25331c9afa6SSasha Levin mm->exe_file,
25431c9afa6SSasha Levin #ifdef CONFIG_MMU_NOTIFIER
255984cfe4eSJason Gunthorpe mm->notifier_subscriptions,
25631c9afa6SSasha Levin #endif
25731c9afa6SSasha Levin #ifdef CONFIG_NUMA_BALANCING
25831c9afa6SSasha Levin mm->numa_next_scan, mm->numa_scan_offset, mm->numa_scan_seq,
25931c9afa6SSasha Levin #endif
26016af97dcSNadav Amit atomic_read(&mm->tlb_flush_pending),
261b8eceeb9SVlastimil Babka mm->def_flags, &mm->def_flags
26231c9afa6SSasha Levin );
26331c9afa6SSasha Levin }
264c2fdc235SSuren Baghdasaryan EXPORT_SYMBOL(dump_mm);
26531c9afa6SSasha Levin
dump_vmg(const struct vma_merge_struct * vmg,const char * reason)266b0d66d82SLorenzo Stoakes void dump_vmg(const struct vma_merge_struct *vmg, const char *reason)
267b0d66d82SLorenzo Stoakes {
268b0d66d82SLorenzo Stoakes if (reason)
269b0d66d82SLorenzo Stoakes pr_warn("vmg %px dumped because: %s\n", vmg, reason);
270b0d66d82SLorenzo Stoakes
271b0d66d82SLorenzo Stoakes if (!vmg) {
272b0d66d82SLorenzo Stoakes pr_warn("vmg %px state: (NULL)\n", vmg);
273b0d66d82SLorenzo Stoakes return;
274b0d66d82SLorenzo Stoakes }
275b0d66d82SLorenzo Stoakes
276b0d66d82SLorenzo Stoakes pr_warn("vmg %px state: mm %px pgoff %lx\n"
277b0d66d82SLorenzo Stoakes "vmi %px [%lx,%lx)\n"
2783a75ccbaSLorenzo Stoakes "prev %px middle %px next %px target %px\n"
279b0d66d82SLorenzo Stoakes "start %lx end %lx flags %lx\n"
280b0d66d82SLorenzo Stoakes "file %px anon_vma %px policy %px\n"
281b0d66d82SLorenzo Stoakes "uffd_ctx %px\n"
282b0d66d82SLorenzo Stoakes "anon_name %px\n"
2836ab2d9c7SLorenzo Stoakes "state %x\n"
284fe3e9cf0SLorenzo Stoakes "just_expand %d\n"
285fe3e9cf0SLorenzo Stoakes "__adjust_middle_start %d __adjust_next_start %d\n"
286fe3e9cf0SLorenzo Stoakes "__remove_middle %d __remove_next %d\n",
287b0d66d82SLorenzo Stoakes vmg, vmg->mm, vmg->pgoff,
288b0d66d82SLorenzo Stoakes vmg->vmi, vmg->vmi ? vma_iter_addr(vmg->vmi) : 0,
289b0d66d82SLorenzo Stoakes vmg->vmi ? vma_iter_end(vmg->vmi) : 0,
2903a75ccbaSLorenzo Stoakes vmg->prev, vmg->middle, vmg->next, vmg->target,
291b0d66d82SLorenzo Stoakes vmg->start, vmg->end, vmg->flags,
292b0d66d82SLorenzo Stoakes vmg->file, vmg->anon_vma, vmg->policy,
293b0d66d82SLorenzo Stoakes #ifdef CONFIG_USERFAULTFD
294b0d66d82SLorenzo Stoakes vmg->uffd_ctx.ctx,
295b0d66d82SLorenzo Stoakes #else
296b0d66d82SLorenzo Stoakes (void *)0,
297b0d66d82SLorenzo Stoakes #endif
298b0d66d82SLorenzo Stoakes vmg->anon_name,
2996ab2d9c7SLorenzo Stoakes (int)vmg->state,
300fe3e9cf0SLorenzo Stoakes vmg->just_expand,
301fe3e9cf0SLorenzo Stoakes vmg->__adjust_middle_start, vmg->__adjust_next_start,
302fe3e9cf0SLorenzo Stoakes vmg->__remove_middle, vmg->__remove_next);
303b0d66d82SLorenzo Stoakes
304b0d66d82SLorenzo Stoakes if (vmg->mm) {
305b0d66d82SLorenzo Stoakes pr_warn("vmg %px mm:\n", vmg);
306b0d66d82SLorenzo Stoakes dump_mm(vmg->mm);
307b0d66d82SLorenzo Stoakes } else {
308b0d66d82SLorenzo Stoakes pr_warn("vmg %px mm: (NULL)\n", vmg);
309b0d66d82SLorenzo Stoakes }
310b0d66d82SLorenzo Stoakes
311b0d66d82SLorenzo Stoakes if (vmg->prev) {
312b0d66d82SLorenzo Stoakes pr_warn("vmg %px prev:\n", vmg);
313b0d66d82SLorenzo Stoakes dump_vma(vmg->prev);
314b0d66d82SLorenzo Stoakes } else {
315b0d66d82SLorenzo Stoakes pr_warn("vmg %px prev: (NULL)\n", vmg);
316b0d66d82SLorenzo Stoakes }
317b0d66d82SLorenzo Stoakes
3183a75ccbaSLorenzo Stoakes if (vmg->middle) {
3193a75ccbaSLorenzo Stoakes pr_warn("vmg %px middle:\n", vmg);
3203a75ccbaSLorenzo Stoakes dump_vma(vmg->middle);
3213a75ccbaSLorenzo Stoakes } else {
3223a75ccbaSLorenzo Stoakes pr_warn("vmg %px middle: (NULL)\n", vmg);
3233a75ccbaSLorenzo Stoakes }
3243a75ccbaSLorenzo Stoakes
325b0d66d82SLorenzo Stoakes if (vmg->next) {
326b0d66d82SLorenzo Stoakes pr_warn("vmg %px next:\n", vmg);
327b0d66d82SLorenzo Stoakes dump_vma(vmg->next);
328b0d66d82SLorenzo Stoakes } else {
329b0d66d82SLorenzo Stoakes pr_warn("vmg %px next: (NULL)\n", vmg);
330b0d66d82SLorenzo Stoakes }
331b0d66d82SLorenzo Stoakes
332b0d66d82SLorenzo Stoakes #ifdef CONFIG_DEBUG_VM_MAPLE_TREE
333b0d66d82SLorenzo Stoakes if (vmg->vmi) {
334b0d66d82SLorenzo Stoakes pr_warn("vmg %px vmi:\n", vmg);
335b0d66d82SLorenzo Stoakes vma_iter_dump_tree(vmg->vmi);
336b0d66d82SLorenzo Stoakes } else {
337b0d66d82SLorenzo Stoakes pr_warn("vmg %px vmi: (NULL)\n", vmg);
338b0d66d82SLorenzo Stoakes }
339b0d66d82SLorenzo Stoakes #endif
340b0d66d82SLorenzo Stoakes }
341b0d66d82SLorenzo Stoakes EXPORT_SYMBOL(dump_vmg);
342b0d66d82SLorenzo Stoakes
343f682a97aSAlexander Duyck static bool page_init_poisoning __read_mostly = true;
344f682a97aSAlexander Duyck
setup_vm_debug(char * str)345f682a97aSAlexander Duyck static int __init setup_vm_debug(char *str)
346f682a97aSAlexander Duyck {
347f682a97aSAlexander Duyck bool __page_init_poisoning = true;
348f682a97aSAlexander Duyck
349f682a97aSAlexander Duyck /*
350f682a97aSAlexander Duyck * Calling vm_debug with no arguments is equivalent to requesting
351f682a97aSAlexander Duyck * to enable all debugging options we can control.
352f682a97aSAlexander Duyck */
353f682a97aSAlexander Duyck if (*str++ != '=' || !*str)
354f682a97aSAlexander Duyck goto out;
355f682a97aSAlexander Duyck
356f682a97aSAlexander Duyck __page_init_poisoning = false;
357f682a97aSAlexander Duyck if (*str == '-')
358f682a97aSAlexander Duyck goto out;
359f682a97aSAlexander Duyck
360f682a97aSAlexander Duyck while (*str) {
361f682a97aSAlexander Duyck switch (tolower(*str)) {
362f682a97aSAlexander Duyck case'p':
363f682a97aSAlexander Duyck __page_init_poisoning = true;
364f682a97aSAlexander Duyck break;
365f682a97aSAlexander Duyck default:
366f682a97aSAlexander Duyck pr_err("vm_debug option '%c' unknown. skipped\n",
367f682a97aSAlexander Duyck *str);
368f682a97aSAlexander Duyck }
369f682a97aSAlexander Duyck
370f682a97aSAlexander Duyck str++;
371f682a97aSAlexander Duyck }
372f682a97aSAlexander Duyck out:
373f682a97aSAlexander Duyck if (page_init_poisoning && !__page_init_poisoning)
374f682a97aSAlexander Duyck pr_warn("Page struct poisoning disabled by kernel command line option 'vm_debug'\n");
375f682a97aSAlexander Duyck
376f682a97aSAlexander Duyck page_init_poisoning = __page_init_poisoning;
377f682a97aSAlexander Duyck
378f682a97aSAlexander Duyck return 1;
379f682a97aSAlexander Duyck }
380f682a97aSAlexander Duyck __setup("vm_debug", setup_vm_debug);
381f682a97aSAlexander Duyck
page_init_poison(struct page * page,size_t size)382f682a97aSAlexander Duyck void page_init_poison(struct page *page, size_t size)
383f682a97aSAlexander Duyck {
384f682a97aSAlexander Duyck if (page_init_poisoning)
385f682a97aSAlexander Duyck memset(page, PAGE_POISON_PATTERN, size);
386f682a97aSAlexander Duyck }
387b50e195fSLiam R. Howlett
vma_iter_dump_tree(const struct vma_iterator * vmi)388b50e195fSLiam R. Howlett void vma_iter_dump_tree(const struct vma_iterator *vmi)
389b50e195fSLiam R. Howlett {
390b50e195fSLiam R. Howlett #if defined(CONFIG_DEBUG_VM_MAPLE_TREE)
391b50e195fSLiam R. Howlett mas_dump(&vmi->mas);
392b50e195fSLiam R. Howlett mt_dump(vmi->mas.tree, mt_dump_hex);
393b50e195fSLiam R. Howlett #endif /* CONFIG_DEBUG_VM_MAPLE_TREE */
394b50e195fSLiam R. Howlett }
395b50e195fSLiam R. Howlett
39682742a3aSSasha Levin #endif /* CONFIG_DEBUG_VM */
397