Como posso verificar quais sinais um processo está ouvindo?

81

Como posso verificar se um processo em execução captura um sinal, ignora-o ou bloqueia-o? Idealmente, eu gostaria de ver uma lista de sinais, ou pelo menos não ter que realmente enviar o sinal para verificar.

Jander
fonte

Respostas:

109

No Linux, você pode encontrar o PID do seu processo e ver /proc/$PID/status. Ele contém linhas que descrevem quais sinais estão bloqueados (SigBlk), ignorados (SigIgn) ou capturados (SigCgt).

# cat /proc/1/status
...
SigBlk: 0000000000000000
SigIgn: fffffffe57f0d8fc
SigCgt: 00000000280b2603
...

O número à direita é uma máscara de bits. Se você o converter de hex para binário, cada 1 bit representa um sinal capturado, contando da direita para a esquerda começando com 1. Portanto, ao interpretar a linha SigCgt, podemos ver que meu initprocesso está captando os seguintes sinais:

00000000280b2603 ==> 101000000010110010011000000011
                     | |       | ||  |  ||       |`->  1 = SIGHUP
                     | |       | ||  |  ||       `-->  2 = SIGINT
                     | |       | ||  |  |`----------> 10 = SIGUSR1
                     | |       | ||  |  `-----------> 11 = SIGSEGV
                     | |       | ||  `--------------> 14 = SIGALRM
                     | |       | |`-----------------> 17 = SIGCHLD
                     | |       | `------------------> 18 = SIGCONT
                     | |       `--------------------> 20 = SIGTSTP
                     | `----------------------------> 28 = SIGWINCH
                     `------------------------------> 30 = SIGPWR

(Encontrei o mapeamento de número para nome executando kill -ldo bash.)

EDIT : E pela demanda popular, um script, no POSIX sh.

sigparse () {
    i=0
    # bits="$(printf "16i 2o %X p" "0x$1" | dc)" # variant for busybox
    bits="$(printf "ibase=16; obase=2; %X\n" "0x$1" | bc)"
    while [ -n "$bits" ] ; do
        i="$(expr "$i" + 1)"
        case "$bits" in
            *1) printf " %s(%s)" "$(kill -l "$i")" "$i" ;;
        esac
        bits="${bits%?}"
    done
}

grep "^Sig...:" "/proc/$1/status" | while read a b ; do
        printf "%s%s\n" "$a" "$(sigparse "$b")"
    done # | fmt -t  # uncomment for pretty-printing
Jander
fonte
2
Se um sinal estiver listado em SigBlk, ele também aparecerá em SigCgt? Porque, ao bloqueá-lo, isso significa apenas que o sinal será reenviado um pouco mais tarde, certo e precisa ser capturado?
CMCDragonkai
Não, você pode bloquear um sinal sem estar pronto para capturá-lo. Se você não captar um sinal, uma ação padrão ocorrerá dependendo do sinal (geralmente a finalização do processo). Se você quiser mais detalhes, deve abrir uma pergunta.
Jander
Para que serve uma versão POSIX de um script de leitura /proc? Só vai funcionar no Linux ... E localnão é o POSIX. Bem, é meio que, mas seu efeito é "não especificado".
Kusalananda
2
@Kusalananda: Linux não implica Bash - por exemplo, pequenas plataformas embarcadas costumam usar o Busybox - mas a conformidade com POSIX é quase uma garantia para qualquer moderno /bin/sh. Você está certo local; Eu vou limpar isso.
Jander
@Jander Fair point. Confesso que faço uma suposição apressada em relação ao Bash e Linux.
Kusalananda
23

No Solaris, execute psigo ID do processo para obter uma lista de sinais e como eles serão tratados.

Por exemplo:

