diff options
Diffstat (limited to 'erts/emulator/sys/unix/sys.c')
| -rw-r--r-- | erts/emulator/sys/unix/sys.c | 80 | 
1 files changed, 76 insertions, 4 deletions
| diff --git a/erts/emulator/sys/unix/sys.c b/erts/emulator/sys/unix/sys.c index 0d1ed17449..f80f136d79 100644 --- a/erts/emulator/sys/unix/sys.c +++ b/erts/emulator/sys/unix/sys.c @@ -139,6 +139,28 @@ volatile int erts_break_requested = 0;  #define ERTS_SET_BREAK_REQUESTED (erts_break_requested = 1)  #define ERTS_UNSET_BREAK_REQUESTED (erts_break_requested = 0)  #endif + +#ifndef ERTS_SMP +static Eterm signalstate_sigterm[] = { +    am_sigint,   /* 0 */ +    am_sighup,   /* 1 */ +    am_sigquit,  /* 2 */ +    am_sigabrt,  /* 3 */ +    am_sigalrm,  /* 4 */ +    am_sigterm,  /* 5 */ +    am_sigusr1,  /* 6 */ +    am_sigusr2,  /* 7 */ +    am_sigchld,  /* 8 */ +    am_sigstop,  /* 9 */ +    am_sigtstp   /* 10 */ +}; + +volatile Uint erts_signal_state = 0; +#define ERTS_SET_SIGNAL_STATE(S) (erts_signal_state |= signum_to_signalstate(S)) +#define ERTS_CLEAR_SIGNAL_STATE (erts_signal_state = 0) +static ERTS_INLINE Uint signum_to_signalstate(int signum); +#endif +  /* set early so the break handler has access to initial mode */  static struct termios initial_tty_mode;  static int replace_intr = 0; @@ -276,6 +298,18 @@ erts_sys_schedule_interrupt_timed(int set, ErtsMonotonicTime timeout_time)  }  #endif +UWord +erts_sys_get_page_size(void) +{ +#if defined(_SC_PAGESIZE) +    return (UWord) sysconf(_SC_PAGESIZE); +#elif defined(HAVE_GETPAGESIZE) +    return (UWord) getpagesize(); +#else +    return (UWord) 4*1024; /* Guess 4 KB */ +#endif +} +  Uint  erts_sys_misc_mem_sz(void)  { @@ -770,13 +804,34 @@ signum_to_signalterm(int signum)      }  } +#ifndef ERTS_SMP +static ERTS_INLINE Uint +signum_to_signalstate(int signum) +{ +    switch (signum) { +    case SIGINT:  return (1 <<  0); +    case SIGHUP:  return (1 <<  1); +    case SIGQUIT: return (1 <<  2); +    case SIGABRT: return (1 <<  3); +    case SIGALRM: return (1 <<  4); +    case SIGTERM: return (1 <<  5); +    case SIGUSR1: return (1 <<  6); +    case SIGUSR2: return (1 <<  7); +    case SIGCHLD: return (1 <<  8); +    case SIGSTOP: return (1 <<  9); +    case SIGTSTP: return (1 << 10); +    default:      return 0; +    } +} +#endif +  static RETSIGTYPE generic_signal_handler(int signum)  {  #ifdef ERTS_SMP      smp_sig_notify(signum);  #else -    Eterm signal = signum_to_signalterm(signum); -    signal_notify_requested(signal); +    ERTS_SET_SIGNAL_STATE(signum); +    ERTS_CHK_IO_AS_INTR(); /* Make sure we don't sleep in poll */  #endif  } @@ -962,6 +1017,23 @@ void erts_do_break_handling(void)      erts_smp_thr_progress_unblock();  } +#ifdef ERTS_SIGNAL_STATE +void erts_handle_signal_state(void) { +    Uint signal_state = ERTS_SIGNAL_STATE; +    Uint i = 0; + +    ERTS_CLEAR_SIGNAL_STATE; + +    while (signal_state) { +        if (signal_state & 0x1) { +            signal_notify_requested(signalstate_sigterm[i]); +        } +        i++; +        signal_state = signal_state >> 1; +    } +} +#endif +  /* Fills in the systems representation of the jam/beam process identifier.  ** The Pid is put in STRING representation in the supplied buffer,  ** no interpretatione of this should be done by the rest of the @@ -1274,8 +1346,8 @@ signal_dispatcher_thread_func(void *unused)          do {              res = read(sig_notify_fds[0], (void *) &sb.buf[i], sizeof(int) - i); -            i += res; -        } while ((i != sizeof(int) && res >= 0) || (res < 0 && errno == EINTR)); +            i += res > 0 ? res : 0; +        } while ((i < sizeof(int) && res >= 0) || (res < 0 && errno == EINTR));  	if (res < 0) {  	    erts_exit(ERTS_ABORT_EXIT, | 
