From: Pedro Alves <palves@redhat.com>
To: Simon Marchi <simon.marchi@polymtl.ca>, paul@mad-scientist.net
Cc: gdb@sourceware.org
Subject: Re: GDB (not) handling SIGINT...?
Date: Thu, 15 Nov 2018 13:55:00 -0000 [thread overview]
Message-ID: <e4b704f2-58f1-f8d9-1ce1-61f2eaed43ab@redhat.com> (raw)
In-Reply-To: <17a7ce8aa190956bd7a8ba9bd7cdea16@polymtl.ca>
On 11/14/2018 10:32 PM, Simon Marchi wrote:
> On 2018-11-13 17:37, Paul Smith wrote:
>> Hi all; I'm using GDB 8.1 on a modern 64bit GNU/Linux system (Ubuntu
>> 18.04LTS).
>>
>> Recently I added a call to sigtimedwait() for SIGINT into my
>> (multithreaded) program and now I'm having an issue with GDB.
>>
>> What I want is that if I attach to my program then continue, then use
>> ^C at the GDB terminal, I should get a (gdb) prompt back but I do NOT
>> want the SIGINT delivered to my program to wake up my sigtimedwait()
>> call (because it will cause my program to do various things that I
>> don't want it to do).
>>
>> I see that SIGINT is set to nopass:
>>
>> Â (gdb) info signals SIGINT
>>  Signal       Stop     Print  Pass to program Description
>>  SIGINT       Yes      Yes    No             Interrupt
>>
>> but yet when I use ^C at the GDB prompt my sigtimedwait() call inside
>> my program does return with "2" (SIGINT), which I don't want.
>>
>> I guess I don't understand what the docs mean when they say that the
>> signal is not passed to the process under debug. Note that in my case
>> I'm attaching to the program from a completely different terminal so
>> there's no issue with process groups etc. and as far as I can
>> understand it, the signal should only be delivered to GDB not my
>> process, so unless there's some weird magic at work here it must be GDB
>> forwarding that signal to my process.
>>
>> Anyone have any thoughts about this?
>>
>> Cheers!
>
> I was able to reproduce it with, it really sounds like a bug.
>
> If others want to try, here's my test program:
>
> #include <stdio.h>
> #include <signal.h>
> #include <stdlib.h>
>
> int main()
> {
> Â Â Â Â sigset_t set;
> Â Â Â Â sigemptyset(&set);
> Â Â Â Â sigaddset(&set, SIGINT);
> Â Â Â Â sigprocmask(SIG_BLOCK, &set, NULL);
> Â Â Â Â for (int i = 0; i < 10; i++) {
> Â Â Â Â Â Â Â int n = sigwaitinfo(&set, NULL);
> Â Â Â Â Â Â Â printf("signal %d\n", n);
> Â Â Â Â }
> }
>
>
> Run in a terminal, attach with GDB in another terminal, continue, then ctrl-C in GDB's terminal. The first call to sigwaitinfo returns -1, I think it's expected as the syscall gets interrupted. But the subsequent calls return 2, showing that indeed the process has received a SIGINT.
>
> Paul, could you please file a bug? You can re-use this test program if you want.
There are a few things going on here, none of it is really new, though:
#1 - If your program blocks SIGINT and then uses sigwait, then GDB won't
ever intercept the SIGINT, because ptrace doesn't ever see the signal.
This is an ancient issue. See:
https://sourceware.org/bugzilla/show_bug.cgi?id=9425
#2 - If you attach to a process instead of running it from GDB, then ctrl-c
reaches gdb first, and then GDB sends/forwards a SIGINT to the child process.
Normally this then causes ptrace to intercept the SIGINT, but, see #1 above.
This case ("attach"), could be handled by GDB instead stopping the
process with SIGTOP or even better, PTRACE_INTERRUPT. Note, TBC, would
only work with attach/a separate terminal.
#3 - If you run the process instead of attaching, in the same terminal
as GDB, then ctrl-c reaches the inferior process first (because
the inferior is put in the foreground), gdb would have no chance to
do PTRACE_INTERRUPT at all. That would be fixable (combined with
PTRACE_INTERRUPT) by my WIP branch here:
https://github.com/palves/gdb/commits/palves/tty-always-separate-session
which effectively makes the "run" case the same as the "attach" case,
by always putting the inferior in a separate terminal session.
Thanks,
Pedro Alves
next prev parent reply other threads:[~2018-11-15 13:55 UTC|newest]
Thread overview: 10+ messages / expand[flat|nested] mbox.gz Atom feed top
2018-11-13 22:37 Paul Smith
2018-11-14 22:32 ` Simon Marchi
2018-11-15 13:55 ` Pedro Alves [this message]
2018-11-15 16:01 ` Paul Smith
2018-11-15 16:24 ` Pedro Alves
2018-11-15 18:01 ` Paul Smith
2018-11-15 18:04 ` Pedro Alves
2018-11-15 18:27 ` Paul Smith
2018-11-19 19:07 ` Paul Smith
2018-11-19 19:44 ` Paul Smith
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=e4b704f2-58f1-f8d9-1ce1-61f2eaed43ab@redhat.com \
--to=palves@redhat.com \
--cc=gdb@sourceware.org \
--cc=paul@mad-scientist.net \
--cc=simon.marchi@polymtl.ca \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox