Index: libc/cmake/modules/LLVMLibCRules.cmake =================================================================== --- libc/cmake/modules/LLVMLibCRules.cmake +++ libc/cmake/modules/LLVMLibCRules.cmake @@ -332,8 +332,7 @@ foreach(dep IN LISTS LIBC_UNITTEST_DEPENDS) get_target_property(dep_type ${dep} "TARGET_TYPE") if (dep_type) - string(COMPARE EQUAL ${dep_type} ${ENTRYPOINT_OBJ_TARGET_TYPE} dep_is_entrypoint) - if(dep_is_entrypoint) + if (${dep_type} STREQUAL ${ENTRYPOINT_OBJ_TARGET_TYPE} OR ${dep_type} STREQUAL "SINGLE_OBJECT") get_target_property(obj_file ${dep} "OBJECT_FILE_RAW") list(APPEND library_deps ${obj_file}) continue() Index: libc/config/linux/api.td =================================================================== --- libc/config/linux/api.td +++ libc/config/linux/api.td @@ -180,9 +180,28 @@ ]; } +def StructSigactionDefn : TypeDecl<"struct sigaction"> { + let Decl = [{ + struct __sigaction { + union { + void (*sa_handler)(int); + void (*sa_action)(int, siginfo_t *, void *); + }; + sigset_t sa_mask; + int sa_flags; + void (*sa_restorer)(void); + }; + }]; +} + def SignalAPI : PublicAPI<"signal.h"> { + let TypeDeclarations = [ + StructSigactionDefn, + ]; + let Functions = [ "raise", + "sigaction", "sigprocmask", "sigemptyset", "sigaddset", Index: libc/config/linux/signal.h.in =================================================================== --- libc/config/linux/signal.h.in +++ libc/config/linux/signal.h.in @@ -9,3 +9,7 @@ %%begin() #include <linux/signal.h> + +#ifndef __LLVM_LIBC_INTERNAL_SIGACTION +#define sigaction __sigaction +#endif Index: libc/lib/CMakeLists.txt =================================================================== --- libc/lib/CMakeLists.txt +++ libc/lib/CMakeLists.txt @@ -18,6 +18,7 @@ # signal.h entrypoints raise + sigaction sigaddset sigemptyset sigprocmask Index: libc/spec/posix.td =================================================================== --- libc/spec/posix.td +++ libc/spec/posix.td @@ -4,6 +4,12 @@ def RestrictSigSetType : RestrictedPtrType<SigSetType>; def ConstRestrictSigSetType : ConstType<RestrictSigSetType>; +def StructSigaction : NamedType<"struct sigaction">; +def StructSigactionPtr : PtrType<StructSigaction>; +def ConstStructSigactionPtr : ConstType<StructSigactionPtr>; +def RestrictStructSigactionPtr : RestrictedPtrType<StructSigaction>; +def ConstRestrictStructSigactionPtr : ConstType<RestrictStructSigactionPtr>; + def POSIX : StandardSpec<"POSIX"> { NamedType OffTType = NamedType<"off_t">; @@ -137,9 +143,19 @@ HeaderSpec Signal = HeaderSpec< "signal.h", [], // Macros - [], // Types + [ + SigSetType, + StructSigaction, + ], [], // Enumerations [ + FunctionSpec< + "sigaction", + RetValSpec<IntType>, + [ArgSpec<IntType>, + ArgSpec<ConstRestrictStructSigactionPtr>, + ArgSpec<RestrictStructSigactionPtr>] + >, FunctionSpec< "sigprocmask", RetValSpec<IntType>, Index: libc/spec/stdc.td =================================================================== --- libc/spec/stdc.td +++ libc/spec/stdc.td @@ -225,7 +225,6 @@ Macro<"SIGTERM"> ], [ - NamedType<"sigset_t">, SizeTType, ], [], // Enumerations Index: libc/src/signal/linux/CMakeLists.txt =================================================================== --- libc/src/signal/linux/CMakeLists.txt +++ libc/src/signal/linux/CMakeLists.txt @@ -12,6 +12,39 @@ signal_h ) +add_custom_target(__restore + COMMAND ${CMAKE_CXX_COMPILER} ${CMAKE_CURRENT_SOURCE_DIR}/__restore.cpp -fomit-frame-pointer + -O3 -c -o ${CMAKE_CURRENT_BINARY_DIR}/__restore.o -I${LIBC_BUILD_DIR}/ + -Wframe-larger-than=0 -Wno-attributes -Werror + + BYPRODUCTS ${CMAKE_CURRENT_BINARY_DIR}/__restore.o + SOURCES + __restore.cpp + DEPENDS + sys_syscall_h + linux_syscall_h +) +set_target_properties(__restore + PROPERTIES + "TARGET_TYPE" "SINGLE_OBJECT" + "OBJECT_FILE" ${CMAKE_CURRENT_BINARY_DIR}/__restore.o + "OBJECT_FILE_RAW" ${CMAKE_CURRENT_BINARY_DIR}/__restore.o +) + +add_entrypoint_object( + sigaction + SRCS + sigaction.cpp + HDRS + signal.h + ../sigaction.h + DEPENDS + __restore + sys_syscall_h + linux_syscall_h + signal_h +) + add_entrypoint_object( sigprocmask SRCS Index: libc/src/signal/linux/__restore.cpp =================================================================== --- /dev/null +++ libc/src/signal/linux/__restore.cpp @@ -0,0 +1,20 @@ +//===----------------- Linux implementation of __restore_rt ---------------===// +// +// 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 implemented seperately from sigaction.cpp so that we can +// strongly control the options this file is compiled with. __restore_rt cannot +// make any stack allocations so we must ensure this. + +#include "config/linux/syscall.h" +#include "include/sys/syscall.h" + +extern "C" void __restore_rt() + __attribute__((no_sanitize("thread", "memory", "undefined", "fuzzer"), + hidden)); + +extern "C" void __restore_rt() { __llvm_libc::syscall(SYS_rt_sigreturn); } Index: libc/src/signal/linux/sigaction.cpp =================================================================== --- /dev/null +++ libc/src/signal/linux/sigaction.cpp @@ -0,0 +1,56 @@ +//===----------------- Linux implementation of sigaction ------------------===// +// +// 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 +// +//===----------------------------------------------------------------------===// + +#define __LLVM_LIBC_INTERNAL_SIGACTION +#include "src/signal/sigaction.h" +#include "src/errno/llvmlibc_errno.h" +#include "src/signal/linux/signal.h" + +#include "src/__support/common.h" + +namespace __llvm_libc { + +// TOOD: Some architectures will have their signal trampoline functions in the +// vdso, use those when available. + +extern "C" void __restore_rt(); + +template <typename T, typename V> +static void copySigaction(T &dest, const V &source) { + dest.sa_handler = source.sa_handler; + dest.sa_mask = source.sa_mask; + dest.sa_flags = source.sa_flags; + dest.sa_restorer = source.sa_restorer; +} + +int LLVM_LIBC_ENTRYPOINT(sigaction)(int signal, + const struct __sigaction *__restrict libc_new, + struct __sigaction *__restrict libc_old) { + struct sigaction kernel_new; + if (libc_new) { + copySigaction(kernel_new, *libc_new); + if (!(kernel_new.sa_flags & SA_RESTORER)) { + kernel_new.sa_flags |= SA_RESTORER; + kernel_new.sa_restorer = __restore_rt; + } + } + + struct sigaction kernel_old; + int ret = syscall(SYS_rt_sigaction, signal, libc_new ? &kernel_new : nullptr, + libc_old ? &kernel_old : nullptr, sizeof(sigset_t)); + if (ret) { + llvmlibc_errno = -ret; + return -1; + } + + if (libc_old) + copySigaction(*libc_old, kernel_old); + return 0; +} + +} // namespace __llvm_libc Index: libc/src/signal/sigaction.h =================================================================== --- /dev/null +++ libc/src/signal/sigaction.h @@ -0,0 +1,22 @@ +//===------------ Implementation header for sigaction --------*- 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 LLVM_LIBC_SRC_SIGNAL_SIGACTION_H +#define LLVM_LIBC_SRC_SIGNAL_SIGACTION_H + +#define __LLVM_LIBC_INTERNAL_SIGACTION +#include "include/signal.h" + +namespace __llvm_libc { + +int sigaction(int signal, const struct __sigaction *__restrict libc_new, + struct __sigaction *__restrict libc_old); + +} // namespace __llvm_libc + +#endif // LLVM_LIBC_SRC_SIGNAL_SIGACTION_H Index: libc/test/src/signal/CMakeLists.txt =================================================================== --- libc/test/src/signal/CMakeLists.txt +++ libc/test/src/signal/CMakeLists.txt @@ -11,6 +11,20 @@ signal_h ) +add_libc_unittest( + sigaction_test + SUITE + libc_signal_unittests + SRCS + sigaction_test.cpp + DEPENDS + sigaction + raise + signal_h + __errno_location + __restore +) + add_libc_unittest( sigprocmask_test SUITE Index: libc/test/src/signal/sigaction_test.cpp =================================================================== --- /dev/null +++ libc/test/src/signal/sigaction_test.cpp @@ -0,0 +1,66 @@ +//===----------------------- Unittests for sigaction ----------------------===// +// +// 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 "include/errno.h" +#define __LLVM_LIBC_INTERNAL_SIGACTION +#include "include/signal.h" +#include "src/signal/raise.h" +#include "src/signal/sigaction.h" + +#include "utils/UnitTest/ErrnoSetterMatcher.h" +#include "utils/UnitTest/Test.h" + +using __llvm_libc::testing::ErrnoSetterMatcher::Fails; +using __llvm_libc::testing::ErrnoSetterMatcher::Succeeds; + +TEST(Sigaction, Invalid) { + // -1 is a much larger signal that NSIG, so this should fail. + EXPECT_THAT(__llvm_libc::sigaction(-1, nullptr, nullptr), Fails(EINVAL)); +} + +// SIGKILL cannot have its action changed, but it can be examined. +TEST(Sigaction, Sigkill) { + struct __sigaction action; + EXPECT_THAT(__llvm_libc::sigaction(SIGKILL, nullptr, &action), Succeeds()); + EXPECT_THAT(__llvm_libc::sigaction(SIGKILL, &action, nullptr), Fails(EINVAL)); +} + +static int sigusr1Count; +static bool correctSignal; + +TEST(Sigaction, CustomAction) { + // Zero this incase tests get run multiple times in the future. + sigusr1Count = 0; + + struct __sigaction action; + EXPECT_THAT(__llvm_libc::sigaction(SIGUSR1, nullptr, &action), Succeeds()); + + action.sa_handler = +[](int signal) { + correctSignal = signal == SIGUSR1; + sigusr1Count++; + }; + EXPECT_THAT(__llvm_libc::sigaction(SIGUSR1, &action, nullptr), Succeeds()); + + __llvm_libc::raise(SIGUSR1); + EXPECT_EQ(sigusr1Count, 1); + EXPECT_TRUE(correctSignal); + + action.sa_handler = SIG_DFL; + EXPECT_THAT(__llvm_libc::sigaction(SIGUSR1, &action, nullptr), Succeeds()); + + EXPECT_DEATH([] { __llvm_libc::raise(SIGUSR1); }, SIGUSR1); +} + +TEST(Sigaction, Ignore) { + struct __sigaction action; + EXPECT_THAT(__llvm_libc::sigaction(SIGUSR1, nullptr, &action), Succeeds()); + action.sa_handler = SIG_IGN; + EXPECT_THAT(__llvm_libc::sigaction(SIGUSR1, &action, nullptr), Succeeds()); + + EXPECT_EXITS([] {__llvm_libc::raise(SIGUSR1); }, 0); +}