mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-01-31 11:37:35 +00:00
7e0a09feec
This change ports APE Loader to Linux AARCH64, so that Raspberry Pi users can run programs like redbean, without the executable needing to modify itself. Progress has also slipped into this change on the issue of making progress better conforming to user expectations and industry standards regarding which symbols we're allowed to declare
261 lines
8.6 KiB
C
261 lines
8.6 KiB
C
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
|
│ Copyright 2021 Justine Alexandra Roberts Tunney │
|
|
│ │
|
|
│ Permission to use, copy, modify, and/or distribute this software for │
|
|
│ any purpose with or without fee is hereby granted, provided that the │
|
|
│ above copyright notice and this permission notice appear in all copies. │
|
|
│ │
|
|
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
|
|
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
|
|
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
|
|
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
|
|
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
|
|
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
|
|
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
|
|
│ PERFORMANCE OF THIS SOFTWARE. │
|
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
|
#include "libc/assert.h"
|
|
#include "libc/calls/calls.h"
|
|
#include "libc/calls/struct/sigset.h"
|
|
#include "libc/dce.h"
|
|
#include "libc/errno.h"
|
|
#include "libc/mem/mem.h"
|
|
#include "libc/stdio/posix_spawn.h"
|
|
#include "libc/stdio/posix_spawn.internal.h"
|
|
#include "libc/sysv/consts/sig.h"
|
|
|
|
/**
|
|
* Initialize posix_spawn() attributes object with default values.
|
|
*
|
|
* @param attr needs to be passed to posix_spawnattr_destroy() later
|
|
* @return 0 on success, or errno on error
|
|
* @raise ENOMEM if we require more vespene gas
|
|
*/
|
|
int posix_spawnattr_init(posix_spawnattr_t *attr) {
|
|
int rc, e = errno;
|
|
struct _posix_spawna *a;
|
|
if ((a = calloc(1, sizeof(struct _posix_spawna)))) {
|
|
*attr = a;
|
|
rc = 0;
|
|
} else {
|
|
rc = errno;
|
|
errno = e;
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
/**
|
|
* Destroys posix_spawn() attributes object.
|
|
*
|
|
* This function is safe to call multiple times.
|
|
*
|
|
* @param attr was initialized by posix_spawnattr_init()
|
|
* @return 0 on success, or errno on error
|
|
*/
|
|
int posix_spawnattr_destroy(posix_spawnattr_t *attr) {
|
|
if (*attr) {
|
|
free(*attr);
|
|
*attr = 0;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Gets posix_spawn() flags.
|
|
*
|
|
* @param attr was initialized by posix_spawnattr_init()
|
|
* @return 0 on success, or errno on error
|
|
*/
|
|
int posix_spawnattr_getflags(const posix_spawnattr_t *attr, short *flags) {
|
|
*flags = (*attr)->flags;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Sets posix_spawn() flags.
|
|
*
|
|
* Setting these flags is needed in order for the other setters in this
|
|
* function to take effect. If a flag is known but unsupported by the
|
|
* host platform, it'll be silently removed from the flags. You can
|
|
* check for this by calling the getter afterwards.
|
|
*
|
|
* @param attr was initialized by posix_spawnattr_init()
|
|
* @param flags may have any of the following
|
|
* - `POSIX_SPAWN_RESETIDS`
|
|
* - `POSIX_SPAWN_SETPGROUP`
|
|
* - `POSIX_SPAWN_SETSIGDEF`
|
|
* - `POSIX_SPAWN_SETSIGMASK`
|
|
* - `POSIX_SPAWN_SETSCHEDPARAM`
|
|
* - `POSIX_SPAWN_SETSCHEDULER`
|
|
* - `POSIX_SPAWN_SETSID`
|
|
* @return 0 on success, or errno on error
|
|
* @raise EINVAL if `flags` has invalid bits
|
|
*/
|
|
int posix_spawnattr_setflags(posix_spawnattr_t *attr, short flags) {
|
|
if (!(IsLinux() || IsFreebsd() || IsNetbsd())) {
|
|
flags &= ~(POSIX_SPAWN_SETSCHEDPARAM | POSIX_SPAWN_SETSCHEDULER);
|
|
}
|
|
if (flags &
|
|
~(POSIX_SPAWN_RESETIDS | POSIX_SPAWN_SETPGROUP | POSIX_SPAWN_SETSIGDEF |
|
|
POSIX_SPAWN_SETSIGMASK | POSIX_SPAWN_SETSCHEDPARAM |
|
|
POSIX_SPAWN_SETSCHEDULER | POSIX_SPAWN_SETSID)) {
|
|
return EINVAL;
|
|
}
|
|
(*attr)->flags = flags;
|
|
return 0;
|
|
}
|
|
|
|
int posix_spawnattr_getpgroup(const posix_spawnattr_t *attr, int *pgroup) {
|
|
*pgroup = (*attr)->pgroup;
|
|
return 0;
|
|
}
|
|
|
|
int posix_spawnattr_setpgroup(posix_spawnattr_t *attr, int pgroup) {
|
|
(*attr)->pgroup = pgroup;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Gets scheduler policy that'll be used for spawned process.
|
|
*
|
|
* If the setter wasn't called then this function will return the
|
|
* scheduling policy of the current process.
|
|
*
|
|
* @param attr was initialized by posix_spawnattr_init()
|
|
* @param schedpolicy receives the result
|
|
* @return 0 on success, or errno on error
|
|
* @raise ENOSYS if platform support isn't available
|
|
*/
|
|
int posix_spawnattr_getschedpolicy(const posix_spawnattr_t *attr,
|
|
int *schedpolicy) {
|
|
int rc, e = errno;
|
|
struct _posix_spawna *a = *(/*unconst*/ posix_spawnattr_t *)attr;
|
|
if (!a->schedpolicy_isset) {
|
|
rc = sched_getscheduler(0);
|
|
if (rc == -1) {
|
|
rc = errno;
|
|
errno = e;
|
|
return rc;
|
|
}
|
|
a->schedpolicy = rc;
|
|
a->schedpolicy_isset = true;
|
|
}
|
|
*schedpolicy = a->schedpolicy;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Specifies scheduler policy override for spawned process.
|
|
*
|
|
* Scheduling policies are inherited by default. Use this to change it.
|
|
*
|
|
* @param attr was initialized by posix_spawnattr_init()
|
|
* @param schedpolicy receives the result
|
|
* @return 0 on success, or errno on error
|
|
*/
|
|
int posix_spawnattr_setschedpolicy(posix_spawnattr_t *attr, int schedpolicy) {
|
|
(*attr)->schedpolicy = schedpolicy;
|
|
(*attr)->schedpolicy_isset = true;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Gets scheduler parameter that'll be used for spawned process.
|
|
*
|
|
* If the setter wasn't called then this function will return the
|
|
* scheduling parameter of the current process.
|
|
*
|
|
* @param attr was initialized by posix_spawnattr_init()
|
|
* @param schedparam receives the result
|
|
* @return 0 on success, or errno on error
|
|
* @raise ENOSYS if platform support isn't available
|
|
*/
|
|
int posix_spawnattr_getschedparam(const posix_spawnattr_t *attr,
|
|
struct sched_param *schedparam) {
|
|
int rc, e = errno;
|
|
struct _posix_spawna *a = *(/*unconst*/ posix_spawnattr_t *)attr;
|
|
if (!a->schedparam_isset) {
|
|
rc = sched_getparam(0, &a->schedparam);
|
|
if (rc == -1) {
|
|
rc = errno;
|
|
errno = e;
|
|
return rc;
|
|
}
|
|
a->schedparam_isset = true;
|
|
}
|
|
*schedparam = a->schedparam;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Specifies scheduler parameter override for spawned process.
|
|
*
|
|
* Scheduling parameters are inherited by default. Use this to change it.
|
|
*
|
|
* @param attr was initialized by posix_spawnattr_init()
|
|
* @param schedparam receives the result
|
|
* @return 0 on success, or errno on error
|
|
*/
|
|
int posix_spawnattr_setschedparam(posix_spawnattr_t *attr,
|
|
const struct sched_param *schedparam) {
|
|
(*attr)->schedparam = *schedparam;
|
|
(*attr)->schedparam_isset = true;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Gets signal mask for sigprocmask() in child process.
|
|
*
|
|
* If the setter wasn't called then this function will return the
|
|
* scheduling parameter of the current process.
|
|
*
|
|
* @return 0 on success, or errno on error
|
|
*/
|
|
int posix_spawnattr_getsigmask(const posix_spawnattr_t *attr,
|
|
sigset_t *sigmask) {
|
|
struct _posix_spawna *a = *(/*unconst*/ posix_spawnattr_t *)attr;
|
|
if (!a->sigmask_isset) {
|
|
npassert(!sigprocmask(SIG_SETMASK, 0, &a->sigmask));
|
|
a->sigmask_isset = true;
|
|
}
|
|
*sigmask = a->sigmask;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Specifies signal mask for sigprocmask() in child process.
|
|
*
|
|
* Signal masks are inherited by default. Use this to change it.
|
|
*
|
|
* @return 0 on success, or errno on error
|
|
*/
|
|
int posix_spawnattr_setsigmask(posix_spawnattr_t *attr,
|
|
const sigset_t *sigmask) {
|
|
(*attr)->sigmask = *sigmask;
|
|
(*attr)->sigmask_isset = true;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Retrieves which signals will be restored to `SIG_DFL`.
|
|
*
|
|
* @return 0 on success, or errno on error
|
|
*/
|
|
int posix_spawnattr_getsigdefault(const posix_spawnattr_t *attr,
|
|
sigset_t *sigdefault) {
|
|
*sigdefault = (*attr)->sigdefault;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Specifies which signals should be restored to `SIG_DFL`.
|
|
*
|
|
* @return 0 on success, or errno on error
|
|
*/
|
|
int posix_spawnattr_setsigdefault(posix_spawnattr_t *attr,
|
|
const sigset_t *sigdefault) {
|
|
(*attr)->sigdefault = *sigdefault;
|
|
return 0;
|
|
}
|