xref: /linux/mm/hugetlb_vmemmap.c (revision 0fc8f6200d2313278fbf4539bbab74677c685531)
1f41f2ed4SMuchun Song // SPDX-License-Identifier: GPL-2.0
2f41f2ed4SMuchun Song /*
3dff03381SMuchun Song  * HugeTLB Vmemmap Optimization (HVO)
4f41f2ed4SMuchun Song  *
5dff03381SMuchun Song  * Copyright (c) 2020, ByteDance. All rights reserved.
6f41f2ed4SMuchun Song  *
7f41f2ed4SMuchun Song  *     Author: Muchun Song <songmuchun@bytedance.com>
8f41f2ed4SMuchun Song  *
9ee65728eSMike Rapoport  * See Documentation/mm/vmemmap_dedup.rst
10f41f2ed4SMuchun Song  */
11e9fdff87SMuchun Song #define pr_fmt(fmt)	"HugeTLB: " fmt
12e9fdff87SMuchun Song 
13998a2997SMuchun Song #include <linux/pgtable.h>
14db5e8d84SVasily Gorbik #include <linux/moduleparam.h>
15998a2997SMuchun Song #include <linux/bootmem_info.h>
16d8f5f7e4SMike Kravetz #include <linux/mmdebug.h>
17fb93ed63SMuchun Song #include <linux/pagewalk.h>
18ad8b2e09SHarry Yoo #include <linux/pgalloc.h>
19ad8b2e09SHarry Yoo 
20998a2997SMuchun Song #include <asm/tlbflush.h>
21f41f2ed4SMuchun Song #include "hugetlb_vmemmap.h"
22622026e8SKiryl Shutsemau #include "internal.h"
23f41f2ed4SMuchun Song 
24998a2997SMuchun Song /**
25998a2997SMuchun Song  * struct vmemmap_remap_walk - walk vmemmap page table
26998a2997SMuchun Song  *
27998a2997SMuchun Song  * @remap_pte:		called for each lowest-level entry (PTE).
28998a2997SMuchun Song  * @nr_walked:		the number of walked pte.
29c0b495b9SKiryl Shutsemau  * @vmemmap_head:	the page to be installed as first in the vmemmap range
30c0b495b9SKiryl Shutsemau  * @vmemmap_tail:	the page to be installed as non-first in the vmemmap range
31998a2997SMuchun Song  * @vmemmap_pages:	the list head of the vmemmap pages that can be freed
32998a2997SMuchun Song  *			or is mapped from.
33f4b7e3efSJoao Martins  * @flags:		used to modify behavior in vmemmap page table walking
34f4b7e3efSJoao Martins  *			operations.
35998a2997SMuchun Song  */
36998a2997SMuchun Song struct vmemmap_remap_walk {
37998a2997SMuchun Song 	void			(*remap_pte)(pte_t *pte, unsigned long addr,
38998a2997SMuchun Song 					     struct vmemmap_remap_walk *walk);
39c0b495b9SKiryl Shutsemau 
40998a2997SMuchun Song 	unsigned long		nr_walked;
41c0b495b9SKiryl Shutsemau 	struct page		*vmemmap_head;
42c0b495b9SKiryl Shutsemau 	struct page		*vmemmap_tail;
43998a2997SMuchun Song 	struct list_head	*vmemmap_pages;
44f4b7e3efSJoao Martins 
45c0b495b9SKiryl Shutsemau 
46f4b7e3efSJoao Martins /* Skip the TLB flush when we split the PMD */
47f4b7e3efSJoao Martins #define VMEMMAP_SPLIT_NO_TLB_FLUSH	BIT(0)
48f13b83fdSJoao Martins /* Skip the TLB flush when we remap the PTE */
49f13b83fdSJoao Martins #define VMEMMAP_REMAP_NO_TLB_FLUSH	BIT(1)
50f4b7e3efSJoao Martins 	unsigned long		flags;
51998a2997SMuchun Song };
52998a2997SMuchun Song 
vmemmap_split_pmd(pmd_t * pmd,struct page * head,unsigned long start,struct vmemmap_remap_walk * walk)53fb93ed63SMuchun Song static int vmemmap_split_pmd(pmd_t *pmd, struct page *head, unsigned long start,
54fb93ed63SMuchun Song 			     struct vmemmap_remap_walk *walk)
55998a2997SMuchun Song {
56998a2997SMuchun Song 	pmd_t __pmd;
57998a2997SMuchun Song 	int i;
58998a2997SMuchun Song 	unsigned long addr = start;
593ce2c24cSMuchun Song 	pte_t *pgtable;
60998a2997SMuchun Song 
613ce2c24cSMuchun Song 	pgtable = pte_alloc_one_kernel(&init_mm);
62998a2997SMuchun Song 	if (!pgtable)
63998a2997SMuchun Song 		return -ENOMEM;
64998a2997SMuchun Song 
65998a2997SMuchun Song 	pmd_populate_kernel(&init_mm, &__pmd, pgtable);
66998a2997SMuchun Song 
67e38f055dSMuchun Song 	for (i = 0; i < PTRS_PER_PTE; i++, addr += PAGE_SIZE) {
68998a2997SMuchun Song 		pte_t entry, *pte;
69998a2997SMuchun Song 		pgprot_t pgprot = PAGE_KERNEL;
70998a2997SMuchun Song 
713ce2c24cSMuchun Song 		entry = mk_pte(head + i, pgprot);
72998a2997SMuchun Song 		pte = pte_offset_kernel(&__pmd, addr);
73998a2997SMuchun Song 		set_pte_at(&init_mm, addr, pte, entry);
74998a2997SMuchun Song 	}
75998a2997SMuchun Song 
76998a2997SMuchun Song 	spin_lock(&init_mm.page_table_lock);
77998a2997SMuchun Song 	if (likely(pmd_leaf(*pmd))) {
78998a2997SMuchun Song 		/*
79998a2997SMuchun Song 		 * Higher order allocations from buddy allocator must be able to
80b6c46600Sjianyun.gao 		 * be treated as independent small pages (as they can be freed
81998a2997SMuchun Song 		 * individually).
82998a2997SMuchun Song 		 */
833ce2c24cSMuchun Song 		if (!PageReserved(head))
843ce2c24cSMuchun Song 			split_page(head, get_order(PMD_SIZE));
85998a2997SMuchun Song 
86998a2997SMuchun Song 		/* Make pte visible before pmd. See comment in pmd_install(). */
87998a2997SMuchun Song 		smp_wmb();
88998a2997SMuchun Song 		pmd_populate_kernel(&init_mm, pmd, pgtable);
89fb93ed63SMuchun Song 		if (!(walk->flags & VMEMMAP_SPLIT_NO_TLB_FLUSH))
90998a2997SMuchun Song 			flush_tlb_kernel_range(start, start + PMD_SIZE);
91998a2997SMuchun Song 	} else {
92998a2997SMuchun Song 		pte_free_kernel(&init_mm, pgtable);
93998a2997SMuchun Song 	}
94998a2997SMuchun Song 	spin_unlock(&init_mm.page_table_lock);
95998a2997SMuchun Song 
96998a2997SMuchun Song 	return 0;
97998a2997SMuchun Song }
98998a2997SMuchun Song 
vmemmap_pmd_entry(pmd_t * pmd,unsigned long addr,unsigned long next,struct mm_walk * walk)99fb93ed63SMuchun Song static int vmemmap_pmd_entry(pmd_t *pmd, unsigned long addr,
100fb93ed63SMuchun Song 			     unsigned long next, struct mm_walk *walk)
101998a2997SMuchun Song {
102be035a2aSMuchun Song 	int ret = 0;
103fb93ed63SMuchun Song 	struct page *head;
104fb93ed63SMuchun Song 	struct vmemmap_remap_walk *vmemmap_walk = walk->private;
105998a2997SMuchun Song 
106fb93ed63SMuchun Song 	/* Only splitting, not remapping the vmemmap pages. */
107fb93ed63SMuchun Song 	if (!vmemmap_walk->remap_pte)
108fb93ed63SMuchun Song 		walk->action = ACTION_CONTINUE;
109fb93ed63SMuchun Song 
110fb93ed63SMuchun Song 	spin_lock(&init_mm.page_table_lock);
111fb93ed63SMuchun Song 	head = pmd_leaf(*pmd) ? pmd_page(*pmd) : NULL;
112be035a2aSMuchun Song 	/*
113be035a2aSMuchun Song 	 * Due to HugeTLB alignment requirements and the vmemmap
114be035a2aSMuchun Song 	 * pages being at the start of the hotplugged memory
115be035a2aSMuchun Song 	 * region in memory_hotplug.memmap_on_memory case. Checking
116be035a2aSMuchun Song 	 * the vmemmap page associated with the first vmemmap page
117be035a2aSMuchun Song 	 * if it is self-hosted is sufficient.
118be035a2aSMuchun Song 	 *
119be035a2aSMuchun Song 	 * [                  hotplugged memory                  ]
120be035a2aSMuchun Song 	 * [        section        ][...][        section        ]
121be035a2aSMuchun Song 	 * [ vmemmap ][              usable memory               ]
122be035a2aSMuchun Song 	 *   ^  | ^                        |
123be035a2aSMuchun Song 	 *   +--+ |                        |
124be035a2aSMuchun Song 	 *        +------------------------+
125be035a2aSMuchun Song 	 */
12647e61d88SMuchun Song 	if (IS_ENABLED(CONFIG_MEMORY_HOTPLUG) && unlikely(!vmemmap_walk->nr_walked)) {
127be035a2aSMuchun Song 		struct page *page = head ? head + pte_index(addr) :
128be035a2aSMuchun Song 				    pte_page(ptep_get(pte_offset_kernel(pmd, addr)));
129be035a2aSMuchun Song 
130be035a2aSMuchun Song 		if (PageVmemmapSelfHosted(page))
131be035a2aSMuchun Song 			ret = -ENOTSUPP;
132be035a2aSMuchun Song 	}
133fb93ed63SMuchun Song 	spin_unlock(&init_mm.page_table_lock);
134be035a2aSMuchun Song 	if (!head || ret)
135be035a2aSMuchun Song 		return ret;
136fb93ed63SMuchun Song 
137fb93ed63SMuchun Song 	return vmemmap_split_pmd(pmd, head, addr & PMD_MASK, vmemmap_walk);
138998a2997SMuchun Song }
139998a2997SMuchun Song 
vmemmap_pte_entry(pte_t * pte,unsigned long addr,unsigned long next,struct mm_walk * walk)140fb93ed63SMuchun Song static int vmemmap_pte_entry(pte_t *pte, unsigned long addr,
141fb93ed63SMuchun Song 			     unsigned long next, struct mm_walk *walk)
142998a2997SMuchun Song {
143fb93ed63SMuchun Song 	struct vmemmap_remap_walk *vmemmap_walk = walk->private;
144f4b7e3efSJoao Martins 
145fb93ed63SMuchun Song 	vmemmap_walk->remap_pte(pte, addr, vmemmap_walk);
146fb93ed63SMuchun Song 	vmemmap_walk->nr_walked++;
147998a2997SMuchun Song 
148998a2997SMuchun Song 	return 0;
149998a2997SMuchun Song }
150998a2997SMuchun Song 
151fb93ed63SMuchun Song static const struct mm_walk_ops vmemmap_remap_ops = {
152fb93ed63SMuchun Song 	.pmd_entry	= vmemmap_pmd_entry,
153fb93ed63SMuchun Song 	.pte_entry	= vmemmap_pte_entry,
154fb93ed63SMuchun Song };
155998a2997SMuchun Song 
vmemmap_remap_range(unsigned long start,unsigned long end,struct vmemmap_remap_walk * walk)156998a2997SMuchun Song static int vmemmap_remap_range(unsigned long start, unsigned long end,
157998a2997SMuchun Song 			       struct vmemmap_remap_walk *walk)
158998a2997SMuchun Song {
159998a2997SMuchun Song 	int ret;
160998a2997SMuchun Song 
161fb93ed63SMuchun Song 	VM_BUG_ON(!PAGE_ALIGNED(start | end));
162fb93ed63SMuchun Song 
16349b960deSMuchun Song 	mmap_read_lock(&init_mm);
16496d81e47SLorenzo Stoakes 	ret = walk_kernel_page_table_range(start, end, &vmemmap_remap_ops,
165fb93ed63SMuchun Song 				    NULL, walk);
16649b960deSMuchun Song 	mmap_read_unlock(&init_mm);
167998a2997SMuchun Song 	if (ret)
168998a2997SMuchun Song 		return ret;
169998a2997SMuchun Song 
170f13b83fdSJoao Martins 	if (walk->remap_pte && !(walk->flags & VMEMMAP_REMAP_NO_TLB_FLUSH))
17111aad263SJoao Martins 		flush_tlb_kernel_range(start, end);
172998a2997SMuchun Song 
173998a2997SMuchun Song 	return 0;
174998a2997SMuchun Song }
175998a2997SMuchun Song 
176998a2997SMuchun Song /*
177998a2997SMuchun Song  * Free a vmemmap page. A vmemmap page can be allocated from the memblock
178998a2997SMuchun Song  * allocator or buddy allocator. If the PG_reserved flag is set, it means
179998a2997SMuchun Song  * that it allocated from the memblock allocator, just free it via the
180998a2997SMuchun Song  * free_bootmem_page(). Otherwise, use __free_page().
181998a2997SMuchun Song  */
free_vmemmap_page(struct page * page)182998a2997SMuchun Song static inline void free_vmemmap_page(struct page *page)
183998a2997SMuchun Song {
18415995a35SSourav Panda 	if (PageReserved(page)) {
1859d857311SPasha Tatashin 		memmap_boot_pages_add(-1);
186998a2997SMuchun Song 		free_bootmem_page(page);
18715995a35SSourav Panda 	} else {
1889d857311SPasha Tatashin 		memmap_pages_add(-1);
189998a2997SMuchun Song 		__free_page(page);
19015995a35SSourav Panda 	}
191998a2997SMuchun Song }
192998a2997SMuchun Song 
193998a2997SMuchun Song /* Free a list of the vmemmap pages */
free_vmemmap_page_list(struct list_head * list)194998a2997SMuchun Song static void free_vmemmap_page_list(struct list_head *list)
195998a2997SMuchun Song {
196998a2997SMuchun Song 	struct page *page, *next;
197998a2997SMuchun Song 
1981cc53a04SMuchun Song 	list_for_each_entry_safe(page, next, list, lru)
199998a2997SMuchun Song 		free_vmemmap_page(page);
200998a2997SMuchun Song }
201998a2997SMuchun Song 
vmemmap_remap_pte(pte_t * pte,unsigned long addr,struct vmemmap_remap_walk * walk)202998a2997SMuchun Song static void vmemmap_remap_pte(pte_t *pte, unsigned long addr,
203998a2997SMuchun Song 			      struct vmemmap_remap_walk *walk)
204998a2997SMuchun Song {
205c33c7948SRyan Roberts 	struct page *page = pte_page(ptep_get(pte));
20611aad263SJoao Martins 	pte_t entry;
207998a2997SMuchun Song 
20811aad263SJoao Martins 	/* Remapping the head page requires r/w */
209c0b495b9SKiryl Shutsemau 	if (unlikely(walk->nr_walked == 0 && walk->vmemmap_head)) {
210c0b495b9SKiryl Shutsemau 		list_del(&walk->vmemmap_head->lru);
21111aad263SJoao Martins 
21211aad263SJoao Martins 		/*
21311aad263SJoao Martins 		 * Makes sure that preceding stores to the page contents from
21411aad263SJoao Martins 		 * vmemmap_remap_free() become visible before the set_pte_at()
21511aad263SJoao Martins 		 * write.
21611aad263SJoao Martins 		 */
21711aad263SJoao Martins 		smp_wmb();
218c0b495b9SKiryl Shutsemau 
219c0b495b9SKiryl Shutsemau 		entry = mk_pte(walk->vmemmap_head, PAGE_KERNEL);
220c0b495b9SKiryl Shutsemau 	} else {
221c0b495b9SKiryl Shutsemau 		/*
222c0b495b9SKiryl Shutsemau 		 * Remap the tail pages as read-only to catch illegal write
223c0b495b9SKiryl Shutsemau 		 * operation to the tail pages.
224c0b495b9SKiryl Shutsemau 		 */
225c0b495b9SKiryl Shutsemau 		entry = mk_pte(walk->vmemmap_tail, PAGE_KERNEL_RO);
22611aad263SJoao Martins 	}
22711aad263SJoao Martins 
22891f386bfSMike Kravetz 	list_add(&page->lru, walk->vmemmap_pages);
229998a2997SMuchun Song 	set_pte_at(&init_mm, addr, pte, entry);
230998a2997SMuchun Song }
231998a2997SMuchun Song 
vmemmap_restore_pte(pte_t * pte,unsigned long addr,struct vmemmap_remap_walk * walk)232998a2997SMuchun Song static void vmemmap_restore_pte(pte_t *pte, unsigned long addr,
233998a2997SMuchun Song 				struct vmemmap_remap_walk *walk)
234998a2997SMuchun Song {
235998a2997SMuchun Song 	struct page *page;
236c0b495b9SKiryl Shutsemau 	struct page *from, *to;
237998a2997SMuchun Song 
238998a2997SMuchun Song 	page = list_first_entry(walk->vmemmap_pages, struct page, lru);
239998a2997SMuchun Song 	list_del(&page->lru);
240c0b495b9SKiryl Shutsemau 
241c0b495b9SKiryl Shutsemau 	/*
242c0b495b9SKiryl Shutsemau 	 * Initialize tail pages in the newly allocated vmemmap page.
243c0b495b9SKiryl Shutsemau 	 *
244c0b495b9SKiryl Shutsemau 	 * There is folio-scope metadata that is encoded in the first few
245c0b495b9SKiryl Shutsemau 	 * tail pages.
246c0b495b9SKiryl Shutsemau 	 *
247c0b495b9SKiryl Shutsemau 	 * Use the value last tail page in the page with the head page
248c0b495b9SKiryl Shutsemau 	 * to initialize the rest of tail pages.
249c0b495b9SKiryl Shutsemau 	 */
250c0b495b9SKiryl Shutsemau 	from = compound_head((struct page *)addr) +
251c0b495b9SKiryl Shutsemau 		PAGE_SIZE / sizeof(struct page) - 1;
252998a2997SMuchun Song 	to = page_to_virt(page);
253c0b495b9SKiryl Shutsemau 	for (int i = 0; i < PAGE_SIZE / sizeof(struct page); i++, to++)
254c0b495b9SKiryl Shutsemau 		*to = *from;
255998a2997SMuchun Song 
256939de63dSMiaohe Lin 	/*
257939de63dSMiaohe Lin 	 * Makes sure that preceding stores to the page contents become visible
258939de63dSMiaohe Lin 	 * before the set_pte_at() write.
259939de63dSMiaohe Lin 	 */
260939de63dSMiaohe Lin 	smp_wmb();
261c0b495b9SKiryl Shutsemau 	set_pte_at(&init_mm, addr, pte, mk_pte(page, PAGE_KERNEL));
262998a2997SMuchun Song }
263998a2997SMuchun Song 
264998a2997SMuchun Song /**
265f4b7e3efSJoao Martins  * vmemmap_remap_split - split the vmemmap virtual address range [@start, @end)
266f4b7e3efSJoao Martins  *                      backing PMDs of the directmap into PTEs
267f4b7e3efSJoao Martins  * @start:     start address of the vmemmap virtual address range that we want
268f4b7e3efSJoao Martins  *             to remap.
269f4b7e3efSJoao Martins  * @end:       end address of the vmemmap virtual address range that we want to
270f4b7e3efSJoao Martins  *             remap.
271f4b7e3efSJoao Martins  * Return: %0 on success, negative error code otherwise.
272f4b7e3efSJoao Martins  */
vmemmap_remap_split(unsigned long start,unsigned long end)273c0b495b9SKiryl Shutsemau static int vmemmap_remap_split(unsigned long start, unsigned long end)
274f4b7e3efSJoao Martins {
275f4b7e3efSJoao Martins 	struct vmemmap_remap_walk walk = {
276f4b7e3efSJoao Martins 		.remap_pte	= NULL,
277f4b7e3efSJoao Martins 		.flags		= VMEMMAP_SPLIT_NO_TLB_FLUSH,
278f4b7e3efSJoao Martins 	};
279f4b7e3efSJoao Martins 
280c0b495b9SKiryl Shutsemau 	return vmemmap_remap_range(start, end, &walk);
281f4b7e3efSJoao Martins }
282f4b7e3efSJoao Martins 
283f4b7e3efSJoao Martins /**
284998a2997SMuchun Song  * vmemmap_remap_free - remap the vmemmap virtual address range [@start, @end)
285c0b495b9SKiryl Shutsemau  *			to use @vmemmap_head/tail, then free vmemmap which
286c0b495b9SKiryl Shutsemau  *			the range are mapped to.
287998a2997SMuchun Song  * @start:	start address of the vmemmap virtual address range that we want
288998a2997SMuchun Song  *		to remap.
289998a2997SMuchun Song  * @end:	end address of the vmemmap virtual address range that we want to
290998a2997SMuchun Song  *		remap.
291c0b495b9SKiryl Shutsemau  * @vmemmap_head: the page to be installed as first in the vmemmap range
292c0b495b9SKiryl Shutsemau  * @vmemmap_tail: the page to be installed as non-first in the vmemmap range
29391f386bfSMike Kravetz  * @vmemmap_pages: list to deposit vmemmap pages to be freed.  It is callers
29491f386bfSMike Kravetz  *		responsibility to free pages.
295f13b83fdSJoao Martins  * @flags:	modifications to vmemmap_remap_walk flags
296998a2997SMuchun Song  *
297998a2997SMuchun Song  * Return: %0 on success, negative error code otherwise.
298998a2997SMuchun Song  */
vmemmap_remap_free(unsigned long start,unsigned long end,struct page * vmemmap_head,struct page * vmemmap_tail,struct list_head * vmemmap_pages,unsigned long flags)299998a2997SMuchun Song static int vmemmap_remap_free(unsigned long start, unsigned long end,
300c0b495b9SKiryl Shutsemau 			      struct page *vmemmap_head,
301c0b495b9SKiryl Shutsemau 			      struct page *vmemmap_tail,
302f13b83fdSJoao Martins 			      struct list_head *vmemmap_pages,
303f13b83fdSJoao Martins 			      unsigned long flags)
304998a2997SMuchun Song {
305998a2997SMuchun Song 	int ret;
306998a2997SMuchun Song 	struct vmemmap_remap_walk walk = {
307998a2997SMuchun Song 		.remap_pte	= vmemmap_remap_pte,
308c0b495b9SKiryl Shutsemau 		.vmemmap_head	= vmemmap_head,
309c0b495b9SKiryl Shutsemau 		.vmemmap_tail	= vmemmap_tail,
31091f386bfSMike Kravetz 		.vmemmap_pages	= vmemmap_pages,
311f13b83fdSJoao Martins 		.flags		= flags,
312998a2997SMuchun Song 	};
313c0b495b9SKiryl Shutsemau 
314c0b495b9SKiryl Shutsemau 	ret = vmemmap_remap_range(start, end, &walk);
315c0b495b9SKiryl Shutsemau 	if (!ret || !walk.nr_walked)
316c0b495b9SKiryl Shutsemau 		return ret;
317c0b495b9SKiryl Shutsemau 
318c0b495b9SKiryl Shutsemau 	end = start + walk.nr_walked * PAGE_SIZE;
31911aad263SJoao Martins 
32011aad263SJoao Martins 	/*
321c0b495b9SKiryl Shutsemau 	 * vmemmap_pages contains pages from the previous vmemmap_remap_range()
322c0b495b9SKiryl Shutsemau 	 * call which failed.  These are pages which were removed from
323c0b495b9SKiryl Shutsemau 	 * the vmemmap. They will be restored in the following call.
324998a2997SMuchun Song 	 */
325998a2997SMuchun Song 	walk = (struct vmemmap_remap_walk) {
326998a2997SMuchun Song 		.remap_pte	= vmemmap_restore_pte,
32791f386bfSMike Kravetz 		.vmemmap_pages	= vmemmap_pages,
328f4b7e3efSJoao Martins 		.flags		= 0,
329998a2997SMuchun Song 	};
330998a2997SMuchun Song 
331c0b495b9SKiryl Shutsemau 	vmemmap_remap_range(start, end, &walk);
332998a2997SMuchun Song 
333998a2997SMuchun Song 	return ret;
334998a2997SMuchun Song }
335998a2997SMuchun Song 
alloc_vmemmap_page_list(unsigned long start,unsigned long end,struct list_head * list)336998a2997SMuchun Song static int alloc_vmemmap_page_list(unsigned long start, unsigned long end,
337eb83f652SPasha Tatashin 				   struct list_head *list)
338998a2997SMuchun Song {
3392eaa6c2aSYuan Can 	gfp_t gfp_mask = GFP_KERNEL | __GFP_RETRY_MAYFAIL;
340998a2997SMuchun Song 	unsigned long nr_pages = (end - start) >> PAGE_SHIFT;
341998a2997SMuchun Song 	int nid = page_to_nid((struct page *)start);
342998a2997SMuchun Song 	struct page *page, *next;
34315995a35SSourav Panda 	int i;
344998a2997SMuchun Song 
34515995a35SSourav Panda 	for (i = 0; i < nr_pages; i++) {
346998a2997SMuchun Song 		page = alloc_pages_node(nid, gfp_mask, 0);
347ace0741aSPasha Tatashin 		if (!page)
348998a2997SMuchun Song 			goto out;
34991f386bfSMike Kravetz 		list_add(&page->lru, list);
350998a2997SMuchun Song 	}
3519d857311SPasha Tatashin 	memmap_pages_add(nr_pages);
35215995a35SSourav Panda 
353998a2997SMuchun Song 	return 0;
354998a2997SMuchun Song out:
355998a2997SMuchun Song 	list_for_each_entry_safe(page, next, list, lru)
356dcc1be11SLorenzo Stoakes 		__free_page(page);
357998a2997SMuchun Song 	return -ENOMEM;
358998a2997SMuchun Song }
359998a2997SMuchun Song 
360998a2997SMuchun Song /**
361998a2997SMuchun Song  * vmemmap_remap_alloc - remap the vmemmap virtual address range [@start, end)
362998a2997SMuchun Song  *			 to the page which is from the @vmemmap_pages
363998a2997SMuchun Song  *			 respectively.
364998a2997SMuchun Song  * @start:	start address of the vmemmap virtual address range that we want
365998a2997SMuchun Song  *		to remap.
366998a2997SMuchun Song  * @end:	end address of the vmemmap virtual address range that we want to
367998a2997SMuchun Song  *		remap.
368c24f188bSMike Kravetz  * @flags:	modifications to vmemmap_remap_walk flags
369998a2997SMuchun Song  *
370998a2997SMuchun Song  * Return: %0 on success, negative error code otherwise.
371998a2997SMuchun Song  */
vmemmap_remap_alloc(unsigned long start,unsigned long end,unsigned long flags)372998a2997SMuchun Song static int vmemmap_remap_alloc(unsigned long start, unsigned long end,
373c0b495b9SKiryl Shutsemau 			       unsigned long flags)
374998a2997SMuchun Song {
375998a2997SMuchun Song 	LIST_HEAD(vmemmap_pages);
376998a2997SMuchun Song 	struct vmemmap_remap_walk walk = {
377998a2997SMuchun Song 		.remap_pte	= vmemmap_restore_pte,
378998a2997SMuchun Song 		.vmemmap_pages	= &vmemmap_pages,
379c24f188bSMike Kravetz 		.flags		= flags,
380998a2997SMuchun Song 	};
381998a2997SMuchun Song 
382eb83f652SPasha Tatashin 	if (alloc_vmemmap_page_list(start, end, &vmemmap_pages))
383998a2997SMuchun Song 		return -ENOMEM;
384998a2997SMuchun Song 
385c0b495b9SKiryl Shutsemau 	return vmemmap_remap_range(start, end, &walk);
386998a2997SMuchun Song }
387998a2997SMuchun Song 
38830152245SMuchun Song static bool vmemmap_optimize_enabled = IS_ENABLED(CONFIG_HUGETLB_PAGE_OPTIMIZE_VMEMMAP_DEFAULT_ON);
hugetlb_vmemmap_optimize_param(char * buf)3895b47c029SFrank van der Linden static int __init hugetlb_vmemmap_optimize_param(char *buf)
3905b47c029SFrank van der Linden {
3915b47c029SFrank van der Linden 	return kstrtobool(buf, &vmemmap_optimize_enabled);
3925b47c029SFrank van der Linden }
3935b47c029SFrank van der Linden early_param("hugetlb_free_vmemmap", hugetlb_vmemmap_optimize_param);
394f41f2ed4SMuchun Song 
__hugetlb_vmemmap_restore_folio(const struct hstate * h,struct folio * folio,unsigned long flags)395ebc20dcaSMuchun Song static int __hugetlb_vmemmap_restore_folio(const struct hstate *h,
396ebc20dcaSMuchun Song 					   struct folio *folio, unsigned long flags)
397ad2fa371SMuchun Song {
398ad2fa371SMuchun Song 	int ret;
399c0b495b9SKiryl Shutsemau 	unsigned long vmemmap_start, vmemmap_end;
400ad2fa371SMuchun Song 
401ebc20dcaSMuchun Song 	VM_WARN_ON_ONCE_FOLIO(!folio_test_hugetlb(folio), folio);
402bd225530SYu Zhao 	VM_WARN_ON_ONCE_FOLIO(folio_ref_count(folio), folio);
403bd225530SYu Zhao 
404c5ad3233SUsama Arif 	if (!folio_test_hugetlb_vmemmap_optimized(folio))
405ad2fa371SMuchun Song 		return 0;
406ad2fa371SMuchun Song 
407c0b495b9SKiryl Shutsemau 	vmemmap_start	= (unsigned long)&folio->page;
4086213834cSMuchun Song 	vmemmap_end	= vmemmap_start + hugetlb_vmemmap_size(h);
409c0b495b9SKiryl Shutsemau 
4106213834cSMuchun Song 	vmemmap_start	+= HUGETLB_VMEMMAP_RESERVE_SIZE;
4115981611dSMuchun Song 
412ad2fa371SMuchun Song 	/*
4136213834cSMuchun Song 	 * The pages which the vmemmap virtual address range [@vmemmap_start,
414c0b495b9SKiryl Shutsemau 	 * @vmemmap_end) are mapped to are freed to the buddy allocator.
415ad2fa371SMuchun Song 	 * When a HugeTLB page is freed to the buddy allocator, previously
416ad2fa371SMuchun Song 	 * discarded vmemmap pages must be allocated and remapping.
417ad2fa371SMuchun Song 	 */
418c0b495b9SKiryl Shutsemau 	ret = vmemmap_remap_alloc(vmemmap_start, vmemmap_end, flags);
419*da3e2d1cSKiryl Shutsemau 	if (!ret)
420c5ad3233SUsama Arif 		folio_clear_hugetlb_vmemmap_optimized(folio);
421ad2fa371SMuchun Song 
422ad2fa371SMuchun Song 	return ret;
423ad2fa371SMuchun Song }
424ad2fa371SMuchun Song 
425cfb8c750SMike Kravetz /**
426c5ad3233SUsama Arif  * hugetlb_vmemmap_restore_folio - restore previously optimized (by
427c5ad3233SUsama Arif  *				hugetlb_vmemmap_optimize_folio()) vmemmap pages which
428c24f188bSMike Kravetz  *				will be reallocated and remapped.
429c24f188bSMike Kravetz  * @h:		struct hstate.
430c5ad3233SUsama Arif  * @folio:     the folio whose vmemmap pages will be restored.
431c24f188bSMike Kravetz  *
432c5ad3233SUsama Arif  * Return: %0 if @folio's vmemmap pages have been reallocated and remapped,
433c24f188bSMike Kravetz  * negative error code otherwise.
434c24f188bSMike Kravetz  */
hugetlb_vmemmap_restore_folio(const struct hstate * h,struct folio * folio)435c5ad3233SUsama Arif int hugetlb_vmemmap_restore_folio(const struct hstate *h, struct folio *folio)
436c24f188bSMike Kravetz {
43701b1d0ffSKiryl Shutsemau 	return __hugetlb_vmemmap_restore_folio(h, folio, 0);
438c24f188bSMike Kravetz }
439c24f188bSMike Kravetz 
440c24f188bSMike Kravetz /**
441cfb8c750SMike Kravetz  * hugetlb_vmemmap_restore_folios - restore vmemmap for every folio on the list.
442cfb8c750SMike Kravetz  * @h:			hstate.
443cfb8c750SMike Kravetz  * @folio_list:		list of folios.
444cfb8c750SMike Kravetz  * @non_hvo_folios:	Output list of folios for which vmemmap exists.
445cfb8c750SMike Kravetz  *
446cfb8c750SMike Kravetz  * Return: number of folios for which vmemmap was restored, or an error code
447cfb8c750SMike Kravetz  *		if an error was encountered restoring vmemmap for a folio.
448cfb8c750SMike Kravetz  *		Folios that have vmemmap are moved to the non_hvo_folios
449cfb8c750SMike Kravetz  *		list.  Processing of entries stops when the first error is
450cfb8c750SMike Kravetz  *		encountered. The folio that experienced the error and all
451cfb8c750SMike Kravetz  *		non-processed folios will remain on folio_list.
452cfb8c750SMike Kravetz  */
hugetlb_vmemmap_restore_folios(const struct hstate * h,struct list_head * folio_list,struct list_head * non_hvo_folios)453cfb8c750SMike Kravetz long hugetlb_vmemmap_restore_folios(const struct hstate *h,
454cfb8c750SMike Kravetz 					struct list_head *folio_list,
455cfb8c750SMike Kravetz 					struct list_head *non_hvo_folios)
456cfb8c750SMike Kravetz {
457cfb8c750SMike Kravetz 	struct folio *folio, *t_folio;
458cfb8c750SMike Kravetz 	long restored = 0;
459cfb8c750SMike Kravetz 	long ret = 0;
46001b1d0ffSKiryl Shutsemau 	unsigned long flags = VMEMMAP_REMAP_NO_TLB_FLUSH;
461bd225530SYu Zhao 
462cfb8c750SMike Kravetz 	list_for_each_entry_safe(folio, t_folio, folio_list, lru) {
463cfb8c750SMike Kravetz 		if (folio_test_hugetlb_vmemmap_optimized(folio)) {
464c2a967f6SYu Zhao 			ret = __hugetlb_vmemmap_restore_folio(h, folio, flags);
465cfb8c750SMike Kravetz 			if (ret)
466cfb8c750SMike Kravetz 				break;
467cfb8c750SMike Kravetz 			restored++;
468cfb8c750SMike Kravetz 		}
469cfb8c750SMike Kravetz 
470cfb8c750SMike Kravetz 		/* Add non-optimized folios to output list */
471cfb8c750SMike Kravetz 		list_move(&folio->lru, non_hvo_folios);
472cfb8c750SMike Kravetz 	}
473cfb8c750SMike Kravetz 
474c24f188bSMike Kravetz 	if (restored)
475c24f188bSMike Kravetz 		flush_tlb_all();
476cfb8c750SMike Kravetz 	if (!ret)
477cfb8c750SMike Kravetz 		ret = restored;
478cfb8c750SMike Kravetz 	return ret;
479cfb8c750SMike Kravetz }
480cfb8c750SMike Kravetz 
4816213834cSMuchun Song /* Return true iff a HugeTLB whose vmemmap should and can be optimized. */
vmemmap_should_optimize_folio(const struct hstate * h,struct folio * folio)482ebc20dcaSMuchun Song static bool vmemmap_should_optimize_folio(const struct hstate *h, struct folio *folio)
48366361095SMuchun Song {
484ebc20dcaSMuchun Song 	if (folio_test_hugetlb_vmemmap_optimized(folio))
48579359d6dSMike Kravetz 		return false;
48679359d6dSMike Kravetz 
487cf5472e5SMuchun Song 	if (!READ_ONCE(vmemmap_optimize_enabled))
4886213834cSMuchun Song 		return false;
4896213834cSMuchun Song 
4906213834cSMuchun Song 	if (!hugetlb_vmemmap_optimizable(h))
4916213834cSMuchun Song 		return false;
49266361095SMuchun Song 
4936213834cSMuchun Song 	return true;
49466361095SMuchun Song }
49566361095SMuchun Song 
vmemmap_get_tail(unsigned int order,struct zone * zone)496622026e8SKiryl Shutsemau static struct page *vmemmap_get_tail(unsigned int order, struct zone *zone)
497622026e8SKiryl Shutsemau {
498622026e8SKiryl Shutsemau 	const unsigned int idx = order - VMEMMAP_TAIL_MIN_ORDER;
499622026e8SKiryl Shutsemau 	struct page *tail, *p;
500622026e8SKiryl Shutsemau 	int node = zone_to_nid(zone);
501622026e8SKiryl Shutsemau 
502622026e8SKiryl Shutsemau 	tail = READ_ONCE(zone->vmemmap_tails[idx]);
503622026e8SKiryl Shutsemau 	if (likely(tail))
504622026e8SKiryl Shutsemau 		return tail;
505622026e8SKiryl Shutsemau 
506622026e8SKiryl Shutsemau 	tail = alloc_pages_node(node, GFP_KERNEL | __GFP_ZERO, 0);
507622026e8SKiryl Shutsemau 	if (!tail)
508622026e8SKiryl Shutsemau 		return NULL;
509622026e8SKiryl Shutsemau 
510622026e8SKiryl Shutsemau 	p = page_to_virt(tail);
511622026e8SKiryl Shutsemau 	for (int i = 0; i < PAGE_SIZE / sizeof(struct page); i++)
512622026e8SKiryl Shutsemau 		init_compound_tail(p + i, NULL, order, zone);
513622026e8SKiryl Shutsemau 
514622026e8SKiryl Shutsemau 	if (cmpxchg(&zone->vmemmap_tails[idx], NULL, tail)) {
515622026e8SKiryl Shutsemau 		__free_page(tail);
516622026e8SKiryl Shutsemau 		tail = READ_ONCE(zone->vmemmap_tails[idx]);
517622026e8SKiryl Shutsemau 	}
518622026e8SKiryl Shutsemau 
519622026e8SKiryl Shutsemau 	return tail;
520622026e8SKiryl Shutsemau }
521622026e8SKiryl Shutsemau 
__hugetlb_vmemmap_optimize_folio(const struct hstate * h,struct folio * folio,struct list_head * vmemmap_pages,unsigned long flags)522c5ad3233SUsama Arif static int __hugetlb_vmemmap_optimize_folio(const struct hstate *h,
523c5ad3233SUsama Arif 					    struct folio *folio,
524f13b83fdSJoao Martins 					    struct list_head *vmemmap_pages,
525f13b83fdSJoao Martins 					    unsigned long flags)
52691f386bfSMike Kravetz {
527c0b495b9SKiryl Shutsemau 	unsigned long vmemmap_start, vmemmap_end;
528c0b495b9SKiryl Shutsemau 	struct page *vmemmap_head, *vmemmap_tail;
529c0b495b9SKiryl Shutsemau 	int nid, ret = 0;
53091f386bfSMike Kravetz 
531ebc20dcaSMuchun Song 	VM_WARN_ON_ONCE_FOLIO(!folio_test_hugetlb(folio), folio);
532bd225530SYu Zhao 	VM_WARN_ON_ONCE_FOLIO(folio_ref_count(folio), folio);
533bd225530SYu Zhao 
534ebc20dcaSMuchun Song 	if (!vmemmap_should_optimize_folio(h, folio))
53591f386bfSMike Kravetz 		return ret;
53691f386bfSMike Kravetz 
537622026e8SKiryl Shutsemau 	nid = folio_nid(folio);
538622026e8SKiryl Shutsemau 	vmemmap_tail = vmemmap_get_tail(h->order, folio_zone(folio));
539622026e8SKiryl Shutsemau 	if (!vmemmap_tail)
540622026e8SKiryl Shutsemau 		return -ENOMEM;
541622026e8SKiryl Shutsemau 
542f13b83fdSJoao Martins 	/*
543f13b83fdSJoao Martins 	 * Very Subtle
544f13b83fdSJoao Martins 	 * If VMEMMAP_REMAP_NO_TLB_FLUSH is set, TLB flushing is not performed
545f13b83fdSJoao Martins 	 * immediately after remapping.  As a result, subsequent accesses
546f13b83fdSJoao Martins 	 * and modifications to struct pages associated with the hugetlb
547f13b83fdSJoao Martins 	 * page could be to the OLD struct pages.  Set the vmemmap optimized
548f13b83fdSJoao Martins 	 * flag here so that it is copied to the new head page.  This keeps
549f13b83fdSJoao Martins 	 * the old and new struct pages in sync.
550f13b83fdSJoao Martins 	 * If there is an error during optimization, we will immediately FLUSH
551f13b83fdSJoao Martins 	 * the TLB and clear the flag below.
552f13b83fdSJoao Martins 	 */
553c5ad3233SUsama Arif 	folio_set_hugetlb_vmemmap_optimized(folio);
55491f386bfSMike Kravetz 
555c0b495b9SKiryl Shutsemau 	vmemmap_head = alloc_pages_node(nid, GFP_KERNEL, 0);
556c0b495b9SKiryl Shutsemau 	if (!vmemmap_head) {
557c0b495b9SKiryl Shutsemau 		ret = -ENOMEM;
558c0b495b9SKiryl Shutsemau 		goto out;
559c0b495b9SKiryl Shutsemau 	}
560c0b495b9SKiryl Shutsemau 
561c0b495b9SKiryl Shutsemau 	copy_page(page_to_virt(vmemmap_head), folio);
562c0b495b9SKiryl Shutsemau 	list_add(&vmemmap_head->lru, vmemmap_pages);
563c0b495b9SKiryl Shutsemau 	memmap_pages_add(1);
564c0b495b9SKiryl Shutsemau 
565c0b495b9SKiryl Shutsemau 	vmemmap_start	= (unsigned long)&folio->page;
56691f386bfSMike Kravetz 	vmemmap_end	= vmemmap_start + hugetlb_vmemmap_size(h);
56791f386bfSMike Kravetz 
56891f386bfSMike Kravetz 	/*
569c0b495b9SKiryl Shutsemau 	 * Remap the vmemmap virtual address range [@vmemmap_start, @vmemmap_end).
570c0b495b9SKiryl Shutsemau 	 * Add pages previously mapping the range to vmemmap_pages list so that
571c0b495b9SKiryl Shutsemau 	 * they can be freed by the caller.
57291f386bfSMike Kravetz 	 */
573c0b495b9SKiryl Shutsemau 	ret = vmemmap_remap_free(vmemmap_start, vmemmap_end,
574c0b495b9SKiryl Shutsemau 				 vmemmap_head, vmemmap_tail,
575f13b83fdSJoao Martins 				 vmemmap_pages, flags);
576c0b495b9SKiryl Shutsemau out:
577*da3e2d1cSKiryl Shutsemau 	if (ret)
578c5ad3233SUsama Arif 		folio_clear_hugetlb_vmemmap_optimized(folio);
57991f386bfSMike Kravetz 
58091f386bfSMike Kravetz 	return ret;
58191f386bfSMike Kravetz }
58291f386bfSMike Kravetz 
5836213834cSMuchun Song /**
584c5ad3233SUsama Arif  * hugetlb_vmemmap_optimize_folio - optimize @folio's vmemmap pages.
5856213834cSMuchun Song  * @h:		struct hstate.
586c5ad3233SUsama Arif  * @folio:     the folio whose vmemmap pages will be optimized.
5876213834cSMuchun Song  *
588c5ad3233SUsama Arif  * This function only tries to optimize @folio's vmemmap pages and does not
5896213834cSMuchun Song  * guarantee that the optimization will succeed after it returns. The caller
590c5ad3233SUsama Arif  * can use folio_test_hugetlb_vmemmap_optimized(@folio) to detect if @folio's
591c5ad3233SUsama Arif  * vmemmap pages have been optimized.
5926213834cSMuchun Song  */
hugetlb_vmemmap_optimize_folio(const struct hstate * h,struct folio * folio)593c5ad3233SUsama Arif void hugetlb_vmemmap_optimize_folio(const struct hstate *h, struct folio *folio)
594f41f2ed4SMuchun Song {
59591f386bfSMike Kravetz 	LIST_HEAD(vmemmap_pages);
596f41f2ed4SMuchun Song 
59701b1d0ffSKiryl Shutsemau 	__hugetlb_vmemmap_optimize_folio(h, folio, &vmemmap_pages, 0);
59891f386bfSMike Kravetz 	free_vmemmap_page_list(&vmemmap_pages);
599f41f2ed4SMuchun Song }
60077490587SMuchun Song 
hugetlb_vmemmap_split_folio(const struct hstate * h,struct folio * folio)601ebc20dcaSMuchun Song static int hugetlb_vmemmap_split_folio(const struct hstate *h, struct folio *folio)
602f4b7e3efSJoao Martins {
603c0b495b9SKiryl Shutsemau 	unsigned long vmemmap_start, vmemmap_end;
604f4b7e3efSJoao Martins 
605ebc20dcaSMuchun Song 	if (!vmemmap_should_optimize_folio(h, folio))
606f4b7e3efSJoao Martins 		return 0;
607f4b7e3efSJoao Martins 
608c0b495b9SKiryl Shutsemau 	vmemmap_start	= (unsigned long)&folio->page;
609f4b7e3efSJoao Martins 	vmemmap_end	= vmemmap_start + hugetlb_vmemmap_size(h);
610f4b7e3efSJoao Martins 
611f4b7e3efSJoao Martins 	/*
612f4b7e3efSJoao Martins 	 * Split PMDs on the vmemmap virtual address range [@vmemmap_start,
613f4b7e3efSJoao Martins 	 * @vmemmap_end]
614f4b7e3efSJoao Martins 	 */
615c0b495b9SKiryl Shutsemau 	return vmemmap_remap_split(vmemmap_start, vmemmap_end);
616f4b7e3efSJoao Martins }
617f4b7e3efSJoao Martins 
__hugetlb_vmemmap_optimize_folios(struct hstate * h,struct list_head * folio_list,bool boot)618752fe17aSFrank van der Linden static void __hugetlb_vmemmap_optimize_folios(struct hstate *h,
619752fe17aSFrank van der Linden 					      struct list_head *folio_list,
620752fe17aSFrank van der Linden 					      bool boot)
62179359d6dSMike Kravetz {
62279359d6dSMike Kravetz 	struct folio *folio;
623752fe17aSFrank van der Linden 	int nr_to_optimize;
62491f386bfSMike Kravetz 	LIST_HEAD(vmemmap_pages);
62501b1d0ffSKiryl Shutsemau 	unsigned long flags = VMEMMAP_REMAP_NO_TLB_FLUSH;
62679359d6dSMike Kravetz 
627752fe17aSFrank van der Linden 	nr_to_optimize = 0;
62891f386bfSMike Kravetz 	list_for_each_entry(folio, folio_list, lru) {
629752fe17aSFrank van der Linden 		int ret;
630752fe17aSFrank van der Linden 		unsigned long spfn, epfn;
631752fe17aSFrank van der Linden 
632752fe17aSFrank van der Linden 		if (boot && folio_test_hugetlb_vmemmap_optimized(folio)) {
633752fe17aSFrank van der Linden 			/*
634752fe17aSFrank van der Linden 			 * Already optimized by pre-HVO, just map the
635752fe17aSFrank van der Linden 			 * mirrored tail page structs RO.
636752fe17aSFrank van der Linden 			 */
637752fe17aSFrank van der Linden 			spfn = (unsigned long)&folio->page;
638752fe17aSFrank van der Linden 			epfn = spfn + pages_per_huge_page(h);
639752fe17aSFrank van der Linden 			vmemmap_wrprotect_hvo(spfn, epfn, folio_nid(folio),
640752fe17aSFrank van der Linden 					HUGETLB_VMEMMAP_RESERVE_SIZE);
641752fe17aSFrank van der Linden 			register_page_bootmem_memmap(pfn_to_section_nr(spfn),
642752fe17aSFrank van der Linden 					&folio->page,
643752fe17aSFrank van der Linden 					HUGETLB_VMEMMAP_RESERVE_SIZE);
644752fe17aSFrank van der Linden 			continue;
645752fe17aSFrank van der Linden 		}
646752fe17aSFrank van der Linden 
647752fe17aSFrank van der Linden 		nr_to_optimize++;
648752fe17aSFrank van der Linden 
649752fe17aSFrank van der Linden 		ret = hugetlb_vmemmap_split_folio(h, folio);
650f4b7e3efSJoao Martins 
651f4b7e3efSJoao Martins 		/*
652b6c46600Sjianyun.gao 		 * Splitting the PMD requires allocating a page, thus let's fail
653f4b7e3efSJoao Martins 		 * early once we encounter the first OOM. No point in retrying
654f4b7e3efSJoao Martins 		 * as it can be dynamically done on remap with the memory
655f4b7e3efSJoao Martins 		 * we get back from the vmemmap deduplication.
656f4b7e3efSJoao Martins 		 */
657f4b7e3efSJoao Martins 		if (ret == -ENOMEM)
658f4b7e3efSJoao Martins 			break;
659f4b7e3efSJoao Martins 	}
660f4b7e3efSJoao Martins 
661752fe17aSFrank van der Linden 	if (!nr_to_optimize)
662752fe17aSFrank van der Linden 		/*
663752fe17aSFrank van der Linden 		 * All pre-HVO folios, nothing left to do. It's ok if
664752fe17aSFrank van der Linden 		 * there is a mix of pre-HVO and not yet HVO-ed folios
665752fe17aSFrank van der Linden 		 * here, as __hugetlb_vmemmap_optimize_folio() will
666752fe17aSFrank van der Linden 		 * skip any folios that already have the optimized flag
667752fe17aSFrank van der Linden 		 * set, see vmemmap_should_optimize_folio().
668752fe17aSFrank van der Linden 		 */
669752fe17aSFrank van der Linden 		goto out;
670752fe17aSFrank van der Linden 
671f4b7e3efSJoao Martins 	flush_tlb_all();
672f4b7e3efSJoao Martins 
673f4b7e3efSJoao Martins 	list_for_each_entry(folio, folio_list, lru) {
674ebc20dcaSMuchun Song 		int ret;
675ebc20dcaSMuchun Song 
676c2a967f6SYu Zhao 		ret = __hugetlb_vmemmap_optimize_folio(h, folio, &vmemmap_pages, flags);
67791f386bfSMike Kravetz 
67891f386bfSMike Kravetz 		/*
67991f386bfSMike Kravetz 		 * Pages to be freed may have been accumulated.  If we
68091f386bfSMike Kravetz 		 * encounter an ENOMEM,  free what we have and try again.
681b6c46600Sjianyun.gao 		 * This can occur in the case that both splitting fails
682f13b83fdSJoao Martins 		 * halfway and head page allocation also failed. In this
683c5ad3233SUsama Arif 		 * case __hugetlb_vmemmap_optimize_folio() would free memory
684f13b83fdSJoao Martins 		 * allowing more vmemmap remaps to occur.
68591f386bfSMike Kravetz 		 */
68691f386bfSMike Kravetz 		if (ret == -ENOMEM && !list_empty(&vmemmap_pages)) {
687f13b83fdSJoao Martins 			flush_tlb_all();
68891f386bfSMike Kravetz 			free_vmemmap_page_list(&vmemmap_pages);
68991f386bfSMike Kravetz 			INIT_LIST_HEAD(&vmemmap_pages);
690c2a967f6SYu Zhao 			__hugetlb_vmemmap_optimize_folio(h, folio, &vmemmap_pages, flags);
69191f386bfSMike Kravetz 		}
69291f386bfSMike Kravetz 	}
69391f386bfSMike Kravetz 
694752fe17aSFrank van der Linden out:
695f13b83fdSJoao Martins 	flush_tlb_all();
69691f386bfSMike Kravetz 	free_vmemmap_page_list(&vmemmap_pages);
69779359d6dSMike Kravetz }
69879359d6dSMike Kravetz 
hugetlb_vmemmap_optimize_folios(struct hstate * h,struct list_head * folio_list)699752fe17aSFrank van der Linden void hugetlb_vmemmap_optimize_folios(struct hstate *h, struct list_head *folio_list)
700752fe17aSFrank van der Linden {
701752fe17aSFrank van der Linden 	__hugetlb_vmemmap_optimize_folios(h, folio_list, false);
702752fe17aSFrank van der Linden }
703752fe17aSFrank van der Linden 
hugetlb_vmemmap_optimize_bootmem_folios(struct hstate * h,struct list_head * folio_list)704752fe17aSFrank van der Linden void hugetlb_vmemmap_optimize_bootmem_folios(struct hstate *h, struct list_head *folio_list)
705752fe17aSFrank van der Linden {
706752fe17aSFrank van der Linden 	__hugetlb_vmemmap_optimize_folios(h, folio_list, true);
707752fe17aSFrank van der Linden }
708752fe17aSFrank van der Linden 
709b1222550SFrank van der Linden #ifdef CONFIG_SPARSEMEM_VMEMMAP_PREINIT
710b1222550SFrank van der Linden 
711b1222550SFrank van der Linden /* Return true of a bootmem allocated HugeTLB page should be pre-HVO-ed */
vmemmap_should_optimize_bootmem_page(struct huge_bootmem_page * m)712b1222550SFrank van der Linden static bool vmemmap_should_optimize_bootmem_page(struct huge_bootmem_page *m)
713b1222550SFrank van der Linden {
714b1222550SFrank van der Linden 	unsigned long section_size, psize, pmd_vmemmap_size;
715b1222550SFrank van der Linden 	phys_addr_t paddr;
716b1222550SFrank van der Linden 
717b1222550SFrank van der Linden 	if (!READ_ONCE(vmemmap_optimize_enabled))
718b1222550SFrank van der Linden 		return false;
719b1222550SFrank van der Linden 
720b1222550SFrank van der Linden 	if (!hugetlb_vmemmap_optimizable(m->hstate))
721b1222550SFrank van der Linden 		return false;
722b1222550SFrank van der Linden 
723b1222550SFrank van der Linden 	psize = huge_page_size(m->hstate);
724b1222550SFrank van der Linden 	paddr = virt_to_phys(m);
725b1222550SFrank van der Linden 
726b1222550SFrank van der Linden 	/*
727b1222550SFrank van der Linden 	 * Pre-HVO only works if the bootmem huge page
728b1222550SFrank van der Linden 	 * is aligned to the section size.
729b1222550SFrank van der Linden 	 */
730b1222550SFrank van der Linden 	section_size = (1UL << PA_SECTION_SHIFT);
731b1222550SFrank van der Linden 	if (!IS_ALIGNED(paddr, section_size) ||
732b1222550SFrank van der Linden 	    !IS_ALIGNED(psize, section_size))
733b1222550SFrank van der Linden 		return false;
734b1222550SFrank van der Linden 
735b1222550SFrank van der Linden 	/*
736b1222550SFrank van der Linden 	 * The pre-HVO code does not deal with splitting PMDS,
737b1222550SFrank van der Linden 	 * so the bootmem page must be aligned to the number
738b1222550SFrank van der Linden 	 * of base pages that can be mapped with one vmemmap PMD.
739b1222550SFrank van der Linden 	 */
740b1222550SFrank van der Linden 	pmd_vmemmap_size = (PMD_SIZE / (sizeof(struct page))) << PAGE_SHIFT;
741b1222550SFrank van der Linden 	if (!IS_ALIGNED(paddr, pmd_vmemmap_size) ||
742b1222550SFrank van der Linden 	    !IS_ALIGNED(psize, pmd_vmemmap_size))
743b1222550SFrank van der Linden 		return false;
744b1222550SFrank van der Linden 
745b1222550SFrank van der Linden 	return true;
746b1222550SFrank van der Linden }
747b1222550SFrank van der Linden 
748b1222550SFrank van der Linden /*
749b1222550SFrank van der Linden  * Initialize memmap section for a gigantic page, HVO-style.
750b1222550SFrank van der Linden  */
hugetlb_vmemmap_init_early(int nid)751b1222550SFrank van der Linden void __init hugetlb_vmemmap_init_early(int nid)
752b1222550SFrank van der Linden {
753b1222550SFrank van der Linden 	unsigned long psize, paddr, section_size;
754b1222550SFrank van der Linden 	unsigned long ns, i, pnum, pfn, nr_pages;
755b1222550SFrank van der Linden 	struct huge_bootmem_page *m = NULL;
756b1222550SFrank van der Linden 	void *map;
757b1222550SFrank van der Linden 
758b1222550SFrank van der Linden 	if (!READ_ONCE(vmemmap_optimize_enabled))
759b1222550SFrank van der Linden 		return;
760b1222550SFrank van der Linden 
761b1222550SFrank van der Linden 	section_size = (1UL << PA_SECTION_SHIFT);
762b1222550SFrank van der Linden 
763b1222550SFrank van der Linden 	list_for_each_entry(m, &huge_boot_pages[nid], list) {
764b1222550SFrank van der Linden 		if (!vmemmap_should_optimize_bootmem_page(m))
765b1222550SFrank van der Linden 			continue;
766b1222550SFrank van der Linden 
767b1222550SFrank van der Linden 		nr_pages = pages_per_huge_page(m->hstate);
768b1222550SFrank van der Linden 		psize = nr_pages << PAGE_SHIFT;
769b1222550SFrank van der Linden 		paddr = virt_to_phys(m);
770b1222550SFrank van der Linden 		pfn = PHYS_PFN(paddr);
771b1222550SFrank van der Linden 		map = pfn_to_page(pfn);
772b1222550SFrank van der Linden 
773b1222550SFrank van der Linden 		pnum = pfn_to_section_nr(pfn);
774b1222550SFrank van der Linden 		ns = psize / section_size;
775b1222550SFrank van der Linden 
776b1222550SFrank van der Linden 		for (i = 0; i < ns; i++) {
777b1222550SFrank van der Linden 			sparse_init_early_section(nid, map, pnum,
778b1222550SFrank van der Linden 					SECTION_IS_VMEMMAP_PREINIT);
779b1222550SFrank van der Linden 			map += section_map_size();
780b1222550SFrank van der Linden 			pnum++;
781b1222550SFrank van der Linden 		}
782b1222550SFrank van der Linden 
783b1222550SFrank van der Linden 		m->flags |= HUGE_BOOTMEM_HVO;
784b1222550SFrank van der Linden 	}
785b1222550SFrank van der Linden }
786b1222550SFrank van der Linden 
pfn_to_zone(unsigned nid,unsigned long pfn)787622026e8SKiryl Shutsemau static struct zone *pfn_to_zone(unsigned nid, unsigned long pfn)
788622026e8SKiryl Shutsemau {
789622026e8SKiryl Shutsemau 	struct zone *zone;
790622026e8SKiryl Shutsemau 	enum zone_type zone_type;
791622026e8SKiryl Shutsemau 
792622026e8SKiryl Shutsemau 	for (zone_type = 0; zone_type < MAX_NR_ZONES; zone_type++) {
793622026e8SKiryl Shutsemau 		zone = &NODE_DATA(nid)->node_zones[zone_type];
794622026e8SKiryl Shutsemau 		if (zone_spans_pfn(zone, pfn))
795622026e8SKiryl Shutsemau 			return zone;
796622026e8SKiryl Shutsemau 	}
797622026e8SKiryl Shutsemau 
798622026e8SKiryl Shutsemau 	return NULL;
799622026e8SKiryl Shutsemau }
800622026e8SKiryl Shutsemau 
hugetlb_vmemmap_init_late(int nid)801b1222550SFrank van der Linden void __init hugetlb_vmemmap_init_late(int nid)
802b1222550SFrank van der Linden {
803b1222550SFrank van der Linden 	struct huge_bootmem_page *m, *tm;
804b1222550SFrank van der Linden 	unsigned long phys, nr_pages, start, end;
805b1222550SFrank van der Linden 	unsigned long pfn, nr_mmap;
806622026e8SKiryl Shutsemau 	struct zone *zone = NULL;
807b1222550SFrank van der Linden 	struct hstate *h;
808b1222550SFrank van der Linden 	void *map;
809b1222550SFrank van der Linden 
810b1222550SFrank van der Linden 	if (!READ_ONCE(vmemmap_optimize_enabled))
811b1222550SFrank van der Linden 		return;
812b1222550SFrank van der Linden 
813b1222550SFrank van der Linden 	list_for_each_entry_safe(m, tm, &huge_boot_pages[nid], list) {
814b1222550SFrank van der Linden 		if (!(m->flags & HUGE_BOOTMEM_HVO))
815b1222550SFrank van der Linden 			continue;
816b1222550SFrank van der Linden 
817b1222550SFrank van der Linden 		phys = virt_to_phys(m);
818b1222550SFrank van der Linden 		h = m->hstate;
819b1222550SFrank van der Linden 		pfn = PHYS_PFN(phys);
820b1222550SFrank van der Linden 		nr_pages = pages_per_huge_page(h);
821209e6d9eSKiryl Shutsemau (Meta) 		map = pfn_to_page(pfn);
822209e6d9eSKiryl Shutsemau (Meta) 		start = (unsigned long)map;
823209e6d9eSKiryl Shutsemau (Meta) 		end = start + nr_pages * sizeof(struct page);
824b1222550SFrank van der Linden 
825b1222550SFrank van der Linden 		if (!hugetlb_bootmem_page_zones_valid(nid, m)) {
826b1222550SFrank van der Linden 			/*
827b1222550SFrank van der Linden 			 * Oops, the hugetlb page spans multiple zones.
828209e6d9eSKiryl Shutsemau (Meta) 			 * Remove it from the list, and populate it normally.
829b1222550SFrank van der Linden 			 */
830b1222550SFrank van der Linden 			list_del(&m->list);
831b1222550SFrank van der Linden 
832209e6d9eSKiryl Shutsemau (Meta) 			vmemmap_populate(start, end, nid, NULL);
833209e6d9eSKiryl Shutsemau (Meta) 			nr_mmap = end - start;
834b1222550SFrank van der Linden 			memmap_boot_pages_add(DIV_ROUND_UP(nr_mmap, PAGE_SIZE));
835b1222550SFrank van der Linden 
836b1222550SFrank van der Linden 			memblock_phys_free(phys, huge_page_size(h));
837b1222550SFrank van der Linden 			continue;
838209e6d9eSKiryl Shutsemau (Meta) 		}
839209e6d9eSKiryl Shutsemau (Meta) 
840622026e8SKiryl Shutsemau 		if (!zone || !zone_spans_pfn(zone, pfn))
841622026e8SKiryl Shutsemau 			zone = pfn_to_zone(nid, pfn);
842622026e8SKiryl Shutsemau 		if (WARN_ON_ONCE(!zone))
843622026e8SKiryl Shutsemau 			continue;
844622026e8SKiryl Shutsemau 
845622026e8SKiryl Shutsemau 		if (vmemmap_populate_hvo(start, end, huge_page_order(h), zone,
846209e6d9eSKiryl Shutsemau (Meta) 					 HUGETLB_VMEMMAP_RESERVE_SIZE) < 0) {
847209e6d9eSKiryl Shutsemau (Meta) 			/* Fallback if HVO population fails */
848209e6d9eSKiryl Shutsemau (Meta) 			vmemmap_populate(start, end, nid, NULL);
849209e6d9eSKiryl Shutsemau (Meta) 			nr_mmap = end - start;
850209e6d9eSKiryl Shutsemau (Meta) 		} else {
851b1222550SFrank van der Linden 			m->flags |= HUGE_BOOTMEM_ZONES_VALID;
852209e6d9eSKiryl Shutsemau (Meta) 			nr_mmap = HUGETLB_VMEMMAP_RESERVE_SIZE;
853209e6d9eSKiryl Shutsemau (Meta) 		}
854209e6d9eSKiryl Shutsemau (Meta) 
855209e6d9eSKiryl Shutsemau (Meta) 		memmap_boot_pages_add(DIV_ROUND_UP(nr_mmap, PAGE_SIZE));
856b1222550SFrank van der Linden 	}
857b1222550SFrank van der Linden }
858b1222550SFrank van der Linden #endif
859b1222550SFrank van der Linden 
8601751f872SJoel Granados static const struct ctl_table hugetlb_vmemmap_sysctls[] = {
86178f39084SMuchun Song 	{
86278f39084SMuchun Song 		.procname	= "hugetlb_optimize_vmemmap",
863cf5472e5SMuchun Song 		.data		= &vmemmap_optimize_enabled,
864f1aa2eb5SOndrej Mosnacek 		.maxlen		= sizeof(vmemmap_optimize_enabled),
86578f39084SMuchun Song 		.mode		= 0644,
866cf5472e5SMuchun Song 		.proc_handler	= proc_dobool,
86778f39084SMuchun Song 	},
86878f39084SMuchun Song };
86978f39084SMuchun Song 
hugetlb_vmemmap_init(void)8706213834cSMuchun Song static int __init hugetlb_vmemmap_init(void)
87178f39084SMuchun Song {
87212318566SMuchun Song 	const struct hstate *h;
873622026e8SKiryl Shutsemau 	struct zone *zone;
87412318566SMuchun Song 
8756213834cSMuchun Song 	/* HUGETLB_VMEMMAP_RESERVE_SIZE should cover all used struct pages */
876fde1c4ecSUsama Arif 	BUILD_BUG_ON(__NR_USED_SUBPAGE > HUGETLB_VMEMMAP_RESERVE_PAGES);
87778f39084SMuchun Song 
878622026e8SKiryl Shutsemau 	for_each_zone(zone) {
879622026e8SKiryl Shutsemau 		for (int i = 0; i < NR_VMEMMAP_TAILS; i++) {
880622026e8SKiryl Shutsemau 			struct page *tail, *p;
881622026e8SKiryl Shutsemau 			unsigned int order;
882622026e8SKiryl Shutsemau 
883622026e8SKiryl Shutsemau 			tail = zone->vmemmap_tails[i];
884622026e8SKiryl Shutsemau 			if (!tail)
885622026e8SKiryl Shutsemau 				continue;
886622026e8SKiryl Shutsemau 
887622026e8SKiryl Shutsemau 			order = i + VMEMMAP_TAIL_MIN_ORDER;
888622026e8SKiryl Shutsemau 			p = page_to_virt(tail);
889622026e8SKiryl Shutsemau 			for (int j = 0; j < PAGE_SIZE / sizeof(struct page); j++)
890622026e8SKiryl Shutsemau 				init_compound_tail(p + j, NULL, order, zone);
891622026e8SKiryl Shutsemau 		}
892622026e8SKiryl Shutsemau 	}
893622026e8SKiryl Shutsemau 
8946213834cSMuchun Song 	for_each_hstate(h) {
8956213834cSMuchun Song 		if (hugetlb_vmemmap_optimizable(h)) {
8966213834cSMuchun Song 			register_sysctl_init("vm", hugetlb_vmemmap_sysctls);
8976213834cSMuchun Song 			break;
8986213834cSMuchun Song 		}
8996213834cSMuchun Song 	}
90078f39084SMuchun Song 	return 0;
90178f39084SMuchun Song }
9026213834cSMuchun Song late_initcall(hugetlb_vmemmap_init);
903