]> git.saurik.com Git - apple/xnu.git/blame - libsyscall/wrappers/spawn/posix_spawn.c
xnu-7195.81.3.tar.gz
[apple/xnu.git] / libsyscall / wrappers / spawn / posix_spawn.c
CommitLineData
39236c6e
A
1/*
2 * Copyright (c) 2006-2012 Apple Inc. All rights reserved.
3 *
4 * @APPLE_LICENSE_HEADER_START@
0a7de745 5 *
39236c6e
A
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. Please obtain a copy of the License at
10 * http://www.opensource.apple.com/apsl/ and read it before using this
11 * file.
0a7de745 12 *
39236c6e
A
13 * The Original Code and all software distributed under the License are
14 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
15 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
16 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
17 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
18 * Please see the License for the specific language governing rights and
19 * limitations under the License.
0a7de745 20 *
39236c6e
A
21 * @APPLE_LICENSE_HEADER_END@
22 */
23
24/*
25 * [SPN] Support for _POSIX_SPAWN
26 */
27
28#define CONFIG_MEMORYSTATUS 1 // <rdar://problem/13604997>
29#include <sys/types.h> /* for user_size_t */
30#include <spawn.h>
31#include <spawn_private.h>
32#include <sys/spawn_internal.h>
33#include <sys/process_policy.h>
34#include <stdlib.h>
35#include <errno.h>
0a7de745 36#include <limits.h> /* for OPEN_MAX, PATH_MAX */
39236c6e
A
37#include <string.h>
38#include <strings.h>
39#include <mach/port.h>
cb323159 40#include <mach/mach_param.h> /* for TASK_PORT_REGISTER_MAX */
39236c6e 41#include <mach/exception_types.h>
3e170ce0 42#include <mach/coalition.h> /* for COALITION_TYPE_MAX */
39037602 43#include <sys/kern_memorystatus.h>
39236c6e
A
44
45/*
46 * posix_spawnattr_init
47 *
48 * Description: Initialize a spawn attributes object attr with default values
49 *
50 * Parameters: attr The spawn attributes object to be
51 * initialized
52 *
53 * Returns: 0 Success
54 * ENOMEM Insufficient memory exists to
55 * initialize the spawn attributes object.
56 *
57 * Note: As an implementation detail, the externally visibily type
58 * posix_spawnattr_t is defined to be a void *, and initialization
59 * involves allocation of a memory object. Subsequent changes to
60 * the spawn attributes may result in reallocation under the
61 * covers.
62 *
63 * Reinitialization of an already initialized spawn attributes
64 * object will result in memory being leaked. Because spawn
65 * attributes are not required to be used in conjunction with a
66 * static initializer, there is no way to distinguish a spawn
67 * attribute with stack garbage from one that's been initialized.
68 * This is arguably an API design error.
69 */
70int
71posix_spawnattr_init(posix_spawnattr_t *attr)
72{
73 _posix_spawnattr_t *psattrp = (_posix_spawnattr_t *)attr;
0a7de745 74 int err = 0;
39236c6e
A
75
76 if ((*psattrp = (_posix_spawnattr_t)malloc(sizeof(struct _posix_spawnattr))) == NULL) {
77 err = ENOMEM;
78 } else {
39236c6e
A
79 /*
80 * The default value of this attribute shall be as if no
81 * flags were set
82 */
83 (*psattrp)->psa_flags = 0;
84
85 /*
86 * The default value of this attribute shall be an empty
87 * signal set
88 */
89 (*psattrp)->psa_sigdefault = 0;
90
91 /* The default value of this attribute is unspecified */
92 (*psattrp)->psa_sigmask = 0;
93
94 /* The default value of this attribute shall be zero */
0a7de745 95 (*psattrp)->psa_pgroup = 0; /* doesn't matter */
39236c6e
A
96
97 /* Default is no binary preferences, i.e. use normal grading */
0a7de745
A
98 memset((*psattrp)->psa_binprefs, 0,
99 sizeof((*psattrp)->psa_binprefs));
f427ee49
A
100 memset((*psattrp)->psa_subcpuprefs, 0xff /* CPU_SUBTYPE_ANY */,
101 sizeof((*psattrp)->psa_subcpuprefs));
39236c6e
A
102
103 /* Default is no port actions to take */
104 (*psattrp)->psa_ports = NULL;
105
106 /*
107 * The default value of this attribute shall be an no
108 * process control on resource starvation
109 */
110 (*psattrp)->psa_pcontrol = 0;
111
112 /*
0a7de745 113 * Initializing the alignment paddings.
39236c6e
A
114 */
115
0a7de745
A
116 (*psattrp)->short_padding = 0;
117 (*psattrp)->flags_padding = 0;
39236c6e 118
fe8ab488
A
119 /* Default is no new apptype requested */
120 (*psattrp)->psa_apptype = POSIX_SPAWN_PROCESS_TYPE_DEFAULT;
39236c6e
A
121
122 /* Jetsam related */
123 (*psattrp)->psa_jetsam_flags = 0;
124 (*psattrp)->psa_priority = -1;
3e170ce0
A
125 (*psattrp)->psa_memlimit_active = -1;
126 (*psattrp)->psa_memlimit_inactive = -1;
39236c6e 127
d9a64523
A
128 /* Default is no thread limit */
129 (*psattrp)->psa_thread_limit = 0;
130
39236c6e
A
131 /* Default is no CPU usage monitor active. */
132 (*psattrp)->psa_cpumonitor_percent = 0;
133 (*psattrp)->psa_cpumonitor_interval = 0;
134
135 /* Default is no MAC policy extensions. */
136 (*psattrp)->psa_mac_extensions = NULL;
fe8ab488 137
3e170ce0
A
138 /* Default is to inherit parent's coalition(s) */
139 (*psattrp)->psa_coalition_info = NULL;
140
490019cf 141 (*psattrp)->psa_persona_info = NULL;
3e170ce0 142
cb323159
A
143 (*psattrp)->psa_posix_cred_info = NULL;
144
3e170ce0
A
145 /*
146 * old coalition field
147 * For backwards compatibility reasons, we set this to 1
148 * which is the first valid coalition id. This will allow
149 * newer user space code to properly spawn processes on
150 * older kernels
151 * (they will just all end up in the same coalition).
152 */
153 (*psattrp)->psa_reserved = 1;
fe8ab488
A
154
155 /* Default is no new clamp */
156 (*psattrp)->psa_qos_clamp = POSIX_SPAWN_PROC_CLAMP_NONE;
3e170ce0
A
157
158 /* Default is no change to role */
159 (*psattrp)->psa_darwin_role = POSIX_SPAWN_DARWIN_ROLE_NONE;
d9a64523
A
160
161 (*psattrp)->psa_max_addr = 0;
f427ee49
A
162
163 (*psattrp)->psa_no_smt = false;
164 (*psattrp)->psa_tecs = false;
165
166 /* Default is no subsystem root path */
167 (*psattrp)->psa_subsystem_root_path = NULL;
168
169 /* Default is no platform given */
170 (*psattrp)->psa_platform = 0;
171
172 /* Default is no option */
173 (*psattrp)->psa_options = PSA_OPTION_NONE;
39236c6e
A
174 }
175
0a7de745 176 return err;
39236c6e
A
177}
178
179
180/*
181 * posix_spawnattr_destroy
182 *
183 * Description: Destroy a spawn attributes object that was previously
184 * initialized via posix_spawnattr_init() by freeing any
185 * memory associated with it and setting it to an invalid value.
186 *
187 * Parameters: attr The spawn attributes object to be
188 * destroyed.
189 *
190 * Returns: 0 Success
191 *
192 * Notes: The destroyed spawn attribute results in the void * pointer
193 * being set to NULL; subsequent use without reinitialization
194 * will result in explicit program failure (rather than merely
195 * "undefined behaviour").
196 *
197 * NOTIMP: Allowed failures (checking NOT required):
198 * EINVAL The value specified by attr is invalid.
199 */
200static int posix_spawn_destroyportactions_np(posix_spawnattr_t *);
3e170ce0 201static int posix_spawn_destroycoalition_info_np(posix_spawnattr_t *);
490019cf 202static int posix_spawn_destroypersona_info_np(posix_spawnattr_t *);
cb323159
A
203static int posix_spawn_destroyposix_cred_info_np(posix_spawnattr_t *);
204static int posix_spawn_destroymacpolicy_info_np(posix_spawnattr_t *);
f427ee49 205static int posix_spawn_destroysubsystem_root_path_np(posix_spawnattr_t *);
39236c6e
A
206
207int
208posix_spawnattr_destroy(posix_spawnattr_t *attr)
209{
210 _posix_spawnattr_t psattr;
211
0a7de745 212 if (attr == NULL || *attr == NULL) {
39236c6e 213 return EINVAL;
0a7de745 214 }
39236c6e
A
215
216 psattr = *(_posix_spawnattr_t *)attr;
217 posix_spawn_destroyportactions_np(attr);
3e170ce0 218 posix_spawn_destroycoalition_info_np(attr);
490019cf 219 posix_spawn_destroypersona_info_np(attr);
cb323159
A
220 posix_spawn_destroyposix_cred_info_np(attr);
221 posix_spawn_destroymacpolicy_info_np(attr);
f427ee49 222 posix_spawn_destroysubsystem_root_path_np(attr);
39236c6e
A
223
224 free(psattr);
225 *attr = NULL;
226
0a7de745 227 return 0;
39236c6e
A
228}
229
230
231/*
232 * posix_spawnattr_setflags
233 *
234 * Description: Set the spawn flags attribute for the spawn attribute object
235 * referred to by 'attr'.
236 *
237 * Parameters: attr The spawn attributes object whose flags
238 * are to be set
239 * flags The flags value to set
240 *
241 * Returns: 0 Success
242 *
243 * NOTIMP: Allowed failures (checking NOT required):
244 * EINVAL The value specified by attr is invalid.
245 * EINVAL The value of the attribute being set is not valid.
246 */
247int
248posix_spawnattr_setflags(posix_spawnattr_t *attr, short flags)
249{
250 _posix_spawnattr_t psattr;
251
0a7de745 252 if (attr == NULL || *attr == NULL) {
39236c6e 253 return EINVAL;
0a7de745 254 }
39236c6e
A
255
256 psattr = *(_posix_spawnattr_t *)attr;
257 psattr->psa_flags = flags;
258
0a7de745 259 return 0;
39236c6e
A
260}
261
262
263/*
264 * posix_spawnattr_getflags
265 *
266 * Description: Retrieve the spawn attributes flag for the spawn attributes
267 * object referenced by 'attr' and place them in the memory
268 * location referenced by 'flagsp'
269 *
270 * Parameters: attr The spawn attributes object whose flags
271 * are to be retrieved
272 * flagsp A pointer to a short value to receive
273 * the flags
274 *
275 * Returns: 0 Success
276 *
277 * Implicit Returns:
278 * *flagps (modified) The flags value from the spawn
279 * attributes object
280 *
281 * NOTIMP: Allowed failures (checking NOT required):
282 * EINVAL The value specified by attr is invalid.
283 * EINVAL The value of the attribute being set is not valid.
284 */
285int
286posix_spawnattr_getflags(const posix_spawnattr_t * __restrict attr,
0a7de745 287 short * __restrict flagsp)
39236c6e
A
288{
289 _posix_spawnattr_t psattr;
290
0a7de745 291 if (attr == NULL || *attr == NULL) {
39236c6e 292 return EINVAL;
0a7de745 293 }
39236c6e
A
294
295 psattr = *(_posix_spawnattr_t *)attr;
296 *flagsp = psattr->psa_flags;
297
0a7de745 298 return 0;
39236c6e
A
299}
300
301
302/*
303 * posix_spawnattr_getsigdefault
304 *
305 * Description: Retrieve the set of signals to be set to default according to
306 * the spawn attribute value referenced by 'attr' and place the
307 * result into the memory containing the sigset_t referenced by
308 * 'sigdefault'
309 *
310 * Parameters: attr The spawn attributes object whose
311 * signal set for default signals is to
312 * be retrieved
313 * sigdefault A pointer to the sigset_t to receive
314 * the signal set
315 *
316 * Returns: 0 Success
317 *
318 * Implicit Returns:
319 * *sigdefault (modified) The signal set of signals to default
320 * from the spawn attributes object
321 */
322int
323posix_spawnattr_getsigdefault(const posix_spawnattr_t * __restrict attr,
0a7de745 324 sigset_t * __restrict sigdefault)
39236c6e
A
325{
326 _posix_spawnattr_t psattr;
327
0a7de745 328 if (attr == NULL || *attr == NULL) {
39236c6e 329 return EINVAL;
0a7de745 330 }
39236c6e
A
331
332 psattr = *(_posix_spawnattr_t *)attr;
333 *sigdefault = psattr->psa_sigdefault;
334
0a7de745 335 return 0;
39236c6e
A
336}
337
338
339/*
340 * posix_spawnattr_getpgroup
341 *
342 * Description: Obtain the value of the spawn process group attribute from the
343 * spawn attributes object referenced by 'attr' and place the
344 * results in the memory location referenced by 'pgroup'
345 *
346 * Parameters: attr The spawn attributes object whose
347 * process group information is to be
348 * retrieved
349 * pgroup A pointer to the pid_t to receive the
350 * process group
351 *
352 * Returns: 0 Success
353 *
354 * Implicit Returns:
355 * *pgroup (modified) The process group information from the
356 * spawn attributes object
357 */
358int
359posix_spawnattr_getpgroup(const posix_spawnattr_t * __restrict attr,
0a7de745 360 pid_t * __restrict pgroup)
39236c6e
A
361{
362 _posix_spawnattr_t psattr;
363
0a7de745 364 if (attr == NULL || *attr == NULL) {
39236c6e 365 return EINVAL;
0a7de745 366 }
39236c6e
A
367
368 psattr = *(_posix_spawnattr_t *)attr;
369 *pgroup = psattr->psa_pgroup;
370
0a7de745 371 return 0;
39236c6e
A
372}
373
374
375/*
376 * posix_spawnattr_getsigmask
377 *
378 * Description: Obtain the value of the spawn signal mask attribute from the
379 * spawn attributes object referenced by 'attr' and place the
380 * result into the memory containing the sigset_t referenced by
381 * 'sigmask'
382 *
383 * Parameters: attr The spawn attributes object whose
384 * signal set for masked signals is to
385 * be retrieved
386 * sigmask A pointer to the sigset_t to receive
387 * the signal set
388 *
389 * Returns: 0 Success
390 *
391 * Implicit Returns:
392 * *sigmask (modified) The signal set of signals to mask
393 * from the spawn attributes object
394 */
395int
396posix_spawnattr_getsigmask(const posix_spawnattr_t * __restrict attr,
0a7de745 397 sigset_t * __restrict sigmask)
39236c6e
A
398{
399 _posix_spawnattr_t psattr;
400
0a7de745 401 if (attr == NULL || *attr == NULL) {
39236c6e 402 return EINVAL;
0a7de745 403 }
39236c6e
A
404
405 psattr = *(_posix_spawnattr_t *)attr;
406 *sigmask = psattr->psa_sigmask;
407
0a7de745 408 return 0;
39236c6e
A
409}
410
411/*
412 * posix_spawnattr_getbinpref_np
413 *
0a7de745
A
414 * Description: Obtain the value of the spawn binary preferences attribute from
415 * the spawn attributes object referenced by 'attr' and place the
39236c6e
A
416 * result into the memory referenced by 'pref'.
417 *
418 * Parameters: attr The spawn attributes object whose
419 * binary preferences are to be retrieved
420 * count The size of the cpu_type_t array
421 * pref An array of cpu types
422 * ocount The actual number copied
423 *
424 * Returns: 0 No binary preferences found
0a7de745
A
425 * > 0 The number of cpu types (less than
426 * count) copied over from 'attr'.
39236c6e
A
427 *
428 * Implicit Returns:
0a7de745 429 * *pref (modified) The binary preferences array
39236c6e
A
430 * from the spawn attributes object
431 */
432int
433posix_spawnattr_getbinpref_np(const posix_spawnattr_t * __restrict attr,
0a7de745 434 size_t count, cpu_type_t *pref, size_t * __restrict ocount)
39236c6e
A
435{
436 _posix_spawnattr_t psattr;
437 int i = 0;
438
f427ee49
A
439 if (attr == NULL || *attr == NULL || pref == NULL) {
440 return EINVAL;
441 }
442
443 psattr = *(_posix_spawnattr_t *)attr;
444 for (i = 0; i < count && i < NBINPREFS; i++) {
445 pref[i] = psattr->psa_binprefs[i];
446 }
447
448 if (ocount) {
449 *ocount = i;
450 }
451 return 0;
452}
453
454/*
455 * posix_spawnattr_getarchpref_np
456 *
457 * Description: Obtain the value of the spawn binary preferences attribute from
458 * the spawn attributes object referenced by 'attr' and place the
459 * result into the memory referenced by 'pref' and 'subpref'.
460 *
461 * Parameters: attr The spawn attributes object whose
462 * binary preferences are to be retrieved
463 * count The size of the cpu_type_t array
464 * pref An array of cpu types
465 * subpref An array of subcpu types
466 * ocount The actual number copied
467 *
468 * Returns: 0 No cpu/subcpu preferences found
469 * > 0 The number of types (less than
470 * count) copied over from 'attr'.
471 *
472 * Implicit Returns:
473 * *pref (modified) The cpu preferences array
474 * from the spawn attributes object
475 * *subpref (modified) The subcpu preferences array
476 * from the spawn attributes object
477 */
478int
479posix_spawnattr_getarchpref_np(const posix_spawnattr_t * __restrict attr,
480 size_t count, cpu_type_t *pref, cpu_subtype_t *subpref, size_t * __restrict ocount)
481{
482 _posix_spawnattr_t psattr;
483 int i = 0;
484
485 if (attr == NULL || *attr == NULL || pref == NULL || subpref == NULL) {
39236c6e 486 return EINVAL;
0a7de745 487 }
39236c6e
A
488
489 psattr = *(_posix_spawnattr_t *)attr;
f427ee49 490 for (i = 0; i < count && i < NBINPREFS; i++) {
39236c6e 491 pref[i] = psattr->psa_binprefs[i];
f427ee49 492 subpref[i] = psattr->psa_subcpuprefs[i];
39236c6e
A
493 }
494
0a7de745 495 if (ocount) {
39236c6e 496 *ocount = i;
0a7de745 497 }
39236c6e
A
498 return 0;
499}
500
501
502/*
503 * posix_spawnattr_getpcontrol_np
504 *
505 * Description: Retrieve the process control property set default according to
506 * the spawn attribute value referenced by 'attr' and place the
507 * result into the memory containing the control referenced by
508 * 'pcontrol'
509 *
510 * Parameters: attr The spawn attributes object whose
511 * signal set for default signals is to
512 * be retrieved
513 * pcontrol A pointer to an int to receive
514 * the process control info
515 *
516 * Returns: 0 Success
517 *
518 * Implicit Returns:
519 * *pcontrol (modified) The signal set of signals to default
520 * from the spawn attributes object
521 */
522int
523posix_spawnattr_getpcontrol_np(const posix_spawnattr_t * __restrict attr,
0a7de745 524 int * __restrict pcontrol)
39236c6e
A
525{
526 _posix_spawnattr_t psattr;
527
0a7de745 528 if (attr == NULL || *attr == NULL) {
39236c6e 529 return EINVAL;
0a7de745 530 }
39236c6e
A
531
532 psattr = *(_posix_spawnattr_t *)attr;
533 *pcontrol = psattr->psa_pcontrol;
534
0a7de745 535 return 0;
39236c6e
A
536}
537
538/*
539 * posix_spawnattr_getprocesstype_np
540 *
541 * Description: Retrieve the process specific behaviors and app launch types
542 * spawn attribute value referenced by 'attr' and place the
543 * result into the memory containing the control referenced by
544 * 'proctype'
545 *
546 * Parameters: attr The spawn attributes object whose
547 * signal set for default signals is to
548 * be retrieved
549 * proctype A pointer to an int to receive
550 * the process type info
551 *
552 * Returns: 0 Success
553 *
554 * Implicit Returns:
555 * *proctype (modified) The process type set to value
556 * from the spawn attributes object
557 */
558int
559posix_spawnattr_getprocesstype_np(const posix_spawnattr_t * __restrict attr,
0a7de745 560 int * __restrict proctype)
39236c6e
A
561{
562 _posix_spawnattr_t psattr;
563
0a7de745 564 if (attr == NULL || *attr == NULL) {
39236c6e 565 return EINVAL;
0a7de745 566 }
39236c6e
A
567
568 psattr = *(_posix_spawnattr_t *)attr;
569 *proctype = psattr->psa_apptype;
570
0a7de745 571 return 0;
39236c6e
A
572}
573/*
574 * posix_spawnattr_setsigdefault
575 *
576 * Description: Set the set of signals to be set to default for the spawn
577 * attribute value referenced by 'attr' from the memory
578 * containing the sigset_t referenced by 'sigdefault'
579 *
580 * Parameters: attr The spawn attributes object whose
581 * signal set for default signals is to
582 * be set
583 * sigdefault A pointer to the sigset_t from which to
584 * obtain the signal set
585 *
586 * Returns: 0 Success
587 */
588int
589posix_spawnattr_setsigdefault(posix_spawnattr_t * __restrict attr,
0a7de745 590 const sigset_t * __restrict sigdefault)
39236c6e
A
591{
592 _posix_spawnattr_t psattr;
593
0a7de745 594 if (attr == NULL || *attr == NULL) {
39236c6e 595 return EINVAL;
0a7de745 596 }
39236c6e
A
597
598 psattr = *(_posix_spawnattr_t *)attr;
599 psattr->psa_sigdefault = *sigdefault;
600
0a7de745 601 return 0;
39236c6e
A
602}
603
604
605/*
606 * posix_spawnattr_setpgroup
607 *
608 * Description: Set the value of the spawn process group attribute for the
609 * spawn attributes object referenced by 'attr' from the value
610 * of 'pgroup'
611 *
612 * Parameters: attr The spawn attributes object for which
613 * the process group information is to be
614 * set
615 * pgroup The process group to set
616 *
617 * Returns: 0 Success
618 */
619int
620posix_spawnattr_setpgroup(posix_spawnattr_t * attr, pid_t pgroup)
621{
622 _posix_spawnattr_t psattr;
623
0a7de745 624 if (attr == NULL || *attr == NULL) {
39236c6e 625 return EINVAL;
0a7de745 626 }
39236c6e
A
627
628 psattr = *(_posix_spawnattr_t *)attr;
629 psattr->psa_pgroup = pgroup;
630
0a7de745 631 return 0;
39236c6e
A
632}
633
634
635/*
636 * posix_spawnattr_setsigmask
637 *
638 * Description: Set the set of signals to be masked for the spawn attribute
639 * value referenced by 'attr' from the memory containing the
640 * sigset_t referenced by 'sigmask'
641 *
642 * Parameters: attr The spawn attributes object whose
643 * signal set for masked signals is to
644 * be set
645 * sigmask A pointer to the sigset_t from which to
646 * obtain the signal set
647 *
648 * Returns: 0 Success
649 */
650int
651posix_spawnattr_setsigmask(posix_spawnattr_t * __restrict attr,
0a7de745 652 const sigset_t * __restrict sigmask)
39236c6e
A
653{
654 _posix_spawnattr_t psattr;
655
0a7de745 656 if (attr == NULL || *attr == NULL) {
39236c6e 657 return EINVAL;
0a7de745 658 }
39236c6e
A
659
660 psattr = *(_posix_spawnattr_t *)attr;
661 psattr->psa_sigmask = *sigmask;
662
0a7de745 663 return 0;
39236c6e
A
664}
665
666
667/*
668 * posix_spawnattr_setbinpref_np
669 *
670 * Description: Set the universal binary preferences for the spawn attribute
671 * value referenced by 'attr' from the memory containing the
672 * cpu_type_t array referenced by 'pref', size of 'count'
673 *
674 * Parameters: attr The spawn attributes object whose
0a7de745
A
675 * binary preferences are to be set
676 * count Size of the array pointed to by 'pref'
677 * pref cpu_type_t array of binary preferences
39236c6e
A
678 * ocount The actual number copied
679 *
680 * Returns: 0 No preferences copied
0a7de745 681 * > 0 Number of preferences copied
39236c6e 682 *
0a7de745
A
683 * Note: The posix_spawnattr_t currently only holds four cpu_type_t's.
684 * If the caller provides more preferences than this limit, they
685 * will be ignored, as reflected in the return value.
39236c6e
A
686 */
687int
688posix_spawnattr_setbinpref_np(posix_spawnattr_t * __restrict attr,
0a7de745 689 size_t count, cpu_type_t *pref, size_t * __restrict ocount)
39236c6e
A
690{
691 _posix_spawnattr_t psattr;
692 int i = 0;
693
f427ee49 694 if (attr == NULL || *attr == NULL || pref == NULL) {
39236c6e 695 return EINVAL;
0a7de745 696 }
39236c6e
A
697
698 psattr = *(_posix_spawnattr_t *)attr;
f427ee49 699 for (i = 0; i < count && i < NBINPREFS; i++) {
39236c6e 700 psattr->psa_binprefs[i] = pref[i];
f427ee49 701 psattr->psa_subcpuprefs[i] = CPU_SUBTYPE_ANY;
39236c6e
A
702 }
703
704 /* return number of binprefs copied over */
0a7de745 705 if (ocount) {
39236c6e 706 *ocount = i;
0a7de745 707 }
f427ee49
A
708
709 for (; i < NBINPREFS; i++) {
710 psattr->psa_binprefs[i] = 0;
711 psattr->psa_subcpuprefs[i] = CPU_SUBTYPE_ANY;
712 }
713
39236c6e
A
714 return 0;
715}
716
f427ee49
A
717/*
718 * posix_spawnattr_setarchpref_np
719 *
720 * Description: Set the universal binary preferences for the spawn attribute
721 * value referenced by 'attr' from the memory containing the
722 * cpu_type_t array referenced by 'pref', the cpu_subtype_t array
723 * referenced by 'subpref' and size of 'count'
724 *
725 * Parameters: attr The spawn attributes object whose
726 * binary preferences are to be set
727 * count Size of the array pointed to by 'pref'
728 * pref cpu_type_t array of cpu binary preferences
729 * subpref cpu_subtype_t array of subcpu binary preferences
730 * ocount The actual number copied
731 *
732 * Returns: 0 No preferences copied
733 * > 0 Number of preferences copied
734 *
735 * Note: The posix_spawnattr_t currently only holds four
736 * cpu_type_t/cpu_subtype_t pairs.
737 * If the caller provides more preferences than this limit, they
738 * will be ignored, as reflected in the return value.
739 */
740int
741posix_spawnattr_setarchpref_np(posix_spawnattr_t * __restrict attr,
742 size_t count, cpu_type_t *pref, cpu_subtype_t *subpref,
743 size_t * __restrict ocount)
744{
745 _posix_spawnattr_t psattr;
746 int i = 0;
747
748 if (attr == NULL || *attr == NULL || pref == NULL || subpref == NULL) {
749 return EINVAL;
750 }
751
752 psattr = *(_posix_spawnattr_t *)attr;
753 for (i = 0; i < count && i < NBINPREFS; i++) {
754 psattr->psa_binprefs[i] = pref[i];
755 psattr->psa_subcpuprefs[i] = subpref[i];
756 }
757
758 /* return number of binprefs copied over */
759 if (ocount) {
760 *ocount = i;
761 }
762
763 for (; i < NBINPREFS; i++) {
764 psattr->psa_binprefs[i] = 0;
765 psattr->psa_subcpuprefs[i] = CPU_SUBTYPE_ANY;
766 }
767
768 return 0;
769}
39236c6e
A
770
771/*
772 * posix_spawnattr_setpcontrol_np
773 *
774 * Description: Set the process control property according to
775 * attribute value referenced by 'attr' from the memory
776 * containing the int value 'pcontrol'
777 *
778 * Parameters: attr The spawn attributes object whose
779 * signal set for default signals is to
780 * be set
781 * pcontrol An int value of the process control info
782 *
783 * Returns: 0 Success
784 */
785int
786posix_spawnattr_setpcontrol_np(posix_spawnattr_t * __restrict attr,
0a7de745 787 const int pcontrol)
39236c6e
A
788{
789 _posix_spawnattr_t psattr;
790
0a7de745 791 if (attr == NULL || *attr == NULL) {
39236c6e 792 return EINVAL;
0a7de745 793 }
39236c6e
A
794
795 psattr = *(_posix_spawnattr_t *)attr;
796 psattr->psa_pcontrol = pcontrol;
797
0a7de745 798 return 0;
39236c6e
A
799}
800
801
802/*
803 * posix_spawnattr_setprocesstype_np
804 *
805 * Description: Set the process specific behaviors and app launch type
806 * attribute value referenced by 'attr' from the memory
807 * containing the int value 'proctype'
808 *
809 * Parameters: attr The spawn attributes object whose
810 * signal set for default signals is to
811 * be set
812 * proctype An int value of the process type info
813 *
814 * Returns: 0 Success
815 */
816int
817posix_spawnattr_setprocesstype_np(posix_spawnattr_t * __restrict attr,
0a7de745 818 const int proctype)
39236c6e
A
819{
820 _posix_spawnattr_t psattr;
821
0a7de745 822 if (attr == NULL || *attr == NULL) {
39236c6e 823 return EINVAL;
0a7de745 824 }
39236c6e
A
825
826 psattr = *(_posix_spawnattr_t *)attr;
827 psattr->psa_apptype = proctype;
828
0a7de745 829 return 0;
39236c6e
A
830}
831
832/*
833 * posix_spawn_createportactions_np
834 * Description: create a new posix_spawn_port_actions struct and link
0a7de745 835 * it into the posix_spawnattr.
39236c6e
A
836 */
837static int
838posix_spawn_createportactions_np(posix_spawnattr_t *attr)
839{
840 _posix_spawnattr_t psattr;
841 _posix_spawn_port_actions_t acts;
842
0a7de745 843 if (attr == NULL || *attr == NULL) {
39236c6e 844 return EINVAL;
0a7de745 845 }
39236c6e
A
846
847 psattr = *(_posix_spawnattr_t *)attr;
848 acts = (_posix_spawn_port_actions_t)malloc(PS_PORT_ACTIONS_SIZE(2));
0a7de745 849 if (acts == NULL) {
39236c6e 850 return ENOMEM;
0a7de745
A
851 }
852
39236c6e
A
853 acts->pspa_alloc = 2;
854 acts->pspa_count = 0;
855
856 psattr->psa_ports = acts;
857 return 0;
858}
859
860/*
861 * posix_spawn_growportactions_np
0a7de745 862 * Description: Enlarge the size of portactions if necessary
39236c6e
A
863 */
864static int
865posix_spawn_growportactions_np(posix_spawnattr_t *attr)
866{
867 _posix_spawnattr_t psattr;
0a7de745 868 _posix_spawn_port_actions_t acts;
39236c6e 869
0a7de745 870 if (attr == NULL || *attr == NULL) {
39236c6e 871 return EINVAL;
0a7de745 872 }
39236c6e
A
873
874 psattr = *(_posix_spawnattr_t *)attr;
0a7de745
A
875 acts = psattr->psa_ports;
876 if (acts == NULL) {
39236c6e 877 return EINVAL;
0a7de745
A
878 }
879
39236c6e 880 /* Double number of port actions allocated for */
5c9f4661 881 int newnum = 0;
0a7de745 882 if (os_mul_overflow(acts->pspa_alloc, 2, &newnum)) {
5c9f4661 883 return ENOMEM;
0a7de745 884 }
5c9f4661 885 size_t newsize = PS_PORT_ACTIONS_SIZE(newnum);
0a7de745 886 if (newsize == 0) {
5c9f4661 887 return ENOMEM;
0a7de745 888 }
5c9f4661
A
889
890 acts = realloc(acts, newsize);
0a7de745 891 if (acts == NULL) {
39236c6e 892 return ENOMEM;
0a7de745
A
893 }
894
39236c6e
A
895 acts->pspa_alloc = newnum;
896 psattr->psa_ports = acts;
897 return 0;
898}
899
900/*
901 * posix_spawn_destroyportactions_np
902 * Description: clean up portactions struct in posix_spawnattr_t attr
903 */
904static int
905posix_spawn_destroyportactions_np(posix_spawnattr_t *attr)
906{
907 _posix_spawnattr_t psattr;
0a7de745 908 _posix_spawn_port_actions_t acts;
39236c6e 909
0a7de745 910 if (attr == NULL || *attr == NULL) {
39236c6e 911 return EINVAL;
0a7de745 912 }
39236c6e
A
913
914 psattr = *(_posix_spawnattr_t *)attr;
0a7de745
A
915 acts = psattr->psa_ports;
916 if (acts == NULL) {
39236c6e 917 return EINVAL;
0a7de745
A
918 }
919
39236c6e
A
920 free(acts);
921 return 0;
922}
923
3e170ce0
A
924/*
925 * posix_spawn_destroycoalition_info_np
926 * Description: clean up coalition_info struct in posix_spawnattr_t attr
927 */
928static int
929posix_spawn_destroycoalition_info_np(posix_spawnattr_t *attr)
930{
931 _posix_spawnattr_t psattr;
932 struct _posix_spawn_coalition_info *coal_info;
933
0a7de745 934 if (attr == NULL || *attr == NULL) {
3e170ce0 935 return EINVAL;
0a7de745 936 }
3e170ce0
A
937
938 psattr = *(_posix_spawnattr_t *)attr;
939 coal_info = psattr->psa_coalition_info;
0a7de745 940 if (coal_info == NULL) {
3e170ce0 941 return EINVAL;
0a7de745 942 }
3e170ce0
A
943
944 psattr->psa_coalition_info = NULL;
945 free(coal_info);
946 return 0;
947}
948
490019cf
A
949/*
950 * posix_spawn_destroypersona_info_np
951 * Description: clean up persona_info struct in posix_spawnattr_t attr
952 */
953static int
954posix_spawn_destroypersona_info_np(posix_spawnattr_t *attr)
955{
956 _posix_spawnattr_t psattr;
957 struct _posix_spawn_persona_info *persona;
958
0a7de745 959 if (attr == NULL || *attr == NULL) {
490019cf 960 return EINVAL;
0a7de745 961 }
490019cf
A
962
963 psattr = *(_posix_spawnattr_t *)attr;
964 persona = psattr->psa_persona_info;
0a7de745 965 if (persona == NULL) {
490019cf 966 return EINVAL;
0a7de745 967 }
490019cf
A
968
969 psattr->psa_persona_info = NULL;
970 free(persona);
971 return 0;
972}
973
cb323159
A
974/*
975 * posix_spawn_destroyposix_cred_info_np
976 * Description: clean up posix_cred_info struct in posix_spawnattr_t attr
977 */
978static int
979posix_spawn_destroyposix_cred_info_np(posix_spawnattr_t *attr)
980{
981 _posix_spawnattr_t psattr;
982 struct _posix_spawn_posix_cred_info *pspci;
983
984 if (attr == NULL || *attr == NULL) {
985 return EINVAL;
986 }
987
988 psattr = *(_posix_spawnattr_t *)attr;
989 pspci = psattr->psa_posix_cred_info;
990 if (pspci == NULL) {
991 return EINVAL;
992 }
993
994 psattr->psa_posix_cred_info = NULL;
995 free(pspci);
996 return 0;
997}
998
f427ee49
A
999/*
1000 * posix_spawn_set_subsystem_root_path
1001 * Description: Set path as the subsystem root path for attr; clears if NULL
1002 */
1003int
1004posix_spawnattr_set_subsystem_root_path_np(posix_spawnattr_t *attr, char *path)
1005{
1006 _posix_spawnattr_t psattr;
1007 char * buf = NULL;
1008 char * old_buf;
1009 size_t bytes;
1010
1011 if (attr == NULL || *attr == NULL) {
1012 return EINVAL;
1013 }
1014
1015 psattr = *(_posix_spawnattr_t *)attr;
1016
1017 if (path) {
1018 buf = malloc(MAXPATHLEN);
1019
1020 if (buf == NULL) {
1021 return ENOMEM;
1022 }
1023
1024 bytes = strlcpy(buf, path, MAXPATHLEN);
1025
1026 if (bytes >= MAXPATHLEN) {
1027 free(buf);
1028 return ENAMETOOLONG;
1029 }
1030 }
1031
1032 old_buf = psattr->psa_subsystem_root_path;
1033 psattr->psa_subsystem_root_path = buf;
1034
1035 free(old_buf);
1036
1037 return 0;
1038}
1039
1040/*
1041 * posix_spawn_destroy_subsystem_root_path_np
1042 * Description: clean up subsystem_root_path string in posix_spawnattr_t attr
1043 */
1044static int
1045posix_spawn_destroysubsystem_root_path_np(posix_spawnattr_t *attr)
1046{
1047 _posix_spawnattr_t psattr;
1048 char * subsystem_root_path;
1049
1050 if (attr == NULL || *attr == NULL) {
1051 return EINVAL;
1052 }
1053
1054 psattr = *(_posix_spawnattr_t *)attr;
1055 subsystem_root_path = psattr->psa_subsystem_root_path;
1056
1057 if (subsystem_root_path == NULL) {
1058 return EINVAL;
1059 }
1060
1061 psattr->psa_subsystem_root_path = NULL;
1062 free(subsystem_root_path);
1063 return 0;
1064}
1065
1066/*
1067 * posix_spawnattr_set_platform_np
1068 * Description: sets the platform in posix_spawnattr_t attr
1069 *
1070 * To be implemented.
1071 */
1072int
1073posix_spawnattr_set_platform_np(posix_spawnattr_t *attr, int platform, uint32_t flags)
1074{
1075 _posix_spawnattr_t psattr;
1076
1077 if (attr == NULL || *attr == NULL) {
1078 return EINVAL;
1079 }
1080
1081 psattr = *(_posix_spawnattr_t *)attr;
1082 psattr->psa_platform = platform;
1083
1084 (void)flags;
1085 return 0;
1086}
1087
1088/*
1089 * posix_spawnattr_disable_ptr_auth_a_keys_np
1090 * Description: Set flag to disable A keys for Ptr Auth
1091 */
1092int
1093posix_spawnattr_disable_ptr_auth_a_keys_np(posix_spawnattr_t *attr, uint32_t flags)
1094{
1095 _posix_spawnattr_t psattr;
1096
1097 if (attr == NULL || *attr == NULL) {
1098 return EINVAL;
1099 }
1100
1101 psattr = *(_posix_spawnattr_t *)attr;
1102
1103 psattr->psa_options |= PSA_OPTION_PLUGIN_HOST_DISABLE_A_KEYS;
1104 (void)flags;
1105 return 0;
1106}
1107
39236c6e
A
1108/*
1109 * posix_spawn_appendportaction_np
1110 * Description: append a port action, grow the array if necessary
1111 */
1112static int
1113posix_spawn_appendportaction_np(posix_spawnattr_t *attr, _ps_port_action_t *act)
1114{
1115 _posix_spawnattr_t psattr;
1116 _posix_spawn_port_actions_t acts;
1117
1118 if (attr == NULL || *attr == NULL || act == NULL) {
1119 return EINVAL;
1120 }
1121
1122 psattr = *(_posix_spawnattr_t *)attr;
1123 acts = psattr->psa_ports;
1124
1125 // Have any port actions been created yet?
1126 if (acts == NULL) {
1127 int err = posix_spawn_createportactions_np(attr);
1128 if (err) {
1129 return err;
1130 }
1131 acts = psattr->psa_ports;
1132 }
1133
1134 // Is there enough room?
1135 if (acts->pspa_alloc == acts->pspa_count) {
1136 int err = posix_spawn_growportactions_np(attr);
1137 if (err) {
1138 return err;
1139 }
1140 acts = psattr->psa_ports;
1141 }
1142
1143 // Add this action to next spot in array
1144 acts->pspa_actions[acts->pspa_count] = *act;
1145 acts->pspa_count++;
1146
1147 return 0;
1148}
1149
1150/*
1151 * posix_spawnattr_setspecialport_np
1152 *
0a7de745 1153 * Description: Set a new value for a mach special port in the spawned task.
39236c6e
A
1154 *
1155 * Parameters: attr The spawn attributes object for the
0a7de745
A
1156 * new process
1157 * new_port The new value for the special port
1158 * which The particular port to be set
1159 * (see task_set_special_port for details)
39236c6e
A
1160 *
1161 * Returns: 0 Success
0a7de745 1162 * ENOMEM Couldn't allocate memory
39236c6e 1163 */
0a7de745 1164int
39236c6e 1165posix_spawnattr_setspecialport_np(
0a7de745
A
1166 posix_spawnattr_t *attr,
1167 mach_port_t new_port,
1168 int which)
39236c6e
A
1169{
1170 _ps_port_action_t action = {
1171 .port_type = PSPA_SPECIAL,
1172 .new_port = new_port,
1173 .which = which,
1174 };
1175 return posix_spawn_appendportaction_np(attr, &action);
1176}
1177
ea3f0419
A
1178/*
1179 * posix_spawnattr_setsuidcredport_np
1180 *
1181 * Description: Set an suid cred port to be used to execute with a different UID.
1182 *
1183 * Parameters: attr The spawn attributes object for the
1184 * new process
1185 * port The suid cred port
1186 *
1187 * Returns: 0 Success
1188 */
1189int
1190posix_spawnattr_setsuidcredport_np(posix_spawnattr_t *attr, mach_port_t port)
1191{
1192 _ps_port_action_t action = {
1193 .port_type = PSPA_SUID_CRED,
1194 .new_port = port,
1195 };
1196 return posix_spawn_appendportaction_np(attr, &action);
1197}
1198
39236c6e
A
1199/*
1200 * posix_spawnattr_setexceptionports_np
1201 *
1202 * Description: Set a new port for a set of exception ports in the spawned task.
1203 *
1204 * Parameters: attr The spawn attributes object for the
0a7de745
A
1205 * new process
1206 * mask A bitfield indicating which exceptions
1207 * to associate the port with
1208 * new_port The new value for the exception port
1209 * behavior The default behavior for the port
1210 * flavor The default flavor for the port
1211 * (see task_set_exception_ports)
39236c6e
A
1212 *
1213 * Returns: 0 Success
1214 */
0a7de745 1215int
39236c6e 1216posix_spawnattr_setexceptionports_np(
0a7de745
A
1217 posix_spawnattr_t *attr,
1218 exception_mask_t mask,
1219 mach_port_t new_port,
1220 exception_behavior_t behavior,
1221 thread_state_flavor_t flavor)
39236c6e
A
1222{
1223 _ps_port_action_t action = {
1224 .port_type = PSPA_EXCEPTION,
1225 .mask = mask,
1226 .new_port = new_port,
1227 .behavior = behavior,
1228 .flavor = flavor,
1229 };
1230 return posix_spawn_appendportaction_np(attr, &action);
1231}
1232
1233/*
1234 * posix_spawnattr_setauditsessionport_np
1235 *
1236 * Description: Set the audit session port rights attribute in the spawned task.
1237 * This is used to securely set the audit session information for
1238 * the new task.
1239 *
1240 * Parameters: attr The spawn attributes object for the
0a7de745
A
1241 * new process
1242 * au_sessionport The audit session send port right
39236c6e
A
1243 *
1244 * Returns: 0 Success
1245 */
0a7de745 1246int
39236c6e 1247posix_spawnattr_setauditsessionport_np(
0a7de745
A
1248 posix_spawnattr_t *attr,
1249 mach_port_t au_sessionport)
39236c6e
A
1250{
1251 _ps_port_action_t action = {
1252 .port_type = PSPA_AU_SESSION,
1253 .new_port = au_sessionport,
1254 };
1255 return posix_spawn_appendportaction_np(attr, &action);
1256}
1257
1258
1259/*
1260 * posix_spawn_file_actions_init
1261 *
1262 * Description: Initialize a spawn file actions object attr with default values
1263 *
1264 * Parameters: file_actions The spawn file actions object to be
1265 * initialized
1266 *
1267 * Returns: 0 Success
1268 * ENOMEM Insufficient memory exists to
1269 * initialize the spawn file actions
1270 * object.
1271 *
1272 * Note: As an implementation detail, the externally visibily type
1273 * posix_spawn_file_actions_t is defined to be a void *, and
1274 * initialization involves allocation of a memory object.
1275 * Subsequent changes to the spawn file actions may result in
1276 * reallocation under the covers.
1277 *
1278 * Reinitialization of an already initialized spawn file actions
1279 * object will result in memory being leaked. Because spawn
1280 * file actions are not required to be used in conjunction with a
1281 * static initializer, there is no way to distinguish a spawn
1282 * file actions with stack garbage from one that's been
1283 * initialized. This is arguably an API design error.
1284 */
1285int
1286posix_spawn_file_actions_init(posix_spawn_file_actions_t *file_actions)
1287{
1288 _posix_spawn_file_actions_t *psactsp = (_posix_spawn_file_actions_t *)file_actions;
0a7de745 1289 int err = 0;
39236c6e
A
1290
1291 if ((*psactsp = (_posix_spawn_file_actions_t)malloc(PSF_ACTIONS_SIZE(PSF_ACTIONS_INIT_COUNT))) == NULL) {
1292 err = ENOMEM;
1293 } else {
1294 (*psactsp)->psfa_act_alloc = PSF_ACTIONS_INIT_COUNT;
1295 (*psactsp)->psfa_act_count = 0;
1296 }
1297
0a7de745 1298 return err;
39236c6e
A
1299}
1300
1301
1302/*
1303 * posix_spawn_file_actions_destroy
1304 *
1305 * Description: Destroy a spawn file actions object that was previously
1306 * initialized via posix_spawn_file_actions_init() by freeing any
1307 * memory associated with it and setting it to an invalid value.
1308 *
1309 * Parameters: attr The spawn file actions object to be
1310 * destroyed.
1311 *
1312 * Returns: 0 Success
1313 *
1314 * Notes: The destroyed spawn file actions results in the void * pointer
1315 * being set to NULL; subsequent use without reinitialization
1316 * will result in explicit program failure (rather than merely
1317 * "undefined behaviour").
1318 *
1319 * NOTIMP: Allowed failures (checking NOT required):
1320 * EINVAL The value specified by file_actions is invalid.
1321 */
1322int
1323posix_spawn_file_actions_destroy(posix_spawn_file_actions_t *file_actions)
1324{
1325 _posix_spawn_file_actions_t psacts;
1326
0a7de745 1327 if (file_actions == NULL || *file_actions == NULL) {
39236c6e 1328 return EINVAL;
0a7de745 1329 }
39236c6e
A
1330
1331 psacts = *(_posix_spawn_file_actions_t *)file_actions;
1332 free(psacts);
1333 *file_actions = NULL;
1334
0a7de745 1335 return 0;
39236c6e
A
1336}
1337
1338
1339/*
1340 * _posix_spawn_file_actions_grow
1341 *
1342 * Description: Grow the available list of file actions associated with the
1343 * pointer to the structure provided; replace the contents of the
1344 * pointer as a side effect.
1345 *
1346 * Parameters: psactsp Pointer to _posix_spawn_file_actions_t
1347 * to grow
1348 *
1349 * Returns: 0 Success
1350 * ENOMEM Insufficient memory for operation
1351 *
1352 * Notes: This code is common to all posix_spawn_file_actions_*()
1353 * functions, since we use a naieve data structure implementation
1354 * at present. Future optimization will likely change this.
1355 */
1356static int
1357_posix_spawn_file_actions_grow(_posix_spawn_file_actions_t *psactsp)
1358{
5c9f4661 1359 int newnum = 0;
0a7de745 1360 if (os_mul_overflow((*psactsp)->psfa_act_alloc, 2, &newnum)) {
5c9f4661 1361 return ENOMEM;
0a7de745 1362 }
5c9f4661
A
1363
1364 size_t newsize = PSF_ACTIONS_SIZE(newnum);
0a7de745 1365 if (newsize == 0) {
5c9f4661 1366 return ENOMEM;
0a7de745 1367 }
39236c6e
A
1368
1369 /*
1370 * XXX may want to impose an administrative limit here; POSIX does
1371 * XXX not provide for an administrative error return in this case,
1372 * XXX so it's probably acceptable to just fail catastrophically
1373 * XXX instead of implementing one.
1374 */
5c9f4661
A
1375 _posix_spawn_file_actions_t new_psacts;
1376 if ((new_psacts = (_posix_spawn_file_actions_t)realloc((*psactsp), newsize)) == NULL) {
1377 return ENOMEM;
39236c6e 1378 }
5c9f4661 1379 new_psacts->psfa_act_alloc = newnum;
39236c6e
A
1380 *psactsp = new_psacts;
1381
5c9f4661 1382 return 0;
39236c6e
A
1383}
1384
1385
1386/*
1387 * posix_spawn_file_actions_addopen
1388 *
1389 * Description: Add an open action to the object referenced by 'file_actions'
1390 * that will cause the file named by 'path' to be attempted to be
1391 * opened with flags 'oflag' and mode 'mode', and, if successful,
1392 * return as descriptor 'filedes' to the spawned process.
1393 *
1394 * Parameters: file_actions File action object to augment
1395 * filedes fd that open is to use
1396 * path path to file to open
1397 * oflag open file flags
1398 * mode open file mode
1399 *
1400 * Returns: 0 Success
1401 * EBADF The value specified by fildes is
1402 * negative or greater than or equal to
1403 * {OPEN_MAX}.
1404 * ENOMEM Insufficient memory exists to add to
1405 * the spawn file actions object.
1406 *
1407 * NOTIMP: Allowed failures (checking NOT required):
1408 * EINVAL The value specified by file_actions is invalid.
1409 */
1410int
1411posix_spawn_file_actions_addopen(
0a7de745
A
1412 posix_spawn_file_actions_t * __restrict file_actions,
1413 int filedes, const char * __restrict path, int oflag,
1414 mode_t mode)
39236c6e
A
1415{
1416 _posix_spawn_file_actions_t *psactsp;
1417 _psfa_action_t *psfileact;
1418
0a7de745 1419 if (file_actions == NULL || *file_actions == NULL) {
39236c6e 1420 return EINVAL;
0a7de745 1421 }
39236c6e
A
1422
1423 psactsp = (_posix_spawn_file_actions_t *)file_actions;
1424 /* Range check; required by POSIX */
0a7de745
A
1425 if (filedes < 0 || filedes >= OPEN_MAX) {
1426 return EBADF;
1427 }
39236c6e
A
1428
1429 /* If we do not have enough slots, grow the structure */
1430 if ((*psactsp)->psfa_act_count == (*psactsp)->psfa_act_alloc) {
1431 /* need to grow file actions structure */
0a7de745
A
1432 if (_posix_spawn_file_actions_grow(psactsp)) {
1433 return ENOMEM;
1434 }
39236c6e
A
1435 }
1436
1437 /*
1438 * Allocate next available slot and fill it out
1439 */
1440 psfileact = &(*psactsp)->psfa_act_acts[(*psactsp)->psfa_act_count++];
1441
1442 psfileact->psfaa_type = PSFA_OPEN;
1443 psfileact->psfaa_filedes = filedes;
1444 psfileact->psfaa_openargs.psfao_oflag = oflag;
1445 psfileact->psfaa_openargs.psfao_mode = mode;
1446 strlcpy(psfileact->psfaa_openargs.psfao_path, path, PATH_MAX);
1447
0a7de745 1448 return 0;
39236c6e
A
1449}
1450
1451
1452/*
1453 * posix_spawn_file_actions_addclose
1454 *
1455 * Description: Add a close action to the object referenced by 'file_actions'
1456 * that will cause the file referenced by 'filedes' to be
1457 * attempted to be closed in the spawned process.
1458 *
1459 * Parameters: file_actions File action object to augment
1460 * filedes fd to close
1461 *
1462 * Returns: 0 Success
1463 * EBADF The value specified by fildes is
1464 * negative or greater than or equal to
1465 * {OPEN_MAX}.
1466 * ENOMEM Insufficient memory exists to add to
1467 * the spawn file actions object.
1468 *
1469 * NOTIMP: Allowed failures (checking NOT required):
1470 * EINVAL The value specified by file_actions is invalid.
1471 */
1472int
1473posix_spawn_file_actions_addclose(posix_spawn_file_actions_t *file_actions,
0a7de745 1474 int filedes)
39236c6e
A
1475{
1476 _posix_spawn_file_actions_t *psactsp;
1477 _psfa_action_t *psfileact;
1478
0a7de745 1479 if (file_actions == NULL || *file_actions == NULL) {
39236c6e 1480 return EINVAL;
0a7de745 1481 }
39236c6e
A
1482
1483 psactsp = (_posix_spawn_file_actions_t *)file_actions;
1484 /* Range check; required by POSIX */
0a7de745
A
1485 if (filedes < 0 || filedes >= OPEN_MAX) {
1486 return EBADF;
1487 }
39236c6e
A
1488
1489 /* If we do not have enough slots, grow the structure */
1490 if ((*psactsp)->psfa_act_count == (*psactsp)->psfa_act_alloc) {
1491 /* need to grow file actions structure */
0a7de745
A
1492 if (_posix_spawn_file_actions_grow(psactsp)) {
1493 return ENOMEM;
1494 }
39236c6e
A
1495 }
1496
1497 /*
1498 * Allocate next available slot and fill it out
1499 */
1500 psfileact = &(*psactsp)->psfa_act_acts[(*psactsp)->psfa_act_count++];
1501
1502 psfileact->psfaa_type = PSFA_CLOSE;
1503 psfileact->psfaa_filedes = filedes;
1504
0a7de745 1505 return 0;
39236c6e
A
1506}
1507
1508
1509/*
1510 * posix_spawn_file_actions_adddup2
1511 *
1512 * Description: Add a dup2 action to the object referenced by 'file_actions'
1513 * that will cause the file referenced by 'filedes' to be
1514 * attempted to be dup2'ed to the descriptor 'newfiledes' in the
1515 * spawned process.
1516 *
1517 * Parameters: file_actions File action object to augment
1518 * filedes fd to dup2
1519 * newfiledes fd to dup2 it to
1520 *
1521 * Returns: 0 Success
1522 * EBADF The value specified by either fildes
1523 * or by newfiledes is negative or greater
1524 * than or equal to {OPEN_MAX}.
1525 * ENOMEM Insufficient memory exists to add to
1526 * the spawn file actions object.
1527 *
1528 * NOTIMP: Allowed failures (checking NOT required):
1529 * EINVAL The value specified by file_actions is invalid.
1530 */
1531int
1532posix_spawn_file_actions_adddup2(posix_spawn_file_actions_t *file_actions,
0a7de745 1533 int filedes, int newfiledes)
39236c6e
A
1534{
1535 _posix_spawn_file_actions_t *psactsp;
1536 _psfa_action_t *psfileact;
1537
0a7de745 1538 if (file_actions == NULL || *file_actions == NULL) {
39236c6e 1539 return EINVAL;
0a7de745 1540 }
39236c6e
A
1541
1542 psactsp = (_posix_spawn_file_actions_t *)file_actions;
1543 /* Range check; required by POSIX */
1544 if (filedes < 0 || filedes >= OPEN_MAX ||
0a7de745
A
1545 newfiledes < 0 || newfiledes >= OPEN_MAX) {
1546 return EBADF;
1547 }
39236c6e
A
1548
1549 /* If we do not have enough slots, grow the structure */
1550 if ((*psactsp)->psfa_act_count == (*psactsp)->psfa_act_alloc) {
1551 /* need to grow file actions structure */
0a7de745
A
1552 if (_posix_spawn_file_actions_grow(psactsp)) {
1553 return ENOMEM;
1554 }
39236c6e
A
1555 }
1556
1557 /*
1558 * Allocate next available slot and fill it out
1559 */
1560 psfileact = &(*psactsp)->psfa_act_acts[(*psactsp)->psfa_act_count++];
1561
1562 psfileact->psfaa_type = PSFA_DUP2;
1563 psfileact->psfaa_filedes = filedes;
cb323159
A
1564 psfileact->psfaa_dup2args.psfad_newfiledes = newfiledes;
1565
1566 return 0;
1567}
1568
1569/*
1570 * posix_spawn_file_actions_add_fileportdup2_np
1571 *
1572 * Description: Add a dup2 action to the object referenced by 'file_actions'
1573 * that will cause the file referenced by 'fileport' to be
1574 * attempted to be dup2'ed to the descriptor 'newfiledes' in the
1575 * spawned process.
1576 *
1577 * Parameters: file_actions File action object to augment
1578 * filedes fileport to dup2
1579 * newfiledes fd to dup2 it to
1580 *
1581 * Returns: 0 Success
1582 * EBADF fileport isn't a valid port, or the
1583 * value specified by newfiledes is
1584 * negative or greater than or equal to
1585 * {OPEN_MAX}.
1586 * ENOMEM Insufficient memory exists to add to
1587 * the spawn file actions object.
1588 *
1589 * NOTIMP: Allowed failures (checking NOT required):
1590 * EINVAL The value specified by file_actions is invalid.
1591 */
1592int
1593posix_spawn_file_actions_add_fileportdup2_np(
1594 posix_spawn_file_actions_t *file_actions,
1595 mach_port_t fileport, int newfiledes)
1596{
1597 _posix_spawn_file_actions_t *psactsp;
1598 _psfa_action_t *psfileact;
1599
1600 if (file_actions == NULL || *file_actions == NULL) {
1601 return EINVAL;
1602 }
1603
1604 psactsp = (_posix_spawn_file_actions_t *)file_actions;
1605 /* Range check; required by POSIX */
1606 if (!MACH_PORT_VALID(fileport) ||
1607 newfiledes < 0 || newfiledes >= OPEN_MAX) {
1608 return EBADF;
1609 }
1610
1611 /* If we do not have enough slots, grow the structure */
1612 if ((*psactsp)->psfa_act_count == (*psactsp)->psfa_act_alloc) {
1613 /* need to grow file actions structure */
1614 if (_posix_spawn_file_actions_grow(psactsp)) {
1615 return ENOMEM;
1616 }
1617 }
1618
1619 /*
1620 * Allocate next available slot and fill it out
1621 */
1622 psfileact = &(*psactsp)->psfa_act_acts[(*psactsp)->psfa_act_count++];
1623
1624 psfileact->psfaa_type = PSFA_FILEPORT_DUP2;
1625 psfileact->psfaa_fileport = fileport;
1626 psfileact->psfaa_dup2args.psfad_newfiledes = newfiledes;
39236c6e 1627
0a7de745 1628 return 0;
39236c6e
A
1629}
1630
1631/*
1632 * posix_spawn_file_actions_addinherit_np
1633 *
1634 * Description: Add the "inherit" action to the object referenced by
1635 * 'file_actions' that will cause the file referenced by
1636 * 'filedes' to continue to be available in the spawned
1637 * process via the same descriptor.
1638 *
1639 * Inheritance is the normal default behaviour for
1640 * file descriptors across exec and spawn; but if the
1641 * POSIX_SPAWN_CLOEXEC_DEFAULT flag is set, the usual
1642 * default is reversed for the purposes of the spawn
1643 * invocation. Any pre-existing descriptors that
1644 * need to be made available to the spawned process can
1645 * be marked explicitly as 'inherit' via this interface.
1646 * Otherwise they will be automatically closed.
1647 *
1648 * Note that any descriptors created via the other file
1649 * actions interfaces are automatically marked as 'inherit'.
1650 *
1651 * Parameters: file_actions File action object to augment
1652 * filedes fd to inherit.
1653 *
1654 * Returns: 0 Success
1655 * EBADF The value specified by fildes is
1656 * negative or greater than or equal to
1657 * {OPEN_MAX}.
1658 * ENOMEM Insufficient memory exists to add to
1659 * the spawn file actions object.
1660 *
1661 * NOTIMP: Allowed failures (checking NOT required):
1662 * EINVAL The value specified by file_actions is invalid.
1663 */
1664int
1665posix_spawn_file_actions_addinherit_np(posix_spawn_file_actions_t *file_actions,
0a7de745 1666 int filedes)
39236c6e
A
1667{
1668 _posix_spawn_file_actions_t *psactsp;
1669 _psfa_action_t *psfileact;
1670
0a7de745
A
1671 if (file_actions == NULL || *file_actions == NULL) {
1672 return EINVAL;
1673 }
39236c6e
A
1674
1675 psactsp = (_posix_spawn_file_actions_t *)file_actions;
1676 /* Range check; required by POSIX */
0a7de745
A
1677 if (filedes < 0 || filedes >= OPEN_MAX) {
1678 return EBADF;
1679 }
39236c6e 1680
0a7de745 1681#if defined(POSIX_SPAWN_CLOEXEC_DEFAULT) // TODO: delete this check
39236c6e
A
1682 /* If we do not have enough slots, grow the structure */
1683 if ((*psactsp)->psfa_act_count == (*psactsp)->psfa_act_alloc) {
1684 /* need to grow file actions structure */
0a7de745
A
1685 if (_posix_spawn_file_actions_grow(psactsp)) {
1686 return ENOMEM;
1687 }
39236c6e
A
1688 }
1689
1690 /*
1691 * Allocate next available slot and fill it out
1692 */
1693 psfileact = &(*psactsp)->psfa_act_acts[(*psactsp)->psfa_act_count++];
1694
1695 psfileact->psfaa_type = PSFA_INHERIT;
1696 psfileact->psfaa_filedes = filedes;
1697#endif
0a7de745 1698 return 0;
39236c6e
A
1699}
1700
cb323159
A
1701
1702/*
1703 * posix_spawn_file_actions_addchdir_np
1704 *
1705 * Description: Add a chdir action to the object referenced by 'file_actions'
1706 * that will cause the current working directory to attempt to be changed
1707 * to that referenced by 'path' in the spawned process.
1708 *
1709 * Parameters: file_actions File action object to augment
1710 * path path of the desired working directory
1711 *
1712 * Returns: 0 Success
1713 * ENOMEM Insufficient memory exists to add to
1714 * the spawn file actions object.
1715 * ENAMETOOLONG The supplied path exceeded PATH_MAX.
1716 *
1717 * NOTIMP: Allowed failures (checking NOT required):
1718 * EINVAL The value specified by file_actions is invalid.
1719 */
1720int
1721posix_spawn_file_actions_addchdir_np(
1722 posix_spawn_file_actions_t * __restrict file_actions,
1723 const char * __restrict path)
1724{
1725 _posix_spawn_file_actions_t *psactsp;
1726 _psfa_action_t *psfileact;
1727
1728 if (file_actions == NULL || *file_actions == NULL) {
1729 return EINVAL;
1730 }
1731
1732 psactsp = (_posix_spawn_file_actions_t *)file_actions;
1733
1734 /* If we do not have enough slots, grow the structure */
1735 if ((*psactsp)->psfa_act_count == (*psactsp)->psfa_act_alloc) {
1736 /* need to grow file actions structure */
1737 if (_posix_spawn_file_actions_grow(psactsp)) {
1738 return ENOMEM;
1739 }
1740 }
1741
1742 /*
1743 * Allocate next available slot and fill it out
1744 */
1745 psfileact = &(*psactsp)->psfa_act_acts[(*psactsp)->psfa_act_count++];
1746
1747 psfileact->psfaa_type = PSFA_CHDIR;
1748 if (strlcpy(psfileact->psfaa_chdirargs.psfac_path, path, PATH_MAX) >= PATH_MAX) {
1749 (*psactsp)->psfa_act_count--;
1750 return ENAMETOOLONG;
1751 }
1752
1753 return 0;
1754}
1755
1756
1757/*
1758 * posix_spawn_file_actions_fchdir_np
1759 *
1760 * Description: Add a fchdir action to the object referenced by 'file_actions'
1761 * that will cause the current working directory to attempt to be changed
1762 * to that referenced by the descriptor 'filedes' in the spawned process.
1763 *
1764 * Parameters: file_actions File action object to augment
1765 * filedes fd to chdir to
1766 *
1767 * Returns: 0 Success
1768 * EBADF The value specified by either fildes is negative or
1769 * greater than or equal to {OPEN_MAX}.
1770 * ENOMEM Insufficient memory exists to add to
1771 * the spawn file actions object.
1772 *
1773 * NOTIMP: Allowed failures (checking NOT required):
1774 * EINVAL The value specified by file_actions is invalid.
1775 */
1776int
1777posix_spawn_file_actions_addfchdir_np(posix_spawn_file_actions_t *file_actions,
1778 int filedes)
1779{
1780 _posix_spawn_file_actions_t *psactsp;
1781 _psfa_action_t *psfileact;
1782
1783 if (file_actions == NULL || *file_actions == NULL) {
1784 return EINVAL;
1785 }
1786
1787 psactsp = (_posix_spawn_file_actions_t *)file_actions;
1788 /* Range check; in spirit of POSIX */
1789 if (filedes < 0 || filedes >= OPEN_MAX) {
1790 return EBADF;
1791 }
1792
1793 /* If we do not have enough slots, grow the structure */
1794 if ((*psactsp)->psfa_act_count == (*psactsp)->psfa_act_alloc) {
1795 /* need to grow file actions structure */
1796 if (_posix_spawn_file_actions_grow(psactsp)) {
1797 return ENOMEM;
1798 }
1799 }
1800
1801 /*
1802 * Allocate next available slot and fill it out
1803 */
1804 psfileact = &(*psactsp)->psfa_act_acts[(*psactsp)->psfa_act_count++];
1805
1806 psfileact->psfaa_type = PSFA_FCHDIR;
1807 psfileact->psfaa_filedes = filedes;
1808
1809 return 0;
1810}
1811
39236c6e
A
1812int
1813posix_spawnattr_setcpumonitor_default(posix_spawnattr_t * __restrict attr)
1814{
0a7de745 1815 return posix_spawnattr_setcpumonitor(attr, PROC_POLICY_CPUMON_DEFAULTS, 0);
39236c6e
A
1816}
1817
1818int
1819posix_spawnattr_setcpumonitor(posix_spawnattr_t * __restrict attr,
0a7de745 1820 uint64_t percent, uint64_t interval)
39236c6e
A
1821{
1822 _posix_spawnattr_t psattr;
1823
0a7de745
A
1824 if (attr == NULL || *attr == NULL) {
1825 return EINVAL;
1826 }
39236c6e
A
1827
1828 psattr = *(_posix_spawnattr_t *)attr;
1829
1830 psattr->psa_cpumonitor_percent = percent;
1831 psattr->psa_cpumonitor_interval = interval;
1832
0a7de745 1833 return 0;
39236c6e
A
1834}
1835
1836int
1837posix_spawnattr_getcpumonitor(posix_spawnattr_t * __restrict attr,
0a7de745 1838 uint64_t *percent, uint64_t *interval)
39236c6e
A
1839{
1840 _posix_spawnattr_t psattr;
1841
0a7de745
A
1842 if (attr == NULL || *attr == NULL) {
1843 return EINVAL;
1844 }
39236c6e
A
1845
1846 psattr = *(_posix_spawnattr_t *)attr;
1847
1848 *percent = psattr->psa_cpumonitor_percent;
1849 *interval = psattr->psa_cpumonitor_interval;
1850
0a7de745 1851 return 0;
39236c6e
A
1852}
1853
cb323159 1854#if (TARGET_OS_IPHONE && !TARGET_OS_SIMULATOR)
5ba3f43e
A
1855/*
1856 * posix_spawnattr_setjetsam
1857 *
1858 * Description: Set jetsam attributes for the spawn attribute object
1859 * referred to by 'attr'.
1860 *
1861 * Parameters: flags The flags value to set
1862 * priority Relative jetsam priority
1863 * memlimit Value in megabytes; a memory footprint
1864 * above this level may result in termination.
1865 * Implies both active and inactive limits.
1866 *
1867 * Returns: 0 Success
1868 *
1869 * Note: to be deprecated (not available on desktop)
1870 *
1871 */
1872int
1873posix_spawnattr_setjetsam(posix_spawnattr_t * __restrict attr,
0a7de745 1874 short flags, int priority, int memlimit)
5ba3f43e
A
1875{
1876 short flags_ext = flags;
1877
0a7de745
A
1878 if (flags & POSIX_SPAWN_JETSAM_MEMLIMIT_FATAL) {
1879 flags_ext |= POSIX_SPAWN_JETSAM_MEMLIMIT_ACTIVE_FATAL;
1880 flags_ext |= POSIX_SPAWN_JETSAM_MEMLIMIT_INACTIVE_FATAL;
1881 } else {
1882 flags_ext &= ~POSIX_SPAWN_JETSAM_MEMLIMIT_ACTIVE_FATAL;
1883 flags_ext &= ~POSIX_SPAWN_JETSAM_MEMLIMIT_INACTIVE_FATAL;
1884 }
5ba3f43e 1885
0a7de745 1886 return posix_spawnattr_setjetsam_ext(attr, flags_ext, priority, memlimit, memlimit);
5ba3f43e 1887}
cb323159 1888#endif /* (TARGET_OS_IPHONE && !TARGET_OS_SIMULATOR) */
39236c6e 1889
39037602
A
1890/*
1891 * posix_spawnattr_setjetsam_ext
1892 *
1893 * Description: Set jetsam attributes for the spawn attribute object
1894 * referred to by 'attr'.
1895 *
1896 * Parameters: flags The flags value to set
1897 * priority Relative jetsam priority
1898 * memlimit_active Value in megabytes; memory footprint
1899 * above this level while process is
1900 * active may result in termination.
1901 * memlimit_inactive Value in megabytes; memory footprint
1902 * above this level while process is
1903 * inactive may result in termination.
1904 *
1905 * Returns: 0 Success
1906 */
1907int
1908posix_spawnattr_setjetsam_ext(posix_spawnattr_t * __restrict attr,
0a7de745 1909 short flags, int priority, int memlimit_active, int memlimit_inactive)
39037602
A
1910{
1911 _posix_spawnattr_t psattr;
1912
0a7de745 1913 if (attr == NULL || *attr == NULL) {
39037602 1914 return EINVAL;
0a7de745 1915 }
39037602
A
1916
1917 psattr = *(_posix_spawnattr_t *)attr;
1918
1919 psattr->psa_jetsam_flags = flags;
1920 psattr->psa_jetsam_flags |= POSIX_SPAWN_JETSAM_SET;
1921 psattr->psa_priority = priority;
1922 psattr->psa_memlimit_active = memlimit_active;
1923 psattr->psa_memlimit_inactive = memlimit_inactive;
1924
0a7de745 1925 return 0;
39037602
A
1926}
1927
d9a64523
A
1928int
1929posix_spawnattr_set_threadlimit_ext(posix_spawnattr_t * __restrict attr,
0a7de745 1930 int thread_limit)
d9a64523
A
1931{
1932 _posix_spawnattr_t psattr;
1933
0a7de745 1934 if (attr == NULL || *attr == NULL) {
d9a64523 1935 return EINVAL;
0a7de745 1936 }
d9a64523
A
1937
1938 psattr = *(_posix_spawnattr_t *)attr;
1939
1940 psattr->psa_thread_limit = thread_limit;
1941
0a7de745 1942 return 0;
d9a64523
A
1943}
1944
39236c6e
A
1945
1946/*
1947 * posix_spawnattr_set_importancewatch_port_np
1948 *
1949 * Description: Mark ports referred to by these rights
0a7de745 1950 * to boost the new task instead of their current task
39236c6e
A
1951 * for the spawn attribute object referred to by 'attr'.
1952 * Ports must be valid at posix_spawn time. They will NOT be
1953 * consumed by the kernel, so they must be deallocated after the spawn returns.
1954 * (If you are SETEXEC-ing, they are cleaned up by the exec operation).
0a7de745 1955 *
39236c6e
A
1956 * The maximum number of watch ports allowed is defined by POSIX_SPAWN_IMPORTANCE_PORT_COUNT.
1957 *
1958 * Parameters: count Number of ports in portarray
1959 * portarray Array of rights
1960 *
1961 * Returns: 0 Success
1962 * EINVAL Bad port count
1963 * ENOMEM Insufficient memory exists to add to
1964 * the spawn port actions object.
1965 */
0a7de745 1966int
39236c6e 1967posix_spawnattr_set_importancewatch_port_np(posix_spawnattr_t * __restrict attr,
0a7de745 1968 int count, mach_port_t portarray[])
39236c6e
A
1969{
1970 int err = 0, i;
1971
1972 if (count < 0 || count > POSIX_SPAWN_IMPORTANCE_PORT_COUNT) {
1973 return EINVAL;
1974 }
1975
1976 for (i = 0; i < count; i++) {
1977 _ps_port_action_t action = {
1978 .port_type = PSPA_IMP_WATCHPORTS,
1979 .new_port = portarray[i],
1980 };
cb323159 1981 err = posix_spawn_appendportaction_np(attr, &action);
39236c6e
A
1982 if (err) {
1983 break;
1984 }
1985 }
1986 return err;
1987}
1988
cb323159
A
1989int
1990posix_spawnattr_set_registered_ports_np(posix_spawnattr_t * __restrict attr,
1991 mach_port_t portarray[], uint32_t count)
1992{
1993 int err = 0;
1994
1995 if (count > TASK_PORT_REGISTER_MAX) {
1996 return EINVAL;
1997 }
1998
1999 for (uint32_t i = 0; i < count; i++) {
2000 _ps_port_action_t action = {
2001 .port_type = PSPA_REGISTERED_PORTS,
2002 .new_port = portarray[i],
2003 };
2004 err = posix_spawn_appendportaction_np(attr, &action);
2005 if (err) {
2006 break;
2007 }
2008 }
2009 return err;
2010}
39236c6e 2011
f427ee49
A
2012int
2013posix_spawnattr_set_ptrauth_task_port_np(posix_spawnattr_t * __restrict attr,
2014 mach_port_t port)
2015{
2016 int err = 0;
2017
2018 _ps_port_action_t action = {
2019 .port_type = PSPA_PTRAUTH_TASK_PORT,
2020 .new_port = port,
2021 };
2022
2023 err = posix_spawn_appendportaction_np(attr, &action);
2024 return err;
2025}
39236c6e
A
2026
2027static
2028_ps_mac_policy_extension_t *
2029posix_spawnattr_macpolicyinfo_lookup(_posix_spawn_mac_policy_extensions_t psmx, const char *policyname)
2030{
2031 int i;
2032
0a7de745 2033 if (psmx == NULL) {
39236c6e 2034 return NULL;
0a7de745
A
2035 }
2036
39236c6e
A
2037 for (i = 0; i < psmx->psmx_count; i++) {
2038 _ps_mac_policy_extension_t *extension = &psmx->psmx_extensions[i];
0a7de745 2039 if (strcmp(extension->policyname, policyname) == 0) {
39236c6e 2040 return extension;
0a7de745 2041 }
39236c6e
A
2042 }
2043 return NULL;
2044}
2045
2046int
2047posix_spawnattr_getmacpolicyinfo_np(const posix_spawnattr_t * __restrict attr,
0a7de745 2048 const char *policyname, void **datap, size_t *datalenp)
39236c6e
A
2049{
2050 _posix_spawnattr_t psattr;
2051 _ps_mac_policy_extension_t *extension;
2052
0a7de745 2053 if (attr == NULL || *attr == NULL || policyname == NULL || datap == NULL) {
39236c6e 2054 return EINVAL;
0a7de745 2055 }
39236c6e
A
2056
2057 psattr = *(_posix_spawnattr_t *)attr;
2058 extension = posix_spawnattr_macpolicyinfo_lookup(psattr->psa_mac_extensions, policyname);
0a7de745 2059 if (extension == NULL) {
39236c6e 2060 return ESRCH;
0a7de745 2061 }
39236c6e 2062 *datap = (void *)(uintptr_t)extension->data;
fe8ab488
A
2063 if (datalenp != NULL) {
2064 *datalenp = (size_t)extension->datalen;
2065 }
39236c6e
A
2066 return 0;
2067}
2068
2069int
2070posix_spawnattr_setmacpolicyinfo_np(posix_spawnattr_t * __restrict attr,
0a7de745 2071 const char *policyname, void *data, size_t datalen)
39236c6e
A
2072{
2073 _posix_spawnattr_t psattr;
2074 _posix_spawn_mac_policy_extensions_t psmx;
2075 _ps_mac_policy_extension_t *extension;
2076
0a7de745 2077 if (attr == NULL || *attr == NULL || policyname == NULL) {
39236c6e 2078 return EINVAL;
0a7de745 2079 }
39236c6e
A
2080
2081 psattr = *(_posix_spawnattr_t *)attr;
2082 psmx = psattr->psa_mac_extensions;
2083 extension = posix_spawnattr_macpolicyinfo_lookup(psattr->psa_mac_extensions, policyname);
2084 if (extension != NULL) {
2085 extension->data = (uintptr_t)data;
2086 extension->datalen = datalen;
2087 return 0;
0a7de745 2088 } else if (psmx == NULL) {
39236c6e 2089 psmx = psattr->psa_mac_extensions = malloc(PS_MAC_EXTENSIONS_SIZE(PS_MAC_EXTENSIONS_INIT_COUNT));
0a7de745 2090 if (psmx == NULL) {
39236c6e 2091 return ENOMEM;
0a7de745 2092 }
39236c6e
A
2093 psmx->psmx_alloc = PS_MAC_EXTENSIONS_INIT_COUNT;
2094 psmx->psmx_count = 0;
0a7de745 2095 } else if (psmx->psmx_count == psmx->psmx_alloc) {
5c9f4661 2096 int newnum = 0;
0a7de745 2097 if (os_mul_overflow(psmx->psmx_alloc, 2, &newnum)) {
5c9f4661 2098 return ENOMEM;
0a7de745 2099 }
5c9f4661 2100 size_t extsize = PS_MAC_EXTENSIONS_SIZE(newnum);
0a7de745 2101 if (extsize == 0) {
5c9f4661 2102 return ENOMEM;
0a7de745 2103 }
5c9f4661 2104 psmx = psattr->psa_mac_extensions = reallocf(psmx, extsize);
0a7de745 2105 if (psmx == NULL) {
39236c6e 2106 return ENOMEM;
0a7de745 2107 }
5c9f4661 2108 psmx->psmx_alloc = newnum;
39236c6e
A
2109 }
2110 extension = &psmx->psmx_extensions[psmx->psmx_count];
2111 strlcpy(extension->policyname, policyname, sizeof(extension->policyname));
2112 extension->data = (uintptr_t)data;
2113 extension->datalen = datalen;
2114 psmx->psmx_count += 1;
2115 return 0;
2116}
2117
cb323159
A
2118/*
2119 * posix_spawn_destroymacpolicy_info_np
2120 * Description: cleanup the macpolicy struct in posix_spawnattr_t attr
2121 */
2122static int
2123posix_spawn_destroymacpolicy_info_np(posix_spawnattr_t *attr)
2124{
2125 _posix_spawnattr_t psattr;
2126 _posix_spawn_mac_policy_extensions_t psmx;
2127
2128 if (attr == NULL || *attr == NULL) {
2129 return EINVAL;
2130 }
2131
2132 psattr = *(_posix_spawnattr_t *)attr;
2133 psmx = psattr->psa_mac_extensions;
2134 if (psmx == NULL) {
2135 return EINVAL;
2136 }
2137
2138 psattr->psa_mac_extensions = NULL;
2139 free(psmx);
2140 return 0;
2141}
2142
0a7de745
A
2143int
2144posix_spawnattr_setcoalition_np(const posix_spawnattr_t * __restrict attr,
2145 uint64_t coalitionid, int type, int role)
fe8ab488
A
2146{
2147 _posix_spawnattr_t psattr;
3e170ce0 2148 struct _posix_spawn_coalition_info *coal_info;
fe8ab488
A
2149
2150 if (attr == NULL || *attr == NULL) {
2151 return EINVAL;
2152 }
0a7de745 2153 if (type < 0 || type > COALITION_TYPE_MAX) {
3e170ce0 2154 return EINVAL;
0a7de745 2155 }
fe8ab488
A
2156
2157 psattr = *(_posix_spawnattr_t *)attr;
3e170ce0
A
2158
2159 coal_info = psattr->psa_coalition_info;
2160 if (!coal_info) {
2161 coal_info = (struct _posix_spawn_coalition_info *)malloc(sizeof(*coal_info));
0a7de745 2162 if (!coal_info) {
3e170ce0 2163 return ENOMEM;
0a7de745 2164 }
3e170ce0
A
2165 memset(coal_info, 0, sizeof(*coal_info));
2166 psattr->psa_coalition_info = coal_info;
2167 }
2168
2169 coal_info->psci_info[type].psci_id = coalitionid;
2170 coal_info->psci_info[type].psci_role = role;
fe8ab488
A
2171
2172 return 0;
2173}
2174
2175
0a7de745
A
2176int
2177posix_spawnattr_set_qos_clamp_np(const posix_spawnattr_t * __restrict attr, uint64_t qos_clamp)
fe8ab488
A
2178{
2179 _posix_spawnattr_t psattr;
2180
2181 if (attr == NULL || *attr == NULL) {
2182 return EINVAL;
2183 }
2184
0a7de745 2185 if (qos_clamp >= POSIX_SPAWN_PROC_CLAMP_LAST) {
fe8ab488 2186 return EINVAL;
0a7de745 2187 }
fe8ab488
A
2188
2189 psattr = *(_posix_spawnattr_t *)attr;
2190 psattr->psa_qos_clamp = qos_clamp;
2191
2192 return 0;
2193}
2194
2195int
2196posix_spawnattr_get_qos_clamp_np(const posix_spawnattr_t * __restrict attr, uint64_t * __restrict qos_clampp)
2197{
2198 _posix_spawnattr_t psattr;
2199
2200 if (attr == NULL || *attr == NULL) {
2201 return EINVAL;
2202 }
2203
2204 psattr = *(_posix_spawnattr_t *)attr;
2205 *qos_clampp = psattr->psa_qos_clamp;
2206
0a7de745 2207 return 0;
fe8ab488
A
2208}
2209
0a7de745
A
2210int
2211posix_spawnattr_set_darwin_role_np(const posix_spawnattr_t * __restrict attr, uint64_t darwin_role)
3e170ce0
A
2212{
2213 _posix_spawnattr_t psattr;
2214
2215 if (attr == NULL || *attr == NULL) {
2216 return EINVAL;
2217 }
2218
2219 psattr = *(_posix_spawnattr_t *)attr;
2220 psattr->psa_darwin_role = darwin_role;
2221
2222 return 0;
2223}
2224
2225int
2226posix_spawnattr_get_darwin_role_np(const posix_spawnattr_t * __restrict attr, uint64_t * __restrict darwin_rolep)
2227{
2228 _posix_spawnattr_t psattr;
2229
2230 if (attr == NULL || *attr == NULL) {
2231 return EINVAL;
2232 }
2233
2234 psattr = *(_posix_spawnattr_t *)attr;
2235 *darwin_rolep = psattr->psa_darwin_role;
2236
0a7de745 2237 return 0;
3e170ce0 2238}
fe8ab488 2239
490019cf
A
2240
2241int
2242posix_spawnattr_set_persona_np(const posix_spawnattr_t * __restrict attr, uid_t persona_id, uint32_t flags)
2243{
2244 _posix_spawnattr_t psattr;
2245 struct _posix_spawn_persona_info *persona;
2246
0a7de745 2247 if (attr == NULL || *attr == NULL) {
490019cf 2248 return EINVAL;
0a7de745 2249 }
490019cf 2250
0a7de745 2251 if (flags & ~POSIX_SPAWN_PERSONA_ALL_FLAGS) {
490019cf 2252 return EINVAL;
0a7de745 2253 }
490019cf
A
2254
2255 psattr = *(_posix_spawnattr_t *)attr;
2256
2257 persona = psattr->psa_persona_info;
2258 if (!persona) {
2259 persona = (struct _posix_spawn_persona_info *)malloc(sizeof(*persona));
0a7de745 2260 if (!persona) {
490019cf 2261 return ENOMEM;
0a7de745 2262 }
490019cf
A
2263 persona->pspi_uid = 0;
2264 persona->pspi_gid = 0;
2265 persona->pspi_ngroups = 0;
2266 persona->pspi_groups[0] = 0;
cb323159 2267 persona->pspi_gmuid = 0;
490019cf
A
2268
2269 psattr->psa_persona_info = persona;
2270 }
2271
2272 persona->pspi_id = persona_id;
2273 persona->pspi_flags = flags;
2274
2275 return 0;
2276}
2277
2278int
2279posix_spawnattr_set_persona_uid_np(const posix_spawnattr_t * __restrict attr, uid_t uid)
2280{
2281 _posix_spawnattr_t psattr;
2282 struct _posix_spawn_persona_info *persona;
2283
0a7de745 2284 if (attr == NULL || *attr == NULL) {
490019cf 2285 return EINVAL;
0a7de745 2286 }
490019cf
A
2287
2288 psattr = *(_posix_spawnattr_t *)attr;
2289 persona = psattr->psa_persona_info;
0a7de745 2290 if (!persona) {
490019cf 2291 return EINVAL;
0a7de745 2292 }
490019cf 2293
0a7de745 2294 if (!(persona->pspi_flags & (POSIX_SPAWN_PERSONA_FLAGS_OVERRIDE | POSIX_SPAWN_PERSONA_FLAGS_VERIFY))) {
490019cf 2295 return EINVAL;
0a7de745 2296 }
490019cf
A
2297
2298 persona->pspi_uid = uid;
2299
2300 persona->pspi_flags |= POSIX_SPAWN_PERSONA_UID;
2301
2302 return 0;
2303}
2304
2305int
2306posix_spawnattr_set_persona_gid_np(const posix_spawnattr_t * __restrict attr, gid_t gid)
2307{
2308 _posix_spawnattr_t psattr;
2309 struct _posix_spawn_persona_info *persona;
2310
0a7de745 2311 if (attr == NULL || *attr == NULL) {
490019cf 2312 return EINVAL;
0a7de745 2313 }
490019cf
A
2314
2315 psattr = *(_posix_spawnattr_t *)attr;
2316 persona = psattr->psa_persona_info;
0a7de745 2317 if (!persona) {
490019cf 2318 return EINVAL;
0a7de745 2319 }
490019cf 2320
0a7de745 2321 if (!(persona->pspi_flags & (POSIX_SPAWN_PERSONA_FLAGS_OVERRIDE | POSIX_SPAWN_PERSONA_FLAGS_VERIFY))) {
490019cf 2322 return EINVAL;
0a7de745 2323 }
490019cf
A
2324
2325 persona->pspi_gid = gid;
2326
2327 persona->pspi_flags |= POSIX_SPAWN_PERSONA_GID;
2328
2329 return 0;
2330}
2331
2332int
2333posix_spawnattr_set_persona_groups_np(const posix_spawnattr_t * __restrict attr, int ngroups, gid_t *gidarray, uid_t gmuid)
2334{
2335 _posix_spawnattr_t psattr;
2336 struct _posix_spawn_persona_info *persona;
2337
0a7de745 2338 if (attr == NULL || *attr == NULL) {
490019cf 2339 return EINVAL;
0a7de745 2340 }
490019cf 2341
0a7de745 2342 if (gidarray == NULL) {
490019cf 2343 return EINVAL;
0a7de745 2344 }
490019cf 2345
0a7de745 2346 if (ngroups > NGROUPS || ngroups < 0) {
490019cf 2347 return EINVAL;
0a7de745 2348 }
490019cf
A
2349
2350 psattr = *(_posix_spawnattr_t *)attr;
2351 persona = psattr->psa_persona_info;
0a7de745 2352 if (!persona) {
490019cf 2353 return EINVAL;
0a7de745 2354 }
490019cf 2355
0a7de745 2356 if (!(persona->pspi_flags & (POSIX_SPAWN_PERSONA_FLAGS_OVERRIDE | POSIX_SPAWN_PERSONA_FLAGS_VERIFY))) {
490019cf 2357 return EINVAL;
0a7de745 2358 }
490019cf
A
2359
2360 persona->pspi_ngroups = ngroups;
0a7de745 2361 for (int i = 0; i < ngroups; i++) {
490019cf 2362 persona->pspi_groups[i] = gidarray[i];
0a7de745 2363 }
490019cf
A
2364
2365 persona->pspi_gmuid = gmuid;
2366
2367 persona->pspi_flags |= POSIX_SPAWN_PERSONA_GROUPS;
2368
2369 return 0;
2370}
2371
d9a64523
A
2372int
2373posix_spawnattr_set_max_addr_np(const posix_spawnattr_t * __restrict attr, uint64_t max_addr)
2374{
2375 _posix_spawnattr_t psattr;
490019cf 2376
d9a64523
A
2377 if (attr == NULL || *attr == NULL) {
2378 return EINVAL;
2379 }
2380
2381 psattr = *(_posix_spawnattr_t *)attr;
2382 psattr->psa_max_addr = max_addr;
2383
2384 return 0;
2385}
490019cf 2386
f427ee49
A
2387int
2388posix_spawnattr_setnosmt_np(const posix_spawnattr_t * __restrict attr)
2389{
2390 _posix_spawnattr_t psattr;
2391
2392 if (attr == NULL || *attr == NULL) {
2393 return EINVAL;
2394 }
2395
2396 psattr = *(_posix_spawnattr_t *)attr;
2397 psattr->psa_no_smt = true;
2398
2399 return 0;
2400}
2401
2402int
2403posix_spawnattr_set_csm_np(const posix_spawnattr_t * __restrict attr, uint32_t flags)
2404{
2405 _posix_spawnattr_t psattr;
2406
2407 if (attr == NULL || *attr == NULL) {
2408 return EINVAL;
2409 }
2410
2411 const uint32_t mask = POSIX_SPAWN_NP_CSM_ALL | POSIX_SPAWN_NP_CSM_TECS | POSIX_SPAWN_NP_CSM_NOSMT;
2412 if ((flags & ~mask) != 0) {
2413 return EINVAL;
2414 }
2415
2416 psattr = *(_posix_spawnattr_t *)attr;
2417
2418 if (flags & (POSIX_SPAWN_NP_CSM_TECS | POSIX_SPAWN_NP_CSM_ALL)) {
2419 psattr->psa_tecs = true;
2420 }
2421 if (flags & (POSIX_SPAWN_NP_CSM_NOSMT | POSIX_SPAWN_NP_CSM_ALL)) {
2422 psattr->psa_no_smt = true;
2423 }
2424
2425 return 0;
2426}
2427
cb323159
A
2428static struct _posix_spawn_posix_cred_info *
2429_posix_spawnattr_get_posix_creds_info(_posix_spawnattr_t psattr)
2430{
2431 struct _posix_spawn_posix_cred_info *pspci = psattr->psa_posix_cred_info;
2432
2433 if (pspci == NULL) {
2434 pspci = malloc(sizeof(struct _posix_spawn_posix_cred_info));
2435 if (pspci != NULL) {
2436 pspci->pspci_flags = 0;
2437 pspci->pspci_uid = 0;
2438 pspci->pspci_gid = 0;
2439 pspci->pspci_ngroups = 0;
2440 pspci->pspci_groups[0] = 0;
2441 pspci->pspci_gmuid = 0;
2442 pspci->pspci_login[0] = '\0';
2443 psattr->psa_posix_cred_info = pspci;
2444 }
2445 }
2446 return pspci;
2447}
2448
2449int
2450posix_spawnattr_set_uid_np(const posix_spawnattr_t *attr, uid_t uid)
2451{
2452 struct _posix_spawn_posix_cred_info *pspci;
2453
2454 if (attr == NULL || *attr == NULL) {
2455 return EINVAL;
2456 }
2457
2458 pspci = _posix_spawnattr_get_posix_creds_info(*(_posix_spawnattr_t *)attr);
2459 if (pspci == NULL) {
2460 return ENOMEM;
2461 }
2462
2463 pspci->pspci_uid = uid;
2464
2465 pspci->pspci_flags |= POSIX_SPAWN_POSIX_CRED_UID;
2466
2467 return 0;
2468}
2469
2470int
2471posix_spawnattr_set_gid_np(const posix_spawnattr_t *attr, gid_t gid)
2472{
2473 struct _posix_spawn_posix_cred_info *pspci;
2474
2475 if (attr == NULL || *attr == NULL) {
2476 return EINVAL;
2477 }
2478
2479 pspci = _posix_spawnattr_get_posix_creds_info(*(_posix_spawnattr_t *)attr);
2480 if (pspci == NULL) {
2481 return ENOMEM;
2482 }
2483
2484 pspci->pspci_gid = gid;
2485
2486 pspci->pspci_flags |= POSIX_SPAWN_POSIX_CRED_GID;
2487
2488 return 0;
2489}
2490
2491int
2492posix_spawnattr_set_groups_np(const posix_spawnattr_t *attr,
2493 int ngroups, gid_t *gidarray, uid_t gmuid)
2494{
2495 struct _posix_spawn_posix_cred_info *pspci;
2496
2497 if (attr == NULL || *attr == NULL) {
2498 return EINVAL;
2499 }
2500
2501 if (gidarray == NULL) {
2502 return EINVAL;
2503 }
2504
2505 if (ngroups > NGROUPS || ngroups < 0) {
2506 return EINVAL;
2507 }
2508
2509 pspci = _posix_spawnattr_get_posix_creds_info(*(_posix_spawnattr_t *)attr);
2510 if (pspci == NULL) {
2511 return ENOMEM;
2512 }
2513
2514 pspci->pspci_ngroups = ngroups;
2515 for (int i = 0; i < ngroups; i++) {
2516 pspci->pspci_groups[i] = gidarray[i];
2517 }
2518
2519 pspci->pspci_gmuid = gmuid;
2520
2521 pspci->pspci_flags |= POSIX_SPAWN_POSIX_CRED_GROUPS;
2522
2523 return 0;
2524}
2525
2526int
2527posix_spawnattr_set_login_np(const posix_spawnattr_t *attr, const char *login)
2528{
2529 struct _posix_spawn_posix_cred_info *pspci;
2530
2531 if (attr == NULL || *attr == NULL) {
2532 return EINVAL;
2533 }
2534
2535 if (strlen(login) > MAXLOGNAME) {
2536 return ERANGE;
2537 }
2538
2539 pspci = _posix_spawnattr_get_posix_creds_info(*(_posix_spawnattr_t *)attr);
2540 if (pspci == NULL) {
2541 return ENOMEM;
2542 }
2543
2544 strlcpy(pspci->pspci_login, login, sizeof(pspci->pspci_login));
2545
2546 pspci->pspci_flags |= POSIX_SPAWN_POSIX_CRED_LOGIN;
2547
2548 return 0;
2549}
2550
2551/*
2552 * posix_spawnattr_set_jetsam_ttr_np
2553 *
2554 * Description: Pass data regarding recent relaunch behavior when jetsammed for the process.
2555 * The recent history is effectively converted into a histogram and the highest
2556 * frequency bucket defines the "type" of the process. The type is passed along
2557 * to the jetsam code as part of psa_jetsam_flags.
2558 *
2559 * Parameters: count Number of entries in the ttrs_millis array
2560 * ttrs_millis Array of raw data for relaunch behavior
2561 *
2562 * Returns: 0 Success
2563 * EINVAL Bad attr pointer or empty data array
2564 */
2565int
2566posix_spawnattr_set_jetsam_ttr_np(const posix_spawnattr_t * __restrict attr, uint32_t count, uint32_t *ttrs_millis)
2567{
2568 _posix_spawnattr_t psattr;
2569
2570 /*
2571 * Define the bucketizing policy which would be used to generate the histogram. These
2572 * values are based on looking at data from various Avg. Joanna runs.
2573 */
2574 static const uint32_t relaunch_buckets_msecs[POSIX_SPAWN_JETSAM_RELAUNCH_BEHAVIOR_BUCKETS] = {
2575 5000,
2576 10000,
2577 UINT32_MAX
2578 };
2579 static const uint32_t relaunch_jetsam_flags[POSIX_SPAWN_JETSAM_RELAUNCH_BEHAVIOR_BUCKETS] = {
2580 POSIX_SPAWN_JETSAM_RELAUNCH_BEHAVIOR_HIGH,
2581 POSIX_SPAWN_JETSAM_RELAUNCH_BEHAVIOR_MED,
2582 POSIX_SPAWN_JETSAM_RELAUNCH_BEHAVIOR_LOW
2583 };
2584
2585 /* Make sure the attr pointer is valid */
2586 if (attr == NULL || *attr == NULL) {
2587 return EINVAL;
2588 }
2589
2590 /* Make sure the count of entries is non-zero */
2591 if (count == 0) {
2592 return EINVAL;
2593 }
2594
2595 psattr = *(_posix_spawnattr_t *)attr;
2596
2597 /* Generate a histogram based on the relaunch data while maintaining highest frequency bucket info */
2598 int relaunch_histogram[POSIX_SPAWN_JETSAM_RELAUNCH_BEHAVIOR_BUCKETS] = {0};
2599 int max_frequency = -1;
2600 int highest_frequency_bucket = -1;
2601
2602 for (uint32_t i = 0; i < count; i++) {
2603 /* For each data point passed in via launchd, find the bucket it lands in */
2604 for (uint32_t bucket = 0; bucket < POSIX_SPAWN_JETSAM_RELAUNCH_BEHAVIOR_BUCKETS; bucket++) {
2605 if (ttrs_millis[i] <= relaunch_buckets_msecs[bucket]) {
2606 relaunch_histogram[bucket]++;
2607
2608 /* Check if the bucket is the highest frequency bucket now */
2609 if (relaunch_histogram[bucket] > max_frequency) {
2610 max_frequency = relaunch_histogram[bucket];
2611 highest_frequency_bucket = bucket;
2612 }
2613 break;
2614 }
2615 }
2616 }
2617 psattr->psa_jetsam_flags |= relaunch_jetsam_flags[highest_frequency_bucket];
2618 return 0;
2619}
2620
39236c6e
A
2621/*
2622 * posix_spawn
2623 *
2624 * Description: Create a new process from the process image corresponding to
2625 * the supplied 'path' argument.
2626 *
2627 * Parameters: pid Pointer to pid_t to receive the
2628 * PID of the spawned process, if
2629 * successful and 'pid' != NULL
2630 * path Path of image file to spawn
2631 * file_actions spawn file actions object which
2632 * describes file actions to be
2633 * performed during the spawn
2634 * attrp spawn attributes object which
2635 * describes attributes to be
2636 * applied during the spawn
2637 * argv argument vector array; NULL
2638 * terminated
2639 * envp environment vector array; NULL
2640 * terminated
2641 *
2642 * Returns: 0 Success
2643 * !0 An errno value indicating the
2644 * cause of the failure to spawn
2645 *
2646 * Notes: Unlike other system calls, the return value of this system
2647 * call is expected to either be a 0 or an errno, rather than a
2648 * 0 or a -1, with the 'errno' variable being set.
2649 */
2650extern int __posix_spawn(pid_t * __restrict, const char * __restrict,
0a7de745
A
2651 struct _posix_spawn_args_desc *,
2652 char *const argv[__restrict], char *const envp[__restrict]);
39236c6e
A
2653
2654int
2655posix_spawn(pid_t * __restrict pid, const char * __restrict path,
0a7de745
A
2656 const posix_spawn_file_actions_t *file_actions,
2657 const posix_spawnattr_t * __restrict attrp,
2658 char *const argv[__restrict], char *const envp[__restrict])
39236c6e
A
2659{
2660 int saveerrno = errno;
2661 int ret;
2662 /*
2663 * Only do extra work if we have file actions or attributes to push
2664 * down. We use a descriptor to push this information down, since we
2665 * want to have size information, which will let us (1) preallocate a
2666 * single chunk of memory for the copyin(), and (2) allow us to do a
2667 * single copyin() per attributes or file actions as a monlithic block.
2668 *
2669 * Note: A future implementation may attempt to do the same
2670 * thing for the argv/envp data, which could potentially
2671 * result in a performance improvement due to increased
2672 * kernel efficiency, even though it would mean copying
2673 * the data in user space.
2674 */
0a7de745
A
2675 if ((file_actions != NULL && (*file_actions != NULL) && (*(_posix_spawn_file_actions_t *)file_actions)->psfa_act_count > 0) || attrp != NULL) {
2676 struct _posix_spawn_args_desc ad;
39236c6e
A
2677
2678 memset(&ad, 0, sizeof(ad));
2679 if (attrp != NULL && *attrp != NULL) {
2680 _posix_spawnattr_t psattr = *(_posix_spawnattr_t *)attrp;
2681 ad.attr_size = sizeof(struct _posix_spawnattr);
2682 ad.attrp = psattr;
2683
2684 if (psattr->psa_ports != NULL) {
5c9f4661
A
2685 size_t psact_size = PS_PORT_ACTIONS_SIZE(psattr->psa_ports->pspa_count);
2686 if (psact_size == 0 && psattr->psa_ports->pspa_count != 0) {
2687 errno = EINVAL;
2688 ret = -1;
2689 goto out;
2690 }
39236c6e 2691 ad.port_actions = psattr->psa_ports;
5c9f4661 2692 ad.port_actions_size = psact_size;
39236c6e
A
2693 }
2694 if (psattr->psa_mac_extensions != NULL) {
5c9f4661
A
2695 size_t macext_size = PS_MAC_EXTENSIONS_SIZE(psattr->psa_mac_extensions->psmx_count);
2696 if (macext_size == 0 && psattr->psa_mac_extensions->psmx_count != 0) {
2697 errno = EINVAL;
2698 ret = -1;
2699 goto out;
2700 }
39236c6e 2701 ad.mac_extensions = psattr->psa_mac_extensions;
5c9f4661 2702 ad.mac_extensions_size = macext_size;
39236c6e 2703 }
3e170ce0
A
2704 if (psattr->psa_coalition_info != NULL) {
2705 ad.coal_info_size = sizeof(struct _posix_spawn_coalition_info);
2706 ad.coal_info = psattr->psa_coalition_info;
2707 }
490019cf
A
2708 if (psattr->psa_persona_info != NULL) {
2709 ad.persona_info_size = sizeof(struct _posix_spawn_persona_info);
2710 ad.persona_info = psattr->psa_persona_info;
2711 }
cb323159
A
2712 if (psattr->psa_posix_cred_info != NULL) {
2713 ad.posix_cred_info_size = sizeof(struct _posix_spawn_posix_cred_info);
2714 ad.posix_cred_info = psattr->psa_posix_cred_info;
2715 }
f427ee49
A
2716 if (psattr->psa_subsystem_root_path != NULL) {
2717 ad.subsystem_root_path_size = MAXPATHLEN;
2718 ad.subsystem_root_path = psattr->psa_subsystem_root_path;
2719 }
39236c6e
A
2720 }
2721 if (file_actions != NULL && *file_actions != NULL) {
2722 _posix_spawn_file_actions_t psactsp =
0a7de745 2723 *(_posix_spawn_file_actions_t *)file_actions;
39236c6e
A
2724
2725 if (psactsp->psfa_act_count > 0) {
5c9f4661
A
2726 size_t fa_size = PSF_ACTIONS_SIZE(psactsp->psfa_act_count);
2727 if (fa_size == 0 && psactsp->psfa_act_count != 0) {
2728 errno = EINVAL;
2729 ret = -1;
2730 goto out;
2731 }
2732 ad.file_actions_size = fa_size;
39236c6e
A
2733 ad.file_actions = psactsp;
2734 }
2735 }
2736
2737 ret = __posix_spawn(pid, path, &ad, argv, envp);
0a7de745 2738 } else {
39236c6e 2739 ret = __posix_spawn(pid, path, NULL, argv, envp);
0a7de745 2740 }
39236c6e 2741
5c9f4661 2742out:
0a7de745 2743 if (ret < 0) {
39236c6e 2744 ret = errno;
0a7de745 2745 }
39236c6e
A
2746 errno = saveerrno;
2747 return ret;
2748}