mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-16 17:46:40 +00:00
107 lines
4.2 KiB
C++
107 lines
4.2 KiB
C++
//===-- asan_interceptors_memintrinsics.cpp -------------------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===---------------------------------------------------------------------===//
|
|
//
|
|
// This file is a part of AddressSanitizer, an address sanity checker.
|
|
//
|
|
// ASan versions of memcpy, memmove, and memset.
|
|
//===---------------------------------------------------------------------===//
|
|
|
|
#define SANITIZER_COMMON_NO_REDEFINE_BUILTINS
|
|
|
|
#include "asan_interceptors_memintrinsics.h"
|
|
|
|
#include "asan_interceptors.h"
|
|
#include "asan_report.h"
|
|
#include "asan_stack.h"
|
|
#include "asan_suppressions.h"
|
|
|
|
using namespace __asan;
|
|
|
|
// memcpy is called during __asan_init() from the internals of printf(...).
|
|
// We do not treat memcpy with to==from as a bug.
|
|
// See http://llvm.org/bugs/show_bug.cgi?id=11763.
|
|
#define ASAN_MEMCPY_IMPL(ctx, to, from, size) \
|
|
do { \
|
|
if (LIKELY(replace_intrin_cached)) { \
|
|
if (LIKELY(to != from)) { \
|
|
CHECK_RANGES_OVERLAP("memcpy", to, size, from, size); \
|
|
} \
|
|
ASAN_READ_RANGE(ctx, from, size); \
|
|
ASAN_WRITE_RANGE(ctx, to, size); \
|
|
} else if (UNLIKELY(!AsanInited())) { \
|
|
return internal_memcpy(to, from, size); \
|
|
} \
|
|
return REAL(memcpy)(to, from, size); \
|
|
} while (0)
|
|
|
|
// memset is called inside Printf.
|
|
#define ASAN_MEMSET_IMPL(ctx, block, c, size) \
|
|
do { \
|
|
if (LIKELY(replace_intrin_cached)) { \
|
|
ASAN_WRITE_RANGE(ctx, block, size); \
|
|
} else if (UNLIKELY(!AsanInited())) { \
|
|
return internal_memset(block, c, size); \
|
|
} \
|
|
return REAL(memset)(block, c, size); \
|
|
} while (0)
|
|
|
|
#define ASAN_MEMMOVE_IMPL(ctx, to, from, size) \
|
|
do { \
|
|
if (LIKELY(replace_intrin_cached)) { \
|
|
ASAN_READ_RANGE(ctx, from, size); \
|
|
ASAN_WRITE_RANGE(ctx, to, size); \
|
|
} \
|
|
return internal_memmove(to, from, size); \
|
|
} while (0)
|
|
|
|
void *__asan_memcpy(void *to, const void *from, uptr size) {
|
|
ASAN_MEMCPY_IMPL(nullptr, to, from, size);
|
|
}
|
|
|
|
void *__asan_memset(void *block, int c, uptr size) {
|
|
ASAN_MEMSET_IMPL(nullptr, block, c, size);
|
|
}
|
|
|
|
void *__asan_memmove(void *to, const void *from, uptr size) {
|
|
ASAN_MEMMOVE_IMPL(nullptr, to, from, size);
|
|
}
|
|
|
|
#if SANITIZER_FUCHSIA
|
|
|
|
// Fuchsia doesn't use sanitizer_common_interceptors.inc, but
|
|
// the only things there it wants are these three. Just define them
|
|
// as aliases here rather than repeating the contents.
|
|
|
|
extern "C" decltype(__asan_memcpy) memcpy[[gnu::alias("__asan_memcpy")]];
|
|
extern "C" decltype(__asan_memmove) memmove[[gnu::alias("__asan_memmove")]];
|
|
extern "C" decltype(__asan_memset) memset[[gnu::alias("__asan_memset")]];
|
|
|
|
#else // SANITIZER_FUCHSIA
|
|
|
|
#define COMMON_INTERCEPTOR_MEMMOVE_IMPL(ctx, to, from, size) \
|
|
do { \
|
|
ASAN_INTERCEPTOR_ENTER(ctx, memmove); \
|
|
ASAN_MEMMOVE_IMPL(ctx, to, from, size); \
|
|
} while (false)
|
|
|
|
#define COMMON_INTERCEPTOR_MEMCPY_IMPL(ctx, to, from, size) \
|
|
do { \
|
|
ASAN_INTERCEPTOR_ENTER(ctx, memcpy); \
|
|
ASAN_MEMCPY_IMPL(ctx, to, from, size); \
|
|
} while (false)
|
|
|
|
#define COMMON_INTERCEPTOR_MEMSET_IMPL(ctx, block, c, size) \
|
|
do { \
|
|
ASAN_INTERCEPTOR_ENTER(ctx, memset); \
|
|
ASAN_MEMSET_IMPL(ctx, block, c, size); \
|
|
} while (false)
|
|
|
|
#include "sanitizer_common/sanitizer_common_interceptors_memintrinsics.inc"
|
|
|
|
#endif // SANITIZER_FUCHSIA
|