mirror of
https://github.com/apache/nuttx.git
synced 2026-06-04 23:03:27 +08:00
mm/kasan: The complete implementation of inline instrumentation functions
1. Use always_inline_function instead of inline, inline does not take effect 2. Activate Kasan acceleration by 1/7 Signed-off-by: wangmingrong1 <wangmingrong1@xiaomi.com>
This commit is contained in:
committed by
Xiang Xiao
parent
27f8f87331
commit
b49eef7f4e
@@ -39,7 +39,6 @@
|
|||||||
#define kasan_init_early() kasan_stop()
|
#define kasan_init_early() kasan_stop()
|
||||||
|
|
||||||
#ifndef CONFIG_MM_KASAN
|
#ifndef CONFIG_MM_KASAN
|
||||||
# define kasan_is_poisoned(addr, size) false
|
|
||||||
# define kasan_poison(addr, size)
|
# define kasan_poison(addr, size)
|
||||||
# define kasan_unpoison(addr, size) addr
|
# define kasan_unpoison(addr, size) addr
|
||||||
# define kasan_register(addr, size)
|
# define kasan_register(addr, size)
|
||||||
@@ -59,23 +58,6 @@ extern "C"
|
|||||||
#define EXTERN extern
|
#define EXTERN extern
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Name: kasan_is_poisoned
|
|
||||||
*
|
|
||||||
* Description:
|
|
||||||
* Check if the memory range is poisoned
|
|
||||||
*
|
|
||||||
* Input Parameters:
|
|
||||||
* addr - range start address
|
|
||||||
* size - range size
|
|
||||||
*
|
|
||||||
* Returned Value:
|
|
||||||
* true if the memory range is poisoned, false otherwise.
|
|
||||||
*
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
bool kasan_is_poisoned(FAR const void *addr, size_t size);
|
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Name: kasan_poison
|
* Name: kasan_poison
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -24,12 +24,10 @@ set(SRCS hook.c)
|
|||||||
if(CONFIG_MM_KASAN)
|
if(CONFIG_MM_KASAN)
|
||||||
list(APPEND FLAGS -fno-builtin)
|
list(APPEND FLAGS -fno-builtin)
|
||||||
if(CONFIG_MM_KASAN_GENERIC)
|
if(CONFIG_MM_KASAN_GENERIC)
|
||||||
list(APPEND SRCS generic.c)
|
|
||||||
list(APPEND FLAGS -fno-sanitize=kernel-address)
|
list(APPEND FLAGS -fno-sanitize=kernel-address)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(CONFIG_MM_KASAN_SW_TAGS)
|
if(CONFIG_MM_KASAN_SW_TAGS)
|
||||||
list(APPEND SRCS sw_tags.c)
|
|
||||||
list(APPEND FLAGS -fno-sanitize=kernel-hwaddress)
|
list(APPEND FLAGS -fno-sanitize=kernel-hwaddress)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
|||||||
@@ -27,12 +27,10 @@ ifeq ($(CONFIG_MM_KASAN),y)
|
|||||||
ifeq ($(CONFIG_ARCH_TOOLCHAIN_GNU),y)
|
ifeq ($(CONFIG_ARCH_TOOLCHAIN_GNU),y)
|
||||||
|
|
||||||
ifeq ($(CONFIG_MM_KASAN_GENERIC),y)
|
ifeq ($(CONFIG_MM_KASAN_GENERIC),y)
|
||||||
CSRCS += generic.c
|
|
||||||
CFLAGS += -fno-sanitize=kernel-address
|
CFLAGS += -fno-sanitize=kernel-address
|
||||||
endif
|
endif
|
||||||
|
|
||||||
ifeq ($(CONFIG_MM_KASAN_SW_TAGS),y)
|
ifeq ($(CONFIG_MM_KASAN_SW_TAGS),y)
|
||||||
CSRCS += sw_tags.c
|
|
||||||
CFLAGS += -fno-sanitize=kernel-hwaddress
|
CFLAGS += -fno-sanitize=kernel-hwaddress
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
|||||||
+55
-53
@@ -26,6 +26,7 @@
|
|||||||
|
|
||||||
#include <nuttx/nuttx.h>
|
#include <nuttx/nuttx.h>
|
||||||
#include <nuttx/mm/kasan.h>
|
#include <nuttx/mm/kasan.h>
|
||||||
|
#include <nuttx/compiler.h>
|
||||||
#include <nuttx/spinlock.h>
|
#include <nuttx/spinlock.h>
|
||||||
|
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
@@ -91,9 +92,9 @@ extern const unsigned char g_globals_region[];
|
|||||||
* Private Functions
|
* Private Functions
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
static FAR uintptr_t *kasan_mem_to_shadow(FAR const void *ptr, size_t size,
|
static inline_function FAR uintptr_t *
|
||||||
FAR unsigned int *bit,
|
kasan_mem_to_shadow(FAR const void *ptr, size_t size,
|
||||||
FAR size_t *align)
|
FAR unsigned int *bit, FAR size_t *align)
|
||||||
{
|
{
|
||||||
FAR struct kasan_region_s *region;
|
FAR struct kasan_region_s *region;
|
||||||
uintptr_t addr = (uintptr_t)ptr;
|
uintptr_t addr = (uintptr_t)ptr;
|
||||||
@@ -131,6 +132,57 @@ static FAR uintptr_t *kasan_mem_to_shadow(FAR const void *ptr, size_t size,
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline_function bool
|
||||||
|
kasan_is_poisoned(FAR const void *addr, size_t size)
|
||||||
|
{
|
||||||
|
FAR uintptr_t *p;
|
||||||
|
unsigned int bit;
|
||||||
|
unsigned int nbit;
|
||||||
|
uintptr_t mask;
|
||||||
|
size_t align;
|
||||||
|
|
||||||
|
p = kasan_mem_to_shadow(addr, size, &bit, &align);
|
||||||
|
if (p == NULL)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (size <= align)
|
||||||
|
{
|
||||||
|
return ((*p >> bit) & 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
nbit = KASAN_BITS_PER_WORD - bit % KASAN_BITS_PER_WORD;
|
||||||
|
mask = KASAN_FIRST_WORD_MASK(bit);
|
||||||
|
size = ALIGN_UP(size, align);
|
||||||
|
size /= align;
|
||||||
|
|
||||||
|
while (size >= nbit)
|
||||||
|
{
|
||||||
|
if ((*p++ & mask) != 0)
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bit += nbit;
|
||||||
|
size -= nbit;
|
||||||
|
|
||||||
|
nbit = KASAN_BITS_PER_WORD;
|
||||||
|
mask = UINTPTR_MAX;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (size)
|
||||||
|
{
|
||||||
|
mask &= KASAN_LAST_WORD_MASK(bit + size);
|
||||||
|
if ((*p & mask) != 0)
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
static void kasan_set_poison(FAR const void *addr, size_t size,
|
static void kasan_set_poison(FAR const void *addr, size_t size,
|
||||||
bool poisoned)
|
bool poisoned)
|
||||||
{
|
{
|
||||||
@@ -195,56 +247,6 @@ FAR void *kasan_reset_tag(FAR const void *addr)
|
|||||||
return (FAR void *)addr;
|
return (FAR void *)addr;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool kasan_is_poisoned(FAR const void *addr, size_t size)
|
|
||||||
{
|
|
||||||
FAR uintptr_t *p;
|
|
||||||
unsigned int bit;
|
|
||||||
unsigned int nbit;
|
|
||||||
uintptr_t mask;
|
|
||||||
size_t align;
|
|
||||||
|
|
||||||
p = kasan_mem_to_shadow(addr, size, &bit, &align);
|
|
||||||
if (p == NULL)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (size <= align)
|
|
||||||
{
|
|
||||||
return ((*p >> bit) & 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
nbit = KASAN_BITS_PER_WORD - bit % KASAN_BITS_PER_WORD;
|
|
||||||
mask = KASAN_FIRST_WORD_MASK(bit);
|
|
||||||
size = ALIGN_UP(size, align);
|
|
||||||
size /= align;
|
|
||||||
|
|
||||||
while (size >= nbit)
|
|
||||||
{
|
|
||||||
if ((*p++ & mask) != 0)
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
bit += nbit;
|
|
||||||
size -= nbit;
|
|
||||||
|
|
||||||
nbit = KASAN_BITS_PER_WORD;
|
|
||||||
mask = UINTPTR_MAX;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (size)
|
|
||||||
{
|
|
||||||
mask &= KASAN_LAST_WORD_MASK(bit + size);
|
|
||||||
if ((*p & mask) != 0)
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
void kasan_poison(FAR const void *addr, size_t size)
|
void kasan_poison(FAR const void *addr, size_t size)
|
||||||
{
|
{
|
||||||
kasan_set_poison(addr, size, true);
|
kasan_set_poison(addr, size, true);
|
||||||
|
|||||||
@@ -31,6 +31,14 @@
|
|||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
|
||||||
|
#ifdef CONFIG_MM_KASAN_GENERIC
|
||||||
|
# include "generic.c"
|
||||||
|
#elif defined(CONFIG_MM_KASAN_SW_TAGS)
|
||||||
|
# include "sw_tags.c"
|
||||||
|
#else
|
||||||
|
# define kasan_is_poisoned(addr, size) false
|
||||||
|
#endif
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Pre-processor Definitions
|
* Pre-processor Definitions
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|||||||
+28
-25
@@ -23,6 +23,7 @@
|
|||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
#include <nuttx/mm/kasan.h>
|
#include <nuttx/mm/kasan.h>
|
||||||
|
#include <nuttx/compiler.h>
|
||||||
#include <nuttx/spinlock.h>
|
#include <nuttx/spinlock.h>
|
||||||
|
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
@@ -74,7 +75,8 @@ static FAR struct kasan_region_s *g_region;
|
|||||||
* Private Functions
|
* Private Functions
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
static FAR uint8_t *kasan_mem_to_shadow(FAR const void *ptr, size_t size)
|
static inline_function FAR uint8_t *
|
||||||
|
kasan_mem_to_shadow(FAR const void *ptr, size_t size)
|
||||||
{
|
{
|
||||||
FAR struct kasan_region_s *region;
|
FAR struct kasan_region_s *region;
|
||||||
uintptr_t addr;
|
uintptr_t addr;
|
||||||
@@ -94,6 +96,31 @@ static FAR uint8_t *kasan_mem_to_shadow(FAR const void *ptr, size_t size)
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline_function bool
|
||||||
|
kasan_is_poisoned(FAR const void *addr, size_t size)
|
||||||
|
{
|
||||||
|
FAR uint8_t *p;
|
||||||
|
uint8_t tag;
|
||||||
|
|
||||||
|
tag = kasan_get_tag(addr);
|
||||||
|
p = kasan_mem_to_shadow(addr, size);
|
||||||
|
if (p == NULL)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
size = KASAN_SHADOW_SIZE(size);
|
||||||
|
while (size--)
|
||||||
|
{
|
||||||
|
if (p[size] != tag)
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
static void kasan_set_poison(FAR const void *addr,
|
static void kasan_set_poison(FAR const void *addr,
|
||||||
size_t size, uint8_t value)
|
size_t size, uint8_t value)
|
||||||
{
|
{
|
||||||
@@ -127,30 +154,6 @@ FAR void *kasan_reset_tag(FAR const void *addr)
|
|||||||
(((uint64_t)(addr)) & ~((uint64_t)0xff << KASAN_TAG_SHIFT));
|
(((uint64_t)(addr)) & ~((uint64_t)0xff << KASAN_TAG_SHIFT));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool kasan_is_poisoned(FAR const void *addr, size_t size)
|
|
||||||
{
|
|
||||||
FAR uint8_t *p;
|
|
||||||
uint8_t tag;
|
|
||||||
|
|
||||||
tag = kasan_get_tag(addr);
|
|
||||||
p = kasan_mem_to_shadow(addr, size);
|
|
||||||
if (p == NULL)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
size = KASAN_SHADOW_SIZE(size);
|
|
||||||
while (size--)
|
|
||||||
{
|
|
||||||
if (p[size] != tag)
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
void kasan_poison(FAR const void *addr, size_t size)
|
void kasan_poison(FAR const void *addr, size_t size)
|
||||||
{
|
{
|
||||||
kasan_set_poison(addr, size, 0xff);
|
kasan_set_poison(addr, size, 0xff);
|
||||||
|
|||||||
Reference in New Issue
Block a user