diff --git a/openmp/libomptarget/DeviceRTL/CMakeLists.txt b/openmp/libomptarget/DeviceRTL/CMakeLists.txt --- a/openmp/libomptarget/DeviceRTL/CMakeLists.txt +++ b/openmp/libomptarget/DeviceRTL/CMakeLists.txt @@ -90,6 +90,7 @@ ${include_directory}/Configuration.h ${include_directory}/Debug.h ${include_directory}/Interface.h + ${include_directory}/LibC.h ${include_directory}/Mapping.h ${include_directory}/State.h ${include_directory}/Synchronization.h @@ -101,6 +102,7 @@ ${source_directory}/Configuration.cpp ${source_directory}/Debug.cpp ${source_directory}/Kernel.cpp + ${source_directory}/LibC.cpp ${source_directory}/Mapping.cpp ${source_directory}/Misc.cpp ${source_directory}/Parallelism.cpp diff --git a/openmp/libomptarget/DeviceRTL/include/LibC.h b/openmp/libomptarget/DeviceRTL/include/LibC.h new file mode 100644 --- /dev/null +++ b/openmp/libomptarget/DeviceRTL/include/LibC.h @@ -0,0 +1,21 @@ +//===--------- LibC.h - Simple implementation of libc functions --- C++ -*-===// +// +// 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 +// +//===----------------------------------------------------------------------===// +// +// +//===----------------------------------------------------------------------===// + +#ifndef OMPTARGET_LIBC_H +#define OMPTARGET_LIBC_H + +#include "Types.h" + +extern "C" { +int memcmp(const void *lhs, const void *rhs, size_t count); +} + +#endif diff --git a/openmp/libomptarget/DeviceRTL/include/Types.h b/openmp/libomptarget/DeviceRTL/include/Types.h --- a/openmp/libomptarget/DeviceRTL/include/Types.h +++ b/openmp/libomptarget/DeviceRTL/include/Types.h @@ -32,6 +32,7 @@ using uint32_t = unsigned int; using int64_t = long; using uint64_t = unsigned long; +using size_t = decltype(sizeof(char)); static_assert(sizeof(int8_t) == 1, "type size mismatch"); static_assert(sizeof(uint8_t) == 1, "type size mismatch"); diff --git a/openmp/libomptarget/DeviceRTL/src/LibC.cpp b/openmp/libomptarget/DeviceRTL/src/LibC.cpp new file mode 100644 --- /dev/null +++ b/openmp/libomptarget/DeviceRTL/src/LibC.cpp @@ -0,0 +1,27 @@ +//===------- LibC.c - Simple implementation of libc functions ----- C -----===// +// +// 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 +// +//===----------------------------------------------------------------------===// + +#include "LibC.h" + +#pragma omp begin declare target device_type(nohost) + +extern "C" { + +int memcmp(const void *lhs, const void *rhs, size_t count) { + auto *L = reinterpret_cast(lhs); + auto *R = reinterpret_cast(rhs); + + for (size_t I = 0; I < count; ++I) + if (L[I] != R[I]) + return (int)L[I] - (int)R[I]; + + return 0; +} +} + +#pragma omp end declare target diff --git a/openmp/libomptarget/DeviceRTL/src/exports b/openmp/libomptarget/DeviceRTL/src/exports --- a/openmp/libomptarget/DeviceRTL/src/exports +++ b/openmp/libomptarget/DeviceRTL/src/exports @@ -1,3 +1,5 @@ omp_* *llvm_* __kmpc_* + +memcmp diff --git a/openmp/libomptarget/test/offloading/test_libc.cpp b/openmp/libomptarget/test/offloading/test_libc.cpp new file mode 100644 --- /dev/null +++ b/openmp/libomptarget/test/offloading/test_libc.cpp @@ -0,0 +1,28 @@ +// RUN: %libomptarget-compilexx-run-and-check-generic + +#include + +extern "C" int printf(const char *, ...); + +// std::equal is lowered to libc function memcmp. +void test_memcpy() { +#pragma omp target + { + int x[2] = {0, 0}; + int y[2] = {0, 0}; + int z[2] = {0, 1}; + bool eq1 = std::equal(x, x + 2, y); + bool eq2 = std::equal(x, x + 2, z); + bool r = eq1 && !eq2; + printf("memcmp: %s\n", r ? "PASS" : "FAIL"); + } +} + +int main(int argc, char *argv[]) { + test_memcpy(); + + return 0; +} + +// CHECK: memcmp: PASS +// CHECK: memcmp: PASS