<p dir="ltr"><br>
On Jan 30, 2013 8:05 AM, &quot;Andrey Vagin&quot; &lt;<a href="mailto:avagin@openvz.org">avagin@openvz.org</a>&gt; wrote:<br>
&gt;<br>
&gt; If signalfd is created with the flag SFD_PEEK, it reads siginfo-s<br>
&gt; without dequeuing signals.<br>
&gt;<br>
&gt; For reading not first siginfo pread(fd, buf, size, pos) can be used,<br>
&gt; where ppos / sizeof(signalfd_siginfo) is a sequence number of a signal<br>
&gt; in a queue.</p>
<p dir="ltr">Andrey,</p>
<p dir="ltr">Is it perhaps worth erroring (EINVAL) if  ((ppos % sizeof (signalfd_siginfo)) != 0) ?</p>
<p dir="ltr">Cheers,</p>
<p dir="ltr">Michael</p>
<p dir="ltr">&gt; This functionality is required for checkpointing pending signals.<br>
&gt;<br>
&gt; v2: * signals can be dumped only from one queue.<br>
&gt;     * treat pos as offset in bytes, not in elements, so pos should be<br>
&gt;       aligned to the size of signalfd_siginfo.<br>
&gt;<br>
&gt; Cc: Oleg Nesterov &lt;<a href="mailto:oleg@redhat.com">oleg@redhat.com</a>&gt;<br>
&gt; Cc: Alexander Viro &lt;<a href="mailto:viro@zeniv.linux.org.uk">viro@zeniv.linux.org.uk</a>&gt;<br>
&gt; Cc: &quot;Paul E. McKenney&quot; &lt;<a href="mailto:paulmck@linux.vnet.ibm.com">paulmck@linux.vnet.ibm.com</a>&gt;<br>
&gt; Cc: David Howells &lt;<a href="mailto:dhowells@redhat.com">dhowells@redhat.com</a>&gt;<br>
&gt; Cc: Dave Jones &lt;<a href="mailto:davej@redhat.com">davej@redhat.com</a>&gt;<br>
&gt; Cc: Andrey Vagin &lt;<a href="mailto:avagin@openvz.org">avagin@openvz.org</a>&gt;<br>
&gt; Cc: Michael Kerrisk &lt;<a href="mailto:mtk.manpages@gmail.com">mtk.manpages@gmail.com</a>&gt;<br>
&gt; Cc: Pavel Emelyanov &lt;<a href="mailto:xemul@parallels.com">xemul@parallels.com</a>&gt;<br>
&gt; CC: Cyrill Gorcunov &lt;<a href="mailto:gorcunov@openvz.org">gorcunov@openvz.org</a>&gt;<br>
&gt; Signed-off-by: Andrey Vagin &lt;<a href="mailto:avagin@openvz.org">avagin@openvz.org</a>&gt;<br>
&gt; ---<br>
&gt;  fs/signalfd.c                 | 61 ++++++++++++++++++++++++++++++++++++++++---<br>
&gt;  include/uapi/linux/signalfd.h |  2 ++<br>
&gt;  2 files changed, 60 insertions(+), 3 deletions(-)<br>
&gt;<br>
&gt; diff --git a/fs/signalfd.c b/fs/signalfd.c<br>
&gt; index 8019ec9..0da6a30 100644<br>
&gt; --- a/fs/signalfd.c<br>
&gt; +++ b/fs/signalfd.c<br>
&gt; @@ -51,6 +51,47 @@ struct signalfd_ctx {<br>
&gt;         sigset_t sigmask;<br>
&gt;  };<br>
&gt;<br>
&gt; +static int peek_signal(struct sigpending *pending, sigset_t *mask,<br>
&gt; +                               siginfo_t *info, loff_t *pseq)<br>
&gt; +{<br>
&gt; +       struct sigqueue *q;<br>
&gt; +       int ret = 0;<br>
&gt; +<br>
&gt; +       spin_lock_irq(&amp;current-&gt;sighand-&gt;siglock);<br>
&gt; +<br>
&gt; +       list_for_each_entry(q, &amp;pending-&gt;list, list) {<br>
&gt; +               if (sigismember(mask, q-&gt;info.si_signo))<br>
&gt; +                       continue;<br>
&gt; +<br>
&gt; +               if ((*pseq)-- == 0) {<br>
&gt; +                       copy_siginfo(info, &amp;q-&gt;info);<br>
&gt; +                       ret = info-&gt;si_signo;<br>
&gt; +                       break;<br>
&gt; +               }<br>
&gt; +       }<br>
&gt; +<br>
&gt; +       spin_unlock_irq(&amp;current-&gt;sighand-&gt;siglock);<br>
&gt; +<br>
&gt; +       return ret;<br>
&gt; +}<br>
&gt; +<br>
&gt; +static ssize_t signalfd_peek(struct signalfd_ctx *ctx,<br>
&gt; +                               siginfo_t *info, loff_t *ppos, int queue_mask)<br>
&gt; +{<br>
&gt; +       loff_t seq = *ppos / sizeof(struct signalfd_siginfo);<br>
&gt; +       int signr = 0;<br>
&gt; +<br>
&gt; +       if (queue_mask &amp; SIGQUEUE_PRIVATE)<br>
&gt; +               signr = peek_signal(&amp;current-&gt;pending,<br>
&gt; +                                       &amp;ctx-&gt;sigmask, info, &amp;seq);<br>
&gt; +       else if (queue_mask &amp; SIGQUEUE_SHARED)<br>
&gt; +               signr = peek_signal(&amp;current-&gt;signal-&gt;shared_pending,<br>
&gt; +                                        &amp;ctx-&gt;sigmask, info, &amp;seq);<br>
&gt; +       (*ppos) += sizeof(struct signalfd_siginfo);<br>
&gt; +<br>
&gt; +       return signr;<br>
&gt; +}<br>
&gt; +<br>
&gt;  static int signalfd_release(struct inode *inode, struct file *file)<br>
&gt;  {<br>
&gt;         kfree(file-&gt;private_data);<br>
&gt; @@ -257,9 +298,15 @@ static ssize_t signalfd_read(struct file *file, char __user *buf, size_t count,<br>
&gt;         if (!count)<br>
&gt;                 return -EINVAL;<br>
&gt;<br>
&gt; +       if (*ppos % sizeof(struct signalfd_siginfo))<br>
&gt; +               return -EINVAL;<br>
&gt; +<br>
&gt;         siginfo = (struct signalfd_siginfo __user *) buf;<br>
&gt;         do {<br>
&gt; -               ret = signalfd_dequeue(ctx, &amp;info, nonblock, qmask);<br>
&gt; +               if (file-&gt;f_flags &amp; SFD_PEEK)<br>
&gt; +                       ret = signalfd_peek(ctx, &amp;info, ppos, qmask);<br>
&gt; +               else<br>
&gt; +                       ret = signalfd_dequeue(ctx, &amp;info, nonblock, qmask);<br>
&gt;<br>
&gt;                 if (unlikely(ret &lt;= 0))<br>
&gt;                         break;<br>
&gt; @@ -315,7 +362,12 @@ SYSCALL_DEFINE4(signalfd4, int, ufd, sigset_t __user *, user_mask,<br>
&gt;         BUILD_BUG_ON(SFD_CLOEXEC != O_CLOEXEC);<br>
&gt;         BUILD_BUG_ON(SFD_NONBLOCK != O_NONBLOCK);<br>
&gt;<br>
&gt; -       if (flags &amp; ~(SFD_CLOEXEC | SFD_NONBLOCK | SFD_RAW | SFD_QUEUES))<br>
&gt; +       if (flags &amp; ~(SFD_CLOEXEC | SFD_NONBLOCK |<br>
&gt; +                       SFD_RAW | SFD_PEEK | SFD_QUEUES))<br>
&gt; +               return -EINVAL;<br>
&gt; +<br>
&gt; +       /* SFD_PEEK can be used for one queue only */<br>
&gt; +       if ((flags &amp; SFD_PEEK) &amp;&amp; ((flags &amp; SFD_QUEUES) == SFD_QUEUES))<br>
&gt;                 return -EINVAL;<br>
&gt;<br>
&gt;         if (sizemask != sizeof(sigset_t) ||<br>
&gt; @@ -352,7 +404,10 @@ SYSCALL_DEFINE4(signalfd4, int, ufd, sigset_t __user *, user_mask,<br>
&gt;                 }<br>
&gt;<br>
&gt;                 file-&gt;f_flags |= (flags &amp; SFD_QUEUES) ? : SFD_QUEUES;<br>
&gt; -               file-&gt;f_flags |= flags &amp; SFD_RAW;<br>
&gt; +               file-&gt;f_flags |= flags &amp; (SFD_RAW | SFD_PEEK);<br>
&gt; +<br>
&gt; +               if (file-&gt;f_flags &amp; SFD_PEEK)<br>
&gt; +                       file-&gt;f_mode |= FMODE_PREAD;<br>
&gt;<br>
&gt;                 fd_install(ufd, file);<br>
&gt;         } else {<br>
&gt; diff --git a/include/uapi/linux/signalfd.h b/include/uapi/linux/signalfd.h<br>
&gt; index 481b658..24c5d2d 100644<br>
&gt; --- a/include/uapi/linux/signalfd.h<br>
&gt; +++ b/include/uapi/linux/signalfd.h<br>
&gt; @@ -20,6 +20,8 @@<br>
&gt;  #define SFD_SHARED_QUEUE O_DIRECTORY<br>
&gt;  /* Read signals from a per-thread queue */<br>
&gt;  #define SFD_PER_THREAD_QUEUE O_EXCL<br>
&gt; +/* Read signals without removing them from a queue */<br>
&gt; +#define SFD_PEEK O_APPEND<br>
&gt;<br>
&gt;  struct signalfd_siginfo {<br>
&gt;         __u32 ssi_signo;<br>
&gt; --<br>
&gt; 1.7.11.7<br>
&gt;<br>
</p>