]> pd.if.org Git - pdclib/blobdiff - functions/signal/raise.c
PDCLib includes with quotes, not <>.
[pdclib] / functions / signal / raise.c
index 29afe151f9b611d6fe0f5f776aa88b5f16496e60..13a8b50b9fa28fa6dfa98ce157494ceae9895461 100644 (file)
@@ -1,5 +1,3 @@
-/* $Id$ */
-
 /* raise( int )
 
    This file is part of the Public Domain C Library (PDCLib).
@@ -10,6 +8,9 @@
 
 #ifndef REGTEST
 
+#include <stdio.h>
+#include <stdlib.h>
+
 extern void (*_PDCLIB_sigabrt)( int );
 extern void (*_PDCLIB_sigfpe)( int );
 extern void (*_PDCLIB_sigill)( int );
@@ -19,42 +20,93 @@ extern void (*_PDCLIB_sigterm)( int );
 
 int raise( int sig )
 {
+    void (*sighandler)( int );
+    char const * message;
     switch ( sig )
     {
         case SIGABRT:
-            _PDCLIB_sigabrt( sig );
+            sighandler = _PDCLIB_sigabrt;
+            message = "Abnormal termination (SIGABRT)";
             break;
         case SIGFPE:
-            _PDCLIB_sigfpe( sig );
+            sighandler = _PDCLIB_sigfpe;
+            message = "Arithmetic exception (SIGFPE)";
             break;
         case SIGILL:
-            _PDCLIB_sigill( sig );
+            sighandler = _PDCLIB_sigill;
+            message = "Illegal instruction (SIGILL)";
             break;
         case SIGINT:
-            _PDCLIB_sigint( sig );
+            sighandler = _PDCLIB_sigint;
+            message = "Interactive attention signal (SIGINT)";
             break;
         case SIGSEGV:
-            _PDCLIB_sigsegv( sig );
+            sighandler = _PDCLIB_sigsegv;
+            message = "Invalid memory access (SIGSEGV)";
             break;
         case SIGTERM:
-            _PDCLIB_sigterm( sig );
+            sighandler = _PDCLIB_sigterm;
+            message = "Termination request (SIGTERM)";
             break;
         default:
-            /* TODO: Implement. */
-            break;
+            fprintf( stderr, "Unknown signal #%d\n", sig );
+            _Exit( EXIT_FAILURE );
+    }
+    if ( sighandler == SIG_DFL )
+    {
+        fputs( message, stderr );
+        _Exit( EXIT_FAILURE );
+    }
+    else if ( sighandler != SIG_IGN )
+    {
+        sighandler = signal( sig, SIG_DFL );
+        sighandler( sig );
     }
-    /* TODO: Implement. */
     return 0;
 }
 
 #endif
 
 #ifdef TEST
-#include <_PDCLIB_test.h>
+#include "_PDCLIB_test.h"
+
+#include <stdlib.h>
+
+static volatile sig_atomic_t flag = 0;
+
+static int expected_signal = 0;
+
+static void test_handler( int sig )
+{
+    TESTCASE( sig == expected_signal );
+    flag = 1;
+}
 
 int main( void )
 {
-    TESTCASE( NO_TESTDRIVER );
+    /* Could be other than SIG_DFL if you changed the implementation. */
+    TESTCASE( signal( SIGABRT, SIG_IGN ) == SIG_DFL );
+    /* Should be ignored. */
+    TESTCASE( raise( SIGABRT ) == 0 );
+    /* Installing test handler, old handler should be returned */
+    TESTCASE( signal( SIGABRT, test_handler ) == SIG_IGN );
+    /* Raising and checking SIGABRT */
+    expected_signal = SIGABRT;
+    TESTCASE( raise( SIGABRT ) == 0 );
+    TESTCASE( flag == 1 );
+    /* Re-installing test handler, should have been reset to default */
+    /* Could be other than SIG_DFL if you changed the implementation. */
+    TESTCASE( signal( SIGABRT, test_handler ) == SIG_DFL );
+    /* Raising and checking SIGABRT */
+    flag = 0;
+    TESTCASE( raise( SIGABRT ) == 0 );
+    TESTCASE( flag == 1 );
+    /* Installing test handler for different signal... */
+    TESTCASE( signal( SIGTERM, test_handler ) == SIG_DFL );
+    /* Raising and checking SIGTERM */
+    expected_signal = SIGTERM;
+    TESTCASE( raise( SIGTERM ) == 0 );
+    TESTCASE( flag == 1 );
     return TEST_RESULTS;
 }
 #endif