// Copyright 2018 The gVisor Authors. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // This test verifies that fork(2) in a signal handler will correctly // restore floating point state after the signal handler returns in both // the child and parent. #include <sys/time.h> #include "gtest/gtest.h" #include "test/util/logging.h" #include "test/util/test_util.h" #include "test/util/thread_util.h" namespace gvisor { namespace testing { namespace { #ifdef __x86_64__ #define GET_XMM(__var, __xmm) \ asm volatile("movq %%" #__xmm ", %0" : "=r"(__var)) #define SET_XMM(__var, __xmm) asm volatile("movq %0, %%" #__xmm : : "r"(__var)) #define GET_FP0(__var) GET_XMM(__var, xmm0) #define SET_FP0(__var) SET_XMM(__var, xmm0) #elif __aarch64__ #define __stringify_1(x...) #x #define __stringify(x...) __stringify_1(x) #define GET_FPREG(var, regname) \ asm volatile("str " __stringify(regname) ", %0" : "=m"(var)) #define SET_FPREG(var, regname) \ asm volatile("ldr " __stringify(regname) ", %0" : "=m"(var)) #define GET_FP0(var) GET_FPREG(var, d0) #define SET_FP0(var) SET_FPREG(var, d0) #endif #define DEFAULT_MXCSR 0x1f80 int parent, child; void sigusr1(int s, siginfo_t* siginfo, void* _uc) { // Fork and clobber %xmm0. The fpstate should be restored by sigreturn(2) // in both parent and child. child = fork(); TEST_CHECK_MSG(child >= 0, "fork failed"); uint64_t val = SIGUSR1; SET_FP0(val); uint64_t got; GET_FP0(got); TEST_CHECK_MSG(val == got, "Basic FP check failed in sigusr1()"); #ifdef __x86_64 uint32_t mxcsr; __asm__("STMXCSR %0" : "=m"(mxcsr)); TEST_CHECK_MSG(mxcsr == DEFAULT_MXCSR, "Unexpected mxcsr"); #endif } TEST(FPSigTest, Fork) { parent = getpid(); pid_t parent_tid = gettid(); struct sigaction sa = {}; sigemptyset(&sa.sa_mask); sa.sa_flags = SA_SIGINFO; sa.sa_sigaction = sigusr1; ASSERT_THAT(sigaction(SIGUSR1, &sa, nullptr), SyscallSucceeds()); // The amd64 ABI specifies that the XMM register set is caller-saved. This // implies that if there is any function call between SET_XMM and GET_XMM the // compiler might save/restore xmm0 implicitly. This defeats the entire // purpose of the test which is to verify that fpstate is restored by // sigreturn(2). // // This is the reason why 'tgkill(getpid(), gettid(), SIGUSR1)' is implemented // in inline assembly below. // // If the OS is broken and registers are clobbered by the child, using tgkill // to signal the current thread increases the likelihood that this thread will // be the one clobbered. uint64_t expected = 0xdeadbeeffacefeed; SET_FP0(expected); #ifdef __x86_64__ asm volatile( "movl %[killnr], %%eax;" "movl %[parent], %%edi;" "movl %[tid], %%esi;" "movl %[sig], %%edx;" "syscall;" : : [ killnr ] "i"(__NR_tgkill), [ parent ] "rm"(parent), [ tid ] "rm"(parent_tid), [ sig ] "i"(SIGUSR1) : "rax", "rdi", "rsi", "rdx", // Clobbered by syscall. "rcx", "r11"); #elif __aarch64__ asm volatile( "mov x8, %0\n" "mov x0, %1\n" "mov x1, %2\n" "mov x2, %3\n" "svc #0\n" ::"r"(__NR_tgkill), "r"(parent), "r"(parent_tid), "r"(SIGUSR1)); #endif uint64_t got; GET_FP0(got); if (getpid() == parent) { // Parent. int status; ASSERT_THAT(waitpid(child, &status, 0), SyscallSucceedsWithValue(child)); EXPECT_TRUE(WIFEXITED(status) && WEXITSTATUS(status) == 0); } // TEST_CHECK_MSG since this may run in the child. TEST_CHECK_MSG(expected == got, "Bad xmm0 value"); if (getpid() != parent) { // Child. _exit(0); } } #ifdef __x86_64__ TEST(FPSigTest, ForkWithZeroMxcsr) { parent = getpid(); pid_t parent_tid = gettid(); struct sigaction sa = {}; sigemptyset(&sa.sa_mask); sa.sa_flags = SA_SIGINFO; sa.sa_sigaction = sigusr1; ASSERT_THAT(sigaction(SIGUSR1, &sa, nullptr), SyscallSucceeds()); // The control bits of the MXCSR register are callee-saved (preserved across // calls), while the status bits are caller-saved (not preserved). uint32_t expected = 0, origin; __asm__("STMXCSR %0" : "=m"(origin)); __asm__("LDMXCSR %0" : : "m"(expected)); asm volatile( "movl %[killnr], %%eax;" "movl %[parent], %%edi;" "movl %[tid], %%esi;" "movl %[sig], %%edx;" "syscall;" : : [killnr] "i"(__NR_tgkill), [parent] "rm"(parent), [tid] "rm"(parent_tid), [sig] "i"(SIGUSR1) : "rax", "rdi", "rsi", "rdx", // Clobbered by syscall. "rcx", "r11"); uint32_t got; __asm__("STMXCSR %0" : "=m"(got)); __asm__("LDMXCSR %0" : : "m"(origin)); if (getpid() == parent) { // Parent. int status; ASSERT_THAT(waitpid(child, &status, 0), SyscallSucceedsWithValue(child)); EXPECT_TRUE(WIFEXITED(status) && WEXITSTATUS(status) == 0); } // TEST_CHECK_MSG since this may run in the child. TEST_CHECK_MSG(expected == got, "Bad mxcsr value"); if (getpid() != parent) { // Child. _exit(0); } } #endif } // namespace } // namespace testing } // namespace gvisor