aboutsummaryrefslogblamecommitdiff
path: root/tools/test/stress2/misc/signal.sh
blob: 0466184b1f14650aa8d8138d5c0e9bddd93763d2 (plain) (tree)






















































































































































































































































                                                                                     
#!/bin/sh

#
# Copyright (c) 2013 EMC Corp.
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions
# are met:
# 1. Redistributions of source code must retain the above copyright
#    notice, this list of conditions and the following disclaimer.
# 2. Redistributions in binary form must reproduce the above copyright
#    notice, this list of conditions and the following disclaimer in the
#    documentation and/or other materials provided with the distribution.
#
# THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
# ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
# SUCH DAMAGE.
#

# Regression test for kern/150138: signal sent to stopped, traced process
# not immediately handled on continue.
# Fixed in r212047.

# Test scenario by Dan McNulty <dkmcnulty gmail.com>

. ../default.cfg

here=`pwd`
wd=/tmp/signal.dir
rm -rf $wd
mkdir -p $wd
cd $wd
cat > waitthread.c <<EOF
#include <unistd.h>
#include <stdlib.h>
#include <pthread.h>
#include <stdio.h>
#include <assert.h>
#include <signal.h>
#include <time.h>
#include <sys/syscall.h>
#include <string.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>

struct threadArg {
    int id;
    pthread_mutex_t *mutex;
};

void *entry(void *arg) {
    struct threadArg *thisArg = (struct threadArg *)arg;

    long lwp_id = thisArg->id;
    if( syscall(SYS_thr_self, &lwp_id) ) {
        perror("syscall");
    }

    printf("%ld waiting on lock\n", lwp_id);

    if( pthread_mutex_lock(thisArg->mutex) != 0 ) {
        perror("pthread_mutex_lock");
        return NULL;
    }

    printf("%ld obtained lock\n", lwp_id);

    if( pthread_mutex_unlock(thisArg->mutex) != 0 ) {
        perror("pthread_mutex_unlock");
        return NULL;
    }

    printf("%ld released lock\n", lwp_id);

    return NULL;
}

int main(int argc, char **argv) {
    if( 2 != argc ) {
        printf("Usage: %s <num. of threads>\n", argv[0]);
        return EXIT_FAILURE;
    }

    printf("%d\n", getpid());

    int numThreads;
    sscanf(argv[1], "%d", &numThreads);
    if( numThreads < 1 ) numThreads = 1;

    pthread_t *threads = (pthread_t *)malloc(sizeof(pthread_t)*numThreads);

    pthread_mutex_t *mutex = (pthread_mutex_t *)malloc(sizeof(pthread_mutex_t));

    if( pthread_mutex_init(mutex, NULL) != 0 ) {
        perror("pthread_mutex_init");
        return EXIT_FAILURE;
    }

    if( pthread_mutex_lock(mutex) != 0 ) {
        perror("pthread_mutex_lock");
        return EXIT_FAILURE;
    }

    int i;
    for(i = 0; i < numThreads; ++i) {
        struct threadArg *arg = (struct threadArg *)malloc(sizeof(struct threadArg));
        arg->id = i;
        arg->mutex = mutex;
        assert( !pthread_create(&threads[i], NULL, &entry, (void *)arg) );
    }

    // Wait on the named pipe
    unlink("/tmp/waitthread");
    if( mkfifo("/tmp/waitthread", S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP) ) {
        perror("mkfifo");
        return EXIT_FAILURE;
    }

    FILE *fifo;
    do {
        if( (fifo = fopen("/tmp/waitthread", "r")) == NULL ) {
            if( errno == EINTR ) continue;

            perror("fopen");
            return EXIT_FAILURE;
        }
        break;
    }while(1);

    unsigned char byte;
    if( fread(&byte, sizeof(unsigned char), 1, fifo) != 1 ) {
        perror("fread");
    }

    fclose(fifo);

    unlink("/tmp/waitthread");

    printf("Received notification\n");

    if( pthread_mutex_unlock(mutex) != 0 ) {
        perror("pthread_mutex_unlock");
        return EXIT_FAILURE;
    }

    printf("Unlocked mutex, joining\n");

    for(i = 0; i < numThreads; ++i ) {
        assert( !pthread_join(threads[i], NULL) );
    }

    return EXIT_SUCCESS;
}
EOF
cat > tkill.c <<EOF
#include <sys/syscall.h>
#include <unistd.h>
#include <stdio.h>
#include <errno.h>
#include <stdlib.h>
#include <signal.h>

int main(int argc, char *argv[]) {
    if( argc != 4 ) {
        printf("Usage: %s <pid> <lwp_id> <sig>\n", argv[0]);
        return EXIT_FAILURE;
    }

    pid_t pid;
    sscanf(argv[1], "%d", &pid);

    long id;
    sscanf(argv[2], "%ld", &id);

    int sig;
    sscanf(argv[3], "%d", &sig);

    if( syscall(SYS_thr_kill2, pid, id, sig) ) {
        perror("syscall");
    }

    return EXIT_SUCCESS;
}
EOF
mycc -o waitthread -Wall -Wextra waitthread.c -lpthread || exit
mycc -o tkill -Wall -Wextra tkill.c || exit
rm -f waitthread.c tkill.c

rm -f gdbfifo gdbout pstat /tmp/waitthread
pkill -9 waitthread
pgrep -q waitthread && { pgrep waitthread | xargs ps -lHp; exit 1; }
mkfifo gdbfifo
trap "rm -f gdbfifo" EXIT INT
sleep 300 > gdbfifo &	# Keep the fifo open
sleep .5
fifopid=$!

(trap - INT; gdb ./waitthread < gdbfifo > gdbout 2>&1) &
sleep .8
gdbpid=$!
echo "set args 8" > gdbfifo
echo "run"        > gdbfifo
sleep .8

for i in `jot 5`; do
	pid=`ps -x | grep -v grep | grep "waitthread 8" |
	    sed 's/^ *//;s/ .*//'`
	[ -n "$pid" ] && break
	sleep 1
done
s=0
if [ -n "$pid" ]; then
	procstat -t $pid > pstat

	t1=`grep fifo  pstat | awk '{print $2}'`
	t2=`grep umtxn pstat | awk '{print $2}' | tail -1`

	set -e
	./tkill $pid $t1 5	# SIGTRAP
	./tkill $pid $t2 2	# SIGINT
	set +e

	echo "c"    > gdbfifo
	echo "quit" > gdbfifo
	sleep 1
	grep -q "signal SIGINT" gdbout || { echo FAIL; cat gdbout; s=1; }
else
	echo "Did not find pid for test program waitthread"
	s=2
fi

kill -9 $fifopid $gdbpid > /dev/null 2>&1
pgrep -q waitthread && pkill -9 waitthread
cd $here
rm -rf $wd /tmp/waitthread
exit $s