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