mirror of
https://github.com/capstone-engine/llvm-capstone.git
synced 2025-04-04 06:12:19 +00:00
[libc] Add memmove benchmarks
This patch enables the benchmarking of `memmove`. Ideally, this should be submitted before D114637. Differential Revision: https://reviews.llvm.org/D114694
This commit is contained in:
parent
a48e05030b
commit
de21f34691
@ -172,11 +172,12 @@ function(add_libc_multi_impl_benchmark name)
|
|||||||
endforeach()
|
endforeach()
|
||||||
endfunction()
|
endfunction()
|
||||||
|
|
||||||
add_libc_multi_impl_benchmark(memcpy)
|
add_libc_multi_impl_benchmark(bcmp)
|
||||||
add_libc_multi_impl_benchmark(memset)
|
|
||||||
add_libc_multi_impl_benchmark(bzero)
|
add_libc_multi_impl_benchmark(bzero)
|
||||||
add_libc_multi_impl_benchmark(memcmp)
|
add_libc_multi_impl_benchmark(memcmp)
|
||||||
add_libc_multi_impl_benchmark(bcmp)
|
add_libc_multi_impl_benchmark(memcpy)
|
||||||
|
add_libc_multi_impl_benchmark(memmove)
|
||||||
|
add_libc_multi_impl_benchmark(memset)
|
||||||
|
|
||||||
#==============================================================================
|
#==============================================================================
|
||||||
# Google Benchmarking tool
|
# Google Benchmarking tool
|
||||||
@ -199,6 +200,7 @@ target_link_libraries(libc.benchmarks.memory_functions.opt_host
|
|||||||
libc.src.string.memcpy_opt_host
|
libc.src.string.memcpy_opt_host
|
||||||
libc.src.string.memset_opt_host
|
libc.src.string.memset_opt_host
|
||||||
libc.src.string.bzero_opt_host
|
libc.src.string.bzero_opt_host
|
||||||
|
libc.src.string.memmove_opt_host
|
||||||
benchmark_main
|
benchmark_main
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -5,6 +5,7 @@
|
|||||||
namespace __llvm_libc {
|
namespace __llvm_libc {
|
||||||
|
|
||||||
extern void *memcpy(void *__restrict, const void *__restrict, size_t);
|
extern void *memcpy(void *__restrict, const void *__restrict, size_t);
|
||||||
|
extern void *memmove(void *, const void *, size_t);
|
||||||
extern void *memset(void *, int, size_t);
|
extern void *memset(void *, int, size_t);
|
||||||
extern void bzero(void *, size_t);
|
extern void bzero(void *, size_t);
|
||||||
extern int memcmp(const void *, const void *, size_t);
|
extern int memcmp(const void *, const void *, size_t);
|
||||||
@ -17,6 +18,7 @@ extern int bcmp(const void *, const void *, size_t);
|
|||||||
using llvm::libc_benchmarks::BzeroConfiguration;
|
using llvm::libc_benchmarks::BzeroConfiguration;
|
||||||
using llvm::libc_benchmarks::MemcmpOrBcmpConfiguration;
|
using llvm::libc_benchmarks::MemcmpOrBcmpConfiguration;
|
||||||
using llvm::libc_benchmarks::MemcpyConfiguration;
|
using llvm::libc_benchmarks::MemcpyConfiguration;
|
||||||
|
using llvm::libc_benchmarks::MemmoveConfiguration;
|
||||||
using llvm::libc_benchmarks::MemsetConfiguration;
|
using llvm::libc_benchmarks::MemsetConfiguration;
|
||||||
|
|
||||||
llvm::ArrayRef<MemcpyConfiguration> getMemcpyConfigurations() {
|
llvm::ArrayRef<MemcpyConfiguration> getMemcpyConfigurations() {
|
||||||
@ -24,6 +26,11 @@ llvm::ArrayRef<MemcpyConfiguration> getMemcpyConfigurations() {
|
|||||||
{__llvm_libc::memcpy, "__llvm_libc::memcpy"}};
|
{__llvm_libc::memcpy, "__llvm_libc::memcpy"}};
|
||||||
return llvm::makeArrayRef(kMemcpyConfigurations);
|
return llvm::makeArrayRef(kMemcpyConfigurations);
|
||||||
}
|
}
|
||||||
|
llvm::ArrayRef<MemmoveConfiguration> getMemmoveConfigurations() {
|
||||||
|
static constexpr MemmoveConfiguration kMemmoveConfigurations[] = {
|
||||||
|
{__llvm_libc::memmove, "__llvm_libc::memmove"}};
|
||||||
|
return llvm::makeArrayRef(kMemmoveConfigurations);
|
||||||
|
}
|
||||||
llvm::ArrayRef<MemcmpOrBcmpConfiguration> getMemcmpConfigurations() {
|
llvm::ArrayRef<MemcmpOrBcmpConfiguration> getMemcmpConfigurations() {
|
||||||
static constexpr MemcmpOrBcmpConfiguration kMemcmpConfiguration[] = {
|
static constexpr MemcmpOrBcmpConfiguration kMemcmpConfiguration[] = {
|
||||||
{__llvm_libc::memcmp, "__llvm_libc::memcmp"}};
|
{__llvm_libc::memcmp, "__llvm_libc::memcmp"}};
|
||||||
|
@ -14,6 +14,12 @@ struct MemcpyConfiguration {
|
|||||||
llvm::StringRef Name;
|
llvm::StringRef Name;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
using MemmoveFunction = void *(*)(void *, const void *, size_t);
|
||||||
|
struct MemmoveConfiguration {
|
||||||
|
MemmoveFunction Function;
|
||||||
|
llvm::StringRef Name;
|
||||||
|
};
|
||||||
|
|
||||||
using MemsetFunction = void *(*)(void *, int, size_t);
|
using MemsetFunction = void *(*)(void *, int, size_t);
|
||||||
struct MemsetConfiguration {
|
struct MemsetConfiguration {
|
||||||
MemsetFunction Function;
|
MemsetFunction Function;
|
||||||
|
@ -108,6 +108,9 @@ CopySetup::CopySetup()
|
|||||||
: ParameterBatch(2), SrcBuffer(ParameterBatch::BufferSize),
|
: ParameterBatch(2), SrcBuffer(ParameterBatch::BufferSize),
|
||||||
DstBuffer(ParameterBatch::BufferSize) {}
|
DstBuffer(ParameterBatch::BufferSize) {}
|
||||||
|
|
||||||
|
MoveSetup::MoveSetup()
|
||||||
|
: ParameterBatch(3), Buffer(ParameterBatch::BufferSize * 3) {}
|
||||||
|
|
||||||
ComparisonSetup::ComparisonSetup()
|
ComparisonSetup::ComparisonSetup()
|
||||||
: ParameterBatch(2), LhsBuffer(ParameterBatch::BufferSize),
|
: ParameterBatch(2), LhsBuffer(ParameterBatch::BufferSize),
|
||||||
RhsBuffer(ParameterBatch::BufferSize) {
|
RhsBuffer(ParameterBatch::BufferSize) {
|
||||||
|
@ -206,6 +206,24 @@ private:
|
|||||||
AlignedBuffer DstBuffer;
|
AlignedBuffer DstBuffer;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/// Provides source and destination buffers for the Move operation as well as
|
||||||
|
/// the associated size distributions.
|
||||||
|
struct MoveSetup : public ParameterBatch {
|
||||||
|
MoveSetup();
|
||||||
|
|
||||||
|
inline static const ArrayRef<MemorySizeDistribution> getDistributions() {
|
||||||
|
return getMemmoveSizeDistributions();
|
||||||
|
}
|
||||||
|
|
||||||
|
inline void *Call(ParameterType Parameter, MemmoveFunction Memmove) {
|
||||||
|
return Memmove(Buffer + ParameterBatch::BufferSize / 3,
|
||||||
|
Buffer + Parameter.OffsetBytes, Parameter.SizeBytes);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
AlignedBuffer Buffer;
|
||||||
|
};
|
||||||
|
|
||||||
/// Provides destination buffer for the Set operation as well as the associated
|
/// Provides destination buffer for the Set operation as well as the associated
|
||||||
/// size distributions.
|
/// size distributions.
|
||||||
struct SetSetup : public ParameterBatch {
|
struct SetSetup : public ParameterBatch {
|
||||||
|
@ -24,6 +24,7 @@
|
|||||||
namespace __llvm_libc {
|
namespace __llvm_libc {
|
||||||
|
|
||||||
extern void *memcpy(void *__restrict, const void *__restrict, size_t);
|
extern void *memcpy(void *__restrict, const void *__restrict, size_t);
|
||||||
|
extern void *memmove(void *, const void *, size_t);
|
||||||
extern void *memset(void *, int, size_t);
|
extern void *memset(void *, int, size_t);
|
||||||
extern void bzero(void *, size_t);
|
extern void bzero(void *, size_t);
|
||||||
extern int memcmp(const void *, const void *, size_t);
|
extern int memcmp(const void *, const void *, size_t);
|
||||||
@ -68,6 +69,9 @@ static cl::opt<uint32_t>
|
|||||||
#if defined(LIBC_BENCHMARK_FUNCTION_MEMCPY)
|
#if defined(LIBC_BENCHMARK_FUNCTION_MEMCPY)
|
||||||
#define LIBC_BENCHMARK_FUNCTION LIBC_BENCHMARK_FUNCTION_MEMCPY
|
#define LIBC_BENCHMARK_FUNCTION LIBC_BENCHMARK_FUNCTION_MEMCPY
|
||||||
using BenchmarkSetup = CopySetup;
|
using BenchmarkSetup = CopySetup;
|
||||||
|
#elif defined(LIBC_BENCHMARK_FUNCTION_MEMMOVE)
|
||||||
|
#define LIBC_BENCHMARK_FUNCTION LIBC_BENCHMARK_FUNCTION_MEMMOVE
|
||||||
|
using BenchmarkSetup = MoveSetup;
|
||||||
#elif defined(LIBC_BENCHMARK_FUNCTION_MEMSET)
|
#elif defined(LIBC_BENCHMARK_FUNCTION_MEMSET)
|
||||||
#define LIBC_BENCHMARK_FUNCTION LIBC_BENCHMARK_FUNCTION_MEMSET
|
#define LIBC_BENCHMARK_FUNCTION LIBC_BENCHMARK_FUNCTION_MEMSET
|
||||||
using BenchmarkSetup = SetSetup;
|
using BenchmarkSetup = SetSetup;
|
||||||
|
@ -17,8 +17,10 @@ using llvm::libc_benchmarks::ComparisonSetup;
|
|||||||
using llvm::libc_benchmarks::CopySetup;
|
using llvm::libc_benchmarks::CopySetup;
|
||||||
using llvm::libc_benchmarks::MemcmpOrBcmpConfiguration;
|
using llvm::libc_benchmarks::MemcmpOrBcmpConfiguration;
|
||||||
using llvm::libc_benchmarks::MemcpyConfiguration;
|
using llvm::libc_benchmarks::MemcpyConfiguration;
|
||||||
|
using llvm::libc_benchmarks::MemmoveConfiguration;
|
||||||
using llvm::libc_benchmarks::MemorySizeDistribution;
|
using llvm::libc_benchmarks::MemorySizeDistribution;
|
||||||
using llvm::libc_benchmarks::MemsetConfiguration;
|
using llvm::libc_benchmarks::MemsetConfiguration;
|
||||||
|
using llvm::libc_benchmarks::MoveSetup;
|
||||||
using llvm::libc_benchmarks::OffsetDistribution;
|
using llvm::libc_benchmarks::OffsetDistribution;
|
||||||
using llvm::libc_benchmarks::SetSetup;
|
using llvm::libc_benchmarks::SetSetup;
|
||||||
|
|
||||||
@ -94,6 +96,10 @@ extern llvm::ArrayRef<MemcpyConfiguration> getMemcpyConfigurations();
|
|||||||
BENCHMARK_MEMORY_FUNCTION(BM_Memcpy, CopySetup, MemcpyConfiguration,
|
BENCHMARK_MEMORY_FUNCTION(BM_Memcpy, CopySetup, MemcpyConfiguration,
|
||||||
getMemcpyConfigurations());
|
getMemcpyConfigurations());
|
||||||
|
|
||||||
|
extern llvm::ArrayRef<MemmoveConfiguration> getMemmoveConfigurations();
|
||||||
|
BENCHMARK_MEMORY_FUNCTION(BM_Memmove, MoveSetup, MemmoveConfiguration,
|
||||||
|
getMemmoveConfigurations());
|
||||||
|
|
||||||
extern llvm::ArrayRef<MemcmpOrBcmpConfiguration> getMemcmpConfigurations();
|
extern llvm::ArrayRef<MemcmpOrBcmpConfiguration> getMemcmpConfigurations();
|
||||||
BENCHMARK_MEMORY_FUNCTION(BM_Memcmp, ComparisonSetup, MemcmpOrBcmpConfiguration,
|
BENCHMARK_MEMORY_FUNCTION(BM_Memcmp, ComparisonSetup, MemcmpOrBcmpConfiguration,
|
||||||
getMemcmpConfigurations());
|
getMemcmpConfigurations());
|
||||||
|
@ -548,6 +548,7 @@ static void Serialize(raw_ostream &Stream,
|
|||||||
Stream << "using llvm::libc_benchmarks::BzeroConfiguration;\n";
|
Stream << "using llvm::libc_benchmarks::BzeroConfiguration;\n";
|
||||||
Stream << "using llvm::libc_benchmarks::MemcmpOrBcmpConfiguration;\n";
|
Stream << "using llvm::libc_benchmarks::MemcmpOrBcmpConfiguration;\n";
|
||||||
Stream << "using llvm::libc_benchmarks::MemcpyConfiguration;\n";
|
Stream << "using llvm::libc_benchmarks::MemcpyConfiguration;\n";
|
||||||
|
Stream << "using llvm::libc_benchmarks::MemmoveConfiguration;\n";
|
||||||
Stream << "using llvm::libc_benchmarks::MemsetConfiguration;\n";
|
Stream << "using llvm::libc_benchmarks::MemsetConfiguration;\n";
|
||||||
Stream << "\n";
|
Stream << "\n";
|
||||||
Stream << "namespace __llvm_libc {\n";
|
Stream << "namespace __llvm_libc {\n";
|
||||||
@ -599,6 +600,11 @@ template <BzeroStub Foo> void Wrap(void *dst, size_t size) {
|
|||||||
}
|
}
|
||||||
)";
|
)";
|
||||||
codegen::configurations::Serialize(Stream, FunctionType::BZERO, Descriptors);
|
codegen::configurations::Serialize(Stream, FunctionType::BZERO, Descriptors);
|
||||||
|
Stream << R"(
|
||||||
|
llvm::ArrayRef<MemmoveConfiguration> getMemmoveConfigurations() {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
)";
|
||||||
Stream << "// Functions : " << Descriptors.size() << "\n";
|
Stream << "// Functions : " << Descriptors.size() << "\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -37,17 +37,6 @@ add_entrypoint_object(
|
|||||||
.string_utils
|
.string_utils
|
||||||
)
|
)
|
||||||
|
|
||||||
add_entrypoint_object(
|
|
||||||
memmove
|
|
||||||
SRCS
|
|
||||||
memmove.cpp
|
|
||||||
HDRS
|
|
||||||
memmove.h
|
|
||||||
DEPENDS
|
|
||||||
libc.src.__support.integer_operations
|
|
||||||
.memory_utils.memcpy_implementation
|
|
||||||
)
|
|
||||||
|
|
||||||
add_entrypoint_object(
|
add_entrypoint_object(
|
||||||
memrchr
|
memrchr
|
||||||
SRCS
|
SRCS
|
||||||
@ -404,6 +393,42 @@ else()
|
|||||||
add_memcpy(memcpy)
|
add_memcpy(memcpy)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
# ------------------------------------------------------------------------------
|
||||||
|
# memmove
|
||||||
|
# ------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
function(add_memmove memmove_name)
|
||||||
|
add_implementation(memmove ${memmove_name}
|
||||||
|
SRCS ${LIBC_SOURCE_DIR}/src/string/memmove.cpp
|
||||||
|
HDRS ${LIBC_SOURCE_DIR}/src/string/memmove.h
|
||||||
|
DEPENDS
|
||||||
|
.memory_utils.memory_utils
|
||||||
|
.memory_utils.memcpy_implementation
|
||||||
|
libc.include.string
|
||||||
|
COMPILE_OPTIONS
|
||||||
|
-fno-builtin
|
||||||
|
${ARGN}
|
||||||
|
)
|
||||||
|
endfunction()
|
||||||
|
|
||||||
|
if(${LIBC_TARGET_ARCHITECTURE_IS_X86})
|
||||||
|
add_memmove(memmove_x86_64_opt_sse2 COMPILE_OPTIONS -march=k8 REQUIRE SSE2)
|
||||||
|
add_memmove(memmove_x86_64_opt_sse4 COMPILE_OPTIONS -march=nehalem REQUIRE SSE4_2)
|
||||||
|
add_memmove(memmove_x86_64_opt_avx2 COMPILE_OPTIONS -march=haswell REQUIRE AVX2)
|
||||||
|
add_memmove(memmove_x86_64_opt_avx512 COMPILE_OPTIONS -march=skylake-avx512 REQUIRE AVX512F)
|
||||||
|
add_memmove(memmove_opt_host COMPILE_OPTIONS ${LIBC_COMPILE_OPTIONS_NATIVE})
|
||||||
|
add_memmove(memmove)
|
||||||
|
elseif(${LIBC_TARGET_ARCHITECTURE_IS_AARCH64})
|
||||||
|
# Disable tail merging as it leads to lower performance.
|
||||||
|
# Note that '-mllvm' needs to be prefixed with 'SHELL:' to prevent CMake flag deduplication.
|
||||||
|
add_memmove(memmove_opt_host COMPILE_OPTIONS ${LIBC_COMPILE_OPTIONS_NATIVE}
|
||||||
|
COMPILE_OPTIONS "SHELL:-mllvm --tail-merge-threshold=0")
|
||||||
|
add_memmove(memmove COMPILE_OPTIONS "SHELL:-mllvm --tail-merge-threshold=0")
|
||||||
|
else()
|
||||||
|
add_memmove(memmove_opt_host COMPILE_OPTIONS ${LIBC_COMPILE_OPTIONS_NATIVE})
|
||||||
|
add_memmove(memmove)
|
||||||
|
endif()
|
||||||
|
|
||||||
# ------------------------------------------------------------------------------
|
# ------------------------------------------------------------------------------
|
||||||
# memset
|
# memset
|
||||||
# ------------------------------------------------------------------------------
|
# ------------------------------------------------------------------------------
|
||||||
|
Loading…
x
Reference in New Issue
Block a user