提交 8fc8f624 编写于 作者: L Liu Shixin 提交者: Zheng Zengkai

arm64: kfence: scale sample_interval to control re-enabling

hulk inclusion
category: feature
bugzilla: 187071, https://gitee.com/openeuler/kernel/issues/I5DLA7

--------------------------------

KFENCE requires linear map to be mapped at page granularity, this
must done in very early time. To save memory of page table, arm64
only map the pages in KFENCE pool itself at page granularity. Thus,
the kfence pool could not allocated by buddy system.

For the flexibility of KFENCE, scale sample_interval to control whether
support to enable kfence after system startup(re-enabling).
Once sample_interval is set to -1 on arm64, memory for kfence pool
will be allocated from early memory no matter KFENCE is enabled or
not.
Signed-off-by: NLiu Shixin <liushixin2@huawei.com>
Reviewed-by: NKefeng Wang <wangkefeng.wang@huawei.com>
Signed-off-by: NZheng Zengkai <zhengzengkai@huawei.com>
上级 6cdb25d2
...@@ -61,6 +61,17 @@ The total memory dedicated to the KFENCE memory pool can be computed as:: ...@@ -61,6 +61,17 @@ The total memory dedicated to the KFENCE memory pool can be computed as::
Using the default config, and assuming a page size of 4 KiB, results in Using the default config, and assuming a page size of 4 KiB, results in
dedicating 2 MiB to the KFENCE memory pool. dedicating 2 MiB to the KFENCE memory pool.
KFENCE allow re-enabling after system startup, but ifndef CONFIG_CONTIG_ALLOC
and KFENCE_NUM_OBJECTS exceeds MAX_ORDER, alloc KFENCE pool after system startup
is not supported.
For arm64, re-enabling KFENCE is kind of conflict with map the ages in KFENCE
pool itself at page granularity. For the flexibility, scale sample_interval to
control whether arm64 supported to enable kfence after system startup.
Once this is set to -1 in boot parameter, kfence_pool will be allocated from
early memory no matter kfence is enabled or not. Otherwise, re-enabling is not
supported on arm64.
Note: On architectures that support huge pages, KFENCE will ensure that the Note: On architectures that support huge pages, KFENCE will ensure that the
pool is using pages of size ``PAGE_SIZE``. This will result in additional page pool is using pages of size ``PAGE_SIZE``. This will result in additional page
tables being allocated. tables being allocated.
......
...@@ -49,6 +49,7 @@ ...@@ -49,6 +49,7 @@
static bool kfence_enabled __read_mostly; static bool kfence_enabled __read_mostly;
static bool disabled_by_warn __read_mostly; static bool disabled_by_warn __read_mostly;
static bool re_enabling __read_mostly;
unsigned long kfence_sample_interval __read_mostly = CONFIG_KFENCE_SAMPLE_INTERVAL; unsigned long kfence_sample_interval __read_mostly = CONFIG_KFENCE_SAMPLE_INTERVAL;
EXPORT_SYMBOL_GPL(kfence_sample_interval); /* Export for test modules. */ EXPORT_SYMBOL_GPL(kfence_sample_interval); /* Export for test modules. */
...@@ -61,16 +62,24 @@ EXPORT_SYMBOL_GPL(kfence_sample_interval); /* Export for test modules. */ ...@@ -61,16 +62,24 @@ EXPORT_SYMBOL_GPL(kfence_sample_interval); /* Export for test modules. */
static int kfence_enable_late(void); static int kfence_enable_late(void);
static int param_set_sample_interval(const char *val, const struct kernel_param *kp) static int param_set_sample_interval(const char *val, const struct kernel_param *kp)
{ {
unsigned long num; long num;
int ret = kstrtoul(val, 0, &num); int ret = kstrtol(val, 0, &num);
if (ret < 0) if (ret < 0)
return ret; return ret;
if (num < -1)
return -ERANGE;
/*
* For architecture that don't require early allocation, always support
* re-enabling. So only need to set num to 0 if num < 0.
*/
num = max_t(long, 0, num);
if (!num) /* Using 0 to indicate KFENCE is disabled. */ if (!num) /* Using 0 to indicate KFENCE is disabled. */
WRITE_ONCE(kfence_enabled, false); WRITE_ONCE(kfence_enabled, false);
*((unsigned long *)kp->arg) = num; *((unsigned long *)kp->arg) = (unsigned long)num;
if (num && !READ_ONCE(kfence_enabled) && system_state != SYSTEM_BOOTING) if (num && !READ_ONCE(kfence_enabled) && system_state != SYSTEM_BOOTING)
return disabled_by_warn ? -EINVAL : kfence_enable_late(); return disabled_by_warn ? -EINVAL : kfence_enable_late();
...@@ -94,11 +103,22 @@ module_param_cb(sample_interval, &sample_interval_param_ops, &kfence_sample_inte ...@@ -94,11 +103,22 @@ module_param_cb(sample_interval, &sample_interval_param_ops, &kfence_sample_inte
#ifdef CONFIG_ARM64 #ifdef CONFIG_ARM64
static int __init parse_sample_interval(char *str) static int __init parse_sample_interval(char *str)
{ {
unsigned long num; long num;
if (kstrtoul(str, 0, &num) < 0) if (kstrtol(str, 0, &num) < 0)
return 0;
if (num < -1)
return 0; return 0;
kfence_sample_interval = num;
/* Using -1 to indicate re-enabling is supported */
if (num == -1) {
re_enabling = true;
pr_err("re-enabling is supported\n");
}
num = max_t(long, 0, num);
kfence_sample_interval = (unsigned long)num;
return 0; return 0;
} }
early_param("kfence.sample_interval", parse_sample_interval); early_param("kfence.sample_interval", parse_sample_interval);
...@@ -948,7 +968,7 @@ static DECLARE_DELAYED_WORK(kfence_timer, toggle_allocation_gate); ...@@ -948,7 +968,7 @@ static DECLARE_DELAYED_WORK(kfence_timer, toggle_allocation_gate);
/* === Public interface ===================================================== */ /* === Public interface ===================================================== */
void __init kfence_early_alloc_pool(void) void __init kfence_early_alloc_pool(void)
{ {
if (!kfence_sample_interval) if (!kfence_sample_interval && !re_enabling)
return; return;
__kfence_pool = memblock_alloc_raw(KFENCE_POOL_SIZE, PAGE_SIZE); __kfence_pool = memblock_alloc_raw(KFENCE_POOL_SIZE, PAGE_SIZE);
...@@ -961,7 +981,7 @@ void __init kfence_early_alloc_pool(void) ...@@ -961,7 +981,7 @@ void __init kfence_early_alloc_pool(void)
void __init kfence_alloc_pool(void) void __init kfence_alloc_pool(void)
{ {
if (!kfence_sample_interval) if (!kfence_sample_interval && !__kfence_pool)
return; return;
if (kfence_dynamic_init()) { if (kfence_dynamic_init()) {
...@@ -996,7 +1016,7 @@ void __init kfence_init(void) ...@@ -996,7 +1016,7 @@ void __init kfence_init(void)
stack_hash_seed = (u32)random_get_entropy(); stack_hash_seed = (u32)random_get_entropy();
/* Setting kfence_sample_interval to 0 on boot disables KFENCE. */ /* Setting kfence_sample_interval to 0 on boot disables KFENCE. */
if (!kfence_sample_interval) if (!kfence_sample_interval && !__kfence_pool)
return; return;
if (!kfence_init_pool_early()) { if (!kfence_init_pool_early()) {
...@@ -1005,6 +1025,9 @@ void __init kfence_init(void) ...@@ -1005,6 +1025,9 @@ void __init kfence_init(void)
} }
kfence_init_enable(); kfence_init_enable();
if (!kfence_sample_interval)
WRITE_ONCE(kfence_enabled, false);
} }
static int kfence_init_late(void) static int kfence_init_late(void)
...@@ -1015,6 +1038,14 @@ static int kfence_init_late(void) ...@@ -1015,6 +1038,14 @@ static int kfence_init_late(void)
struct page *pages; struct page *pages;
#endif #endif
/*
* For kfence re_enabling on ARM64, kfence_pool should be allocated
* at startup instead of here. So just return -EINVAL here which means
* re_enabling is not supported.
*/
if (IS_ENABLED(CONFIG_ARM64))
return -EINVAL;
if (kfence_dynamic_init()) if (kfence_dynamic_init())
return -ENOMEM; return -ENOMEM;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册