diff --git a/arch/arm64/include/mte.h b/arch/arm64/include/mte.h new file mode 100644 index 0000000000000..e7dd167d8a539 --- /dev/null +++ b/arch/arm64/include/mte.h @@ -0,0 +1,58 @@ +/**************************************************************************** + * arch/arm64/include/mte.h + * + * SPDX-License-Identifier: Apache-2.0 + * + * Licensed to the Apache Software Foundation (ASF) under one or more + * contributor license agreements. See the NOTICE file distributed with + * this work for additional information regarding copyright ownership. The + * ASF licenses this file to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance with the + * License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + * + ****************************************************************************/ + +#ifndef ___ARCH_ARM64_SRC_COMMON_ARM64_MTE_H +#define ___ARCH_ARM64_SRC_COMMON_ARM64_MTE_H + +/**************************************************************************** + * Public Function Prototypes + ****************************************************************************/ + +/* Initialize MTE settings and enable memory tagging */ + +void arm64_mte_init(void); + +/* Enable MTE by setting the TCF1 bit in SCTLR_EL1 */ + +void arm64_mte_enable(void); + +/* Disable MTE by clearing the TCF1 bit in SCTLR_EL1 */ + +void arm64_mte_disable(void); + +/* Set memory tags for a given memory range */ + +void arm64_mte_set_tag(const void *addr, size_t size); + +/* Get a random label based on the address through the mte register */ + +uint8_t arm64_mte_get_random_tag(const void *addr); + +/* Get the address without label */ + +FAR void *arm64_mte_get_untagged_addr(const void *addr); + +/* Get the address with label */ + +FAR void *arm64_mte_get_tagged_addr(const void *addr, uint8_t tag); + +#endif /* ___ARCH_ARM64_SRC_COMMON_ARM64_MTE_H */ diff --git a/arch/arm64/src/common/arm64_arch.h b/arch/arm64/src/common/arm64_arch.h index 3a32703cadb32..349104007c571 100644 --- a/arch/arm64/src/common/arm64_arch.h +++ b/arch/arm64/src/common/arm64_arch.h @@ -506,12 +506,6 @@ uint64_t arm64_get_mpid(int cpu); int arm64_get_cpuid(uint64_t mpid); #endif -#ifdef CONFIG_ARM64_MTE -void arm64_enable_mte(void); -#else -#define arm64_enable_mte() -#endif - #endif /* __ASSEMBLY__ */ #endif /* ___ARCH_ARM64_SRC_COMMON_ARM64_ARCH_H */ diff --git a/arch/arm64/src/common/arm64_mmu.h b/arch/arm64/src/common/arm64_mmu.h index a0677568c530d..1ff165a49078b 100644 --- a/arch/arm64/src/common/arm64_mmu.h +++ b/arch/arm64/src/common/arm64_mmu.h @@ -170,8 +170,10 @@ * in the address range [59:55] = 0b00000 are unchecked accesses. */ -#define TCR_TCMA0 (1ULL << 57) -#define TCR_TCMA1 (1ULL << 58) +#define TCR_TCMA1 BIT(56) +#define TCR_TCMA0 BIT(57) +#define TCR_MTX0_SHIFT BIT(60) +#define TCR_MTX1_SHIFT BIT(61) #define TCR_PS_BITS_4GB 0x0ULL #define TCR_PS_BITS_64GB 0x1ULL diff --git a/arch/arm64/src/common/arm64_mte.c b/arch/arm64/src/common/arm64_mte.c index 6a2230f90f085..2701e6c3a1e01 100644 --- a/arch/arm64/src/common/arm64_mte.c +++ b/arch/arm64/src/common/arm64_mte.c @@ -29,6 +29,7 @@ #include #include "arm64_arch.h" +#include "arm64_mmu.h" /**************************************************************************** * Pre-processor Definitions @@ -36,11 +37,17 @@ #define GCR_EL1_VAL 0x10001 +/* The alignment length of the MTE must be a multiple of sixteen */ + +#define MTE_MM_AILGN 16 + +#define MTE_TAG_SHIFT 56 + /**************************************************************************** * Private Functions ****************************************************************************/ -static int arm64_mte_is_support(void) +static int mte_is_support(void) { int supported; __asm__ volatile ( @@ -53,15 +60,86 @@ static int arm64_mte_is_support(void) return supported != 0; } +static void mte_set_tcf(bool enable) +{ + uint64_t val = read_sysreg(sctlr_el1); + + if (enable) + { + val |= SCTLR_TCF1_BIT; + } + else + { + val &= ~SCTLR_TCF1_BIT; + } + + write_sysreg(val, sctlr_el1); +} + +static inline uint8_t mte_get_ptr_tag(const void *ptr) +{ + return 0xf0 | (uint8_t)(((uint64_t)(ptr)) >> MTE_TAG_SHIFT); +} + /**************************************************************************** * Public Functions ****************************************************************************/ -void arm64_enable_mte(void) +uint8_t arm64_mte_get_random_tag(const void *addr) +{ + asm("irg %0, %0" : "=r" (addr)); + + return mte_get_ptr_tag(addr); +} + +FAR void *arm64_mte_get_untagged_addr(const void *addr) +{ + return (FAR void *) + (((uint64_t)(addr)) & ~((uint64_t)0xff << MTE_TAG_SHIFT)); +} + +FAR void *arm64_mte_get_tagged_addr(const void *addr, uint8_t tag) +{ + return (FAR void *) + (((uint64_t)(addr)) | ((uint64_t)tag << MTE_TAG_SHIFT)); +} + +/* Disable MTE by clearing the TCF1 bit in SCTLR_EL1 */ + +void arm64_mte_disable(void) +{ + mte_set_tcf(false); +} + +/* Enable MTE by setting the TCF1 bit in SCTLR_EL1 */ + +void arm64_mte_enable(void) +{ + mte_set_tcf(true); +} + +/* Set memory tags for a given memory range */ + +void arm64_mte_set_tag(const void *addr, size_t size) +{ + size_t i; + + DEBUGASSERT((uintptr_t)addr % MTE_MM_AILGN == 0); + DEBUGASSERT(size % MTE_MM_AILGN == 0); + + for (i = 0; i < size; i += MTE_MM_AILGN) + { + asm("stg %0, [%0]" : : "r"(addr + i)); + } +} + +/* Initialize MTE settings and enable memory tagging */ + +void arm64_mte_init(void) { uint64_t val; - if (!arm64_mte_is_support()) + if (!mte_is_support()) { return; } @@ -78,6 +156,14 @@ void arm64_enable_mte(void) assert(!(read_sysreg(ttbr0_el1) & TTBR_CNP_BIT)); assert(!(read_sysreg(ttbr1_el1) & TTBR_CNP_BIT)); + /* Controls the default value for skipping high bytes */ + + val = read_sysreg(tcr_el1); + val |= TCR_TCMA1; + write_sysreg(val, tcr_el1); + + /* Enable the MTE function */ + val = read_sysreg(sctlr_el1); val |= SCTLR_ATA_BIT | SCTLR_TCF1_BIT; write_sysreg(val, sctlr_el1); diff --git a/arch/arm64/src/qemu/qemu_boot.c b/arch/arm64/src/qemu/qemu_boot.c index 4686c60b24c0d..ae7523e74bbe5 100644 --- a/arch/arm64/src/qemu/qemu_boot.c +++ b/arch/arm64/src/qemu/qemu_boot.c @@ -36,6 +36,7 @@ # include #endif +#include #include #ifdef CONFIG_SMP @@ -161,7 +162,9 @@ void arm64_chip_boot(void) arm64_mmu_init(true); - arm64_enable_mte(); +#ifdef CONFIG_ARM64_MTE + arm64_mte_init(); +#endif #ifdef CONFIG_DEVICE_TREE fdt_register((const char *)0x40000000); diff --git a/include/nuttx/mm/kasan.h b/include/nuttx/mm/kasan.h index a349eb9675bb7..9ba72d8dd105c 100644 --- a/include/nuttx/mm/kasan.h +++ b/include/nuttx/mm/kasan.h @@ -46,6 +46,8 @@ # define kasan_stop() # define kasan_debugpoint(t,a,s) 0 # define kasan_init_early() +# define kasan_hw_open() +# define kasan_hw_close() #else # define kasan_init_early() kasan_stop() @@ -201,6 +203,23 @@ void kasan_stop(void); int kasan_debugpoint(int type, FAR void *addr, size_t size); +#ifndef CONFIG_MM_KASAN_MTE_TAGS +# define kasan_hw_open() +# define kasan_hw_close() +#else +/**************************************************************************** + * Name: kasan_hw_open + ****************************************************************************/ + +void kasan_hw_open(void); + +/**************************************************************************** + * Name: kasan_hw_open + ****************************************************************************/ + +void kasan_hw_close(void); +#endif + #undef EXTERN #ifdef __cplusplus } diff --git a/mm/Kconfig b/mm/Kconfig index 167f679c84ea4..cced34b7149be 100644 --- a/mm/Kconfig +++ b/mm/Kconfig @@ -348,10 +348,22 @@ config MM_KASAN_SW_TAGS ---help--- KAsan based on software tags +config MM_KASAN_MTE_TAGS + bool "KAsan MTE tags" + depends on ARM64_MTE + select MM_KASAN_NO_INSTRUMENT + ---help--- + KAsan based on hardware tags + endchoice +config MM_KASAN_NO_INSTRUMENT + bool + default n + config MM_KASAN_ALL bool "Enable KASan for the entire image" + depends on !MM_KASAN_NO_INSTRUMENT default y ---help--- This option activates address sanitizer for the entire image. @@ -366,6 +378,7 @@ config MM_KASAN_REGIONS config MM_KASAN_WATCHPOINT int "Kasan watchpoint maximum number" + depends on !MM_KASAN_NO_INSTRUMENT default 0 ---help--- The maximum number of watchpoints that can be set by KASan. diff --git a/mm/kasan/hook.c b/mm/kasan/hook.c index d4dc19c21295d..baeddbba95f3d 100644 --- a/mm/kasan/hook.c +++ b/mm/kasan/hook.c @@ -42,6 +42,8 @@ # include "generic.c" #elif defined(CONFIG_MM_KASAN_SW_TAGS) # include "sw_tags.c" +#elif defined(CONFIG_MM_KASAN_MTE_TAGS) +# include "mte_tags.c" #else # define kasan_is_poisoned(addr, size) false #endif @@ -123,6 +125,8 @@ static uint32_t g_region_init; * Private Functions ****************************************************************************/ +#ifndef CONFIG_MM_KASAN_MTE_TAGS + static void kasan_show_memory(FAR const uint8_t *addr, size_t size, size_t dumpsize) { @@ -252,6 +256,10 @@ static inline void kasan_check_report(FAR const void *addr, size_t size, # endif #endif } +#else +#define kasan_check_report(addr, size, is_write, return_address) +#define kasan_report(addr, size, is_write, return_address) +#endif /**************************************************************************** * Public Functions diff --git a/mm/kasan/mte_tags.c b/mm/kasan/mte_tags.c new file mode 100644 index 0000000000000..71882029467dd --- /dev/null +++ b/mm/kasan/mte_tags.c @@ -0,0 +1,95 @@ +/**************************************************************************** + * mm/kasan/mte_tags.c + * + * Licensed to the Apache Software Foundation (ASF) under one or more + * contributor license agreements. See the NOTICE file distributed with + * this work for additional information regarding copyright ownership. The + * ASF licenses this file to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance with the + * License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + * + ****************************************************************************/ + +/**************************************************************************** + * Included Files + ****************************************************************************/ + +#include +#include + +#include + +#include +#include +#include + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +static void *kasan_set_poison(FAR const void *addr, size_t size, uint8_t tag) +{ + FAR const void *tag_addr; + + /* Get random labels and the addresses after labeling */ + + tag_addr = arm64_mte_get_tagged_addr(addr, tag); + + /* Add MTE hardware label to memory block */ + + arm64_mte_set_tag(tag_addr, size); + + return (void *)tag_addr; +} + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +void kasan_hw_open(void) +{ + arm64_mte_enable(); +} + +void kasan_hw_close(void) +{ + arm64_mte_disable(); +} + +FAR void *kasan_reset_tag(FAR const void *addr) +{ + return arm64_mte_get_untagged_addr(addr); +} + +void kasan_poison(FAR const void *addr, size_t size) +{ + uint8_t tag = arm64_mte_get_random_tag(addr); + + kasan_set_poison(addr, size, tag); +} + +FAR void *kasan_unpoison(FAR const void *addr, size_t size) +{ + uint8_t tag = arm64_mte_get_random_tag(addr); + + return kasan_set_poison(addr, size, tag); +} + +void kasan_register(FAR void *addr, FAR size_t *size) +{ + uint8_t tag = arm64_mte_get_random_tag(addr); + + kasan_set_poison(addr, *size, tag); +} + +void kasan_unregister(FAR void *addr) +{ +} diff --git a/mm/mm_heap/mm_free.c b/mm/mm_heap/mm_free.c index 19a8a744a1b8c..e8bfa79bf829c 100644 --- a/mm/mm_heap/mm_free.c +++ b/mm/mm_heap/mm_free.c @@ -231,6 +231,7 @@ void mm_delayfree(FAR struct mm_heap_s *heap, FAR void *mem, bool delay) void mm_free(FAR struct mm_heap_s *heap, FAR void *mem) { + kasan_hw_close(); minfo("Freeing %p\n", mem); /* Protect against attempts to free a NULL reference */ @@ -253,4 +254,5 @@ void mm_free(FAR struct mm_heap_s *heap, FAR void *mem) #endif mm_delayfree(heap, mem, CONFIG_MM_FREE_DELAYCOUNT_MAX > 0); + kasan_hw_open(); } diff --git a/mm/mm_heap/mm_initialize.c b/mm/mm_heap/mm_initialize.c index 61b1b91a6c41b..f2f88081767be 100644 --- a/mm/mm_heap/mm_initialize.c +++ b/mm/mm_heap/mm_initialize.c @@ -109,6 +109,8 @@ void mm_addregion(FAR struct mm_heap_s *heap, FAR void *heapstart, #if CONFIG_MM_REGIONS > 1 int idx; + kasan_hw_close(); + DEBUGVERIFY(mm_lock(heap)); idx = heap->mm_nregions; @@ -209,6 +211,7 @@ void mm_addregion(FAR struct mm_heap_s *heap, FAR void *heapstart, sched_note_heap(NOTE_HEAP_ADD, heap, heapstart, heapsize, heap->mm_curused); mm_unlock(heap); + kasan_hw_open(); } /**************************************************************************** @@ -237,6 +240,8 @@ FAR struct mm_heap_s *mm_initialize(FAR const char *name, uintptr_t heap_adj; int i; + kasan_hw_close(); + minfo("Heap: name=%s, start=%p size=%zu\n", name, heapstart, heapsize); /* First ensure the memory to be used is aligned */ @@ -291,7 +296,7 @@ FAR struct mm_heap_s *mm_initialize(FAR const char *name, procfs_register_meminfo(&heap->mm_procfs); # endif #endif - + kasan_hw_open(); return heap; } @@ -302,11 +307,12 @@ mm_initialize_pool(FAR const char *name, FAR const struct mempool_init_s *init) { FAR struct mm_heap_s *heap; - #if CONFIG_MM_HEAP_MEMPOOL_THRESHOLD > 0 size_t poolsize[MEMPOOL_NPOOLS]; struct mempool_init_s def; + kasan_hw_close(); + if (init == NULL) { /* Initialize the multiple mempool default parameter */ @@ -349,6 +355,7 @@ mm_initialize_pool(FAR const char *name, init->dict_expendsize); } + kasan_hw_open(); return heap; } #endif @@ -371,6 +378,8 @@ void mm_uninitialize(FAR struct mm_heap_s *heap) { int i; + kasan_hw_close(); + #ifdef CONFIG_MM_HEAP_MEMPOOL mempool_multiple_deinit(heap->mm_mpool); #endif @@ -389,4 +398,5 @@ void mm_uninitialize(FAR struct mm_heap_s *heap) # endif #endif nxmutex_destroy(&heap->mm_lock); + kasan_hw_open(); } diff --git a/mm/mm_heap/mm_malloc.c b/mm/mm_heap/mm_malloc.c index dc201b16f5b65..d4f63c6181569 100644 --- a/mm/mm_heap/mm_malloc.c +++ b/mm/mm_heap/mm_malloc.c @@ -175,6 +175,8 @@ FAR void *mm_malloc(FAR struct mm_heap_s *heap, size_t size) FAR void *ret = NULL; int ndx; + kasan_hw_close(); + /* Free the delay list first */ free_delaylist(heap, false); @@ -409,5 +411,6 @@ FAR void *mm_malloc(FAR struct mm_heap_s *heap, size_t size) #endif DEBUGASSERT(ret == NULL || ((uintptr_t)ret) % MM_ALIGN == 0); + kasan_hw_open(); return ret; }