From owner-freebsd-hackers Sat Jan 5 13:19:48 2002 Delivered-To: freebsd-hackers@freebsd.org Received: from snipe.prod.itd.earthlink.net (snipe.mail.pas.earthlink.net [207.217.120.62]) by hub.freebsd.org (Postfix) with ESMTP id B4D3037B41F for ; Sat, 5 Jan 2002 13:18:24 -0800 (PST) Received: from pool0211.cvx21-bradley.dialup.earthlink.net ([209.179.192.211] helo=mindspring.com) by snipe.prod.itd.earthlink.net with esmtp (Exim 3.33 #1) id 16MyCm-0005nO-00; Sat, 05 Jan 2002 13:18:17 -0800 Message-ID: <3C376D8D.CBEA1BF@mindspring.com> Date: Sat, 05 Jan 2002 13:18:05 -0800 From: Terry Lambert X-Mailer: Mozilla 4.7 [en]C-CCK-MCD {Sony} (Win98; U) X-Accept-Language: en MIME-Version: 1.0 To: fernando@cursosvirtuales.com.ar Cc: freebsd-hackers@freebsd.org Subject: Re: SA_NODEFER and signal nesting References: <200201052102.g05L2iW67469@servidor1.cursosvirtuales.com.ar> Content-Type: text/plain; charset=us-ascii Content-Transfer-Encoding: 7bit Sender: owner-freebsd-hackers@FreeBSD.ORG Precedence: bulk List-ID: List-Archive: (Web Archive) List-Help: (List Instructions) List-Subscribe: List-Unsubscribe: X-Loop: FreeBSD.ORG "Fernando P. Schapachnik" wrote: > I'm trying to do Async I/O using O_ASYNC on sockets and handling > SIGIO. My testing shows that even if I unblock SIGIO at the begining of the > handler the kernel only delivers one level of nested signals. Ie: while the > first SIGIO is being handled a second might arrive, but a third delivered > signal does not reach the process. Yes. Signals are persistant conditions, not events. > The same happens if I catch the signals with sigaction and specify > SA_NODEFER. Same program on Linux can handle up to 23 nested signals. Depending on Linux's behaviour, which is contrary to conformance to the POSIX standard, will make your code non-portable. > Is this a known behavior? Yes. Even better, it's standard. > Is there any way to change it? When you get the first SIGIO, set a flag (volatile) in the signal handler. In your main loop, check for the flag, and if it is present, use poll/select to verify that there is data pending, and while there is data pending, retrieve it. At most, you will recheck one extra time for a run of N events (N+1 times), and since you are concerned about N >= 23, this overhead is practically non-existant. If you want to avoid the "extra" system call, then switch to using non-blocking I/O, and when the number of bytes processed by the call is 0, then you are done with that iteration. This approach is portable between all versions of UNIX that support select or poll (you can conditionally compile the code based on the presence or absences of FD_SET_SIZE, or other manifest constants associated solely with the poll or select system calls, so the resulting code will be 100% portable source code). See also the source for the "init" program, which has to be able to reap child processes, and so has to do SIGCHLD or SIGCLD (depending on your flavor of UNIX) handling the same way. -- Terry To Unsubscribe: send mail to majordomo@FreeBSD.org with "unsubscribe freebsd-hackers" in the body of the message