Blame view

fs/signalfd.c 8.96 KB
b24413180   Greg Kroah-Hartman   License cleanup: ...
1
  // SPDX-License-Identifier: GPL-2.0
fba2afaae   Davide Libenzi   signal/timer/even...
2
3
4
5
6
7
8
9
10
11
12
13
14
  /*
   *  fs/signalfd.c
   *
   *  Copyright (C) 2003  Linus Torvalds
   *
   *  Mon Mar 5, 2007: Davide Libenzi <davidel@xmailserver.org>
   *      Changed ->read() to return a siginfo strcture instead of signal number.
   *      Fixed locking in ->poll().
   *      Added sighand-detach notification.
   *      Added fd re-use in sys_signalfd() syscall.
   *      Now using anonymous inode source.
   *      Thanks to Oleg Nesterov for useful code review and suggestions.
   *      More comments and suggestions from Arnd Bergmann.
b8fceee17   Davide Libenzi   signalfd simplifi...
15
   *  Sat May 19, 2007: Davi E. M. Arnaut <davi@haxent.com.br>
b3762bfc8   Davi Arnaut   signalfd: retriev...
16
   *      Retrieve multiple signals with one read() call
b8fceee17   Davide Libenzi   signalfd simplifi...
17
18
   *  Sun Jul 15, 2007: Davide Libenzi <davidel@xmailserver.org>
   *      Attach to the sighand only during read() and poll().
fba2afaae   Davide Libenzi   signal/timer/even...
19
20
21
22
23
24
25
   */
  
  #include <linux/file.h>
  #include <linux/poll.h>
  #include <linux/init.h>
  #include <linux/fs.h>
  #include <linux/sched.h>
5a0e3ad6a   Tejun Heo   include cleanup: ...
26
  #include <linux/slab.h>
fba2afaae   Davide Libenzi   signal/timer/even...
27
28
29
30
31
  #include <linux/kernel.h>
  #include <linux/signal.h>
  #include <linux/list.h>
  #include <linux/anon_inodes.h>
  #include <linux/signalfd.h>
7ec37dfd4   Adrian Bunk   fs/signalfd.c sho...
32
  #include <linux/syscalls.h>
138d22b58   Cyrill Gorcunov   fs, epoll: add pr...
33
  #include <linux/proc_fs.h>
