1 // SPDX-License-Identifier: GPL-2.0
3 * Copyright (C) 2002 Richard Henderson
4 * Copyright (C) 2001 Rusty Russell, 2002, 2010 Rusty Russell IBM.
6 * Copyright (C) 2024 Mike Rapoport IBM.
10 #include <linux/vmalloc.h>
11 #include <linux/execmem.h>
12 #include <linux/moduleloader.h>
14 static struct execmem_info *execmem_info __ro_after_init;
15 static struct execmem_info default_execmem_info __ro_after_init;
17 static void *__execmem_alloc(struct execmem_range *range, size_t size)
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;
22 unsigned long start = range->start;
23 unsigned long end = range->end;
24 unsigned int align = range->alignment;
25 pgprot_t pgprot = range->pgprot;
29 vm_flags |= VM_DEFER_KMEMLEAK;
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));
43 pr_warn_ratelimited("execmem: unable to allocate memory\n");
47 if (kasan && (kasan_alloc_module_shadow(p, size, GFP_KERNEL) < 0)) {
52 return kasan_reset_tag(p);
55 void *execmem_alloc(enum execmem_type type, size_t size)
57 struct execmem_range *range = &execmem_info->ranges[type];
59 return __execmem_alloc(range, size);
62 void execmem_free(void *ptr)
65 * This memory may be RO, and freeing RO memory in an interrupt is not
66 * supported by vmalloc.
68 WARN_ON(in_interrupt());
72 static bool execmem_validate(struct execmem_info *info)
74 struct execmem_range *r = &info->ranges[EXECMEM_DEFAULT];
76 if (!r->alignment || !r->start || !r->end || !pgprot_val(r->pgprot)) {
77 pr_crit("Invalid parameters for execmem allocator, module loading will fail");
84 static void execmem_init_missing(struct execmem_info *info)
86 struct execmem_range *default_range = &info->ranges[EXECMEM_DEFAULT];
88 for (int i = EXECMEM_DEFAULT + 1; i < EXECMEM_TYPE_MAX; i++) {
89 struct execmem_range *r = &info->ranges[i];
92 if (i == EXECMEM_MODULE_DATA)
93 r->pgprot = PAGE_KERNEL;
95 r->pgprot = default_range->pgprot;
96 r->alignment = default_range->alignment;
97 r->start = default_range->start;
98 r->end = default_range->end;
99 r->flags = default_range->flags;
100 r->fallback_start = default_range->fallback_start;
101 r->fallback_end = default_range->fallback_end;
106 struct execmem_info * __weak execmem_arch_setup(void)
111 static void __init __execmem_init(void)
113 struct execmem_info *info = execmem_arch_setup();
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;
123 if (!execmem_validate(info))
126 execmem_init_missing(info);
131 #ifdef CONFIG_ARCH_WANTS_EXECMEM_LATE
132 static int __init execmem_late_init(void)
137 core_initcall(execmem_late_init);
139 void __init execmem_init(void)