hc
2023-12-09 b22da3d8526a935aa31e086e63f60ff3246cb61c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
// SPDX-License-Identifier: GPL-2.0
/*
 * Copyright (C) 2020 - Google LLC
 * Author: Quentin Perret <qperret@google.com>
 */
 
#include <linux/kvm_host.h>
#include <linux/memblock.h>
#include <linux/sort.h>
 
#include <asm/kvm_host.h>
 
#include <nvhe/memory.h>
#include <nvhe/mm.h>
 
static struct memblock_region *hyp_memory = kvm_nvhe_sym(hyp_memory);
static unsigned int *hyp_memblock_nr_ptr = &kvm_nvhe_sym(hyp_memblock_nr);
 
phys_addr_t hyp_mem_base;
phys_addr_t hyp_mem_size;
 
static int cmp_hyp_memblock(const void *p1, const void *p2)
{
   const struct memblock_region *r1 = p1;
   const struct memblock_region *r2 = p2;
 
   return r1->base < r2->base ? -1 : (r1->base > r2->base);
}
 
static void __init sort_memblock_regions(void)
{
   sort(hyp_memory,
        *hyp_memblock_nr_ptr,
        sizeof(struct memblock_region),
        cmp_hyp_memblock,
        NULL);
}
 
static int __init register_memblock_regions(void)
{
   struct memblock_region *reg;
 
   for_each_mem_region(reg) {
       if (*hyp_memblock_nr_ptr >= HYP_MEMBLOCK_REGIONS)
           return -ENOMEM;
 
       hyp_memory[*hyp_memblock_nr_ptr] = *reg;
       (*hyp_memblock_nr_ptr)++;
   }
   sort_memblock_regions();
 
   return 0;
}
 
void __init kvm_hyp_reserve(void)
{
   u64 nr_pages, prev, hyp_mem_pages = 0;
   int ret;
 
   if (!is_hyp_mode_available() || is_kernel_in_hyp_mode())
       return;
 
   if (kvm_get_mode() != KVM_MODE_PROTECTED)
       return;
 
   ret = register_memblock_regions();
   if (ret) {
       *hyp_memblock_nr_ptr = 0;
       kvm_err("Failed to register hyp memblocks: %d\n", ret);
       return;
   }
 
   hyp_mem_pages += hyp_s1_pgtable_pages();
   hyp_mem_pages += host_s2_mem_pgtable_pages();
   hyp_mem_pages += host_s2_dev_pgtable_pages();
 
   /*
    * The hyp_vmemmap needs to be backed by pages, but these pages
    * themselves need to be present in the vmemmap, so compute the number
    * of pages needed by looking for a fixed point.
    */
   nr_pages = 0;
   do {
       prev = nr_pages;
       nr_pages = hyp_mem_pages + prev;
       nr_pages = DIV_ROUND_UP(nr_pages * sizeof(struct hyp_page), PAGE_SIZE);
       nr_pages += __hyp_pgtable_max_pages(nr_pages);
   } while (nr_pages != prev);
   hyp_mem_pages += nr_pages;
 
   /*
    * Try to allocate a PMD-aligned region to reduce TLB pressure once
    * this is unmapped from the host stage-2, and fallback to PAGE_SIZE.
    */
   hyp_mem_size = hyp_mem_pages << PAGE_SHIFT;
   hyp_mem_base = memblock_find_in_range(0, memblock_end_of_DRAM(),
                         ALIGN(hyp_mem_size, PMD_SIZE),
                         PMD_SIZE);
   if (!hyp_mem_base)
       hyp_mem_base = memblock_find_in_range(0, memblock_end_of_DRAM(),
                             hyp_mem_size, PAGE_SIZE);
   else
       hyp_mem_size = ALIGN(hyp_mem_size, PMD_SIZE);
 
   if (!hyp_mem_base) {
       kvm_err("Failed to reserve hyp memory\n");
       return;
   }
   memblock_reserve(hyp_mem_base, hyp_mem_size);
 
   kvm_info("Reserved %lld MiB at 0x%llx\n", hyp_mem_size >> 20,
        hyp_mem_base);
}