7d197ed4a   Al Viro   switch signalfd{,...
34
  #include <linux/compat.h>
fba2afaae   Davide Libenzi   signal/timer/even...
35

d80e731ec   Oleg Nesterov   epoll: introduce ...
36
37
38
  void signalfd_cleanup(struct sighand_struct *sighand)
  {
  	wait_queue_head_t *wqh = &sighand->signalfd_wqh;
971316f05   Oleg Nesterov   epoll: ep_unregis...
39
40
41
  	/*
  	 * The lockless check can race with remove_wait_queue() in progress,
  	 * but in this case its caller should run under rcu_read_lock() and
5f0d5a3ae   Paul E. McKenney   mm: Rename SLAB_D...
42
  	 * sighand_cachep is SLAB_TYPESAFE_BY_RCU, we can safely return.
971316f05   Oleg Nesterov   epoll: ep_unregis...
43
  	 */
d80e731ec   Oleg Nesterov   epoll: introduce ...
44
45
  	if (likely(!waitqueue_active(wqh)))
  		return;
ac6424b98   Ingo Molnar   sched/wait: Renam...
46
  	/* wait_queue_entry_t->func(POLLFREE) should do remove_wait_queue() */
a9a08845e   Linus Torvalds   vfs: do bulk POLL...
47
  	wake_up_poll(wqh, EPOLLHUP | POLLFREE);
d80e731ec   Oleg Nesterov   epoll: introduce ...
48
  }
fba2afaae   Davide Libenzi   signal/timer/even...
49
  struct signalfd_ctx {
fba2afaae   Davide Libenzi   signal/timer/even...
50
  	sigset_t sigmask;
fba2afaae   Davide Libenzi   signal/timer/even...
51
  };
fba2afaae   Davide Libenzi   signal/timer/even...
52
53
  static int signalfd_release(struct inode *inode, struct file *file)
  {
b8fceee17   Davide Libenzi   signalfd simplifi...
54
  	kfree(file->private_data);
fba2afaae   Davide Libenzi   signal/timer/even...
55
56
  	return 0;
  }
076ccb76e   Al Viro   fs: annotate ->po...
57
  static __poll_t signalfd_poll(struct file *file, poll_table *wait)
fba2afaae   Davide Libenzi   signal/timer/even...
58
59
  {
  	struct signalfd_ctx *ctx = file->private_data;
076ccb76e   Al Viro   fs: annotate ->po...
60
  	__poll_t events = 0;
fba2afaae   Davide Libenzi   signal/timer/even...
61

b8fceee17   Davide Libenzi   signalfd simplifi...
62
  	poll_wait(file, &current->sighand->signalfd_wqh, wait);
fba2afaae   Davide Libenzi   signal/timer/even...
63

b8fceee17   Davide Libenzi   signalfd simplifi...
64
65
66
67
  	spin_lock_irq(&current->sighand->siglock);
  	if (next_signal(&current->pending, &ctx->sigmask) ||
  	    next_signal(&current->signal->shared_pending,
  			&ctx->sigmask))
a9a08845e   Linus Torvalds   vfs: do bulk POLL...
68
  		events |= EPOLLIN;
b8fceee17   Davide Libenzi   signalfd simplifi...
69
  	spin_unlock_irq(&current->sighand->siglock);
fba2afaae   Davide Libenzi   signal/timer/even...
70
71
72
73
74
75
76
77
78
79
  
  	return events;
  }
  
  /*
   * Copied from copy_siginfo_to_user() in kernel/signal.c
   */
  static int signalfd_copyinfo(struct signalfd_siginfo __user *uinfo,
  			     siginfo_t const *kinfo)
  {
5611f55ee   Eric W. Biederman   signal/signalfd: ...
80
  	struct signalfd_siginfo new;
fba2afaae   Davide Libenzi   signal/timer/even...
81
82
83
84
  
  	BUILD_BUG_ON(sizeof(struct signalfd_siginfo) != 128);
  
  	/*
14e4a0f2b   Robert P. J. Day   Fix a small numbe...
85
  	 * Unused members should be zero ...
fba2afaae   Davide Libenzi   signal/timer/even...
86
  	 */
5611f55ee   Eric W. Biederman   signal/signalfd: ...
87
  	memset(&new, 0, sizeof(new));
fba2afaae   Davide Libenzi   signal/timer/even...
88
89
90
91
92
  
  	/*
  	 * If you change siginfo_t structure, please be sure
  	 * this code is fixed accordingly.
  	 */
5611f55ee   Eric W. Biederman   signal/signalfd: ...
93
94
95
  	new.ssi_signo = kinfo->si_signo;
  	new.ssi_errno = kinfo->si_errno;
  	new.ssi_code  = kinfo->si_code;
cc731525f   Eric W. Biederman   signal: Remove ke...
96
97
  	switch (siginfo_layout(kinfo->si_signo, kinfo->si_code)) {
  	case SIL_KILL:
5611f55ee   Eric W. Biederman   signal/signalfd: ...
98
99
  		new.ssi_pid = kinfo->si_pid;
  		new.ssi_uid = kinfo->si_uid;
fba2afaae   Davide Libenzi   signal/timer/even...
100
  		break;
cc731525f   Eric W. Biederman   signal: Remove ke...
101
  	case SIL_TIMER:
5611f55ee   Eric W. Biederman   signal/signalfd: ...
102
103
104
105
  		new.ssi_tid = kinfo->si_tid;
  		new.ssi_overrun = kinfo->si_overrun;
  		new.ssi_ptr = (long) kinfo->si_ptr;
  		new.ssi_int = kinfo->si_int;
fba2afaae   Davide Libenzi   signal/timer/even...
106
  		break;
cc731525f   Eric W. Biederman   signal: Remove ke...
107
  	case SIL_POLL:
5611f55ee   Eric W. Biederman   signal/signalfd: ...
108
109
  		new.ssi_band = kinfo->si_band;
  		new.ssi_fd   = kinfo->si_fd;
fba2afaae   Davide Libenzi   signal/timer/even...
110
  		break;
31931c93d   Eric W. Biederman   signal: Extend si...
111
112
113
114
115
116
117
118
119
  	case SIL_FAULT_BNDERR:
  	case SIL_FAULT_PKUERR:
  		/*
  		 * Fall through to the SIL_FAULT case.  Both SIL_FAULT_BNDERR
  		 * and SIL_FAULT_PKUERR are only generated by faults that
  		 * deliver them synchronously to userspace.  In case someone
  		 * injects one of these signals and signalfd catches it treat
  		 * it as SIL_FAULT.
  		 */
cc731525f   Eric W. Biederman   signal: Remove ke...
120
  	case SIL_FAULT:
5611f55ee   Eric W. Biederman   signal/signalfd: ...
121
  		new.ssi_addr = (long) kinfo->si_addr;
fba2afaae   Davide Libenzi   signal/timer/even...
122
  #ifdef __ARCH_SI_TRAPNO
5611f55ee   Eric W. Biederman   signal/signalfd: ...
123
  		new.ssi_trapno = kinfo->si_trapno;
fba2afaae   Davide Libenzi   signal/timer/even...
124
  #endif
31931c93d   Eric W. Biederman   signal: Extend si...
125
126
127
128
129
130
131
  		break;
  	case SIL_FAULT_MCEERR:
  		new.ssi_addr = (long) kinfo->si_addr;
  #ifdef __ARCH_SI_TRAPNO
  		new.ssi_trapno = kinfo->si_trapno;
  #endif
  		new.ssi_addr_lsb = (short) kinfo->si_addr_lsb;
fba2afaae   Davide Libenzi   signal/timer/even...
132
  		break;
cc731525f   Eric W. Biederman   signal: Remove ke...
133
  	case SIL_CHLD:
5611f55ee   Eric W. Biederman   signal/signalfd: ...
134
135
136
137
138
  		new.ssi_pid    = kinfo->si_pid;
  		new.ssi_uid    = kinfo->si_uid;
  		new.ssi_status = kinfo->si_status;
  		new.ssi_utime  = kinfo->si_utime;
  		new.ssi_stime  = kinfo->si_stime;
fba2afaae   Davide Libenzi   signal/timer/even...
139
  		break;
cc731525f   Eric W. Biederman   signal: Remove ke...
140
  	case SIL_RT:
0859ab59a   Davide Libenzi   signalfd: fix for...
141
142
143
  		/*
  		 * This case catches also the signals queued by sigqueue().
  		 */
5611f55ee   Eric W. Biederman   signal/signalfd: ...
144
145
146
147
  		new.ssi_pid = kinfo->si_pid;
  		new.ssi_uid = kinfo->si_uid;
  		new.ssi_ptr = (long) kinfo->si_ptr;
  		new.ssi_int = kinfo->si_int;
fba2afaae   Davide Libenzi   signal/timer/even...
148
  		break;
76b7f6707   Eric W. Biederman   signal/signalfd: ...
149
150
151
152
153
  	case SIL_SYS:
  		new.ssi_call_addr = (long) kinfo->si_call_addr;
  		new.ssi_syscall   = kinfo->si_syscall;
  		new.ssi_arch      = kinfo->si_arch;
  		break;
fba2afaae   Davide Libenzi   signal/timer/even...
154
  	}
5611f55ee   Eric W. Biederman   signal/signalfd: ...
155
156
157
158
  	if (copy_to_user(uinfo, &new, sizeof(struct signalfd_siginfo)))
  		return -EFAULT;
  
  	return sizeof(*uinfo);
fba2afaae   Davide Libenzi   signal/timer/even...
159
  }
b3762bfc8   Davi Arnaut   signalfd: retriev...
160
161
162
163
  static ssize_t signalfd_dequeue(struct signalfd_ctx *ctx, siginfo_t *info,
  				int nonblock)
  {
  	ssize_t ret;
b3762bfc8   Davi Arnaut   signalfd: retriev...
164
  	DECLARE_WAITQUEUE(wait, current);
b8fceee17   Davide Libenzi   signalfd simplifi...
165
166
  	spin_lock_irq(&current->sighand->siglock);
  	ret = dequeue_signal(current, &ctx->sigmask, info);
b3762bfc8   Davi Arnaut   signalfd: retriev...
167
168
169
170
171
172
  	switch (ret) {
  	case 0:
  		if (!nonblock)
  			break;
  		ret = -EAGAIN;
  	default:
b8fceee17   Davide Libenzi   signalfd simplifi...
173
  		spin_unlock_irq(&current->sighand->siglock);
b3762bfc8   Davi Arnaut   signalfd: retriev...
174
175
  		return ret;
  	}
b8fceee17   Davide Libenzi   signalfd simplifi...
176
  	add_wait_queue(&current->sighand->signalfd_wqh, &wait);
b3762bfc8   Davi Arnaut   signalfd: retriev...
177
178
  	for (;;) {
  		set_current_state(TASK_INTERRUPTIBLE);
b8fceee17   Davide Libenzi   signalfd simplifi...
179
  		ret = dequeue_signal(current, &ctx->sigmask, info);
b3762bfc8   Davi Arnaut   signalfd: retriev...
180
181
182
183
184
185
  		if (ret != 0)
  			break;
  		if (signal_pending(current)) {
  			ret = -ERESTARTSYS;
  			break;
  		}
b8fceee17   Davide Libenzi   signalfd simplifi...
186
  		spin_unlock_irq(&current->sighand->siglock);
b3762bfc8   Davi Arnaut   signalfd: retriev...
187
  		schedule();
b8fceee17   Davide Libenzi   signalfd simplifi...
188
  		spin_lock_irq(&current->sighand->siglock);
b3762bfc8   Davi Arnaut   signalfd: retriev...
189
  	}
b8fceee17   Davide Libenzi   signalfd simplifi...
190
  	spin_unlock_irq(&current->sighand->siglock);
b3762bfc8   Davi Arnaut   signalfd: retriev...
191

b8fceee17   Davide Libenzi   signalfd simplifi...
192
  	remove_wait_queue(&current->sighand->signalfd_wqh, &wait);
b3762bfc8   Davi Arnaut   signalfd: retriev...
193
194
195
196
  	__set_current_state(TASK_RUNNING);
  
  	return ret;
  }
fba2afaae   Davide Libenzi   signal/timer/even...
197
  /*
b8fceee17   Davide Libenzi   signalfd simplifi...
198
199
200
   * Returns a multiple of the size of a "struct signalfd_siginfo", or a negative
   * error code. The "count" parameter must be at least the size of a
   * "struct signalfd_siginfo".
fba2afaae   Davide Libenzi   signal/timer/even...
201
202
203
204
205
   */
  static ssize_t signalfd_read(struct file *file, char __user *buf, size_t count,
  			     loff_t *ppos)
  {
  	struct signalfd_ctx *ctx = file->private_data;
b3762bfc8   Davi Arnaut   signalfd: retriev...
206
207
208
  	struct signalfd_siginfo __user *siginfo;
  	int nonblock = file->f_flags & O_NONBLOCK;
  	ssize_t ret, total = 0;
fba2afaae   Davide Libenzi   signal/timer/even...
209
  	siginfo_t info;
fba2afaae   Davide Libenzi   signal/timer/even...
210

b3762bfc8   Davi Arnaut   signalfd: retriev...
211
212
  	count /= sizeof(struct signalfd_siginfo);
  	if (!count)
fba2afaae   Davide Libenzi   signal/timer/even...
213
  		return -EINVAL;
fba2afaae   Davide Libenzi   signal/timer/even...
214

b3762bfc8   Davi Arnaut   signalfd: retriev...
215
  	siginfo = (struct signalfd_siginfo __user *) buf;
b3762bfc8   Davi Arnaut   signalfd: retriev...
216
217
218
219
220
221
222
223
224
225
226
  	do {
  		ret = signalfd_dequeue(ctx, &info, nonblock);
  		if (unlikely(ret <= 0))
  			break;
  		ret = signalfd_copyinfo(siginfo, &info);
  		if (ret < 0)
  			break;
  		siginfo++;
  		total += ret;
  		nonblock = 1;
  	} while (--count);
b8fceee17   Davide Libenzi   signalfd simplifi...
227
  	return total ? total: ret;
fba2afaae   Davide Libenzi   signal/timer/even...
228
  }
138d22b58   Cyrill Gorcunov   fs, epoll: add pr...
229
  #ifdef CONFIG_PROC_FS
a3816ab0e   Joe Perches   fs: Convert show_...
230
  static void signalfd_show_fdinfo(struct seq_file *m, struct file *f)
138d22b58   Cyrill Gorcunov   fs, epoll: add pr...
231
232
233
234
235
236
237
  {
  	struct signalfd_ctx *ctx = f->private_data;
  	sigset_t sigmask;
  
  	sigmask = ctx->sigmask;
  	signotset(&sigmask);
  	render_sigset_t(m, "sigmask:\t", &sigmask);
138d22b58   Cyrill Gorcunov   fs, epoll: add pr...
238
239
  }
  #endif
fba2afaae   Davide Libenzi   signal/timer/even...
240
  static const struct file_operations signalfd_fops = {
138d22b58   Cyrill Gorcunov   fs, epoll: add pr...
241
242
243
  #ifdef CONFIG_PROC_FS
  	.show_fdinfo	= signalfd_show_fdinfo,
  #endif
fba2afaae   Davide Libenzi   signal/timer/even...
244
245
246
  	.release	= signalfd_release,
  	.poll		= signalfd_poll,
  	.read		= signalfd_read,
6038f373a   Arnd Bergmann   llseek: automatic...
247
  	.llseek		= noop_llseek,
fba2afaae   Davide Libenzi   signal/timer/even...
248
  };
5ed0127fc   Al Viro   signalfd: lift si...
249
  static int do_signalfd4(int ufd, sigset_t *mask, int flags)
fba2afaae   Davide Libenzi   signal/timer/even...
250
  {
fba2afaae   Davide Libenzi   signal/timer/even...
251
  	struct signalfd_ctx *ctx;
fba2afaae   Davide Libenzi   signal/timer/even...
252

e38b36f32   Ulrich Drepper   flag parameters: ...
253
254
255
  	/* Check the SFD_* constants for consistency.  */
  	BUILD_BUG_ON(SFD_CLOEXEC != O_CLOEXEC);
  	BUILD_BUG_ON(SFD_NONBLOCK != O_NONBLOCK);
5fb5e0492   Ulrich Drepper   flag parameters: ...
256
  	if (flags & ~(SFD_CLOEXEC | SFD_NONBLOCK))
9deb27bae   Ulrich Drepper   flag parameters: ...
257
  		return -EINVAL;
5ed0127fc   Al Viro   signalfd: lift si...
258
259
  	sigdelsetmask(mask, sigmask(SIGKILL) | sigmask(SIGSTOP));
  	signotset(mask);
fba2afaae   Davide Libenzi   signal/timer/even...
260
261
262
263
264
  
  	if (ufd == -1) {
  		ctx = kmalloc(sizeof(*ctx), GFP_KERNEL);
  		if (!ctx)
  			return -ENOMEM;
5ed0127fc   Al Viro   signalfd: lift si...
265
  		ctx->sigmask = *mask;
fba2afaae   Davide Libenzi   signal/timer/even...
266
267
268
269
270
  
  		/*
  		 * When we call this, the initialization must be complete, since
  		 * anon_inode_getfd() will install the fd.
  		 */
7d9dbca34   Ulrich Drepper   flag parameters: ...
271
  		ufd = anon_inode_getfd("[signalfd]", &signalfd_fops, ctx,
628ff7c1d   Roland Dreier   anonfd: Allow mak...
272
  				       O_RDWR | (flags & (O_CLOEXEC | O_NONBLOCK)));
2030a42ce   Al Viro   [PATCH] sanitize ...
273
274
  		if (ufd < 0)
  			kfree(ctx);
fba2afaae   Davide Libenzi   signal/timer/even...
275
  	} else {
2903ff019   Al Viro   switch simple cas...
276
277
  		struct fd f = fdget(ufd);
  		if (!f.file)
fba2afaae   Davide Libenzi   signal/timer/even...
278
  			return -EBADF;
2903ff019   Al Viro   switch simple cas...
279
280
281
  		ctx = f.file->private_data;
  		if (f.file->f_op != &signalfd_fops) {
  			fdput(f);
fba2afaae   Davide Libenzi   signal/timer/even...
282
283
  			return -EINVAL;
  		}
b8fceee17   Davide Libenzi   signalfd simplifi...
284
  		spin_lock_irq(&current->sighand->siglock);
5ed0127fc   Al Viro   signalfd: lift si...
285
  		ctx->sigmask = *mask;
b8fceee17   Davide Libenzi   signalfd simplifi...
286
287
288
  		spin_unlock_irq(&current->sighand->siglock);
  
  		wake_up(&current->sighand->signalfd_wqh);
2903ff019   Al Viro   switch simple cas...
289
  		fdput(f);
fba2afaae   Davide Libenzi   signal/timer/even...
290
291
292
  	}
  
  	return ufd;
fba2afaae   Davide Libenzi   signal/timer/even...
293
  }
9deb27bae   Ulrich Drepper   flag parameters: ...
294

52fb6db0f   Dominik Brodowski   fs: add do_signal...
295
296
297
  SYSCALL_DEFINE4(signalfd4, int, ufd, sigset_t __user *, user_mask,
  		size_t, sizemask, int, flags)
  {
5ed0127fc   Al Viro   signalfd: lift si...
298
299
300
301
302
303
  	sigset_t mask;
  
  	if (sizemask != sizeof(sigset_t) ||
  	    copy_from_user(&mask, user_mask, sizeof(mask)))
  		return -EINVAL;
  	return do_signalfd4(ufd, &mask, flags);
52fb6db0f   Dominik Brodowski   fs: add do_signal...
304
  }
836f92adf   Heiko Carstens   [CVE-2009-0029] S...
305
306
  SYSCALL_DEFINE3(signalfd, int, ufd, sigset_t __user *, user_mask,
  		size_t, sizemask)
9deb27bae   Ulrich Drepper   flag parameters: ...
307
  {
5ed0127fc   Al Viro   signalfd: lift si...
308
309
310
311
312
313
  	sigset_t mask;
  
  	if (sizemask != sizeof(sigset_t) ||
  	    copy_from_user(&mask, user_mask, sizeof(mask)))
  		return -EINVAL;
  	return do_signalfd4(ufd, &mask, 0);
9deb27bae   Ulrich Drepper   flag parameters: ...
314
  }
7d197ed4a   Al Viro   switch signalfd{,...
315
316
  
  #ifdef CONFIG_COMPAT
570484bfe   Dominik Brodowski   fs: add do_compat...
317
  static long do_compat_signalfd4(int ufd,
5ed0127fc   Al Viro   signalfd: lift si...
318
  			const compat_sigset_t __user *user_mask,
570484bfe   Dominik Brodowski   fs: add do_compat...
319
  			compat_size_t sigsetsize, int flags)
7d197ed4a   Al Viro   switch signalfd{,...
320
  {
5ed0127fc   Al Viro   signalfd: lift si...
321
  	sigset_t mask;
7d197ed4a   Al Viro   switch signalfd{,...
322
323
324
  
  	if (sigsetsize != sizeof(compat_sigset_t))
  		return -EINVAL;
5ed0127fc   Al Viro   signalfd: lift si...
325
  	if (get_compat_sigset(&mask, user_mask))
7d197ed4a   Al Viro   switch signalfd{,...
326
  		return -EFAULT;
5ed0127fc   Al Viro   signalfd: lift si...
327
  	return do_signalfd4(ufd, &mask, flags);
7d197ed4a   Al Viro   switch signalfd{,...
328
  }
570484bfe   Dominik Brodowski   fs: add do_compat...
329
  COMPAT_SYSCALL_DEFINE4(signalfd4, int, ufd,
5ed0127fc   Al Viro   signalfd: lift si...
330
  		     const compat_sigset_t __user *, user_mask,
570484bfe   Dominik Brodowski   fs: add do_compat...
331
332
333
  		     compat_size_t, sigsetsize,
  		     int, flags)
  {
5ed0127fc   Al Viro   signalfd: lift si...
334
  	return do_compat_signalfd4(ufd, user_mask, sigsetsize, flags);
570484bfe   Dominik Brodowski   fs: add do_compat...
335
  }
7d197ed4a   Al Viro   switch signalfd{,...
336
  COMPAT_SYSCALL_DEFINE3(signalfd, int, ufd,
5ed0127fc   Al Viro   signalfd: lift si...
337
  		     const compat_sigset_t __user *, user_mask,
7d197ed4a   Al Viro   switch signalfd{,...
338
339
  		     compat_size_t, sigsetsize)
  {
5ed0127fc   Al Viro   signalfd: lift si...
340
  	return do_compat_signalfd4(ufd, user_mask, sigsetsize, 0);
7d197ed4a   Al Viro   switch signalfd{,...
341
342
  }
  #endif