]> git.saurik.com Git - apple/xnu.git/blame - bsd/kern/sys_pipe.c
xnu-7195.81.3.tar.gz
[apple/xnu.git] / bsd / kern / sys_pipe.c
CommitLineData
91447636
A
1/*
2 * Copyright (c) 1996 John S. Dyson
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * 1. Redistributions of source code must retain the above copyright
9 * notice immediately at the beginning of the file, without modification,
10 * this list of conditions, and the following disclaimer.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * 3. Absolutely no warranty of function or purpose is made by the author
15 * John S. Dyson.
16 * 4. Modifications may be freely made to this file if the above conditions
17 * are met.
18 */
19/*
f427ee49 20 * Copyright (c) 2003-2020 Apple Inc. All rights reserved.
91447636 21 *
2d21ac55 22 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
0a7de745 23 *
2d21ac55
A
24 * This file contains Original Code and/or Modifications of Original Code
25 * as defined in and that are subject to the Apple Public Source License
26 * Version 2.0 (the 'License'). You may not use this file except in
27 * compliance with the License. The rights granted to you under the License
28 * may not be used to create, or enable the creation or redistribution of,
29 * unlawful or unlicensed copies of an Apple operating system, or to
30 * circumvent, violate, or enable the circumvention or violation of, any
31 * terms of an Apple operating system software license agreement.
0a7de745 32 *
2d21ac55
A
33 * Please obtain a copy of the License at
34 * http://www.opensource.apple.com/apsl/ and read it before using this file.
0a7de745 35 *
2d21ac55
A
36 * The Original Code and all software distributed under the License are
37 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
8f6c56a5
A
38 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
39 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
2d21ac55
A
40 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
41 * Please see the License for the specific language governing rights and
42 * limitations under the License.
0a7de745 43 *
2d21ac55
A
44 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
45 */
46/*
47 * NOTICE: This file was modified by SPARTA, Inc. in 2005 to introduce
48 * support for mandatory and extensible security protections. This notice
49 * is included in support of clause 2.2 (b) of the Apple Public License,
50 * Version 2.0.
91447636
A
51 */
52
53/*
54 * This file contains a high-performance replacement for the socket-based
55 * pipes scheme originally used in FreeBSD/4.4Lite. It does not support
56 * all features of sockets, but does do everything that pipes normally
57 * do.
316670eb
A
58 *
59 * Pipes are implemented as circular buffers. Following are the valid states in pipes operations
0a7de745 60 *
316670eb
A
61 * _________________________________
62 * 1. |_________________________________| r=w, c=0
0a7de745 63 *
316670eb
A
64 * _________________________________
65 * 2. |__r:::::wc_______________________| r <= w , c > 0
66 *
67 * _________________________________
68 * 3. |::::wc_____r:::::::::::::::::::::| r>w , c > 0
69 *
70 * _________________________________
71 * 4. |:::::::wrc:::::::::::::::::::::::| w=r, c = Max size
72 *
73 *
74 * Nomenclature:-
75 * a-z define the steps in a program flow
76 * 1-4 are the states as defined aboe
77 * Action: is what file operation is done on the pipe
0a7de745 78 *
316670eb
A
79 * Current:None Action: initialize with size M=200
80 * a. State 1 ( r=0, w=0, c=0)
0a7de745 81 *
316670eb
A
82 * Current: a Action: write(100) (w < M)
83 * b. State 2 (r=0, w=100, c=100)
0a7de745 84 *
316670eb
A
85 * Current: b Action: write(100) (w = M-w)
86 * c. State 4 (r=0,w=0,c=200)
0a7de745 87 *
316670eb
A
88 * Current: b Action: read(70) ( r < c )
89 * d. State 2(r=70,w=100,c=30)
0a7de745 90 *
316670eb
A
91 * Current: d Action: write(75) ( w < (m-w))
92 * e. State 2 (r=70,w=175,c=105)
0a7de745 93 *
316670eb
A
94 * Current: d Action: write(110) ( w > (m-w))
95 * f. State 3 (r=70,w=10,c=140)
0a7de745 96 *
316670eb
A
97 * Current: d Action: read(30) (r >= c )
98 * g. State 1 (r=100,w=100,c=0)
0a7de745 99 *
91447636
A
100 */
101
102/*
316670eb
A
103 * This code create half duplex pipe buffers for facilitating file like
104 * operations on pipes. The initial buffer is very small, but this can
105 * dynamically change to larger sizes based on usage. The buffer size is never
106 * reduced. The total amount of kernel memory used is governed by maxpipekva.
107 * In case of dynamic expansion limit is reached, the output thread is blocked
0a7de745 108 * until the pipe buffer empties enough to continue.
91447636
A
109 *
110 * In order to limit the resource use of pipes, two sysctls exist:
111 *
112 * kern.ipc.maxpipekva - This is a hard limit on the amount of pageable
0a7de745 113 * address space available to us in pipe_map.
91447636
A
114 *
115 * Memory usage may be monitored through the sysctls
316670eb 116 * kern.ipc.pipes, kern.ipc.pipekva.
91447636
A
117 *
118 */
119
120#include <sys/param.h>
121#include <sys/systm.h>
122#include <sys/filedesc.h>
123#include <sys/kernel.h>
124#include <sys/vnode.h>
125#include <sys/proc_internal.h>
126#include <sys/kauth.h>
127#include <sys/file_internal.h>
128#include <sys/stat.h>
129#include <sys/ioctl.h>
130#include <sys/fcntl.h>
131#include <sys/malloc.h>
132#include <sys/syslog.h>
133#include <sys/unistd.h>
134#include <sys/resourcevar.h>
135#include <sys/aio_kern.h>
136#include <sys/signalvar.h>
137#include <sys/pipe.h>
138#include <sys/sysproto.h>
0c530ab8 139#include <sys/proc_info.h>
91447636 140
b0d623f7 141#include <security/audit/audit.h>
91447636
A
142
143#include <sys/kdebug.h>
144
145#include <kern/zalloc.h>
316670eb 146#include <kern/kalloc.h>
91447636
A
147#include <vm/vm_kern.h>
148#include <libkern/OSAtomic.h>
5ba3f43e
A
149#include <libkern/section_keywords.h>
150
151#if CONFIG_MACF
152#include <security/mac_framework.h>
153#endif
91447636 154
f427ee49
A
155#define f_flag fp_glob->fg_flag
156#define f_ops fp_glob->fg_ops
157#define f_data fp_glob->fg_data
91447636 158
cb323159
A
159struct pipepair {
160 lck_mtx_t pp_mtx;
161 struct pipe pp_rpipe;
162 struct pipe pp_wpipe;
f427ee49 163 uint64_t pp_pipe_id; /* unique ID shared by both pipe ends */
cb323159
A
164};
165
166#define PIPE_PAIR(pipe) \
167 __container_of(PIPE_MTX(pipe), struct pipepair, pp_mtx)
168
91447636 169/*
0a7de745 170 * interfaces to the outside world exported through file operations
91447636
A
171 */
172static int pipe_read(struct fileproc *fp, struct uio *uio,
0a7de745 173 int flags, vfs_context_t ctx);
91447636 174static int pipe_write(struct fileproc *fp, struct uio *uio,
0a7de745 175 int flags, vfs_context_t ctx);
2d21ac55 176static int pipe_close(struct fileglob *fg, vfs_context_t ctx);
2d21ac55 177static int pipe_select(struct fileproc *fp, int which, void * wql,
0a7de745 178 vfs_context_t ctx);
2d21ac55 179static int pipe_kqfilter(struct fileproc *fp, struct knote *kn,
cb323159 180 struct kevent_qos_s *kev);
2d21ac55 181static int pipe_ioctl(struct fileproc *fp, u_long cmd, caddr_t data,
0a7de745
A
182 vfs_context_t ctx);
183static int pipe_drain(struct fileproc *fp, vfs_context_t ctx);
b0d623f7 184
39236c6e 185static const struct fileops pipeops = {
cb323159
A
186 .fo_type = DTYPE_PIPE,
187 .fo_read = pipe_read,
188 .fo_write = pipe_write,
189 .fo_ioctl = pipe_ioctl,
190 .fo_select = pipe_select,
191 .fo_close = pipe_close,
192 .fo_drain = pipe_drain,
39037602 193 .fo_kqfilter = pipe_kqfilter,
39236c6e 194};
91447636 195
39037602 196static void filt_pipedetach(struct knote *kn);
91447636 197
cb323159
A
198static int filt_pipenotsup(struct knote *kn, long hint);
199static int filt_pipenotsuptouch(struct knote *kn, struct kevent_qos_s *kev);
200static int filt_pipenotsupprocess(struct knote *kn, struct kevent_qos_s *kev);
201
39037602 202static int filt_piperead(struct knote *kn, long hint);
cb323159
A
203static int filt_pipereadtouch(struct knote *kn, struct kevent_qos_s *kev);
204static int filt_pipereadprocess(struct knote *kn, struct kevent_qos_s *kev);
39037602
A
205
206static int filt_pipewrite(struct knote *kn, long hint);
cb323159
A
207static int filt_pipewritetouch(struct knote *kn, struct kevent_qos_s *kev);
208static int filt_pipewriteprocess(struct knote *kn, struct kevent_qos_s *kev);
209
210SECURITY_READ_ONLY_EARLY(struct filterops) pipe_nfiltops = {
211 .f_isfd = 1,
212 .f_detach = filt_pipedetach,
213 .f_event = filt_pipenotsup,
214 .f_touch = filt_pipenotsuptouch,
215 .f_process = filt_pipenotsupprocess,
216};
39037602 217
5ba3f43e 218SECURITY_READ_ONLY_EARLY(struct filterops) pipe_rfiltops = {
cb323159
A
219 .f_isfd = 1,
220 .f_detach = filt_pipedetach,
221 .f_event = filt_piperead,
222 .f_touch = filt_pipereadtouch,
39037602 223 .f_process = filt_pipereadprocess,
b0d623f7 224};
316670eb 225
5ba3f43e 226SECURITY_READ_ONLY_EARLY(struct filterops) pipe_wfiltops = {
cb323159
A
227 .f_isfd = 1,
228 .f_detach = filt_pipedetach,
229 .f_event = filt_pipewrite,
230 .f_touch = filt_pipewritetouch,
39037602 231 .f_process = filt_pipewriteprocess,
b0d623f7 232};
91447636 233
f427ee49 234#if PIPE_SYSCTLS
316670eb 235static int nbigpipe; /* for compatibility sake. no longer used */
f427ee49 236#endif
316670eb
A
237static int amountpipes; /* total number of pipes in system */
238static int amountpipekva; /* total memory used by pipes */
91447636 239
f427ee49
A
240static _Atomic uint64_t pipe_unique_id = 1;
241
39236c6e 242int maxpipekva __attribute__((used)) = PIPE_KVAMAX; /* allowing 16MB max. */
91447636
A
243
244#if PIPE_SYSCTLS
245SYSCTL_DECL(_kern_ipc);
246
0a7de745
A
247SYSCTL_INT(_kern_ipc, OID_AUTO, maxpipekva, CTLFLAG_RD | CTLFLAG_LOCKED,
248 &maxpipekva, 0, "Pipe KVA limit");
249SYSCTL_INT(_kern_ipc, OID_AUTO, maxpipekvawired, CTLFLAG_RW | CTLFLAG_LOCKED,
250 &maxpipekvawired, 0, "Pipe KVA wired limit");
251SYSCTL_INT(_kern_ipc, OID_AUTO, pipes, CTLFLAG_RD | CTLFLAG_LOCKED,
252 &amountpipes, 0, "Current # of pipes");
253SYSCTL_INT(_kern_ipc, OID_AUTO, bigpipes, CTLFLAG_RD | CTLFLAG_LOCKED,
254 &nbigpipe, 0, "Current # of big pipes");
255SYSCTL_INT(_kern_ipc, OID_AUTO, pipekva, CTLFLAG_RD | CTLFLAG_LOCKED,
256 &amountpipekva, 0, "Pipe KVA usage");
257SYSCTL_INT(_kern_ipc, OID_AUTO, pipekvawired, CTLFLAG_RD | CTLFLAG_LOCKED,
258 &amountpipekvawired, 0, "Pipe wired KVA usage");
91447636
A
259#endif
260
cb323159 261static int pipepair_alloc(struct pipe **rpipe, struct pipe **wpipe);
91447636
A
262static void pipeclose(struct pipe *cpipe);
263static void pipe_free_kmem(struct pipe *cpipe);
316670eb
A
264static int pipespace(struct pipe *cpipe, int size);
265static int choose_pipespace(unsigned long current, unsigned long expected);
266static int expand_pipespace(struct pipe *p, int target_size);
91447636 267static void pipeselwakeup(struct pipe *cpipe, struct pipe *spipe);
316670eb
A
268static __inline int pipeio_lock(struct pipe *cpipe, int catch);
269static __inline void pipeio_unlock(struct pipe *cpipe);
91447636 270
f427ee49
A
271static LCK_GRP_DECLARE(pipe_mtx_grp, "pipe");
272static ZONE_DECLARE(pipe_zone, "pipe zone", sizeof(struct pipepair), ZC_NONE);
91447636 273
0a7de745 274#define MAX_PIPESIZE(pipe) ( MAX(PIPE_SIZE, (pipe)->pipe_buffer.size) )
316670eb 275
91447636
A
276SYSINIT(vfs, SI_SUB_VFS, SI_ORDER_ANY, pipeinit, NULL);
277
f427ee49 278#if defined(XNU_TARGET_OS_OSX)
2d21ac55 279/* Bitmap for things to touch in pipe_touch() */
0a7de745
A
280#define PIPE_ATIME 0x00000001 /* time of last access */
281#define PIPE_MTIME 0x00000002 /* time of last modification */
282#define PIPE_CTIME 0x00000004 /* time of last status change */
2d21ac55
A
283
284static void
285pipe_touch(struct pipe *tpipe, int touch)
286{
5ba3f43e 287 struct timespec now;
2d21ac55 288
5ba3f43e 289 nanotime(&now);
2d21ac55
A
290
291 if (touch & PIPE_ATIME) {
292 tpipe->st_atimespec.tv_sec = now.tv_sec;
5ba3f43e 293 tpipe->st_atimespec.tv_nsec = now.tv_nsec;
2d21ac55
A
294 }
295
296 if (touch & PIPE_MTIME) {
297 tpipe->st_mtimespec.tv_sec = now.tv_sec;
5ba3f43e 298 tpipe->st_mtimespec.tv_nsec = now.tv_nsec;
2d21ac55
A
299 }
300
301 if (touch & PIPE_CTIME) {
302 tpipe->st_ctimespec.tv_sec = now.tv_sec;
5ba3f43e 303 tpipe->st_ctimespec.tv_nsec = now.tv_nsec;
2d21ac55
A
304 }
305}
5ba3f43e 306#endif
2d21ac55 307
0a7de745 308static const unsigned int pipesize_blocks[] = {512, 1024, 2048, 4096, 4096 * 2, PIPE_SIZE, PIPE_SIZE * 4 };
316670eb 309
0a7de745
A
310/*
311 * finds the right size from possible sizes in pipesize_blocks
312 * returns the size which matches max(current,expected)
316670eb 313 */
0a7de745 314static int
316670eb
A
315choose_pipespace(unsigned long current, unsigned long expected)
316{
0a7de745 317 int i = sizeof(pipesize_blocks) / sizeof(unsigned int) - 1;
316670eb
A
318 unsigned long target;
319
3e170ce0
A
320 /*
321 * assert that we always get an atomic transaction sized pipe buffer,
322 * even if the system pipe buffer high-water mark has been crossed.
323 */
324 assert(PIPE_BUF == pipesize_blocks[0]);
325
0a7de745 326 if (expected > current) {
316670eb 327 target = expected;
0a7de745 328 } else {
316670eb 329 target = current;
0a7de745 330 }
316670eb 331
0a7de745
A
332 while (i > 0 && pipesize_blocks[i - 1] > target) {
333 i = i - 1;
316670eb 334 }
0a7de745 335
316670eb
A
336 return pipesize_blocks[i];
337}
91447636
A
338
339
316670eb
A
340/*
341 * expand the size of pipe while there is data to be read,
342 * and then free the old buffer once the current buffered
343 * data has been transferred to new storage.
344 * Required: PIPE_LOCK and io lock to be held by caller.
345 * returns 0 on success or no expansion possible
346 */
0a7de745 347static int
316670eb
A
348expand_pipespace(struct pipe *p, int target_size)
349{
350 struct pipe tmp, oldpipe;
351 int error;
352 tmp.pipe_buffer.buffer = 0;
0a7de745 353
316670eb
A
354 if (p->pipe_buffer.size >= (unsigned) target_size) {
355 return 0; /* the existing buffer is max size possible */
356 }
0a7de745 357
316670eb
A
358 /* create enough space in the target */
359 error = pipespace(&tmp, target_size);
0a7de745
A
360 if (error != 0) {
361 return error;
362 }
316670eb
A
363
364 oldpipe.pipe_buffer.buffer = p->pipe_buffer.buffer;
365 oldpipe.pipe_buffer.size = p->pipe_buffer.size;
0a7de745 366
316670eb 367 memcpy(tmp.pipe_buffer.buffer, p->pipe_buffer.buffer, p->pipe_buffer.size);
0a7de745 368 if (p->pipe_buffer.cnt > 0 && p->pipe_buffer.in <= p->pipe_buffer.out) {
316670eb
A
369 /* we are in State 3 and need extra copying for read to be consistent */
370 memcpy(&tmp.pipe_buffer.buffer[p->pipe_buffer.size], p->pipe_buffer.buffer, p->pipe_buffer.size);
371 p->pipe_buffer.in += p->pipe_buffer.size;
372 }
373
374 p->pipe_buffer.buffer = tmp.pipe_buffer.buffer;
375 p->pipe_buffer.size = tmp.pipe_buffer.size;
376
377
378 pipe_free_kmem(&oldpipe);
379 return 0;
380}
381
91447636
A
382/*
383 * The pipe system call for the DTYPE_PIPE type of pipes
0a7de745 384 *
316670eb 385 * returns:
0a7de745 386 * FREAD | fd0 | -->[struct rpipe] --> |~~buffer~~| \
316670eb 387 * (pipe_mutex)
0a7de745 388 * FWRITE | fd1 | -->[struct wpipe] --X /
91447636
A
389 */
390
391/* ARGSUSED */
392int
b0d623f7 393pipe(proc_t p, __unused struct pipe_args *uap, int32_t *retval)
91447636
A
394{
395 struct fileproc *rf, *wf;
396 struct pipe *rpipe, *wpipe;
cb323159 397 int error;
91447636 398
cb323159
A
399 error = pipepair_alloc(&rpipe, &wpipe);
400 if (error) {
401 return error;
0a7de745
A
402 }
403
0a7de745 404 /*
cb323159
A
405 * for now we'll create half-duplex pipes(refer returns section above).
406 * this is what we've always supported..
0a7de745 407 */
91447636 408
cb323159 409 error = falloc(p, &rf, &retval[0], vfs_context_current());
91447636 410 if (error) {
0a7de745 411 goto freepipes;
91447636 412 }
91447636 413 rf->f_flag = FREAD;
91447636
A
414 rf->f_data = (caddr_t)rpipe;
415 rf->f_ops = &pipeops;
416
cb323159 417 error = falloc(p, &wf, &retval[1], vfs_context_current());
91447636
A
418 if (error) {
419 fp_free(p, retval[0], rf);
0a7de745 420 goto freepipes;
91447636
A
421 }
422 wf->f_flag = FWRITE;
91447636
A
423 wf->f_data = (caddr_t)wpipe;
424 wf->f_ops = &pipeops;
425
6601e61a
A
426 rpipe->pipe_peer = wpipe;
427 wpipe->pipe_peer = rpipe;
2d21ac55 428
2d21ac55 429#if CONFIG_MACF
91447636
A
430 /*
431 * XXXXXXXX SHOULD NOT HOLD FILE_LOCK() XXXXXXXXXXXX
432 *
433 * struct pipe represents a pipe endpoint. The MAC label is shared
2d21ac55
A
434 * between the connected endpoints. As a result mac_pipe_label_init() and
435 * mac_pipe_label_associate() should only be called on one of the endpoints
91447636
A
436 * after they have been connected.
437 */
2d21ac55
A
438 mac_pipe_label_init(rpipe);
439 mac_pipe_label_associate(kauth_cred_get(), rpipe);
440 wpipe->pipe_label = rpipe->pipe_label;
91447636 441#endif
2d21ac55 442 proc_fdlock_spin(p);
6601e61a
A
443 procfdtbl_releasefd(p, retval[0], NULL);
444 procfdtbl_releasefd(p, retval[1], NULL);
91447636
A
445 fp_drop(p, retval[0], rf, 1);
446 fp_drop(p, retval[1], wf, 1);
447 proc_fdunlock(p);
0a7de745 448 return 0;
91447636
A
449
450freepipes:
0a7de745
A
451 pipeclose(rpipe);
452 pipeclose(wpipe);
0a7de745 453 return error;
91447636
A
454}
455
91447636 456int
2d21ac55 457pipe_stat(struct pipe *cpipe, void *ub, int isstat64)
91447636 458{
2d21ac55 459#if CONFIG_MACF
0a7de745 460 int error;
91447636 461#endif
0a7de745
A
462 int pipe_size = 0;
463 int pipe_count;
464 struct stat *sb = (struct stat *)0; /* warning avoidance ; protected by isstat64 */
2d21ac55 465 struct stat64 * sb64 = (struct stat64 *)0; /* warning avoidance ; protected by isstat64 */
91447636 466
0a7de745
A
467 if (cpipe == NULL) {
468 return EBADF;
469 }
91447636 470 PIPE_LOCK(cpipe);
2d21ac55
A
471
472#if CONFIG_MACF
473 error = mac_pipe_check_stat(kauth_cred_get(), cpipe);
474 if (error) {
475 PIPE_UNLOCK(cpipe);
0a7de745 476 return error;
2d21ac55 477 }
91447636
A
478#endif
479 if (cpipe->pipe_buffer.buffer == 0) {
0a7de745
A
480 /* must be stat'ing the write fd */
481 if (cpipe->pipe_peer) {
482 /* the peer still exists, use it's info */
483 pipe_size = MAX_PIPESIZE(cpipe->pipe_peer);
2d21ac55
A
484 pipe_count = cpipe->pipe_peer->pipe_buffer.cnt;
485 } else {
486 pipe_count = 0;
487 }
488 } else {
0a7de745 489 pipe_size = MAX_PIPESIZE(cpipe);
2d21ac55 490 pipe_count = cpipe->pipe_buffer.cnt;
91447636 491 }
2d21ac55
A
492 /*
493 * since peer's buffer is setup ouside of lock
494 * we might catch it in transient state
495 */
0a7de745 496 if (pipe_size == 0) {
316670eb 497 pipe_size = MAX(PIPE_SIZE, pipesize_blocks[0]);
0a7de745 498 }
91447636 499
2d21ac55 500 if (isstat64 != 0) {
0a7de745 501 sb64 = (struct stat64 *)ub;
91447636 502
2d21ac55
A
503 bzero(sb64, sizeof(*sb64));
504 sb64->st_mode = S_IFIFO | S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP;
505 sb64->st_blksize = pipe_size;
506 sb64->st_size = pipe_count;
507 sb64->st_blocks = (sb64->st_size + sb64->st_blksize - 1) / sb64->st_blksize;
0a7de745 508
2d21ac55
A
509 sb64->st_uid = kauth_getuid();
510 sb64->st_gid = kauth_getgid();
0a7de745 511
2d21ac55
A
512 sb64->st_atimespec.tv_sec = cpipe->st_atimespec.tv_sec;
513 sb64->st_atimespec.tv_nsec = cpipe->st_atimespec.tv_nsec;
0a7de745 514
2d21ac55
A
515 sb64->st_mtimespec.tv_sec = cpipe->st_mtimespec.tv_sec;
516 sb64->st_mtimespec.tv_nsec = cpipe->st_mtimespec.tv_nsec;
91447636 517
2d21ac55
A
518 sb64->st_ctimespec.tv_sec = cpipe->st_ctimespec.tv_sec;
519 sb64->st_ctimespec.tv_nsec = cpipe->st_ctimespec.tv_nsec;
91447636 520
2d21ac55 521 /*
0a7de745
A
522 * Return a relatively unique inode number based on the current
523 * address of this pipe's struct pipe. This number may be recycled
524 * relatively quickly.
525 */
cb323159 526 sb64->st_ino = (ino64_t)VM_KERNEL_ADDRHASH((uintptr_t)cpipe);
2d21ac55 527 } else {
0a7de745 528 sb = (struct stat *)ub;
2d21ac55
A
529
530 bzero(sb, sizeof(*sb));
531 sb->st_mode = S_IFIFO | S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP;
532 sb->st_blksize = pipe_size;
533 sb->st_size = pipe_count;
534 sb->st_blocks = (sb->st_size + sb->st_blksize - 1) / sb->st_blksize;
0a7de745 535
2d21ac55
A
536 sb->st_uid = kauth_getuid();
537 sb->st_gid = kauth_getgid();
0a7de745 538
2d21ac55
A
539 sb->st_atimespec.tv_sec = cpipe->st_atimespec.tv_sec;
540 sb->st_atimespec.tv_nsec = cpipe->st_atimespec.tv_nsec;
0a7de745 541
2d21ac55
A
542 sb->st_mtimespec.tv_sec = cpipe->st_mtimespec.tv_sec;
543 sb->st_mtimespec.tv_nsec = cpipe->st_mtimespec.tv_nsec;
544
545 sb->st_ctimespec.tv_sec = cpipe->st_ctimespec.tv_sec;
546 sb->st_ctimespec.tv_nsec = cpipe->st_ctimespec.tv_nsec;
547
548 /*
0a7de745
A
549 * Return a relatively unique inode number based on the current
550 * address of this pipe's struct pipe. This number may be recycled
551 * relatively quickly.
552 */
cb323159 553 sb->st_ino = (ino_t)VM_KERNEL_ADDRHASH((uintptr_t)cpipe);
2d21ac55
A
554 }
555 PIPE_UNLOCK(cpipe);
91447636
A
556
557 /*
2d21ac55
A
558 * POSIX: Left as 0: st_dev, st_nlink, st_rdev, st_flags, st_gen,
559 * st_uid, st_gid.
560 *
561 * XXX (st_dev) should be unique, but there is no device driver that
562 * XXX is associated with pipes, since they are implemented via a
563 * XXX struct fileops indirection rather than as FS objects.
91447636 564 */
0a7de745 565 return 0;
91447636
A
566}
567
f427ee49
A
568uint64_t
569pipe_id(struct pipe *p)
570{
571 return PIPE_PAIR(p)->pp_pipe_id;
572}
91447636
A
573
574/*
575 * Allocate kva for pipe circular buffer, the space is pageable
576 * This routine will 'realloc' the size of a pipe safely, if it fails
577 * it will retain the old buffer.
578 * If it fails it will return ENOMEM.
579 */
580static int
581pipespace(struct pipe *cpipe, int size)
582{
583 vm_offset_t buffer;
584
0a7de745
A
585 if (size <= 0) {
586 return EINVAL;
587 }
91447636 588
f427ee49
A
589 buffer = (vm_offset_t)kheap_alloc(KHEAP_DATA_BUFFERS, size, Z_WAITOK);
590 if (!buffer) {
0a7de745
A
591 return ENOMEM;
592 }
91447636
A
593
594 /* free old resources if we're resizing */
595 pipe_free_kmem(cpipe);
596 cpipe->pipe_buffer.buffer = (caddr_t)buffer;
597 cpipe->pipe_buffer.size = size;
598 cpipe->pipe_buffer.in = 0;
599 cpipe->pipe_buffer.out = 0;
600 cpipe->pipe_buffer.cnt = 0;
601
b0d623f7
A
602 OSAddAtomic(1, &amountpipes);
603 OSAddAtomic(cpipe->pipe_buffer.size, &amountpipekva);
91447636 604
0a7de745 605 return 0;
91447636
A
606}
607
608/*
609 * initialize and allocate VM and memory for pipe
610 */
611static int
cb323159 612pipepair_alloc(struct pipe **rp_out, struct pipe **wp_out)
91447636 613{
cb323159
A
614 struct pipepair *pp = zalloc(pipe_zone);
615 struct pipe *rpipe = &pp->pp_rpipe;
616 struct pipe *wpipe = &pp->pp_wpipe;
91447636 617
cb323159 618 if (pp == NULL) {
0a7de745
A
619 return ENOMEM;
620 }
91447636
A
621
622 /*
623 * protect so pipespace or pipeclose don't follow a junk pointer
624 * if pipespace() fails.
625 */
cb323159 626 bzero(pp, sizeof(struct pipepair));
f427ee49
A
627 pp->pp_pipe_id = os_atomic_inc_orig(&pipe_unique_id, relaxed);
628 lck_mtx_init(&pp->pp_mtx, &pipe_mtx_grp, LCK_ATTR_NULL);
cb323159
A
629
630 rpipe->pipe_mtxp = &pp->pp_mtx;
631 wpipe->pipe_mtxp = &pp->pp_mtx;
632
f427ee49 633#if defined(XNU_TARGET_OS_OSX)
2d21ac55 634 /* Initial times are all the time of creation of the pipe */
cb323159
A
635 pipe_touch(rpipe, PIPE_ATIME | PIPE_MTIME | PIPE_CTIME);
636 pipe_touch(wpipe, PIPE_ATIME | PIPE_MTIME | PIPE_CTIME);
5ba3f43e 637#endif
cb323159
A
638
639 /*
640 * allocate the space for the normal I/O direction up
641 * front... we'll delay the allocation for the other
642 * direction until a write actually occurs (most likely it won't)...
643 */
644 int error = pipespace(rpipe, choose_pipespace(rpipe->pipe_buffer.size, 0));
645 if (__improbable(error)) {
f427ee49 646 lck_mtx_destroy(&pp->pp_mtx, &pipe_mtx_grp);
cb323159
A
647 zfree(pipe_zone, pp);
648 return error;
649 }
650
651 *rp_out = rpipe;
652 *wp_out = wpipe;
0a7de745 653 return 0;
91447636
A
654}
655
cb323159
A
656static void
657pipepair_destroy_pipe(struct pipepair *pp, struct pipe *cpipe)
658{
659 bool can_free;
660
661 pipe_free_kmem(cpipe);
662
663 lck_mtx_lock(&pp->pp_mtx);
664 if (__improbable(cpipe->pipe_state & PIPE_DEAD)) {
665 panic("double free of pipe %p in pair %p", cpipe, pp);
666 }
667
668 cpipe->pipe_state |= PIPE_DEAD;
669
670 can_free = (pp->pp_rpipe.pipe_state & PIPE_DEAD) &&
671 (pp->pp_wpipe.pipe_state & PIPE_DEAD);
672 lck_mtx_unlock(&pp->pp_mtx);
673
674 if (can_free) {
f427ee49 675 lck_mtx_destroy(&pp->pp_mtx, &pipe_mtx_grp);
cb323159
A
676 zfree(pipe_zone, pp);
677 }
678}
91447636
A
679
680/*
681 * lock a pipe for I/O, blocking other access
682 */
2d21ac55 683static inline int
316670eb 684pipeio_lock(struct pipe *cpipe, int catch)
91447636
A
685{
686 int error;
91447636
A
687 while (cpipe->pipe_state & PIPE_LOCKFL) {
688 cpipe->pipe_state |= PIPE_LWANT;
91447636 689 error = msleep(cpipe, PIPE_MTX(cpipe), catch ? (PRIBIO | PCATCH) : PRIBIO,
0a7de745
A
690 "pipelk", 0);
691 if (error != 0) {
692 return error;
693 }
91447636
A
694 }
695 cpipe->pipe_state |= PIPE_LOCKFL;
0a7de745 696 return 0;
91447636
A
697}
698
699/*
700 * unlock a pipe I/O lock
701 */
2d21ac55 702static inline void
316670eb 703pipeio_unlock(struct pipe *cpipe)
91447636 704{
91447636 705 cpipe->pipe_state &= ~PIPE_LOCKFL;
91447636
A
706 if (cpipe->pipe_state & PIPE_LWANT) {
707 cpipe->pipe_state &= ~PIPE_LWANT;
708 wakeup(cpipe);
709 }
710}
711
316670eb
A
712/*
713 * wakeup anyone whos blocked in select
714 */
91447636 715static void
2d21ac55 716pipeselwakeup(struct pipe *cpipe, struct pipe *spipe)
91447636 717{
91447636
A
718 if (cpipe->pipe_state & PIPE_SEL) {
719 cpipe->pipe_state &= ~PIPE_SEL;
720 selwakeup(&cpipe->pipe_sel);
721 }
cb323159
A
722
723 KNOTE(&cpipe->pipe_sel.si_note, 1);
91447636 724
91447636 725 if (spipe && (spipe->pipe_state & PIPE_ASYNC) && spipe->pipe_pgid) {
0a7de745
A
726 if (spipe->pipe_pgid < 0) {
727 gsignal(-spipe->pipe_pgid, SIGIO);
728 } else {
729 proc_signal(spipe->pipe_pgid, SIGIO);
730 }
731 }
91447636
A
732}
733
316670eb
A
734/*
735 * Read n bytes from the buffer. Semantics are similar to file read.
736 * returns: number of bytes read from the buffer
737 */
91447636
A
738/* ARGSUSED */
739static int
2d21ac55 740pipe_read(struct fileproc *fp, struct uio *uio, __unused int flags,
0a7de745 741 __unused vfs_context_t ctx)
91447636
A
742{
743 struct pipe *rpipe = (struct pipe *)fp->f_data;
744 int error;
745 int nread = 0;
746 u_int size;
747
748 PIPE_LOCK(rpipe);
749 ++rpipe->pipe_busy;
750
316670eb 751 error = pipeio_lock(rpipe, 1);
0a7de745 752 if (error) {
91447636 753 goto unlocked_error;
0a7de745 754 }
91447636 755
2d21ac55
A
756#if CONFIG_MACF
757 error = mac_pipe_check_read(kauth_cred_get(), rpipe);
0a7de745 758 if (error) {
91447636 759 goto locked_error;
0a7de745 760 }
91447636
A
761#endif
762
316670eb 763
91447636
A
764 while (uio_resid(uio)) {
765 /*
766 * normal pipe buffer receive
767 */
768 if (rpipe->pipe_buffer.cnt > 0) {
316670eb
A
769 /*
770 * # bytes to read is min( bytes from read pointer until end of buffer,
0a7de745 771 * total unread bytes,
316670eb
A
772 * user requested byte count)
773 */
91447636 774 size = rpipe->pipe_buffer.size - rpipe->pipe_buffer.out;
0a7de745 775 if (size > rpipe->pipe_buffer.cnt) {
91447636 776 size = rpipe->pipe_buffer.cnt;
0a7de745 777 }
f427ee49
A
778
779 size = (u_int) MIN(INT_MAX, MIN((user_size_t)size,
780 (user_size_t)uio_resid(uio)));
91447636 781
316670eb 782 PIPE_UNLOCK(rpipe); /* we still hold io lock.*/
91447636 783 error = uiomove(
0a7de745
A
784 &rpipe->pipe_buffer.buffer[rpipe->pipe_buffer.out],
785 size, uio);
91447636 786 PIPE_LOCK(rpipe);
0a7de745 787 if (error) {
91447636 788 break;
0a7de745 789 }
91447636
A
790
791 rpipe->pipe_buffer.out += size;
0a7de745 792 if (rpipe->pipe_buffer.out >= rpipe->pipe_buffer.size) {
91447636 793 rpipe->pipe_buffer.out = 0;
0a7de745 794 }
91447636
A
795
796 rpipe->pipe_buffer.cnt -= size;
0a7de745 797
91447636
A
798 /*
799 * If there is no more to read in the pipe, reset
800 * its pointers to the beginning. This improves
801 * cache hit stats.
802 */
803 if (rpipe->pipe_buffer.cnt == 0) {
804 rpipe->pipe_buffer.in = 0;
805 rpipe->pipe_buffer.out = 0;
806 }
807 nread += size;
91447636
A
808 } else {
809 /*
810 * detect EOF condition
811 * read returns 0 on EOF, no need to set error
812 */
cb323159
A
813 if ((rpipe->pipe_state & (PIPE_DRAIN | PIPE_EOF)) ||
814 (fileproc_get_vflags(fp) & FPV_DRAIN)) {
91447636 815 break;
b0d623f7 816 }
91447636
A
817
818 /*
819 * If the "write-side" has been blocked, wake it up now.
820 */
821 if (rpipe->pipe_state & PIPE_WANTW) {
822 rpipe->pipe_state &= ~PIPE_WANTW;
823 wakeup(rpipe);
824 }
825
826 /*
316670eb 827 * Break if some data was read in previous iteration.
91447636 828 */
0a7de745 829 if (nread > 0) {
91447636 830 break;
0a7de745 831 }
91447636
A
832
833 /*
0a7de745 834 * Unlock the pipe buffer for our remaining processing.
91447636
A
835 * We will either break out with an error or we will
836 * sleep and relock to loop.
837 */
316670eb 838 pipeio_unlock(rpipe);
91447636
A
839
840 /*
841 * Handle non-blocking mode operation or
842 * wait for more data.
843 */
844 if (fp->f_flag & FNONBLOCK) {
845 error = EAGAIN;
846 } else {
847 rpipe->pipe_state |= PIPE_WANTR;
91447636 848 error = msleep(rpipe, PIPE_MTX(rpipe), PRIBIO | PCATCH, "piperd", 0);
0a7de745
A
849 if (error == 0) {
850 error = pipeio_lock(rpipe, 1);
851 }
91447636 852 }
0a7de745 853 if (error) {
91447636 854 goto unlocked_error;
0a7de745 855 }
91447636
A
856 }
857 }
2d21ac55 858#if CONFIG_MACF
91447636
A
859locked_error:
860#endif
316670eb 861 pipeio_unlock(rpipe);
91447636
A
862
863unlocked_error:
864 --rpipe->pipe_busy;
865
866 /*
867 * PIPE_WANT processing only makes sense if pipe_busy is 0.
868 */
869 if ((rpipe->pipe_busy == 0) && (rpipe->pipe_state & PIPE_WANT)) {
0a7de745 870 rpipe->pipe_state &= ~(PIPE_WANT | PIPE_WANTW);
91447636 871 wakeup(rpipe);
316670eb 872 } else if (rpipe->pipe_buffer.cnt < rpipe->pipe_buffer.size) {
91447636
A
873 /*
874 * Handle write blocking hysteresis.
875 */
876 if (rpipe->pipe_state & PIPE_WANTW) {
877 rpipe->pipe_state &= ~PIPE_WANTW;
878 wakeup(rpipe);
879 }
880 }
881
0a7de745 882 if ((rpipe->pipe_buffer.size - rpipe->pipe_buffer.cnt) > 0) {
91447636 883 pipeselwakeup(rpipe, rpipe->pipe_peer);
0a7de745 884 }
91447636 885
f427ee49 886#if defined(XNU_TARGET_OS_OSX)
2d21ac55
A
887 /* update last read time */
888 pipe_touch(rpipe, PIPE_ATIME);
5ba3f43e 889#endif
2d21ac55 890
91447636
A
891 PIPE_UNLOCK(rpipe);
892
0a7de745 893 return error;
91447636
A
894}
895
91447636 896/*
0a7de745 897 * perform a write of n bytes into the read side of buffer. Since
316670eb 898 * pipes are unidirectional a write is meant to be read by the otherside only.
91447636 899 */
91447636 900static int
2d21ac55 901pipe_write(struct fileproc *fp, struct uio *uio, __unused int flags,
0a7de745 902 __unused vfs_context_t ctx)
91447636
A
903{
904 int error = 0;
f427ee49 905 size_t orig_resid;
91447636
A
906 int pipe_size;
907 struct pipe *wpipe, *rpipe;
316670eb 908 // LP64todo - fix this!
f427ee49
A
909 orig_resid = (size_t)uio_resid(uio);
910 if (orig_resid > LONG_MAX) {
911 return EINVAL;
912 }
316670eb 913 int space;
91447636
A
914
915 rpipe = (struct pipe *)fp->f_data;
916
917 PIPE_LOCK(rpipe);
918 wpipe = rpipe->pipe_peer;
919
920 /*
921 * detect loss of pipe read side, issue SIGPIPE if lost.
922 */
cb323159
A
923 if (wpipe == NULL || (wpipe->pipe_state & (PIPE_DRAIN | PIPE_EOF)) ||
924 (fileproc_get_vflags(fp) & FPV_DRAIN)) {
91447636 925 PIPE_UNLOCK(rpipe);
0a7de745 926 return EPIPE;
91447636 927 }
2d21ac55
A
928#if CONFIG_MACF
929 error = mac_pipe_check_write(kauth_cred_get(), wpipe);
91447636
A
930 if (error) {
931 PIPE_UNLOCK(rpipe);
0a7de745 932 return error;
91447636
A
933 }
934#endif
935 ++wpipe->pipe_busy;
936
937 pipe_size = 0;
938
91447636 939 /*
316670eb
A
940 * need to allocate some storage... we delay the allocation
941 * until the first write on fd[0] to avoid allocating storage for both
942 * 'pipe ends'... most pipes are half-duplex with the writes targeting
943 * fd[1], so allocating space for both ends is a waste...
91447636 944 */
91447636 945
0a7de745
A
946 if (wpipe->pipe_buffer.buffer == 0 || (
947 (unsigned)orig_resid > wpipe->pipe_buffer.size - wpipe->pipe_buffer.cnt &&
948 amountpipekva < maxpipekva)) {
949 pipe_size = choose_pipespace(wpipe->pipe_buffer.size, wpipe->pipe_buffer.cnt + orig_resid);
91447636
A
950 }
951 if (pipe_size) {
0a7de745 952 /*
91447636 953 * need to do initial allocation or resizing of pipe
0a7de745 954 * holding both structure and io locks.
91447636 955 */
316670eb 956 if ((error = pipeio_lock(wpipe, 1)) == 0) {
0a7de745 957 if (wpipe->pipe_buffer.cnt == 0) {
316670eb 958 error = pipespace(wpipe, pipe_size);
0a7de745 959 } else {
316670eb 960 error = expand_pipespace(wpipe, pipe_size);
0a7de745
A
961 }
962
316670eb 963 pipeio_unlock(wpipe);
0a7de745 964
316670eb 965 /* allocation failed */
0a7de745
A
966 if (wpipe->pipe_buffer.buffer == 0) {
967 error = ENOMEM;
968 }
91447636
A
969 }
970 if (error) {
0a7de745 971 /*
91447636
A
972 * If an error occurred unbusy and return, waking up any pending
973 * readers.
974 */
0a7de745
A
975 --wpipe->pipe_busy;
976 if ((wpipe->pipe_busy == 0) &&
91447636 977 (wpipe->pipe_state & PIPE_WANT)) {
0a7de745 978 wpipe->pipe_state &= ~(PIPE_WANT | PIPE_WANTR);
91447636
A
979 wakeup(wpipe);
980 }
981 PIPE_UNLOCK(rpipe);
0a7de745 982 return error;
91447636
A
983 }
984 }
91447636
A
985
986 while (uio_resid(uio)) {
0a7de745 987retrywrite:
91447636
A
988 space = wpipe->pipe_buffer.size - wpipe->pipe_buffer.cnt;
989
316670eb 990 /* Writes of size <= PIPE_BUF must be atomic. */
0a7de745 991 if ((space < uio_resid(uio)) && (orig_resid <= PIPE_BUF)) {
91447636 992 space = 0;
0a7de745 993 }
91447636
A
994
995 if (space > 0) {
0a7de745 996 if ((error = pipeio_lock(wpipe, 1)) == 0) {
f427ee49
A
997 size_t size; /* Transfer size */
998 size_t segsize; /* first segment to transfer */
91447636 999
cb323159
A
1000 if ((wpipe->pipe_state & (PIPE_DRAIN | PIPE_EOF)) ||
1001 (fileproc_get_vflags(fp) & FPV_DRAIN)) {
316670eb 1002 pipeio_unlock(wpipe);
0a7de745 1003 error = EPIPE;
91447636
A
1004 break;
1005 }
0a7de745 1006 /*
316670eb 1007 * If a process blocked in pipeio_lock, our
91447636
A
1008 * value for space might be bad... the mutex
1009 * is dropped while we're blocked
1010 */
0a7de745 1011 if (space > (int)(wpipe->pipe_buffer.size -
91447636 1012 wpipe->pipe_buffer.cnt)) {
316670eb 1013 pipeio_unlock(wpipe);
91447636
A
1014 goto retrywrite;
1015 }
1016
1017 /*
1018 * Transfer size is minimum of uio transfer
1019 * and free space in pipe buffer.
1020 */
1021 // LP64todo - fix this!
0a7de745 1022 if (space > uio_resid(uio)) {
f427ee49
A
1023 size = (size_t)uio_resid(uio);
1024 if (size > LONG_MAX) {
1025 panic("size greater than LONG_MAX");
1026 }
0a7de745 1027 } else {
91447636 1028 size = space;
0a7de745 1029 }
91447636 1030 /*
0a7de745 1031 * First segment to transfer is minimum of
91447636
A
1032 * transfer size and contiguous space in
1033 * pipe buffer. If first segment to transfer
1034 * is less than the transfer size, we've got
1035 * a wraparound in the buffer.
1036 */
0a7de745
A
1037 segsize = wpipe->pipe_buffer.size -
1038 wpipe->pipe_buffer.in;
1039 if (segsize > size) {
91447636 1040 segsize = size;
0a7de745
A
1041 }
1042
91447636
A
1043 /* Transfer first segment */
1044
1045 PIPE_UNLOCK(rpipe);
0a7de745 1046 error = uiomove(&wpipe->pipe_buffer.buffer[wpipe->pipe_buffer.in],
f427ee49 1047 (int)segsize, uio);
91447636 1048 PIPE_LOCK(rpipe);
0a7de745 1049
91447636 1050 if (error == 0 && segsize < size) {
0a7de745 1051 /*
91447636
A
1052 * Transfer remaining part now, to
1053 * support atomic writes. Wraparound
316670eb 1054 * happened. (State 3)
91447636 1055 */
0a7de745
A
1056 if (wpipe->pipe_buffer.in + segsize !=
1057 wpipe->pipe_buffer.size) {
91447636
A
1058 panic("Expected pipe buffer "
1059 "wraparound disappeared");
0a7de745
A
1060 }
1061
91447636
A
1062 PIPE_UNLOCK(rpipe);
1063 error = uiomove(
0a7de745 1064 &wpipe->pipe_buffer.buffer[0],
f427ee49 1065 (int)(size - segsize), uio);
91447636
A
1066 PIPE_LOCK(rpipe);
1067 }
0a7de745 1068 /*
316670eb
A
1069 * readers never know to read until count is updated.
1070 */
91447636
A
1071 if (error == 0) {
1072 wpipe->pipe_buffer.in += size;
316670eb 1073 if (wpipe->pipe_buffer.in >
91447636
A
1074 wpipe->pipe_buffer.size) {
1075 if (wpipe->pipe_buffer.in !=
1076 size - segsize +
0a7de745 1077 wpipe->pipe_buffer.size) {
91447636
A
1078 panic("Expected "
1079 "wraparound bad");
0a7de745 1080 }
f427ee49
A
1081 wpipe->pipe_buffer.in = (unsigned int)(size -
1082 segsize);
91447636 1083 }
0a7de745 1084
91447636
A
1085 wpipe->pipe_buffer.cnt += size;
1086 if (wpipe->pipe_buffer.cnt >
0a7de745 1087 wpipe->pipe_buffer.size) {
91447636 1088 panic("Pipe buffer overflow");
0a7de745 1089 }
91447636 1090 }
316670eb 1091 pipeio_unlock(wpipe);
91447636 1092 }
0a7de745 1093 if (error) {
91447636 1094 break;
0a7de745 1095 }
91447636
A
1096 } else {
1097 /*
1098 * If the "read-side" has been blocked, wake it up now.
1099 */
1100 if (wpipe->pipe_state & PIPE_WANTR) {
1101 wpipe->pipe_state &= ~PIPE_WANTR;
1102 wakeup(wpipe);
1103 }
cb323159 1104
91447636 1105 /*
cb323159
A
1106 * If read side wants to go away, we just issue a signal
1107 * to ourselves.
91447636 1108 */
cb323159
A
1109 if ((wpipe->pipe_state & (PIPE_DRAIN | PIPE_EOF)) ||
1110 (fileproc_get_vflags(fp) & FPV_DRAIN)) {
1111 error = EPIPE;
91447636
A
1112 break;
1113 }
6d2010ae
A
1114
1115 /*
cb323159
A
1116 * don't block on non-blocking I/O
1117 * we'll do the pipeselwakeup on the way out
6d2010ae 1118 */
cb323159
A
1119 if (fp->f_flag & FNONBLOCK) {
1120 error = EAGAIN;
6d2010ae 1121 break;
0a7de745 1122 }
6d2010ae 1123
91447636
A
1124 /*
1125 * We have no more space and have something to offer,
1126 * wake up select/poll.
1127 */
1128 pipeselwakeup(wpipe, wpipe);
1129
1130 wpipe->pipe_state |= PIPE_WANTW;
1131
1132 error = msleep(wpipe, PIPE_MTX(wpipe), PRIBIO | PCATCH, "pipewr", 0);
1133
0a7de745 1134 if (error != 0) {
91447636 1135 break;
0a7de745 1136 }
91447636
A
1137 }
1138 }
1139 --wpipe->pipe_busy;
1140
1141 if ((wpipe->pipe_busy == 0) && (wpipe->pipe_state & PIPE_WANT)) {
1142 wpipe->pipe_state &= ~(PIPE_WANT | PIPE_WANTR);
1143 wakeup(wpipe);
1144 }
1145 if (wpipe->pipe_buffer.cnt > 0) {
1146 /*
1147 * If there are any characters in the buffer, we wake up
1148 * the reader if it was blocked waiting for data.
1149 */
1150 if (wpipe->pipe_state & PIPE_WANTR) {
1151 wpipe->pipe_state &= ~PIPE_WANTR;
1152 wakeup(wpipe);
1153 }
1154 /*
1155 * wake up thread blocked in select/poll or post the notification
1156 */
1157 pipeselwakeup(wpipe, wpipe);
1158 }
2d21ac55 1159
f427ee49 1160#if defined(XNU_TARGET_OS_OSX)
2d21ac55
A
1161 /* Update modification, status change (# of bytes in pipe) times */
1162 pipe_touch(rpipe, PIPE_MTIME | PIPE_CTIME);
1163 pipe_touch(wpipe, PIPE_MTIME | PIPE_CTIME);
5ba3f43e 1164#endif
91447636
A
1165 PIPE_UNLOCK(rpipe);
1166
0a7de745 1167 return error;
91447636
A
1168}
1169
1170/*
1171 * we implement a very minimal set of ioctls for compatibility with sockets.
1172 */
1173/* ARGSUSED 3 */
1174static int
2d21ac55 1175pipe_ioctl(struct fileproc *fp, u_long cmd, caddr_t data,
0a7de745 1176 __unused vfs_context_t ctx)
91447636
A
1177{
1178 struct pipe *mpipe = (struct pipe *)fp->f_data;
2d21ac55 1179#if CONFIG_MACF
91447636
A
1180 int error;
1181#endif
1182
1183 PIPE_LOCK(mpipe);
1184
2d21ac55
A
1185#if CONFIG_MACF
1186 error = mac_pipe_check_ioctl(kauth_cred_get(), mpipe, cmd);
91447636
A
1187 if (error) {
1188 PIPE_UNLOCK(mpipe);
1189
0a7de745 1190 return error;
91447636
A
1191 }
1192#endif
1193
1194 switch (cmd) {
91447636
A
1195 case FIONBIO:
1196 PIPE_UNLOCK(mpipe);
0a7de745 1197 return 0;
91447636
A
1198
1199 case FIOASYNC:
1200 if (*(int *)data) {
1201 mpipe->pipe_state |= PIPE_ASYNC;
1202 } else {
1203 mpipe->pipe_state &= ~PIPE_ASYNC;
1204 }
1205 PIPE_UNLOCK(mpipe);
0a7de745 1206 return 0;
91447636
A
1207
1208 case FIONREAD:
316670eb 1209 *(int *)data = mpipe->pipe_buffer.cnt;
91447636 1210 PIPE_UNLOCK(mpipe);
0a7de745 1211 return 0;
91447636
A
1212
1213 case TIOCSPGRP:
1214 mpipe->pipe_pgid = *(int *)data;
1215
1216 PIPE_UNLOCK(mpipe);
0a7de745 1217 return 0;
91447636
A
1218
1219 case TIOCGPGRP:
1220 *(int *)data = mpipe->pipe_pgid;
1221
1222 PIPE_UNLOCK(mpipe);
0a7de745 1223 return 0;
91447636
A
1224 }
1225 PIPE_UNLOCK(mpipe);
0a7de745 1226 return ENOTTY;
91447636
A
1227}
1228
1229
1230static int
2d21ac55 1231pipe_select(struct fileproc *fp, int which, void *wql, vfs_context_t ctx)
91447636
A
1232{
1233 struct pipe *rpipe = (struct pipe *)fp->f_data;
1234 struct pipe *wpipe;
1235 int retnum = 0;
1236
0a7de745
A
1237 if (rpipe == NULL || rpipe == (struct pipe *)-1) {
1238 return retnum;
1239 }
91447636
A
1240
1241 PIPE_LOCK(rpipe);
1242
1243 wpipe = rpipe->pipe_peer;
0a7de745 1244
91447636 1245
2d21ac55
A
1246#if CONFIG_MACF
1247 /*
1248 * XXX We should use a per thread credential here; minimally, the
1249 * XXX process credential should have a persistent reference on it
1250 * XXX before being passed in here.
1251 */
1252 if (mac_pipe_check_select(vfs_context_ucred(ctx), rpipe, which)) {
1253 PIPE_UNLOCK(rpipe);
0a7de745 1254 return 0;
2d21ac55
A
1255 }
1256#endif
0a7de745
A
1257 switch (which) {
1258 case FREAD:
91447636
A
1259 if ((rpipe->pipe_state & PIPE_DIRECTW) ||
1260 (rpipe->pipe_buffer.cnt > 0) ||
cb323159
A
1261 (rpipe->pipe_state & (PIPE_DRAIN | PIPE_EOF)) ||
1262 (fileproc_get_vflags(fp) & FPV_DRAIN)) {
0a7de745 1263 retnum = 1;
91447636 1264 } else {
0a7de745
A
1265 rpipe->pipe_state |= PIPE_SEL;
1266 selrecord(vfs_context_proc(ctx), &rpipe->pipe_sel, wql);
91447636
A
1267 }
1268 break;
1269
0a7de745
A
1270 case FWRITE:
1271 if (wpipe) {
ebb1b9f4 1272 wpipe->pipe_state |= PIPE_WSELECT;
0a7de745 1273 }
b0d623f7 1274 if (wpipe == NULL || (wpipe->pipe_state & (PIPE_DRAIN | PIPE_EOF)) ||
cb323159 1275 (fileproc_get_vflags(fp) & FPV_DRAIN) ||
91447636 1276 (((wpipe->pipe_state & PIPE_DIRECTW) == 0) &&
0a7de745
A
1277 (MAX_PIPESIZE(wpipe) - wpipe->pipe_buffer.cnt) >= PIPE_BUF)) {
1278 retnum = 1;
91447636 1279 } else {
0a7de745 1280 wpipe->pipe_state |= PIPE_SEL;
2d21ac55 1281 selrecord(vfs_context_proc(ctx), &wpipe->pipe_sel, wql);
91447636
A
1282 }
1283 break;
0a7de745
A
1284 case 0:
1285 rpipe->pipe_state |= PIPE_SEL;
2d21ac55 1286 selrecord(vfs_context_proc(ctx), &rpipe->pipe_sel, wql);
91447636 1287 break;
0a7de745 1288 }
91447636
A
1289 PIPE_UNLOCK(rpipe);
1290
0a7de745 1291 return retnum;
91447636
A
1292}
1293
1294
1295/* ARGSUSED 1 */
1296static int
2d21ac55 1297pipe_close(struct fileglob *fg, __unused vfs_context_t ctx)
91447636 1298{
0a7de745 1299 struct pipe *cpipe;
91447636 1300
2d21ac55 1301 proc_fdlock_spin(vfs_context_proc(ctx));
91447636
A
1302 cpipe = (struct pipe *)fg->fg_data;
1303 fg->fg_data = NULL;
2d21ac55 1304 proc_fdunlock(vfs_context_proc(ctx));
0a7de745
A
1305 if (cpipe) {
1306 pipeclose(cpipe);
1307 }
91447636 1308
0a7de745 1309 return 0;
91447636
A
1310}
1311
1312static void
1313pipe_free_kmem(struct pipe *cpipe)
1314{
91447636 1315 if (cpipe->pipe_buffer.buffer != NULL) {
b0d623f7
A
1316 OSAddAtomic(-(cpipe->pipe_buffer.size), &amountpipekva);
1317 OSAddAtomic(-1, &amountpipes);
f427ee49 1318 kheap_free(KHEAP_DATA_BUFFERS, cpipe->pipe_buffer.buffer,
0a7de745 1319 cpipe->pipe_buffer.size);
91447636 1320 cpipe->pipe_buffer.buffer = NULL;
316670eb 1321 cpipe->pipe_buffer.size = 0;
91447636 1322 }
ebb1b9f4
A
1323}
1324
91447636
A
1325/*
1326 * shutdown the pipe
1327 */
1328static void
1329pipeclose(struct pipe *cpipe)
1330{
1331 struct pipe *ppipe;
1332
cb323159 1333 PIPE_LOCK(cpipe);
91447636
A
1334
1335 /*
1336 * If the other side is blocked, wake it up saying that
1337 * we want to close it down.
1338 */
b0d623f7 1339 cpipe->pipe_state &= ~PIPE_DRAIN;
2d21ac55
A
1340 cpipe->pipe_state |= PIPE_EOF;
1341 pipeselwakeup(cpipe, cpipe);
0a7de745 1342
91447636 1343 while (cpipe->pipe_busy) {
2d21ac55 1344 cpipe->pipe_state |= PIPE_WANT;
91447636
A
1345
1346 wakeup(cpipe);
0a7de745 1347 msleep(cpipe, PIPE_MTX(cpipe), PRIBIO, "pipecl", 0);
91447636
A
1348 }
1349
2d21ac55
A
1350#if CONFIG_MACF
1351 /*
1352 * Free the shared pipe label only after the two ends are disconnected.
1353 */
0a7de745 1354 if (cpipe->pipe_label != NULL && cpipe->pipe_peer == NULL) {
2d21ac55 1355 mac_pipe_label_destroy(cpipe);
0a7de745 1356 }
91447636
A
1357#endif
1358
1359 /*
1360 * Disconnect from peer
1361 */
1362 if ((ppipe = cpipe->pipe_peer) != NULL) {
b0d623f7 1363 ppipe->pipe_state &= ~(PIPE_DRAIN);
91447636
A
1364 ppipe->pipe_state |= PIPE_EOF;
1365
1366 pipeselwakeup(ppipe, ppipe);
1367 wakeup(ppipe);
1368
cb323159 1369 KNOTE(&ppipe->pipe_sel.si_note, 1);
91447636 1370
91447636
A
1371 ppipe->pipe_peer = NULL;
1372 }
91447636
A
1373
1374 /*
1375 * free resources
1376 */
cb323159
A
1377
1378 PIPE_UNLOCK(cpipe);
1379
1380 pipepair_destroy_pipe(PIPE_PAIR(cpipe), cpipe);
1381}
1382
1383static int64_t
1384filt_pipelowwat(struct knote *kn, struct pipe *rpipe, int64_t def_lowwat)
1385{
1386 if ((kn->kn_sfflags & NOTE_LOWAT) == 0) {
1387 return def_lowwat;
91447636 1388 }
cb323159
A
1389 if (rpipe->pipe_buffer.size && kn->kn_sdata > MAX_PIPESIZE(rpipe)) {
1390 return MAX_PIPESIZE(rpipe);
ebb1b9f4 1391 }
cb323159 1392 return MAX(kn->kn_sdata, def_lowwat);
91447636
A
1393}
1394
91447636 1395static int
cb323159 1396filt_pipe_draincommon(struct knote *kn, struct pipe *rpipe)
91447636 1397{
cb323159 1398 struct pipe *wpipe = rpipe->pipe_peer;
39037602 1399
39037602
A
1400 if ((rpipe->pipe_state & (PIPE_DRAIN | PIPE_EOF)) ||
1401 (wpipe == NULL) || (wpipe->pipe_state & (PIPE_DRAIN | PIPE_EOF))) {
1402 kn->kn_flags |= EV_EOF;
cb323159
A
1403 return 1;
1404 }
1405
1406 return 0;
1407}
1408
1409static int
1410filt_pipenotsup(struct knote *kn, long hint)
1411{
1412#pragma unused(hint)
1413 struct pipe *rpipe = kn->kn_hook;
1414
1415 return filt_pipe_draincommon(kn, rpipe);
1416}
1417
1418static int
1419filt_pipenotsuptouch(struct knote *kn, struct kevent_qos_s *kev)
1420{
1421 struct pipe *rpipe = kn->kn_hook;
1422 int res;
1423
1424 PIPE_LOCK(rpipe);
1425
1426 /* accept new kevent data (and save off lowat threshold and flag) */
1427 kn->kn_sfflags = kev->fflags;
1428 kn->kn_sdata = kev->data;
1429
1430 /* determine if any event is now deemed fired */
1431 res = filt_pipe_draincommon(kn, rpipe);
1432
1433 PIPE_UNLOCK(rpipe);
1434
1435 return res;
1436}
1437
1438static int
1439filt_pipenotsupprocess(struct knote *kn, struct kevent_qos_s *kev)
1440{
1441 struct pipe *rpipe = kn->kn_hook;
1442 int res;
1443
1444 PIPE_LOCK(rpipe);
1445 res = filt_pipe_draincommon(kn, rpipe);
1446 if (res) {
1447 knote_fill_kevent(kn, kev, 0);
1448 }
1449 PIPE_UNLOCK(rpipe);
1450
1451 return res;
1452}
1453
1454/*ARGSUSED*/
1455static int
1456filt_piperead_common(struct knote *kn, struct kevent_qos_s *kev, struct pipe *rpipe)
1457{
1458 int64_t data = rpipe->pipe_buffer.cnt;
1459 int res = 0;
1460
1461 if (filt_pipe_draincommon(kn, rpipe)) {
1462 res = 1;
39037602 1463 } else {
cb323159 1464 res = data >= filt_pipelowwat(kn, rpipe, 1);
39037602 1465 }
cb323159
A
1466 if (res && kev) {
1467 knote_fill_kevent(kn, kev, data);
1468 }
1469 return res;
39037602 1470}
91447636 1471
39037602
A
1472static int
1473filt_piperead(struct knote *kn, long hint)
1474{
1475#pragma unused(hint)
cb323159 1476 struct pipe *rpipe = kn->kn_hook;
39037602 1477
cb323159 1478 return filt_piperead_common(kn, NULL, rpipe);
39037602 1479}
d9a64523 1480
39037602 1481static int
cb323159 1482filt_pipereadtouch(struct knote *kn, struct kevent_qos_s *kev)
39037602 1483{
cb323159 1484 struct pipe *rpipe = kn->kn_hook;
39037602
A
1485 int retval;
1486
1487 PIPE_LOCK(rpipe);
1488
1489 /* accept new inputs (and save the low water threshold and flag) */
1490 kn->kn_sdata = kev->data;
1491 kn->kn_sfflags = kev->fflags;
39037602
A
1492
1493 /* identify if any events are now fired */
cb323159 1494 retval = filt_piperead_common(kn, NULL, rpipe);
39037602
A
1495
1496 PIPE_UNLOCK(rpipe);
1497
1498 return retval;
1499}
1500
1501static int
cb323159 1502filt_pipereadprocess(struct knote *kn, struct kevent_qos_s *kev)
39037602 1503{
cb323159 1504 struct pipe *rpipe = kn->kn_hook;
39037602
A
1505 int retval;
1506
1507 PIPE_LOCK(rpipe);
cb323159 1508 retval = filt_piperead_common(kn, kev, rpipe);
39037602
A
1509 PIPE_UNLOCK(rpipe);
1510
0a7de745 1511 return retval;
39037602
A
1512}
1513
1514/*ARGSUSED*/
1515static int
cb323159 1516filt_pipewrite_common(struct knote *kn, struct kevent_qos_s *kev, struct pipe *rpipe)
39037602 1517{
cb323159
A
1518 int64_t data = 0;
1519 int res = 0;
39037602 1520
cb323159
A
1521 if (filt_pipe_draincommon(kn, rpipe)) {
1522 res = 1;
1523 } else {
1524 data = MAX_PIPESIZE(rpipe) - rpipe->pipe_buffer.cnt;
1525 res = data >= filt_pipelowwat(kn, rpipe, PIPE_BUF);
39037602 1526 }
cb323159
A
1527 if (res && kev) {
1528 knote_fill_kevent(kn, kev, data);
39037602 1529 }
cb323159 1530 return res;
39037602
A
1531}
1532
1533/*ARGSUSED*/
1534static int
1535filt_pipewrite(struct knote *kn, long hint)
1536{
1537#pragma unused(hint)
cb323159 1538 struct pipe *rpipe = kn->kn_hook;
39037602 1539
cb323159 1540 return filt_pipewrite_common(kn, NULL, rpipe);
39037602
A
1541}
1542
1543
1544static int
cb323159 1545filt_pipewritetouch(struct knote *kn, struct kevent_qos_s *kev)
39037602 1546{
cb323159 1547 struct pipe *rpipe = kn->kn_hook;
39037602
A
1548 int res;
1549
1550 PIPE_LOCK(rpipe);
1551
1552 /* accept new kevent data (and save off lowat threshold and flag) */
1553 kn->kn_sfflags = kev->fflags;
1554 kn->kn_sdata = kev->data;
39037602
A
1555
1556 /* determine if any event is now deemed fired */
cb323159 1557 res = filt_pipewrite_common(kn, NULL, rpipe);
39037602
A
1558
1559 PIPE_UNLOCK(rpipe);
1560
1561 return res;
1562}
1563
1564static int
cb323159 1565filt_pipewriteprocess(struct knote *kn, struct kevent_qos_s *kev)
39037602 1566{
cb323159 1567 struct pipe *rpipe = kn->kn_hook;
39037602
A
1568 int res;
1569
1570 PIPE_LOCK(rpipe);
cb323159 1571 res = filt_pipewrite_common(kn, kev, rpipe);
39037602
A
1572 PIPE_UNLOCK(rpipe);
1573
1574 return res;
1575}
1576
1577/*ARGSUSED*/
1578static int
cb323159
A
1579pipe_kqfilter(struct fileproc *fp, struct knote *kn,
1580 __unused struct kevent_qos_s *kev)
39037602 1581{
cb323159
A
1582 struct pipe *cpipe = (struct pipe *)fp->f_data;
1583 struct pipe *rpipe = &PIPE_PAIR(cpipe)->pp_rpipe;
39037602 1584 int res;
91447636
A
1585
1586 PIPE_LOCK(cpipe);
2d21ac55
A
1587#if CONFIG_MACF
1588 /*
1589 * XXX We should use a per thread credential here; minimally, the
1590 * XXX process credential should have a persistent reference on it
1591 * XXX before being passed in here.
1592 */
cb323159
A
1593 kauth_cred_t cred = vfs_context_ucred(vfs_context_current());
1594 if (mac_pipe_check_kqfilter(cred, kn, cpipe) != 0) {
2d21ac55 1595 PIPE_UNLOCK(cpipe);
cb323159 1596 knote_set_error(kn, EPERM);
39037602 1597 return 0;
2d21ac55
A
1598 }
1599#endif
91447636 1600
cb323159
A
1601 /*
1602 * FreeBSD will fail the attach with EPIPE if the peer pipe is detached,
1603 * however, this isn't a programming error as the other side closing
1604 * could race with the kevent registration.
1605 *
1606 * Attach should only fail for programming mistakes else it will break
1607 * libdispatch.
1608 *
1609 * Like FreeBSD, have a "Neutered" filter that will not fire until
1610 * the pipe dies if the wrong filter is attached to the wrong end.
1611 *
1612 * Knotes are always attached to the "rpipe".
1613 */
91447636
A
1614 switch (kn->kn_filter) {
1615 case EVFILT_READ:
cb323159
A
1616 if (fp->f_flag & FREAD) {
1617 kn->kn_filtid = EVFILTID_PIPE_R;
1618 res = filt_piperead_common(kn, NULL, rpipe);
1619 } else {
1620 kn->kn_filtid = EVFILTID_PIPE_N;
1621 res = filt_pipe_draincommon(kn, rpipe);
1622 }
91447636 1623 break;
39037602 1624
91447636 1625 case EVFILT_WRITE:
cb323159
A
1626 if (fp->f_flag & FWRITE) {
1627 kn->kn_filtid = EVFILTID_PIPE_W;
1628 res = filt_pipewrite_common(kn, NULL, rpipe);
1629 } else {
1630 kn->kn_filtid = EVFILTID_PIPE_N;
1631 res = filt_pipe_draincommon(kn, rpipe);
0a7de745 1632 }
91447636 1633 break;
cb323159 1634
91447636 1635 default:
0a7de745 1636 PIPE_UNLOCK(cpipe);
cb323159 1637 knote_set_error(kn, EINVAL);
39037602 1638 return 0;
91447636
A
1639 }
1640
cb323159
A
1641 kn->kn_hook = rpipe;
1642 KNOTE_ATTACH(&rpipe->pipe_sel.si_note, kn);
91447636
A
1643
1644 PIPE_UNLOCK(cpipe);
39037602 1645 return res;
91447636
A
1646}
1647
1648static void
1649filt_pipedetach(struct knote *kn)
1650{
1651 struct pipe *cpipe = (struct pipe *)kn->kn_fp->f_data;
cb323159 1652 struct pipe *rpipe = &PIPE_PAIR(cpipe)->pp_rpipe;
91447636
A
1653
1654 PIPE_LOCK(cpipe);
cb323159 1655 KNOTE_DETACH(&rpipe->pipe_sel.si_note, kn);
91447636
A
1656 PIPE_UNLOCK(cpipe);
1657}
1658
0c530ab8
A
1659int
1660fill_pipeinfo(struct pipe * cpipe, struct pipe_info * pinfo)
1661{
2d21ac55 1662#if CONFIG_MACF
0a7de745 1663 int error;
0c530ab8 1664#endif
5ba3f43e 1665 struct timespec now;
2d21ac55
A
1666 struct vinfo_stat * ub;
1667 int pipe_size = 0;
1668 int pipe_count;
0c530ab8 1669
0a7de745
A
1670 if (cpipe == NULL) {
1671 return EBADF;
1672 }
0c530ab8 1673 PIPE_LOCK(cpipe);
2d21ac55
A
1674
1675#if CONFIG_MACF
1676 error = mac_pipe_check_stat(kauth_cred_get(), cpipe);
1677 if (error) {
1678 PIPE_UNLOCK(cpipe);
0a7de745 1679 return error;
2d21ac55 1680 }
0c530ab8
A
1681#endif
1682 if (cpipe->pipe_buffer.buffer == 0) {
0a7de745 1683 /*
0c530ab8
A
1684 * must be stat'ing the write fd
1685 */
0a7de745
A
1686 if (cpipe->pipe_peer) {
1687 /*
2d21ac55
A
1688 * the peer still exists, use it's info
1689 */
0a7de745 1690 pipe_size = MAX_PIPESIZE(cpipe->pipe_peer);
2d21ac55
A
1691 pipe_count = cpipe->pipe_peer->pipe_buffer.cnt;
1692 } else {
1693 pipe_count = 0;
1694 }
1695 } else {
0a7de745 1696 pipe_size = MAX_PIPESIZE(cpipe);
2d21ac55 1697 pipe_count = cpipe->pipe_buffer.cnt;
0c530ab8 1698 }
2d21ac55
A
1699 /*
1700 * since peer's buffer is setup ouside of lock
1701 * we might catch it in transient state
1702 */
0a7de745 1703 if (pipe_size == 0) {
2d21ac55 1704 pipe_size = PIPE_SIZE;
0a7de745 1705 }
0c530ab8
A
1706
1707 ub = &pinfo->pipe_stat;
1708
1709 bzero(ub, sizeof(*ub));
2d21ac55
A
1710 ub->vst_mode = S_IFIFO | S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP;
1711 ub->vst_blksize = pipe_size;
1712 ub->vst_size = pipe_count;
0a7de745 1713 if (ub->vst_blksize != 0) {
2d21ac55 1714 ub->vst_blocks = (ub->vst_size + ub->vst_blksize - 1) / ub->vst_blksize;
0a7de745 1715 }
2d21ac55 1716 ub->vst_nlink = 1;
0c530ab8 1717
2d21ac55
A
1718 ub->vst_uid = kauth_getuid();
1719 ub->vst_gid = kauth_getgid();
0c530ab8 1720
5ba3f43e 1721 nanotime(&now);
2d21ac55 1722 ub->vst_atime = now.tv_sec;
5ba3f43e 1723 ub->vst_atimensec = now.tv_nsec;
0c530ab8 1724
2d21ac55 1725 ub->vst_mtime = now.tv_sec;
5ba3f43e 1726 ub->vst_mtimensec = now.tv_nsec;
0c530ab8 1727
2d21ac55 1728 ub->vst_ctime = now.tv_sec;
5ba3f43e 1729 ub->vst_ctimensec = now.tv_nsec;
0c530ab8
A
1730
1731 /*
1732 * Left as 0: st_dev, st_ino, st_nlink, st_rdev, st_flags, st_gen, st_uid, st_gid.
1733 * XXX (st_dev, st_ino) should be unique.
1734 */
1735
cb323159
A
1736 pinfo->pipe_handle = (uint64_t)VM_KERNEL_ADDRHASH((uintptr_t)cpipe);
1737 pinfo->pipe_peerhandle = (uint64_t)VM_KERNEL_ADDRHASH((uintptr_t)(cpipe->pipe_peer));
0c530ab8 1738 pinfo->pipe_status = cpipe->pipe_state;
2d21ac55
A
1739
1740 PIPE_UNLOCK(cpipe);
1741
0a7de745 1742 return 0;
0c530ab8 1743}
b0d623f7
A
1744
1745
0a7de745 1746static int
b0d623f7
A
1747pipe_drain(struct fileproc *fp, __unused vfs_context_t ctx)
1748{
b0d623f7 1749 /* Note: fdlock already held */
f427ee49 1750 struct pipe *ppipe, *cpipe = (struct pipe *)(fp->fp_glob->fg_data);
cb323159
A
1751 boolean_t drain_pipe = FALSE;
1752
1753 /* Check if the pipe is going away */
f427ee49
A
1754 lck_mtx_lock_spin(&fp->fp_glob->fg_lock);
1755 if (os_ref_get_count_raw(&fp->fp_glob->fg_count) == 1) {
cb323159
A
1756 drain_pipe = TRUE;
1757 }
f427ee49 1758 lck_mtx_unlock(&fp->fp_glob->fg_lock);
b0d623f7
A
1759
1760 if (cpipe) {
1761 PIPE_LOCK(cpipe);
cb323159
A
1762
1763 if (drain_pipe) {
1764 cpipe->pipe_state |= PIPE_DRAIN;
1765 cpipe->pipe_state &= ~(PIPE_WANTR | PIPE_WANTW);
1766 }
b0d623f7 1767 wakeup(cpipe);
0a7de745 1768
b0d623f7
A
1769 /* Must wake up peer: a writer sleeps on the read side */
1770 if ((ppipe = cpipe->pipe_peer)) {
cb323159
A
1771 if (drain_pipe) {
1772 ppipe->pipe_state |= PIPE_DRAIN;
1773 ppipe->pipe_state &= ~(PIPE_WANTR | PIPE_WANTW);
1774 }
b0d623f7
A
1775 wakeup(ppipe);
1776 }
0a7de745 1777
b0d623f7
A
1778 PIPE_UNLOCK(cpipe);
1779 return 0;
1780 }
1781
1782 return 1;
1783}