bash-4.2$ psig $$
11088:  bash
HUP     caught  termsig_sighandler  0   HUP,INT,ILL,TRAP,ABRT,EMT,FPE,BUS,SEGV,SYS,PIPE,ALRM,TERM,USR1,USR2,VTALRM,XCPU,XFSZ,LOST
INT     caught  sigint_sighandler   0
QUIT    ignored
ILL     caught  termsig_sighandler  0   HUP,INT,ILL,TRAP,ABRT,EMT,FPE,BUS,SEGV,SYS,PIPE,ALRM,TERM,USR1,USR2,VTALRM,XCPU,XFSZ,LOST
TRAP    caught  termsig_sighandler  0   HUP,INT,ILL,TRAP,ABRT,EMT,FPE,BUS,SEGV,SYS,PIPE,ALRM,TERM,USR1,USR2,VTALRM,XCPU,XFSZ,LOST
ABRT    caught  termsig_sighandler  0   HUP,INT,ILL,TRAP,ABRT,EMT,FPE,BUS,SEGV,SYS,PIPE,ALRM,TERM,USR1,USR2,VTALRM,XCPU,XFSZ,LOST
EMT     caught  termsig_sighandler  0   HUP,INT,ILL,TRAP,ABRT,EMT,FPE,BUS,SEGV,SYS,PIPE,ALRM,TERM,USR1,USR2,VTALRM,XCPU,XFSZ,LOST
FPE     caught  termsig_sighandler  0   HUP,INT,ILL,TRAP,ABRT,EMT,FPE,BUS,SEGV,SYS,PIPE,ALRM,TERM,USR1,USR2,VTALRM,XCPU,XFSZ,LOST
KILL    default
BUS     caught  termsig_sighandler  0   HUP,INT,ILL,TRAP,ABRT,EMT,FPE,BUS,SEGV,SYS,PIPE,ALRM,TERM,USR1,USR2,VTALRM,XCPU,XFSZ,LOST
SEGV    caught  termsig_sighandler  0   HUP,INT,ILL,TRAP,ABRT,EMT,FPE,BUS,SEGV,SYS,PIPE,ALRM,TERM,USR1,USR2,VTALRM,XCPU,XFSZ,LOST
SYS     caught  termsig_sighandler  0   HUP,INT,ILL,TRAP,ABRT,EMT,FPE,BUS,SEGV,SYS,PIPE,ALRM,TERM,USR1,USR2,VTALRM,XCPU,XFSZ,LOST
PIPE    caught  termsig_sighandler  0   HUP,INT,ILL,TRAP,ABRT,EMT,FPE,BUS,SEGV,SYS,PIPE,ALRM,TERM,USR1,USR2,VTALRM,XCPU,XFSZ,LOST
ALRM    caught  termsig_sighandler  0   HUP,INT,ILL,TRAP,ABRT,EMT,FPE,BUS,SEGV,SYS,PIPE,ALRM,TERM,USR1,USR2,VTALRM,XCPU,XFSZ,LOST
TERM    ignored
USR1    caught  termsig_sighandler  0   HUP,INT,ILL,TRAP,ABRT,EMT,FPE,BUS,SEGV,SYS,PIPE,ALRM,TERM,USR1,USR2,VTALRM,XCPU,XFSZ,LOST
USR2    caught  termsig_sighandler  0   HUP,INT,ILL,TRAP,ABRT,EMT,FPE,BUS,SEGV,SYS,PIPE,ALRM,TERM,USR1,USR2,VTALRM,XCPU,XFSZ,LOST
CLD     blocked,caught  0x4898e8    RESTART
PWR     default
WINCH   caught  sigwinch_sighandler 0
[...]

que mostra que SIGHUP, SIGILL etc. serão capturados pela mesma função de manipulador de sinal termsig_sighandler, que será executada sem o uso de nenhum dos sinalizadores que poderiam ser configurados via sigactione todos os sinais que serão temporariamente mascarados enquanto o manipulador de sinal estiver executando (neste caso, todos usando o mesmo manipulador de sinal, para que ele não seja reinserido enquanto estiver em execução). Você também pode ver que SIGQUIT & SIGTERM serão ignorados, SIGKILL & SIGPWR usam as ações de sinal padrão do sistema e SIGCLD especifica o sinalizador RESTART, portanto, se seu manipulador de sinais interromper uma chamada de sistema, o syscall será reiniciado.

alanc
fonte
Impressionante! Eu esperava que alguém adicionasse uma resposta que não fosse Linux.
Jander
4

(Essa resposta é semelhante à resposta de @ user18096, na medida em que cria um script em torno da resposta de @ Jander.)

Eu escrevi a psig scriptpara pegar um PID (ou todos os PIDs) e criar uma saída legível por humanos a partir das máscaras de sinal /proc/<PID>/status.

Exemplo de saída:

