]>
Commit | Line | Data |
---|---|---|
12af2b83 MRI |
1 | // SPDX-License-Identifier: GPL-2.0 |
2 | /* | |
3 | * Copyright (C) 2002 Richard Henderson | |
4 | * Copyright (C) 2001 Rusty Russell, 2002, 2010 Rusty Russell IBM. | |
5 | * Copyright (C) 2023 Luis Chamberlain <[email protected]> | |
6 | * Copyright (C) 2024 Mike Rapoport IBM. | |
7 | */ | |
8 | ||
9 | #include <linux/mm.h> | |
10 | #include <linux/vmalloc.h> | |
11 | #include <linux/execmem.h> | |
12 | #include <linux/moduleloader.h> | |
13 | ||
f6bec26c | 14 | static struct execmem_info *execmem_info __ro_after_init; |
223b5e57 | 15 | static struct execmem_info default_execmem_info __ro_after_init; |
f6bec26c MRI |
16 | |
17 | static void *__execmem_alloc(struct execmem_range *range, size_t size) | |
12af2b83 | 18 | { |
223b5e57 MRI |
19 | bool kasan = range->flags & EXECMEM_KASAN_SHADOW; |
20 | unsigned long vm_flags = VM_FLUSH_RESET_PERMS; | |
21 | gfp_t gfp_flags = GFP_KERNEL | __GFP_NOWARN; | |
f6bec26c MRI |
22 | unsigned long start = range->start; |
23 | unsigned long end = range->end; | |
24 | unsigned int align = range->alignment; | |
25 | pgprot_t pgprot = range->pgprot; | |
223b5e57 MRI |
26 | void *p; |
27 | ||
28 | if (kasan) | |
29 | vm_flags |= VM_DEFER_KMEMLEAK; | |
30 | ||
31 | p = __vmalloc_node_range(size, align, start, end, gfp_flags, | |
32 | pgprot, vm_flags, NUMA_NO_NODE, | |
33 | __builtin_return_address(0)); | |
34 | if (!p && range->fallback_start) { | |
35 | start = range->fallback_start; | |
36 | end = range->fallback_end; | |
37 | p = __vmalloc_node_range(size, align, start, end, gfp_flags, | |
38 | pgprot, vm_flags, NUMA_NO_NODE, | |
39 | __builtin_return_address(0)); | |
40 | } | |
41 | ||
42 | if (!p) { | |
43 | pr_warn_ratelimited("execmem: unable to allocate memory\n"); | |
44 | return NULL; | |
45 | } | |
46 | ||
47 | if (kasan && (kasan_alloc_module_shadow(p, size, GFP_KERNEL) < 0)) { | |
48 | vfree(p); | |
49 | return NULL; | |
50 | } | |
f6bec26c | 51 | |
223b5e57 | 52 | return kasan_reset_tag(p); |
12af2b83 MRI |
53 | } |
54 | ||
55 | void *execmem_alloc(enum execmem_type type, size_t size) | |
56 | { | |
223b5e57 | 57 | struct execmem_range *range = &execmem_info->ranges[type]; |
f6bec26c MRI |
58 | |
59 | return __execmem_alloc(range, size); | |
12af2b83 MRI |
60 | } |
61 | ||
62 | void execmem_free(void *ptr) | |
63 | { | |
64 | /* | |
65 | * This memory may be RO, and freeing RO memory in an interrupt is not | |
66 | * supported by vmalloc. | |
67 | */ | |
68 | WARN_ON(in_interrupt()); | |
69 | vfree(ptr); | |
70 | } | |
f6bec26c MRI |
71 | |
72 | static bool execmem_validate(struct execmem_info *info) | |
73 | { | |
74 | struct execmem_range *r = &info->ranges[EXECMEM_DEFAULT]; | |
75 | ||
76 | if (!r->alignment || !r->start || !r->end || !pgprot_val(r->pgprot)) { | |
77 | pr_crit("Invalid parameters for execmem allocator, module loading will fail"); | |
78 | return false; | |
79 | } | |
80 | ||
81 | return true; | |
82 | } | |
83 | ||
84 | static void execmem_init_missing(struct execmem_info *info) | |
85 | { | |
86 | struct execmem_range *default_range = &info->ranges[EXECMEM_DEFAULT]; | |
87 | ||
88 | for (int i = EXECMEM_DEFAULT + 1; i < EXECMEM_TYPE_MAX; i++) { | |
89 | struct execmem_range *r = &info->ranges[i]; | |
90 | ||
91 | if (!r->start) { | |
223b5e57 MRI |
92 | if (i == EXECMEM_MODULE_DATA) |
93 | r->pgprot = PAGE_KERNEL; | |
94 | else | |
95 | r->pgprot = default_range->pgprot; | |
f6bec26c MRI |
96 | r->alignment = default_range->alignment; |
97 | r->start = default_range->start; | |
98 | r->end = default_range->end; | |
223b5e57 MRI |
99 | r->flags = default_range->flags; |
100 | r->fallback_start = default_range->fallback_start; | |
101 | r->fallback_end = default_range->fallback_end; | |
f6bec26c MRI |
102 | } |
103 | } | |
104 | } | |
105 | ||
106 | struct execmem_info * __weak execmem_arch_setup(void) | |
107 | { | |
108 | return NULL; | |
109 | } | |
110 | ||
223b5e57 | 111 | static void __init __execmem_init(void) |
f6bec26c MRI |
112 | { |
113 | struct execmem_info *info = execmem_arch_setup(); | |
114 | ||
223b5e57 MRI |
115 | if (!info) { |
116 | info = execmem_info = &default_execmem_info; | |
117 | info->ranges[EXECMEM_DEFAULT].start = VMALLOC_START; | |
118 | info->ranges[EXECMEM_DEFAULT].end = VMALLOC_END; | |
119 | info->ranges[EXECMEM_DEFAULT].pgprot = PAGE_KERNEL_EXEC; | |
120 | info->ranges[EXECMEM_DEFAULT].alignment = 1; | |
121 | } | |
122 | ||
123 | if (!execmem_validate(info)) | |
f6bec26c MRI |
124 | return; |
125 | ||
126 | execmem_init_missing(info); | |
127 | ||
128 | execmem_info = info; | |
129 | } | |
223b5e57 MRI |
130 | |
131 | #ifdef CONFIG_ARCH_WANTS_EXECMEM_LATE | |
132 | static int __init execmem_late_init(void) | |
133 | { | |
134 | __execmem_init(); | |
135 | return 0; | |
136 | } | |
137 | core_initcall(execmem_late_init); | |
138 | #else | |
139 | void __init execmem_init(void) | |
140 | { | |
141 | __execmem_init(); | |
142 | } | |
143 | #endif |