11.1Skamil# Copyright (c) 2018 The NetBSD Foundation, Inc.
21.1Skamil# All rights reserved.
31.1Skamil#
41.1Skamil# This code is derived from software contributed to The NetBSD Foundation
51.1Skamil# by Yang Zheng.
61.1Skamil#
71.1Skamil# Redistribution and use in source and binary forms, with or without
81.1Skamil# modification, are permitted provided that the following conditions
91.1Skamil# are met:
101.1Skamil# 1. Redistributions of source code must retain the above copyright
111.1Skamil#    notice, this list of conditions and the following disclaimer.
121.1Skamil# 2. Redistributions in binary form must reproduce the above copyright
131.1Skamil#    notice, this list of conditions and the following disclaimer in the
141.1Skamil#    documentation and/or other materials provided with the distribution.
151.1Skamil#
161.1Skamil# THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
171.1Skamil# ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
181.1Skamil# TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
191.1Skamil# PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
201.1Skamil# BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
211.1Skamil# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
221.1Skamil# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
231.1Skamil# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
241.1Skamil# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
251.1Skamil# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
261.1Skamil# POSSIBILITY OF SUCH DAMAGE.
271.1Skamil#
281.1Skamil
291.4Smartintsan_available_archs()
301.1Skamil{
311.4Smartin	atf_set "require.arch" "x86_64"
321.1Skamil}
331.1Skamil
341.1Skamilatf_test_case signal_errno
351.1Skamilsignal_errno_head() {
361.1Skamil	atf_set "descr" "Test thread sanitizer for errno modification in signal condition"
371.1Skamil	atf_set "require.progs" "c++ paxctl"
381.4Smartin	tsan_available_archs
391.1Skamil}
401.1Skamil
411.1Skamilatf_test_case signal_errno_profile
421.1Skamilsignal_errno_profile_head() {
431.1Skamil	atf_set "descr" "Test thread sanitizer for errno modification in signal with profiling option"
441.1Skamil	atf_set "require.progs" "c++ paxctl"
451.4Smartin	tsan_available_archs
461.1Skamil}
471.1Skamilatf_test_case signal_errno_pic
481.1Skamilsignal_errno_pic_head() {
491.1Skamil	atf_set "descr" "Test thread sanitizer for errno modification in signal with position independent code (PIC) flag"
501.1Skamil	atf_set "require.progs" "c++ paxctl"
511.4Smartin	tsan_available_archs
521.1Skamil}
531.1Skamilatf_test_case signal_errno_pie
541.1Skamilsignal_errno_pie_head() {
551.1Skamil	atf_set "descr" "Test thread sanitizer for errno modification in signal with position independent execution (PIE) flag"
561.1Skamil	atf_set "require.progs" "c++ paxctl"
571.4Smartin	tsan_available_archs
581.1Skamil}
591.1Skamil
601.1Skamilsignal_errno_body(){
611.1Skamil	cat > test.cc << EOF
621.1Skamil#include <pthread.h>
631.1Skamil#include <signal.h>
641.1Skamil#include <stdlib.h>
651.1Skamil#include <errno.h>
661.1Skamil
671.1Skamilpthread_t mainth;
681.1Skamilstatic void MyHandler(int a, siginfo_t *s, void *c) { errno = 1; }
691.1Skamilstatic void* sendsignal(void *p) { pthread_kill(mainth, SIGPROF); return NULL; }
701.1Skamilint main() {
711.1Skamil  mainth = pthread_self();
721.1Skamil  struct sigaction act = {};
731.1Skamil  act.sa_sigaction = &MyHandler;
741.1Skamil  sigaction(SIGPROF, &act, 0);
751.1Skamil  pthread_t th;
761.1Skamil  pthread_create(&th, 0, sendsignal, 0);
771.1Skamil  pthread_join(th, 0);
781.1Skamil  return 0;
791.1Skamil}
801.1SkamilEOF
811.1Skamil
821.1Skamil	c++ -fsanitize=thread -o test test.cc
831.1Skamil	paxctl +a test
841.1Skamil	atf_check -s ignore -o ignore -e match:"WARNING: ThreadSanitizer: signal handler spoils errno" ./test
851.1Skamil}
861.1Skamil
871.1Skamilsignal_errno_profile_body(){
881.5Sgson	atf_expect_fail "PR toolchain/55760"
891.1Skamil	cat > test.cc << EOF
901.1Skamil#include <pthread.h>
911.1Skamil#include <signal.h>
921.1Skamil#include <stdlib.h>
931.1Skamil#include <errno.h>
941.1Skamil
951.1Skamilpthread_t mainth;
961.1Skamilstatic void MyHandler(int a, siginfo_t *s, void *c) { errno = 1; }
971.1Skamilstatic void* sendsignal(void *p) { pthread_kill(mainth, SIGPROF); return NULL; }
981.1Skamilint main() {
991.1Skamil  mainth = pthread_self();
1001.1Skamil  struct sigaction act = {};
1011.1Skamil  act.sa_sigaction = &MyHandler;
1021.1Skamil  sigaction(SIGPROF, &act, 0);
1031.1Skamil  pthread_t th;
1041.1Skamil  pthread_create(&th, 0, sendsignal, 0);
1051.1Skamil  pthread_join(th, 0);
1061.1Skamil  return 0;
1071.1Skamil}
1081.1SkamilEOF
1091.1Skamil
1101.7Sskrll	c++ -fsanitize=thread -static -o test -pg test.cc
1111.1Skamil	paxctl +a test
1121.1Skamil	atf_check -s ignore -o ignore -e match:"WARNING: ThreadSanitizer: signal handler spoils errno" ./test
1131.1Skamil}
1141.1Skamil
1151.1Skamilsignal_errno_pic_body(){
1161.1Skamil	cat > test.cc << EOF
1171.1Skamil#include <stdio.h>
1181.1Skamil#include <stdlib.h>
1191.1Skamilint help(int);
1201.1Skamilint main(int argc, char **argv) {return help(argc);}
1211.1SkamilEOF
1221.1Skamil
1231.1Skamil	cat > pic.cc << EOF
1241.1Skamil#include <pthread.h>
1251.1Skamil#include <signal.h>
1261.1Skamil#include <stdlib.h>
1271.1Skamil#include <errno.h>
1281.1Skamil
1291.1Skamilpthread_t mainth;
1301.1Skamilstatic void MyHandler(int a, siginfo_t *s, void *c) { errno = 1; }
1311.1Skamilstatic void* sendsignal(void *p) { pthread_kill(mainth, SIGPROF); return NULL; }
1321.1Skamilint help(int argc) {
1331.1Skamil  mainth = pthread_self();
1341.1Skamil  struct sigaction act = {};
1351.1Skamil  act.sa_sigaction = &MyHandler;
1361.1Skamil  sigaction(SIGPROF, &act, 0);
1371.1Skamil  pthread_t th;
1381.1Skamil  pthread_create(&th, 0, sendsignal, 0);
1391.1Skamil  pthread_join(th, 0);
1401.1Skamil  return 0;
1411.1Skamil}
1421.1SkamilEOF
1431.1Skamil
1441.1Skamil	c++ -fsanitize=thread -fPIC -shared -o libtest.so pic.cc
1451.1Skamil	c++ -o test test.cc -fsanitize=thread -L. -ltest
1461.1Skamil	paxctl +a test
1471.1Skamil
1481.1Skamil	export LD_LIBRARY_PATH=.
1491.1Skamil	atf_check -s ignore -o ignore -e match:"WARNING: ThreadSanitizer: signal handler spoils errno" ./test
1501.1Skamil}
1511.1Skamilsignal_errno_pie_body(){
1521.6Sskrll
1531.1Skamil	#check whether -pie flag is supported on this architecture
1541.6Sskrll	if ! c++ -pie -dM -E - < /dev/null 2>/dev/null >/dev/null; then
1551.1Skamil		atf_set_skip "c++ -pie not supported on this architecture"
1561.1Skamil	fi
1571.1Skamil	cat > test.cc << EOF
1581.1Skamil#include <pthread.h>
1591.1Skamil#include <signal.h>
1601.1Skamil#include <stdlib.h>
1611.1Skamil#include <errno.h>
1621.1Skamil
1631.1Skamilpthread_t mainth;
1641.1Skamilstatic void MyHandler(int a, siginfo_t *s, void *c) { errno = 1; }
1651.1Skamilstatic void* sendsignal(void *p) { pthread_kill(mainth, SIGPROF); return NULL; }
1661.1Skamilint main() {
1671.1Skamil  mainth = pthread_self();
1681.1Skamil  struct sigaction act = {};
1691.1Skamil  act.sa_sigaction = &MyHandler;
1701.1Skamil  sigaction(SIGPROF, &act, 0);
1711.1Skamil  pthread_t th;
1721.1Skamil  pthread_create(&th, 0, sendsignal, 0);
1731.1Skamil  pthread_join(th, 0);
1741.1Skamil  return 0;
1751.1Skamil}
1761.1SkamilEOF
1771.1Skamil
1781.1Skamil	c++ -fsanitize=thread -o test -fpie -pie test.cc
1791.1Skamil	paxctl +a test
1801.1Skamil	atf_check -s ignore -o ignore -e match:"WARNING: ThreadSanitizer: signal handler spoils errno" ./test
1811.1Skamil}
1821.1Skamil
1831.1Skamilatf_init_test_cases()
1841.1Skamil{
1851.1Skamil	atf_add_test_case signal_errno
1861.1Skamil	atf_add_test_case signal_errno_profile
1871.1Skamil	atf_add_test_case signal_errno_pie
1881.1Skamil	atf_add_test_case signal_errno_pic
1891.1Skamil}
190