summaryrefslogtreecommitdiff
path: root/lib/mlibc/options/lsb
diff options
context:
space:
mode:
authorIan Moffett <ian@osmora.org>2024-03-07 17:28:00 -0500
committerIan Moffett <ian@osmora.org>2024-03-07 17:28:32 -0500
commitbd5969fc876a10b18613302db7087ef3c40f18e1 (patch)
tree7c2b8619afe902abf99570df2873fbdf40a4d1a1 /lib/mlibc/options/lsb
parenta95b38b1b92b172e6cc4e8e56a88a30cc65907b0 (diff)
lib: Add mlibc
Signed-off-by: Ian Moffett <ian@osmora.org>
Diffstat (limited to 'lib/mlibc/options/lsb')
-rw-r--r--lib/mlibc/options/lsb/generic/auxv.cpp59
-rw-r--r--lib/mlibc/options/lsb/generic/dso_exit.cpp42
-rw-r--r--lib/mlibc/options/lsb/generic/tls.cpp23
-rw-r--r--lib/mlibc/options/lsb/include/sys/auxv.h26
-rw-r--r--lib/mlibc/options/lsb/meson.build14
5 files changed, 164 insertions, 0 deletions
diff --git a/lib/mlibc/options/lsb/generic/auxv.cpp b/lib/mlibc/options/lsb/generic/auxv.cpp
new file mode 100644
index 0000000..a4d2c8f
--- /dev/null
+++ b/lib/mlibc/options/lsb/generic/auxv.cpp
@@ -0,0 +1,59 @@
+
+#include <errno.h>
+#include <stdint.h>
+#include <sys/auxv.h>
+
+#include <bits/ensure.h>
+
+extern "C" uintptr_t *__dlapi_entrystack();
+
+int peekauxval(unsigned long type, unsigned long *out) {
+ // Find the auxiliary vector by skipping args and environment.
+ auto aux = __dlapi_entrystack();
+ aux += *aux + 1; // Skip argc and all arguments
+ __ensure(!*aux);
+ aux++;
+ while(*aux) // Now, we skip the environment.
+ aux++;
+ aux++;
+
+ // Parse the auxiliary vector.
+ while(true) {
+ auto value = aux + 1;
+ if(*aux == AT_NULL) {
+ errno = ENOENT;
+ return -1;
+ }else if(*aux == type) {
+ *out = *value;
+ return 0;
+ }
+ aux += 2;
+ }
+}
+
+unsigned long getauxval(unsigned long type) {
+ unsigned long value = 0;
+ if(peekauxval(type, &value))
+ return 0;
+ return value;
+}
+
+// XXX(qookie):
+// This is here because libgcc will call into __getauxval on glibc Linux
+// (which is what it believes we are due to the aarch64-linux-gnu toolchain)
+// in order to find AT_HWCAP to discover if LSE atomics are supported.
+//
+// This is not necessary on a custom Linux toolchain and is purely an artifact of
+// using the host toolchain.
+
+// __gnu_linux__ is the define checked by libgcc
+#if defined(__aarch64__) && defined(__gnu_linux__)
+
+extern "C" unsigned long __getauxval(unsigned long type) {
+ unsigned long value = 0;
+ if(peekauxval(type, &value))
+ return 0;
+ return value;
+}
+
+#endif
diff --git a/lib/mlibc/options/lsb/generic/dso_exit.cpp b/lib/mlibc/options/lsb/generic/dso_exit.cpp
new file mode 100644
index 0000000..b8b239d
--- /dev/null
+++ b/lib/mlibc/options/lsb/generic/dso_exit.cpp
@@ -0,0 +1,42 @@
+
+// for memcpy()
+#include <string.h>
+
+#include <bits/ensure.h>
+#include <mlibc/allocator.hpp>
+
+#include <frg/eternal.hpp>
+#include <frg/vector.hpp>
+
+struct ExitHandler {
+ void (*function)(void *);
+ void *argument;
+ void *dsoHandle;
+};
+
+using ExitQueue = frg::vector<ExitHandler, MemoryAllocator>;
+
+ExitQueue &getExitQueue() {
+ // use frg::eternal to prevent the compiler from scheduling the destructor
+ // by generating a call to __cxa_atexit().
+ static frg::eternal<ExitQueue> singleton(getAllocator());
+ return singleton.get();
+}
+
+extern "C" int __cxa_atexit(void (*function)(void *), void *argument, void *handle) {
+ ExitHandler handler;
+ handler.function = function;
+ handler.argument = argument;
+ handler.dsoHandle = handle;
+ getExitQueue().push(handler);
+ return 0;
+}
+
+void __mlibc_do_finalize() {
+ ExitQueue &eq = getExitQueue();
+ for(size_t i = eq.size(); i > 0; i--) {
+ auto handler = &eq[i - 1];
+ handler->function(handler->argument);
+ }
+}
+
diff --git a/lib/mlibc/options/lsb/generic/tls.cpp b/lib/mlibc/options/lsb/generic/tls.cpp
new file mode 100644
index 0000000..1d7cc30
--- /dev/null
+++ b/lib/mlibc/options/lsb/generic/tls.cpp
@@ -0,0 +1,23 @@
+#include <internal-config.h>
+#include <mlibc/thread.hpp>
+#include <mlibc/rtdl-abi.hpp>
+
+#if defined(__riscv) && defined(MLIBC_STATIC_BUILD)
+ // On RISC-V, linker optimisation is not guaranteed and so we may still get
+ // calls to this function in statically linked binaries.
+ // TODO: This will break dlopen calls from statically linked programs.
+ extern "C" void *__tls_get_addr(struct __abi_tls_entry *entry) {
+ Tcb *tcbPtr = mlibc::get_current_tcb();
+ auto dtvPtr = reinterpret_cast<char *>(tcbPtr->dtvPointers[0]);
+ return reinterpret_cast<void *>(dtvPtr + entry->offset + TLS_DTV_OFFSET);
+ }
+#elif defined(__i386__)
+ extern "C" __attribute__((regparm(1))) void *___tls_get_addr(struct __abi_tls_entry *entry) {
+ return __dlapi_get_tls(entry);
+ }
+#else
+ extern "C" void *__tls_get_addr(struct __abi_tls_entry *entry) {
+ return __dlapi_get_tls(entry);
+ }
+#endif
+
diff --git a/lib/mlibc/options/lsb/include/sys/auxv.h b/lib/mlibc/options/lsb/include/sys/auxv.h
new file mode 100644
index 0000000..a3e028c
--- /dev/null
+++ b/lib/mlibc/options/lsb/include/sys/auxv.h
@@ -0,0 +1,26 @@
+#ifndef _SYS_AUXV_H
+#define _SYS_AUXV_H
+
+#define AT_NULL 0
+#include <abi-bits/auxv.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#ifndef __MLIBC_ABI_ONLY
+
+// mlibc extension: Like getauxval but handles errors in a sane way.
+// Success: Return 0.
+// Failure: Return -1 and set errno.
+int peekauxval(unsigned long type, unsigned long *value);
+
+unsigned long getauxval(unsigned long type);
+
+#endif /* !__MLIBC_ABI_ONLY */
+
+#ifdef __cplusplus
+} // extern "C"
+#endif
+
+#endif
diff --git a/lib/mlibc/options/lsb/meson.build b/lib/mlibc/options/lsb/meson.build
new file mode 100644
index 0000000..322a912
--- /dev/null
+++ b/lib/mlibc/options/lsb/meson.build
@@ -0,0 +1,14 @@
+
+lsb_sources = files(
+ 'generic/auxv.cpp',
+ 'generic/dso_exit.cpp',
+ 'generic/tls.cpp',
+)
+
+if not no_headers
+ install_headers(
+ 'include/sys/auxv.h',
+ subdir: 'sys'
+ )
+endif
+