/*
* COPYRIGHT (c) 1989-2009.
* On-Line Applications Research Corporation (OAR).
*
* The license and distribution terms for this file may be
* found in the file LICENSE in this distribution or at
* http://www.rtems.com/license/LICENSE.
*
* $Id$
*/
#if defined(USE_USER_SIGNALS_PROCESS)
#define TEST_NAME "03"
#define TEST_STRING "User Signals to Process"
#define SIGNAL_ONE SIGUSR1
#define SIGNAL_TWO SIGUSR2
#define SEND_SIGNAL(_sig) kill( getpid(), _sig )
#define TO_PROCESS
#elif defined(USE_REAL_TIME_SIGNALS_PROCESS)
#define TEST_NAME "04"
#define TEST_STRING "Real-Time Signals to Process"
#define SIGNAL_ONE SIGRTMIN
#define SIGNAL_TWO SIGRTMAX
#define SEND_SIGNAL(_sig) kill( getpid(), _sig )
#define TO_PROCESS
#elif defined(USE_USER_SIGNALS_THREAD)
#define TEST_NAME "05"
#define TEST_STRING "User Signals to Thread"
#define SIGNAL_ONE SIGUSR1
#define SIGNAL_TWO SIGUSR2
#define SEND_SIGNAL(_sig) pthread_kill( id, _sig )
#define TO_THREAD
#elif defined(USE_REAL_TIME_SIGNALS_THREAD)
#define TEST_NAME "05"
#define TEST_STRING "Real-Time Signals to Thread"
#define SIGNAL_ONE SIGRTMIN
#define SIGNAL_TWO SIGRTMAX
#define SEND_SIGNAL(_sig) pthread_kill( id, _sig )
#define TO_THREAD
#else
#error "Test Mode not defined"
#endif
#include <pmacros.h>
#include <signal.h>
#include <errno.h>
#include <pthread.h>
#include <sched.h>
volatile bool Signal_occurred;
volatile pthread_t Signal_thread;
void Signal_handler(
int signo,
siginfo_t *info,
void *arg
)
{
Signal_occurred = true;
Signal_thread = pthread_self();
}
const char *signal_name(int signo)
{
if (signo == SIGUSR1)
return "SIGUSR1";
if (signo == SIGUSR2)
return "SIGUSR2";
if (signo == SIGRTMIN)
return "SIGRTMIN";
if (signo == SIGRTMAX)
return "SIGRTMAX";
return "unknown-signal";
}
void *Test_Thread(void *arg)
{
bool blocked = *((bool *)arg);
const char *name;
int sc;
sigset_t mask;
sigset_t wait_mask;
siginfo_t info;
if ( blocked )
name = "SignalBlocked";
else
name = "SignalNotBlocked";
/* build unblocked mask */
sc = sigemptyset( &mask );
rtems_test_assert( !sc );
printf( "%s - Unblock %s\n", name, signal_name(SIGNAL_ONE) );
sc = sigaddset( &mask, SIGNAL_ONE );
rtems_test_assert( !sc );
if ( !blocked ) {
printf( "%s - Unblock %s\n", name, signal_name(SIGNAL_TWO) );
sc = sigaddset( &mask, SIGNAL_TWO );
rtems_test_assert( !sc );
}
/* unblocked signals */
sc = pthread_sigmask( SIG_UNBLOCK, &mask, NULL );
rtems_test_assert( !sc );
/* build wait mask */
sc = sigemptyset( &wait_mask );
rtems_test_assert( !sc );
sc = sigaddset( &wait_mask, SIGNAL_ONE );
rtems_test_assert( !sc );
/* wait for a signal */
memset( &info, 0, sizeof(info) );
printf( "%s - Wait for %s unblocked\n", name, signal_name(SIGNAL_ONE) );
sigwaitinfo( &wait_mask, &info );
rtems_test_assert( !sc );
printf( "%s - siginfo.si_signo=%d\n", name, info.si_signo );
printf( "%s - siginfo.si_code=%d\n", name, info.si_code );
/* FIXME: Instead of casting to (uintptr_t) and using PRIxPTR, we
* likely should use %p. However, this would render this test's
* behavior non-deterministic, because %p's behavior is
* "implementation defined" */
printf( "%s - siginfo.si_value=0x%08" PRIxPTR "\n", name, (uintptr_t) info.si_value.sival_ptr );
rtems_test_assert( info.si_signo == SIGNAL_TWO );
rtems_test_assert( info.si_code == SI_USER );
printf( "%s - exiting\n", name );
return NULL;
}
void *POSIX_Init(
void *argument
)
{
int sc;
pthread_t id;
struct sigaction act;
bool trueArg = true;
bool falseArg = false;
struct timespec delay_request;
puts( "\n\n*** POSIX TEST SIGNAL " TEST_NAME " ***" );
puts( "Init - Variation is: " TEST_STRING );
Signal_occurred = false;
act.sa_handler = NULL;
act.sa_sigaction = Signal_handler;
act.sa_flags = SA_SIGINFO;
sigaction( SIGNAL_ONE, &act, NULL );
sigaction( SIGNAL_TWO, &act, NULL );
/* create threads */
sc = pthread_create( &id, NULL, Test_Thread, &falseArg );
rtems_test_assert( !sc );
sc = pthread_create( &id, NULL, Test_Thread, &trueArg );
rtems_test_assert( !sc );
puts( "Init - sleep - let threads settle - OK" );
delay_request.tv_sec = 0;
delay_request.tv_nsec = 5 * 100000000;
sc = nanosleep( &delay_request, NULL );
rtems_test_assert( !sc );
puts( "Init - sleep - SignalBlocked thread settle - OK" );
sc = nanosleep( &delay_request, NULL );
rtems_test_assert( !sc );
printf( "Init - sending %s - deliver to one thread\n",
signal_name(SIGNAL_TWO));
sc = SEND_SIGNAL( SIGNAL_TWO );
rtems_test_assert( !sc );
printf( "Init - sending %s - deliver to other thread\n",
signal_name(SIGNAL_TWO));
sc = SEND_SIGNAL( SIGNAL_TWO );
rtems_test_assert( !sc );
#if defined(TO_PROCESS)
printf( "Init - sending %s - expect EAGAIN\n", signal_name(SIGNAL_TWO) );
sc = SEND_SIGNAL( SIGNAL_TWO );
rtems_test_assert( sc == -1 );
rtems_test_assert( errno == EAGAIN );
#endif
puts( "Init - sleep - let thread report if it unblocked - OK" );
usleep(500000);
/* we are just sigwait'ing the signal, not delivering it */
rtems_test_assert( Signal_occurred == true );
puts( "*** END OF POSIX TEST SIGNAL " TEST_NAME " ***" );
rtems_test_exit(0);
return NULL; /* just so the compiler thinks we returned something */
}
/* configuration information */
#define CONFIGURE_APPLICATION_NEEDS_CONSOLE_DRIVER
#define CONFIGURE_APPLICATION_NEEDS_CLOCK_DRIVER
#define CONFIGURE_MAXIMUM_POSIX_THREADS 3
#define CONFIGURE_MAXIMUM_POSIX_QUEUED_SIGNALS 1
#define CONFIGURE_POSIX_INIT_THREAD_TABLE
#define CONFIGURE_INIT
#include <rtems/confdefs.h>