% ./psig -a
[     1] Signals Queued: 8/773737
[     1] Signals Pending:
[     1] Signals Pending (Shared):
[     1] Signals Blocked:
[     1] Signals Ignored: SIGPIPE
[     1] Signals Caught: SIGHUP,SIGINT,SIGABRT,SIGUSR1,SIGSEGV,SIGALRM,SIGTERM,SIGCHLD,SIGPWR
...
[ 31001] Signals Queued: 0/773737
[ 31001] Signals Pending:
[ 31001] Signals Pending (Shared):
[ 31001] Signals Blocked: SIGHUP,SIGINT,SIGQUIT,SIGILL,SIGTRAP,SIGABRT,SIGBUS,SIGFPE,SIGUSR1,SIGUSR2,SIGPIPE,SIGALRM,SIGTERM,SIGSTKFLT,SIGCHLD,SIGCONT,SIGTSTP,SIGTTIN,SIGTTOU,SIGURG,SIGXCPU,SIGXFSZ,SIGPROF,SIGWINCH,SIGIO,SIGPWR,SIGSYS,SIGRTMIN,SIGRTMIN+1,SIGRTMIN+2,SIGRTMIN+3,SIGRTMIN+4,SIGRTMIN+5,SIGRTMIN+6,SIGRTMIN+7,SIGRTMIN+8,SIGRTMIN+9,SIGRTMIN+10,SIGRTMIN+11,SIGRTMIN+12,SIGRTMIN+13,SIGRTMIN+14,SIGRTMIN+15,SIGRTMAX-14,SIGRTMAX-13,SIGRTMAX-12,SIGRTMAX-11,SIGRTMAX-10,SIGRTMAX-9,SIGRTMAX-8,SIGRTMAX-7,SIGRTMAX-6,SIGRTMAX-5,SIGRTMAX-4,SIGRTMAX-3,SIGRTMAX-2,SIGRTMAX-1,SIGRTMAX
[ 31001] Signals Ignored: SIGHUP,SIGINT,SIGQUIT,SIGPIPE,SIGXFSZ
[ 31001] Signals Caught: SIGBUS,SIGUSR1,SIGSEGV,SIGUSR2,SIGALRM,SIGTERM,SIGVTALRM

Ressalvas:

  • Esta é uma resposta específica do Linux.
  • Pode precisar de uma versão Python relativamente nova para executar o script, ele usa withe OrderedDict.
erik.weathers
fonte
2

Continuo voltando à bonita resposta do @ Jander, esperando um decodificador de copiar e colar quando confrontado com coisas como:

user@machine:~$ grep Sig...: /proc/18475/status
SigPnd: 0000000000000000
SigBlk: fffffffe7dfbfaff
SigIgn: 0000000000001000
SigCgt: 0000000182006e47
user@machine:~$ 

Acho que vou ter que bater alguma coisa ... diga:

user@machine:~$ ruby -wn - /proc/18475/status <<'EOF'
if $_.match(/Sig(Pnd|Blk|Ign|Cgt):\s([0-9a-f]{16})/) == nil
  next
end
field = $1
mask = $2.to_i(16)
names = []
Signal.list().each_pair() {
  |name, number|
  if number == 0
    # "EXIT" => 0
    next
  end
  if (mask & (1 << (number - 1))) == 0
    next
  end
  names << name
}
puts("Sig#{field}: #{names.join(" | ")}")
EOF
SigPnd: 
SigBlk: HUP | INT | QUIT | ILL | TRAP | IOT | ABRT | FPE | BUS | SYS | PIPE | ALRM | TERM | URG | TSTP | CONT | CHLD | CLD | TTIN | TTOU | IO | XCPU | XFSZ | PROF | WINCH | USR1 | USR2 | PWR | POLL
SigIgn: PIPE
SigCgt: HUP | INT | QUIT | BUS | SEGV | ALRM | TERM | VTALRM | USR1 | USR2
user@machine:~$ 

Eu queria que fosse um pouco legível, mas isso tornou um pouco mais desajeitado invocar do que eu gostaria, então, graças à sugestão de @ alanc, vou salvá-lo como ~ / bin / psig.

Martin Dorey
fonte
2

Usar esta(link quebrado) nesta biblioteca para obter informações sobre os trabalhos em execução.

Existe um campo especial struct Jobpara os sinais, chamadosigCgt

Você pode usar algo como isto:

#include"read_proc.h"
int main(void)
{
    struct Root * rt=read_proc();
    struct Job * jb=rt->first->job;
    printf("%ull\n",jb->sigCgt);
    return 0;
}
LittleByBlue
fonte
Eu adoraria, mas o link está quebrado.
Michael Fox
1
@MichaelFox see my edit. O usuário excluiu sua conta. O novo link aponta para o mesmo projeto
LittleByBlue 7/17
1

No FreeBSD, use procstat -i <PID>para ver quais sinais são ignorados pelo processo. Da mesma forma, procstat -j <PID>para ver quais sinais estão bloqueados pelos encadeamentos do processo. Ambos os comandos mostram se um sinal está pendente.

Saída de amostra:

$ procstat -i 38540 PID COMM SIG FLAGS 38540 nsulfd HUP -I- 38540 nsulfd INT -I- 38540 nsulfd QUIT -I- 38540 nsulfd ILL --- 38540 nsulfd TRAP --- ...

$ procstat -j 38540 PID TID COMM SIG FLAGS 38540 101220 nsulfd HUP -- 38540 101220 nsulfd INT -- 38540 101220 nsulfd QUIT -B 38540 101220 nsulfd ILL -- 38540 101220 nsulfd TRAP -- ...

Veja procstat (1) .

Deepak
fonte