diff options
author | Daniel Bertalan <dani@danielbertalan.dev> | 2021-07-31 15:54:48 +0200 |
---|---|---|
committer | Andreas Kling <kling@serenityos.org> | 2021-08-08 10:55:36 +0200 |
commit | 15e217ea68f5f83f953ff6c41a7c2529704f1f11 (patch) | |
tree | 3b6729696b01fa96ae0c9f9135d50343f813e627 /Toolchain | |
parent | 5d617be462d7ec01eb845cf34914bbdbf796201c (diff) | |
download | serenity-15e217ea68f5f83f953ff6c41a7c2529704f1f11.zip |
Toolchain: Add LLVM patch and script for building it
This contains all the bits and pieces necessary to build a Clang binary
that will correctly compile SerenityOS.
I had some trouble with getting LLVM building with a single command, so
for now, I decided to build each LLVM component in a separate command
invocation. In the future, we can also make the main llvm build step
architecture-independent, but that would come with extra work to make
library and include paths work.
The binutils build invocation and related boilerplate is duplicated
because we only use `objdump` from GNU binutils in the Clang toolchain,
so most features can be disabled.
Diffstat (limited to 'Toolchain')
-rwxr-xr-x | Toolchain/BuildClang.sh | 353 | ||||
-rwxr-xr-x | Toolchain/BuildIt.sh | 1 | ||||
-rw-r--r-- | Toolchain/Patches/llvm.patch | 933 |
3 files changed, 1287 insertions, 0 deletions
diff --git a/Toolchain/BuildClang.sh b/Toolchain/BuildClang.sh new file mode 100755 index 0000000000..27f6d3f392 --- /dev/null +++ b/Toolchain/BuildClang.sh @@ -0,0 +1,353 @@ +#!/usr/bin/env bash +set -eo pipefail + +# === CONFIGURATION AND SETUP === + +DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd )" + +echo "$DIR" + +ARCH=${ARCH:-"i686"} +LLVM_ARCH= +[ "$ARCH" = "i686" ] && LLVM_ARCH="i386" || LLVM_ARCH="$ARCH" +LLVM_TARGET="$LLVM_ARCH-pc-serenity" +PREFIX="$DIR/Local/clang/$ARCH" +BUILD="$DIR/../Build/$ARCH" +SYSROOT="$BUILD/Root" + +MD5SUM="md5sum" +REALPATH="realpath" +MAKE="make" +NPROC="nproc" + +SYSTEM_NAME="$(uname -s)" + +if [ "$SYSTEM_NAME" = "OpenBSD" ]; then + MD5SUM="md5 -q" + REALPATH="readlink -f" + MAKE="gmake" + NPROC="sysctl -n hw.ncpuonline" + export CC=egcc + export CXX=eg++ + export LDFLAGS=-Wl,-z,notext +elif [ "$SYSTEM_NAME" = "FreeBSD" ]; then + MD5SUM="md5 -q" + MAKE="gmake" + NPROC="sysctl -n hw.ncpu" +fi + +if [ -z "$MAKEJOBS" ]; then + MAKEJOBS=$($NPROC) +fi + + +if [ ! -d "$BUILD" ]; then + mkdir -p "$BUILD" +fi +BUILD=$($REALPATH "$BUILD") + +dev= +while [ "$1" != "" ]; do + case $1 in + --dev ) dev=1 + ;; + esac + shift +done + +echo PREFIX is "$PREFIX" +echo SYSROOT is "$SYSROOT" + +mkdir -p "$DIR/Tarballs" + +LLVM_VERSION="12.0.1" +LLVM_MD5SUM="c28061313a4f1b7d74cd491a19f569b4" +LLVM_NAME="llvm-project-$LLVM_VERSION.src" +LLVM_PKG="$LLVM_NAME.tar.xz" +LLVM_URL="https://github.com/llvm/llvm-project/releases/download/llvmorg-$LLVM_VERSION/$LLVM_PKG" + +# We need GNU binutils because we use a feature that llvm-objdump doesn't support yet. +BINUTILS_VERSION="2.37" +BINUTILS_MD5SUM="1e55743d73c100b7a0d67ffb32398cdb" +BINUTILS_NAME="binutils-$BINUTILS_VERSION" +BINUTILS_PKG="${BINUTILS_NAME}.tar.gz" +BINUTILS_BASE_URL="https://ftp.gnu.org/gnu/binutils" + +buildstep() { + NAME=$1 + shift + "$@" 2>&1 | sed $'s|^|\x1b[34m['"${NAME}"$']\x1b[39m |' +} + +# === DEPENDENCIES === +buildstep dependencies echo "Checking whether 'make' is available..." +if ! command -v ${MAKE:-make} >/dev/null; then + buildstep dependencies echo "Please make sure to install GNU Make (for the '${MAKE:-make}' tool)." + exit 1 +fi + +buildstep dependencies echo "Checking whether CMake is available..." +if ! command -v cmake >/dev/null; then + buildstep dependencies echo "Please make sure to install CMake." + exit 1 +fi + +buildstep dependencies echo "Checking whether 'patch' is available..." +if ! command -v patch >/dev/null; then + buildstep dependencies echo "Please make sure to install GNU patch." +fi + +buildstep dependencies echo "Checking whether your C compiler works..." +if ! ${CC:-cc} -o /dev/null -xc - >/dev/null <<'PROGRAM' +int main() {} +PROGRAM +then + buildstep dependencies echo "Please make sure to install a working C compiler." + exit 1 +fi + +buildstep dependencies echo "Checking whether your C++ compiler works..." +if ! ${CXX:-c++} -o /dev/null -xc - >/dev/null <<'PROGRAM' +int main() {} +PROGRAM +then + buildstep dependencies echo "Please make sure to install a working C++ compiler." + exit 1 +fi + +# === DOWNLOAD AND PATCH === + +pushd "$DIR/Tarballs" + md5="" + if [ -e "$LLVM_PKG" ]; then + md5="$($MD5SUM ${LLVM_PKG} | cut -f1 -d' ')" + echo "llvm md5='$md5'" + fi + + if [ "$md5" != "$LLVM_MD5SUM" ] ; then + rm -f "$LLVM_PKG" + curl -LO "$LLVM_URL" + else + echo "Skipped downloading LLVM" + fi + + if [ -d "$LLVM_NAME" ]; then + # Drop the previously patched extracted dir + rm -rf "${LLVM_NAME}" + # Also drop the build dir + rm -rf "$DIR/Build/clang/$ARCH" + fi + echo "Extracting LLVM..." + tar -xJf "$LLVM_PKG" + + pushd "$LLVM_NAME" + if [ "$dev" = "1" ]; then + git init > /dev/null + git add . > /dev/null + git commit -am "BASE" > /dev/null + git apply "$DIR"/Patches/llvm.patch > /dev/null + else + patch -p1 < "$DIR/Patches/llvm.patch" > /dev/null + fi + $MD5SUM "$DIR/Patches/llvm.patch" > .patch.applied + popd + + md5="" + if [ -e "$BINUTILS_PKG" ]; then + md5="$($MD5SUM $BINUTILS_PKG | cut -f1 -d' ')" + echo "bu md5='$md5'" + fi + + if [ "$md5" != "$BINUTILS_MD5SUM" ]; then + rm -f "$BINUTILS_PKG" + curl -LO "$BINUTILS_BASE_URL/$BINUTILS_PKG" + else + echo "Skipped downloading GNU binutils" + fi + + if [ -d "$BINUTILS_NAME" ]; then + rm -rf "$BINUTILS_NAME" + rm -rf "$DIR/Build/clang/$ARCH/binutils" + fi + echo "Extracting GNU binutils" + + + tar -xzf "$BINUTILS_PKG" + pushd "$BINUTILS_NAME" + if [ "$dev" = "1" ]; then + git init > /dev/null + git add . > /dev/null + git commit -am "BASE" > /dev/null + git apply "$DIR"/Patches/binutils.patch > /dev/null + else + patch -p1 < "$DIR/Patches/binutils.patch" > /dev/null + fi + $MD5SUM "$DIR/Patches/binutils.patch" > .patch.applied + popd +popd + +# === COPY SERENITYOS HEADERS === + +mkdir -p "$BUILD" +pushd "$BUILD" + mkdir -p Root/usr/include/ + SRC_ROOT=$($REALPATH "$DIR"/..) + FILES=$(find "$SRC_ROOT"/Userland/Libraries/LibC "$SRC_ROOT"/Userland/Libraries/LibM "$SRC_ROOT"/Userland/Libraries/LibPthread "$SRC_ROOT"/Userland/Libraries/LibDl -name '*.h' -print) + for header in $FILES; do + target=$(echo "$header" | sed -e "s@$SRC_ROOT/Userland/Libraries/LibC@@" -e "s@$SRC_ROOT/Userland/Libraries/LibM@@" -e "s@$SRC_ROOT/Userland/Libraries/LibPthread@@" -e "s@$SRC_ROOT/Userland/Libraries/LibDl@@") + buildstep "system_headers" install -D "$header" "Root/usr/include/$target" + done + unset SRC_ROOT +popd + +# === COMPILE AND INSTALL === + +rm -rf "$PREFIX" +mkdir -p "$PREFIX" + +mkdir -p "$DIR/Build/clang/$ARCH" + +pushd "$DIR/Build/clang/$ARCH" + + mkdir -p llvm + pushd llvm + buildstep "llvm+clang/configure" cmake "$DIR/Tarballs/llvm-project-$LLVM_VERSION.src/llvm" \ + -G Ninja \ + -DCMAKE_BUILD_TYPE="MinSizeRel" \ + -DCMAKE_INSTALL_PREFIX="$PREFIX" \ + -DLLVM_DEFAULT_TARGET_TRIPLE="$LLVM_TARGET" \ + -DLLVM_TARGETS_TO_BUILD=X86 \ + -DLLVM_ENABLE_BINDINGS=OFF \ + -DLLVM_ENABLE_PER_TARGET_RUNTIME_DIR=OFF \ + -DLLVM_ENABLE_PROJECTS="clang;lld" \ + -DLLVM_INCLUDE_BENCHMARKS=OFF \ + -DLLVM_INCLUDE_TESTS=OFF \ + ${dev:+"-DLLVM_CCACHE_BUILD=ON"} || exit 1 + + buildstep "llvm+clang/build" ninja -j "$MAKEJOBS" || exit 1 + buildstep "llvm+clang/install" ninja install || exit 1 + popd + + LLVM_COMPILER_FLAGS="-target $LLVM_TARGET --sysroot $SYSROOT -ftls-model=initial-exec -ffreestanding -nostdlib -nostdlib++" + + mkdir -p "compiler-rt" + pushd compiler-rt + buildstep "compiler-rt/configure" cmake "$DIR/Tarballs/llvm-project-$LLVM_VERSION.src/compiler-rt" \ + -GNinja \ + -DCMAKE_SYSROOT="$SYSROOT" \ + -DCMAKE_INSTALL_PREFIX="$PREFIX/lib/clang/${LLVM_VERSION}" \ + -DCMAKE_BUILD_TYPE="Release" \ + -DCMAKE_AR="$PREFIX/bin/llvm-ar" \ + -DCMAKE_C_COMPILER="$PREFIX/bin/clang" \ + -DCMAKE_C_COMPILER_TARGET="$LLVM_TARGET" \ + -DCMAKE_C_FLAGS="$LLVM_COMPILER_FLAGS" \ + -DCMAKE_CXX_COMPILER="$PREFIX/bin/clang++" \ + -DCMAKE_CXX_FLAGS="$LLVM_COMPILER_FLAGS" \ + -DCMAKE_CXX_COMPILER_TARGET="$LLVM_TARGET" \ + -DCMAKE_ASM_COMPILER="$PREFIX/bin/clang" \ + -DCMAKE_ASM_COMPILER_TARGET="$LLVM_TARGET" \ + -DCMAKE_ASM_FLAGS="$LLVM_COMPILER_FLAGS" \ + -DLLVM_CONFIG_PATH="$PREFIX/bin/llvm-config" \ + -DCOMPILER_RT_EXCLUDE_ATOMIC_BUILTIN=OFF \ + -DCOMPILER_RT_OS_DIR="serenity" \ + -DCOMPILER_RT_DEFAULT_TARGET_ONLY=ON \ + -DCOMPILER_RT_BUILD_BUILTINS=ON \ + -DCOMPILER_RT_BUILD_CRT=ON \ + -DCOMPILER_RT_BUILD_LIBFUZZER=OFF \ + -DCOMPILER_RT_BUILD_MEMPROF=OFF \ + -DCOMPILER_RT_BUILD_PROFILE=OFF \ + -DCOMPILER_RT_BUILD_SANITIZERS=OFF \ + -DCOMPILER_RT_BUILD_XRAY=OFF || exit 1 + + buildstep "compiler-rt/build" ninja -j "$MAKEJOBS" || exit 1 + buildstep "compiler-rt/install" ninja install || exit 1 + popd + + mkdir -p libunwind + pushd libunwind + buildstep "libunwind/configure" cmake "$DIR/Tarballs/llvm-project-$LLVM_VERSION.src/libunwind" \ + -GNinja \ + -DCMAKE_BUILD_TYPE="Release" \ + -DCMAKE_SYSROOT="$SYSROOT" \ + -DCMAKE_C_COMPILER="$PREFIX/bin/clang" \ + -DCMAKE_C_COMPILER_TARGET="$LLVM_TARGET" \ + -DCMAKE_C_FLAGS="$LLVM_COMPILER_FLAGS" \ + -DCMAKE_CXX_COMPILER="$PREFIX/bin/clang++" \ + -DCMAKE_CXX_COMPILER_TARGET="$LLVM_TARGET" \ + -DCMAKE_CXX_FLAGS="$LLVM_COMPILER_FLAGS" \ + -DCMAKE_INSTALL_PREFIX="$PREFIX" \ + -DLLVM_CONFIG_PATH="$PREFIX/bin/llvm-config" \ + -DLIBUNWIND_TARGET_TRIPLE="$LLVM_TARGET" \ + -DLIBUNWIND_SYSROOT="$SYSROOT" || exit 1 + + buildstep "libunwind/build" ninja -j "$MAKEJOBS" || exit 1 + buildstep "libunwind/install" ninja install || exit 1 + popd + + mkdir -p libcxxabi + pushd libcxxabi + buildstep "libcxxabi/configure" cmake "$DIR/Tarballs/llvm-project-$LLVM_VERSION.src/libcxxabi" \ + -GNinja \ + -DCMAKE_BUILD_TYPE="Release" \ + -DCMAKE_SYSROOT="$SYSROOT" \ + -DCMAKE_C_COMPILER="$PREFIX/bin/clang" \ + -DCMAKE_C_COMPILER_TARGET="$LLVM_TARGET" \ + -DCMAKE_C_FLAGS="$LLVM_COMPILER_FLAGS" \ + -DCMAKE_CXX_COMPILER="$PREFIX/bin/clang++" \ + -DCMAKE_CXX_COMPILER_TARGET="$LLVM_TARGET" \ + -DCMAKE_CXX_FLAGS="$LLVM_COMPILER_FLAGS" \ + -DCMAKE_INSTALL_PREFIX="$PREFIX" \ + -DLLVM_CONFIG_PATH="$PREFIX/bin/llvm-config" \ + -DLIBCXXABI_STANDALONE_BUILD=ON \ + -DLIBCXXABI_ENABLE_EXCEPTIONS=ON \ + -DLIBCXXABI_STANDALONE_BUILD=ON \ + -DLIBCXXABI_ENABLE_ASSERTIONS=OFF \ + -DLIBCXXABI_BAREMETAL=ON || exit 1 + + buildstep "libcxxabi/build" ninja -j "$MAKEJOBS" || exit 1 + buildstep "libcxxabi/install" ninja install || exit 1 + popd + + mkdir -p libcxx + pushd libcxx + buildstep "libcxx/configure" cmake "$DIR/Tarballs/llvm-project-$LLVM_VERSION.src/libcxx" \ + -G Ninja \ + -DCMAKE_BUILD_TYPE="Release" \ + -DCMAKE_BINARY_DIR="$PREFIX/usr" \ + -DCMAKE_SYSROOT="$SYSROOT" \ + -DCMAKE_C_COMPILER="$PREFIX/bin/clang" \ + -DCMAKE_C_FLAGS="$LLVM_COMPILER_FLAGS" \ + -DCMAKE_CXX_COMPILER="$PREFIX/bin/clang++" \ + -DCMAKE_CXX_FLAGS="$LLVM_COMPILER_FLAGS" \ + -DCMAKE_INSTALL_PREFIX="$PREFIX" \ + -DLLVM_CONFIG_PATH="$PREFIX/bin/llvm-config" \ + -DLLVM_BINARY_DIR="$PREFIX/usr" \ + -DLIBCXX_INSTALL_LIBRARY=ON \ + -DLIBCXX_ENABLE_LOCALIZATION=OFF \ + -DLIBCXX_ENABLE_FILESYSTEM=OFF \ + -DLIBCXX_INSTALL_HEADERS=ON \ + -DLIBCXX_ENABLE_SHARED=ON \ + -DLIBCXX_ENABLE_LOCALIZATION=OFF \ + -DLIBCXX_ENABLE_STATIC=ON \ + -DLIBCXX_CXX_ABI="libcxxabi" \ + -DLIBCXX_INCLUDE_BENCHMARKS=OFF || exit 1 + + buildstep "libcxx/build" ninja -j "$MAKEJOBS" || exit 1 + buildstep "libcxx/install" ninja install || exit 1 + buildstep "libcxx/install-headers" ninja install-cxx-headers || exit 1 + popd + + mkdir -p binutils + pushd binutils + buildstep "binutils/configure" "$DIR/Tarballs/$BINUTILS_NAME/configure" --prefix="$PREFIX/binutils" \ + --target="$ARCH-pc-serenity" \ + --disable-nls \ + --disable-gas \ + --disable-gold \ + --disable-ld \ + --disable-gprof \ + --enable-shared || exit 1 + buildstep "binutils/build" "$MAKE" -j "$MAKEJOBS" || exit 1 + buildstep "binutils/install" "$MAKE" install || exit 1 + popd +popd diff --git a/Toolchain/BuildIt.sh b/Toolchain/BuildIt.sh index f4931cb4c5..6b4eba77d4 100755 --- a/Toolchain/BuildIt.sh +++ b/Toolchain/BuildIt.sh @@ -71,6 +71,7 @@ echo SYSROOT is "$SYSROOT" mkdir -p "$DIR/Tarballs" +# Note: The version number and hash in BuildClang.sh needs to be kept in sync with this. BINUTILS_VERSION="2.37" BINUTILS_MD5SUM="1e55743d73c100b7a0d67ffb32398cdb" BINUTILS_NAME="binutils-$BINUTILS_VERSION" diff --git a/Toolchain/Patches/llvm.patch b/Toolchain/Patches/llvm.patch new file mode 100644 index 0000000000..6fc16786ab --- /dev/null +++ b/Toolchain/Patches/llvm.patch @@ -0,0 +1,933 @@ +diff --git a/clang/lib/Basic/Targets.cpp b/clang/lib/Basic/Targets.cpp +index 90a67d03b..ad21af415 100644 +--- a/clang/lib/Basic/Targets.cpp ++++ b/clang/lib/Basic/Targets.cpp +@@ -527,6 +527,8 @@ TargetInfo *AllocateTarget(const llvm::Triple &Triple, + return new MCUX86_32TargetInfo(Triple, Opts); + case llvm::Triple::Hurd: + return new HurdTargetInfo<X86_32TargetInfo>(Triple, Opts); ++ case llvm::Triple::Serenity: ++ return new SerenityTargetInfo<X86_32TargetInfo>(Triple, Opts); + default: + return new X86_32TargetInfo(Triple, Opts); + } +@@ -579,6 +581,8 @@ TargetInfo *AllocateTarget(const llvm::Triple &Triple, + return new NaClTargetInfo<X86_64TargetInfo>(Triple, Opts); + case llvm::Triple::PS4: + return new PS4OSTargetInfo<X86_64TargetInfo>(Triple, Opts); ++ case llvm::Triple::Serenity: ++ return new SerenityTargetInfo<X86_64TargetInfo>(Triple, Opts); + default: + return new X86_64TargetInfo(Triple, Opts); + } +diff --git a/clang/lib/Basic/Targets/OSTargets.h b/clang/lib/Basic/Targets/OSTargets.h +index 67fa1a537..9dd9039bb 100644 +--- a/clang/lib/Basic/Targets/OSTargets.h ++++ b/clang/lib/Basic/Targets/OSTargets.h +@@ -947,6 +947,22 @@ public: + : WebAssemblyOSTargetInfo<Target>(Triple, Opts) {} + }; + ++// SerenityOS Target ++template <typename Target> ++class LLVM_LIBRARY_VISIBILITY SerenityTargetInfo : public OSTargetInfo<Target> { ++protected: ++ void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, ++ MacroBuilder &Builder) const override { ++ Builder.defineMacro("__serenity__"); ++ Builder.defineMacro("__unix__"); ++ Builder.defineMacro("__ELF__"); ++ } ++ ++public: ++ SerenityTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) ++ : OSTargetInfo<Target>(Triple, Opts) {} ++}; ++ + } // namespace targets + } // namespace clang + #endif // LLVM_CLANG_LIB_BASIC_TARGETS_OSTARGETS_H +diff --git a/clang/lib/Driver/CMakeLists.txt b/clang/lib/Driver/CMakeLists.txt +index aeffcf0bb..43bec8fd7 100644 +--- a/clang/lib/Driver/CMakeLists.txt ++++ b/clang/lib/Driver/CMakeLists.txt +@@ -66,6 +66,7 @@ add_clang_library(clangDriver + ToolChains/OpenBSD.cpp + ToolChains/PS4CPU.cpp + ToolChains/RISCVToolchain.cpp ++ ToolChains/Serenity.cpp + ToolChains/Solaris.cpp + ToolChains/TCE.cpp + ToolChains/VEToolchain.cpp +diff --git a/clang/lib/Driver/Driver.cpp b/clang/lib/Driver/Driver.cpp +index 418e1d3e8..672aadc4d 100644 +--- a/clang/lib/Driver/Driver.cpp ++++ b/clang/lib/Driver/Driver.cpp +@@ -41,6 +41,7 @@ + #include "ToolChains/PPCLinux.h" + #include "ToolChains/PS4CPU.h" + #include "ToolChains/RISCVToolchain.h" ++#include "ToolChains/Serenity.h" + #include "ToolChains/Solaris.h" + #include "ToolChains/TCE.h" + #include "ToolChains/VEToolchain.h" +@@ -5128,6 +5129,9 @@ const ToolChain &Driver::getToolChain(const ArgList &Args, + case llvm::Triple::Fuchsia: + TC = std::make_unique<toolchains::Fuchsia>(*this, Target, Args); + break; ++ case llvm::Triple::Serenity: ++ TC = std::make_unique<toolchains::Serenity>(*this, Target, Args); ++ break; + case llvm::Triple::Solaris: + TC = std::make_unique<toolchains::Solaris>(*this, Target, Args); + break; +diff --git a/clang/lib/Driver/ToolChain.cpp b/clang/lib/Driver/ToolChain.cpp +index b2ddef141..48880a612 100644 +--- a/clang/lib/Driver/ToolChain.cpp ++++ b/clang/lib/Driver/ToolChain.cpp +@@ -398,6 +398,8 @@ StringRef ToolChain::getOSLibName() const { + return "sunos"; + case llvm::Triple::AIX: + return "aix"; ++ case llvm::Triple::Serenity: ++ return "serenity"; + default: + return getOS(); + } +diff --git a/clang/lib/Driver/ToolChains/Serenity.cpp b/clang/lib/Driver/ToolChains/Serenity.cpp +new file mode 100644 +index 000000000..8840188eb +--- /dev/null ++++ b/clang/lib/Driver/ToolChains/Serenity.cpp +@@ -0,0 +1,220 @@ ++//===--- Serenity.cpp - SerenityOS ToolChain Implementations ----*- C++ -*-===// ++// ++// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. ++// See https://llvm.org/LICENSE.txt for license information. ++// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception ++// ++//===----------------------------------------------------------------------===// ++ ++#include "Serenity.h" ++#include "CommonArgs.h" ++#include "InputInfo.h" ++#include "clang/Driver/Compilation.h" ++#include "clang/Driver/Driver.h" ++#include "clang/Driver/Options.h" ++#include "clang/Driver/SanitizerArgs.h" ++#include "llvm/Option/ArgList.h" ++#include "llvm/Support/VirtualFileSystem.h" ++ ++using namespace clang::driver; ++using namespace clang::driver::toolchains; ++using namespace clang; ++using namespace llvm::opt; ++ ++static bool getPIE(const ArgList &Args, const ToolChain &TC) { ++ if (Args.hasArg(options::OPT_static, options::OPT_shared, ++ options::OPT_static_pie)) ++ return false; ++ Arg *Last = Args.getLastArg(options::OPT_pie, options::OPT_no_pie, ++ options::OPT_nopie); ++ return Last ? Last->getOption().matches(options::OPT_pie) : TC.isPIEDefault(); ++} ++ ++void tools::serenity::Linker::ConstructJob(Compilation &C, const JobAction &JA, ++ const InputInfo &Output, ++ const InputInfoList &Inputs, ++ const ArgList &Args, ++ const char *LinkingOutput) const { ++ const auto &TC = getToolChain(); ++ const auto &D = TC.getDriver(); ++ const bool IsShared = Args.hasArg(options::OPT_shared); ++ const bool IsStatic = ++ Args.hasArg(options::OPT_static) && !Args.hasArg(options::OPT_static_pie); ++ const bool IsRdynamic = Args.hasArg(options::OPT_rdynamic); ++ const bool IsStaticPIE = Args.hasArg(options::OPT_static_pie); ++ const bool IsPIE = getPIE(Args, TC); ++ ArgStringList CmdArgs; ++ ++ if (!D.SysRoot.empty()) ++ CmdArgs.push_back(Args.MakeArgString("--sysroot=" + D.SysRoot)); ++ ++ if (IsPIE || IsStaticPIE) ++ CmdArgs.push_back("-pie"); ++ ++ if (IsShared) ++ CmdArgs.push_back("-shared"); ++ ++ if (IsStatic || IsStaticPIE) ++ CmdArgs.push_back("-static"); ++ ++ if (IsStaticPIE) { ++ CmdArgs.push_back("--no-dynamic-linker"); ++ CmdArgs.push_back("-z"); ++ CmdArgs.push_back("text"); ++ } ++ ++ if (!IsStatic && !IsStaticPIE) { ++ if (IsRdynamic) ++ CmdArgs.push_back("-export-dynamic"); ++ CmdArgs.push_back("-dynamic-linker"); ++ CmdArgs.push_back("/usr/lib/Loader.so"); ++ } ++ ++ if (!IsStatic || IsStaticPIE) ++ CmdArgs.push_back("--eh-frame-hdr"); ++ ++ if (Output.isFilename()) { ++ CmdArgs.push_back("-o"); ++ CmdArgs.push_back(Output.getFilename()); ++ } ++ ++ if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nostartfiles)) { ++ CmdArgs.push_back(Args.MakeArgString( ++ TC.GetFilePath((IsShared) ? "crt0_shared.o" : "crt0.o"))); ++ CmdArgs.push_back(Args.MakeArgString(TC.GetFilePath("crti.o"))); ++ ++ std::string crtbegin_path; ++ if (TC.GetRuntimeLibType(Args) == ToolChain::RLT_CompilerRT) { ++ std::string crtbegin = ++ TC.getCompilerRT(Args, "crtbegin", ToolChain::FT_Object); ++ if (TC.getVFS().exists(crtbegin)) ++ crtbegin_path = crtbegin; ++ } ++ if (crtbegin_path.empty()) { ++ const char *crtbegin = (IsShared || IsPIE) ? "crtbeginS.o" : "crtbegin.o"; ++ crtbegin_path = TC.GetFilePath(crtbegin); ++ } ++ CmdArgs.push_back(Args.MakeArgString(crtbegin_path)); ++ } ++ ++ Args.AddAllArgs(CmdArgs, options::OPT_L); ++ Args.AddAllArgs(CmdArgs, options::OPT_u); ++ ++ TC.AddFilePathLibArgs(Args, CmdArgs); ++ ++ if (D.isUsingLTO()) { ++ assert(!Inputs.empty() && "Must have at least one input."); ++ addLTOOptions(TC, Args, CmdArgs, Output, Inputs[0], ++ D.getLTOMode() == LTOK_Thin); ++ } ++ ++ Args.AddAllArgs(CmdArgs, options::OPT_T_Group); ++ Args.AddAllArgs(CmdArgs, options::OPT_e); ++ Args.AddAllArgs(CmdArgs, options::OPT_s); ++ Args.AddAllArgs(CmdArgs, options::OPT_t); ++ Args.AddAllArgs(CmdArgs, options::OPT_Z_Flag); ++ Args.AddAllArgs(CmdArgs, options::OPT_r); ++ ++ addLinkerCompressDebugSectionsOption(TC, Args, CmdArgs); ++ ++ AddLinkerInputs(TC, Inputs, Args, CmdArgs, JA); ++ ++ if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nodefaultlibs)) { ++ AddRunTimeLibs(TC, D, CmdArgs, Args); ++ ++ // We supply our own sanitizer runtimes ++ // FIXME: What if we want to use Clang-supplied ones as well? ++ const SanitizerArgs &Sanitize = TC.getSanitizerArgs(); ++ if (Sanitize.needsUbsanRt()) ++ CmdArgs.push_back("-lubsan"); ++ } ++ ++ if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nodefaultlibs)) { ++ if (D.CCCIsCXX()) { ++ if (TC.ShouldLinkCXXStdlib(Args)) ++ TC.AddCXXStdlibLibArgs(Args, CmdArgs); ++ CmdArgs.push_back("-lm"); ++ } ++ } ++ ++ if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nodefaultlibs)) { ++ if (Args.hasArg(options::OPT_pthread, options::OPT_pthreads)) ++ CmdArgs.push_back("-lpthread"); ++ ++ if (!Args.hasArg(options::OPT_nolibc)) ++ CmdArgs.push_back("-lc"); ++ } ++ ++ if (!Args.hasArg(options::OPT_nostdlib, options::OPT_nostartfiles)) { ++ std::string crtend_path; ++ if (TC.GetRuntimeLibType(Args) == ToolChain::RLT_CompilerRT) { ++ std::string crtend = ++ TC.getCompilerRT(Args, "crtend", ToolChain::FT_Object); ++ if (TC.getVFS().exists(crtend)) ++ crtend_path = crtend; ++ } ++ if (crtend_path.empty()) { ++ const char *crtend = (IsShared || IsPIE) ? "crtendS.o" : "crtend.o"; ++ crtend_path = TC.GetFilePath(crtend); ++ } ++ CmdArgs.push_back(Args.MakeArgString(crtend_path)); ++ ++ CmdArgs.push_back(Args.MakeArgString(TC.GetFilePath("crtn.o"))); ++ } ++ ++ Args.AddAllArgs(CmdArgs, options::OPT_T); ++ ++ const char *Exec = Args.MakeArgString(TC.GetLinkerPath()); ++ C.addCommand(std::make_unique<Command>(JA, *this, ++ ResponseFileSupport::AtFileCurCP(), ++ Exec, CmdArgs, Inputs, Output)); ++} ++ ++Serenity::Serenity(const Driver &D, const llvm::Triple &Triple, ++ const ArgList &Args) ++ : Generic_ELF(D, Triple, Args) { ++ getFilePaths().push_back(getDriver().Dir + "/../lib"); ++ if (!getDriver().SysRoot.empty()) ++ getFilePaths().push_back(getDriver().SysRoot + "/usr/lib"); ++} ++ ++Tool *Serenity::buildLinker() const { ++ return new tools::serenity::Linker(*this); ++} ++ ++void Serenity::AddClangSystemIncludeArgs(const ArgList &DriverArgs, ++ ArgStringList &CC1Args) const { ++ const Driver &D = getDriver(); ++ ++ if (DriverArgs.hasArg(options::OPT_nostdinc)) ++ return; ++ ++ if (!DriverArgs.hasArg(options::OPT_nobuiltininc)) { ++ addSystemInclude(DriverArgs, CC1Args, D.ResourceDir + "/include"); ++ } ++ ++ if (DriverArgs.hasArg(options::OPT_nostdlibinc)) ++ return; ++ ++ if (!D.SysRoot.empty()) { ++ addExternCSystemInclude(DriverArgs, CC1Args, D.SysRoot + "/usr/include"); ++ } ++} ++ ++void Serenity::AddClangCXXStdlibIncludeArgs(const ArgList &DriverArgs, ++ ArgStringList &CC1Args) const { ++ if (DriverArgs.hasArg(options::OPT_nostdlibinc, options::OPT_nostdincxx)) ++ return; ++ ++ switch (GetCXXStdlibType(DriverArgs)) { ++ case ToolChain::CST_Libcxx: { ++ addSystemInclude(DriverArgs, CC1Args, ++ getDriver().SysRoot + "/usr/include/c++/v1"); ++ addSystemInclude(DriverArgs, CC1Args, ++ getDriver().Dir + "/../include/c++/v1"); ++ break; ++ } ++ default: ++ llvm_unreachable("invalid stdlib name"); ++ } ++} +diff --git a/clang/lib/Driver/ToolChains/Serenity.h b/clang/lib/Driver/ToolChains/Serenity.h +new file mode 100644 +index 000000000..a45a0c853 +--- /dev/null ++++ b/clang/lib/Driver/ToolChains/Serenity.h +@@ -0,0 +1,78 @@ ++//===--- Serenity.h - SerenityOS ToolChain Implementations ------*- C++ -*-===// ++// ++// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. ++// See https://llvm.org/LICENSE.txt for license information. ++// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception ++// ++//===----------------------------------------------------------------------===// ++ ++#ifndef LLVM_CLANG_LIB_DRIVER_TOOLCHAINS_SERENITY_H ++#define LLVM_CLANG_LIB_DRIVER_TOOLCHAINS_SERENITY_H ++ ++#include "Gnu.h" ++#include "clang/Driver/Tool.h" ++#include "clang/Driver/ToolChain.h" ++ ++namespace clang { ++namespace driver { ++namespace tools { ++/// Serenity -- Directly call GNU Binutils assembler and linker ++namespace serenity { ++ ++class LLVM_LIBRARY_VISIBILITY Linker : public Tool { ++public: ++ Linker(const ToolChain &TC) : Tool("serenity::Linker", "linker", TC) {} ++ ++ bool hasIntegratedCPP() const override { return false; } ++ bool isLinkJob() const override { return true; } ++ ++ void ConstructJob(Compilation &C, const JobAction &JA, ++ const InputInfo &Output, const InputInfoList &Inputs, ++ const llvm::opt::ArgList &TCArgs, ++ const char *LinkingOutput) const override; ++}; ++} // end namespace serenity ++} // end namespace tools ++ ++namespace toolchains { ++ ++class LLVM_LIBRARY_VISIBILITY Serenity : public Generic_ELF { ++public: ++ Serenity(const Driver &D, const llvm::Triple &Triple, ++ const llvm::opt::ArgList &Args); ++ ++ void ++ AddClangSystemIncludeArgs(const llvm::opt::ArgList &DriverArgs, ++ llvm::opt::ArgStringList &CC1Args) const override; ++ ++ void AddClangCXXStdlibIncludeArgs( ++ const llvm::opt::ArgList &DriverArgs, ++ llvm::opt::ArgStringList &CC1Args) const override; ++ ++ RuntimeLibType GetDefaultRuntimeLibType() const override { ++ return ToolChain::RLT_CompilerRT; ++ } ++ ++ UnwindLibType GetDefaultUnwindLibType() const override { ++ return ToolChain::UNW_CompilerRT; ++ } ++ ++ CXXStdlibType GetDefaultCXXStdlibType() const override { ++ return ToolChain::CST_Libcxx; ++ } ++ const char *getDefaultLinker() const override { return "ld.lld"; } ++ ++ bool HasNativeLLVMSupport() const override { return true; } ++ bool IsIntegratedAssemblerDefault() const override { return true; } ++ ++ unsigned GetDefaultDwarfVersion() const override { return 5; } ++ ++protected: ++ Tool *buildLinker() const override; ++}; ++ ++} // end namespace toolchains ++} // end namespace driver ++} // end namespace clang ++ ++#endif // LLVM_CLANG_LIB_DRIVER_TOOLCHAINS_SERENITY_H +diff --git a/clang/lib/Frontend/InitHeaderSearch.cpp b/clang/lib/Frontend/InitHeaderSearch.cpp +index bc31445d6..a4a3fbc89 100644 +--- a/clang/lib/Frontend/InitHeaderSearch.cpp ++++ b/clang/lib/Frontend/InitHeaderSearch.cpp +@@ -229,6 +229,7 @@ void InitHeaderSearch::AddDefaultCIncludePaths(const llvm::Triple &triple, + case llvm::Triple::PS4: + case llvm::Triple::ELFIAMCU: + case llvm::Triple::Fuchsia: ++ case llvm::Triple::Serenity: + break; + case llvm::Triple::Win32: + if (triple.getEnvironment() != llvm::Triple::Cygnus) +@@ -428,6 +429,7 @@ void InitHeaderSearch::AddDefaultIncludePaths(const LangOptions &Lang, + case llvm::Triple::Solaris: + case llvm::Triple::WASI: + case llvm::Triple::AIX: ++ case llvm::Triple::Serenity: + return; + + case llvm::Triple::Win32: +diff --git a/libcxx/include/__config b/libcxx/include/__config +index a3838c89e..8a20731ef 100644 +--- a/libcxx/include/__config ++++ b/libcxx/include/__config +@@ -1132,6 +1132,7 @@ extern "C" _LIBCPP_FUNC_VIS void __sanitizer_annotate_contiguous_container( + defined(__CloudABI__) || \ + defined(__sun__) || \ + defined(__MVS__) || \ ++ defined(__serenity__) || \ + (defined(__MINGW32__) && __has_include(<pthread.h>)) + # define _LIBCPP_HAS_THREAD_API_PTHREAD + # elif defined(__Fuchsia__) +diff --git a/libcxx/include/__string b/libcxx/include/__string +index d8b672e4c..638cf1698 100644 +--- a/libcxx/include/__string ++++ b/libcxx/include/__string +@@ -451,6 +451,7 @@ char_traits<char>::find(const char_type* __s, size_t __n, const char_type& __a) + } + + ++#ifndef __serenity__ + // char_traits<wchar_t> + + template <> +@@ -531,6 +532,7 @@ char_traits<wchar_t>::compare(const char_type* __s1, const char_type* __s2, size + #endif + } + ++#endif + + template <class _Traits> + _LIBCPP_INLINE_VISIBILITY +@@ -543,6 +545,7 @@ inline size_t __char_traits_length_checked(const typename _Traits::char_type* __ + #endif + } + ++#ifndef __serenity__ + inline _LIBCPP_CONSTEXPR_AFTER_CXX14 + size_t + char_traits<wchar_t>::length(const char_type* __s) _NOEXCEPT +@@ -579,6 +582,7 @@ char_traits<wchar_t>::find(const char_type* __s, size_t __n, const char_type& __ + return nullptr; + #endif + } ++#endif + + + #ifndef _LIBCPP_NO_HAS_CHAR8_T +diff --git a/libcxx/include/cwchar b/libcxx/include/cwchar +index 451c621f9..1e83b8e60 100644 +--- a/libcxx/include/cwchar ++++ b/libcxx/include/cwchar +@@ -112,11 +112,13 @@ size_t wcsrtombs(char* restrict dst, const wchar_t** restrict src, size_t len, + + _LIBCPP_BEGIN_NAMESPACE_STD + +-using ::mbstate_t; + using ::size_t; ++using ::FILE; ++ ++#ifndef __serenity__ ++using ::mbstate_t; + using ::tm; + using ::wint_t; +-using ::FILE; + using ::fwprintf; + using ::fwscanf; + using ::swprintf; +@@ -186,6 +188,7 @@ using ::putwchar; + using ::vwprintf; + using ::wprintf; + #endif ++#endif + + _LIBCPP_END_NAMESPACE_STD + +diff --git a/libcxx/include/cwctype b/libcxx/include/cwctype +index 575fd5661..c401b6cf0 100644 +--- a/libcxx/include/cwctype ++++ b/libcxx/include/cwctype +@@ -59,6 +59,7 @@ wctrans_t wctrans(const char* property); + + _LIBCPP_BEGIN_NAMESPACE_STD + ++#ifndef __serenity__ + using ::wint_t; + using ::wctrans_t; + using ::wctype_t; +@@ -80,6 +81,7 @@ using ::towlower; + using ::towupper; + using ::towctrans; + using ::wctrans; ++#endif + + _LIBCPP_END_NAMESPACE_STD + +diff --git a/libcxx/include/initializer_list b/libcxx/include/initializer_list +index 893736f57..70a0d1ccf 100644 +--- a/libcxx/include/initializer_list ++++ b/libcxx/include/initializer_list +@@ -43,7 +43,9 @@ template<class E> const E* end(initializer_list<E> il) noexcept; // constexpr in + */ + + #include <__config> ++#ifndef KERNEL + #include <cstddef> ++#endif + + #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER) + #pragma GCC system_header +diff --git a/libcxx/include/iosfwd b/libcxx/include/iosfwd +index 0a0de99ff..790dc4023 100644 +--- a/libcxx/include/iosfwd ++++ b/libcxx/include/iosfwd +@@ -92,7 +92,11 @@ typedef fpos<char_traits<wchar_t>::state_type> wstreampos; + */ + + #include <__config> ++#if 0 + #include <wchar.h> // for mbstate_t ++#endif ++ ++#define mbstate_t void + + #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER) + #pragma GCC system_header +@@ -109,7 +113,9 @@ template<> struct char_traits<char8_t>; + #endif + template<> struct char_traits<char16_t>; + template<> struct char_traits<char32_t>; ++#if 0 + template<> struct char_traits<wchar_t>; ++#endif + + template<class _Tp> class _LIBCPP_TEMPLATE_VIS allocator; + +diff --git a/libcxx/include/new b/libcxx/include/new +index 0562cef45..82f4ca8be 100644 +--- a/libcxx/include/new ++++ b/libcxx/include/new +@@ -301,6 +301,8 @@ inline _LIBCPP_INLINE_VISIBILITY void __libcpp_deallocate_unsized(void* __ptr, s + #endif + } + ++#define _LIBCPP_HAS_NO_LIBRARY_ALIGNED_ALLOCATION ++ + #if !defined(_LIBCPP_HAS_NO_LIBRARY_ALIGNED_ALLOCATION) + // Low-level helpers to call the aligned allocation and deallocation functions + // on the target platform. This is used to implement libc++'s own memory +diff --git a/libcxx/include/string b/libcxx/include/string +index 687795c79..f3a216647 100644 +--- a/libcxx/include/string ++++ b/libcxx/include/string +@@ -1728,8 +1728,10 @@ _LIBCPP_STRING_UNSTABLE_EXTERN_TEMPLATE_LIST(_LIBCPP_EXTERN_TEMPLATE, char) + _LIBCPP_STRING_UNSTABLE_EXTERN_TEMPLATE_LIST(_LIBCPP_EXTERN_TEMPLATE, wchar_t) + #else + _LIBCPP_STRING_V1_EXTERN_TEMPLATE_LIST(_LIBCPP_EXTERN_TEMPLATE, char) ++#ifndef __serenity__ + _LIBCPP_STRING_V1_EXTERN_TEMPLATE_LIST(_LIBCPP_EXTERN_TEMPLATE, wchar_t) + #endif ++#endif + + + #ifndef _LIBCPP_HAS_NO_DEDUCTION_GUIDES +@@ -4527,11 +4529,13 @@ inline namespace literals + return basic_string<char> (__str, __len); + } + ++#ifndef __serenity__ + inline _LIBCPP_INLINE_VISIBILITY + basic_string<wchar_t> operator "" s( const wchar_t *__str, size_t __len ) + { + return basic_string<wchar_t> (__str, __len); + } ++#endif + + #ifndef _LIBCPP_NO_HAS_CHAR8_T + inline _LIBCPP_INLINE_VISIBILITY +diff --git a/libcxx/include/string_view b/libcxx/include/string_view +index bc0245cf2..55f9b62c4 100644 +--- a/libcxx/include/string_view ++++ b/libcxx/include/string_view +@@ -204,7 +204,9 @@ typedef basic_string_view<char8_t> u8string_view; + #endif + typedef basic_string_view<char16_t> u16string_view; + typedef basic_string_view<char32_t> u32string_view; ++#ifndef __serenity__ + typedef basic_string_view<wchar_t> wstring_view; ++#endif + + template<class _CharT, class _Traits> + class +@@ -214,7 +216,9 @@ class + #endif + _LIBCPP_PREFERRED_NAME(u16string_view) + _LIBCPP_PREFERRED_NAME(u32string_view) ++#ifndef __serenity__ + _LIBCPP_PREFERRED_NAME(wstring_view) ++#endif + basic_string_view { + public: + // types +@@ -836,11 +840,13 @@ inline namespace literals + return basic_string_view<char> (__str, __len); + } + ++#ifndef __serenity__ + inline _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR + basic_string_view<wchar_t> operator "" sv(const wchar_t *__str, size_t __len) _NOEXCEPT + { + return basic_string_view<wchar_t> (__str, __len); + } ++#endif + + #ifndef _LIBCPP_NO_HAS_CHAR8_T + inline _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR +diff --git a/libcxx/include/wchar.h b/libcxx/include/wchar.h +index b21a78968..ae20183ed 100644 +--- a/libcxx/include/wchar.h ++++ b/libcxx/include/wchar.h +@@ -132,6 +132,7 @@ size_t wcsrtombs(char* restrict dst, const wchar_t** restrict src, size_t len, + #endif + + #if defined(__cplusplus) && !defined(_LIBCPP_WCHAR_H_HAS_CONST_OVERLOADS) && defined(_LIBCPP_PREFERRED_OVERLOAD) ++#ifndef __serenity__ + extern "C++" { + inline _LIBCPP_INLINE_VISIBILITY + wchar_t* __libcpp_wcschr(const wchar_t* __s, wchar_t __c) {return (wchar_t*)wcschr(__s, __c);} +@@ -178,5 +179,6 @@ size_t wcsnrtombs(char *__restrict dst, const wchar_t **__restrict src, + size_t nwc, size_t len, mbstate_t *__restrict ps); + } // extern "C++" + #endif // __cplusplus && _LIBCPP_MSVCRT ++#endif + + #endif // _LIBCPP_WCHAR_H +diff --git a/libcxx/src/string.cpp b/libcxx/src/string.cpp +index 5105594cf..ca85269db 100644 +--- a/libcxx/src/string.cpp ++++ b/libcxx/src/string.cpp +@@ -131,6 +131,7 @@ as_integer( const string& func, const string& s, size_t* idx, int base ) + return as_integer_helper<unsigned long long>( func, s, idx, base, strtoull ); + } + ++#ifndef __serenity__ + // wstring + template<> + inline +@@ -175,6 +176,7 @@ as_integer( const string& func, const wstring& s, size_t* idx, int base ) + { + return as_integer_helper<unsigned long long>( func, s, idx, base, wcstoull ); + } ++#endif + + // as_float + +@@ -226,6 +228,8 @@ as_float( const string& func, const string& s, size_t* idx ) + return as_float_helper<long double>( func, s, idx, strtold ); + } + ++ ++#ifndef __serenity__ + template<> + inline + float +@@ -249,6 +253,7 @@ as_float( const string& func, const wstring& s, size_t* idx ) + { + return as_float_helper<long double>( func, s, idx, wcstold ); + } ++#endif + + } // unnamed namespace + +@@ -258,11 +263,13 @@ stoi(const string& str, size_t* idx, int base) + return as_integer<int>( "stoi", str, idx, base ); + } + ++#ifndef __serenity__ + int + stoi(const wstring& str, size_t* idx, int base) + { + return as_integer<int>( "stoi", str, idx, base ); + } ++#endif + + long + stol(const string& str, size_t* idx, int base) +@@ -270,11 +277,13 @@ stol(const string& str, size_t* idx, int base) + return as_integer<long>( "stol", str, idx, base ); + } + ++#ifndef __serenity__ + long + stol(const wstring& str, size_t* idx, int base) + { + return as_integer<long>( "stol", str, idx, base ); + } ++#endif + + unsigned long + stoul(const string& str, size_t* idx, int base) +@@ -282,11 +291,13 @@ stoul(const string& str, size_t* idx, int base) + return as_integer<unsigned long>( "stoul", str, idx, base ); + } + ++#ifndef __serenity__ + unsigned long + stoul(const wstring& str, size_t* idx, int base) + { + return as_integer<unsigned long>( "stoul", str, idx, base ); + } ++#endif + + long long + stoll(const string& str, size_t* idx, int base) +@@ -294,11 +305,13 @@ stoll(const string& str, size_t* idx, int base) + return as_integer<long long>( "stoll", str, idx, base ); + } + ++#ifndef __serenity__ + long long + stoll(const wstring& str, size_t* idx, int base) + { + return as_integer<long long>( "stoll", str, idx, base ); + } ++#endif + + unsigned long long + stoull(const string& str, size_t* idx, int base) +@@ -306,11 +319,13 @@ stoull(const string& str, size_t* idx, int base) + return as_integer<unsigned long long>( "stoull", str, idx, base ); + } + ++#ifndef __serenity__ + unsigned long long + stoull(const wstring& str, size_t* idx, int base) + { + return as_integer<unsigned long long>( "stoull", str, idx, base ); + } ++#endif + + float + stof(const string& str, size_t* idx) +@@ -318,11 +333,13 @@ stof(const string& str, size_t* idx) + return as_float<float>( "stof", str, idx ); + } + ++#ifndef __serenity__ + float + stof(const wstring& str, size_t* idx) + { + return as_float<float>( "stof", str, idx ); + } ++#endif + + double + stod(const string& str, size_t* idx) +@@ -330,11 +347,13 @@ stod(const string& str, size_t* idx) + return as_float<double>( "stod", str, idx ); + } + ++#ifndef __serenity__ + double + stod(const wstring& str, size_t* idx) + { + return as_float<double>( "stod", str, idx ); + } ++#endif + + long double + stold(const string& str, size_t* idx) +@@ -342,11 +361,13 @@ stold(const string& str, size_t* idx) + return as_float<long double>( "stold", str, idx ); + } + ++#ifndef __serenity__ + long double + stold(const wstring& str, size_t* idx) + { + return as_float<long double>( "stold", str, idx ); + } ++#endif + + // to_string + +@@ -397,6 +418,7 @@ struct initial_string<string> + } + }; + ++#ifndef __serenity__ + template <> + struct initial_string<wstring> + { +@@ -408,7 +430,9 @@ struct initial_string<wstring> + return s; + } + }; ++#endif + ++#ifndef __serenity__ + typedef int (*wide_printf)(wchar_t* __restrict, size_t, const wchar_t*__restrict, ...); + + inline +@@ -421,6 +445,7 @@ get_swprintf() + return static_cast<int (__cdecl*)(wchar_t* __restrict, size_t, const wchar_t*__restrict, ...)>(_snwprintf); + #endif + } ++#endif + + template <typename S, typename V> + S i_to_string(const V v) +@@ -444,20 +469,24 @@ string to_string (unsigned val) { return i_to_string< string>(val); } + string to_string (unsigned long val) { return i_to_string< string>(val); } + string to_string (unsigned long long val) { return i_to_string< string>(val); } + ++#ifndef __serenity__ + wstring to_wstring(int val) { return i_to_string<wstring>(val); } + wstring to_wstring(long val) { return i_to_string<wstring>(val); } + wstring to_wstring(long long val) { return i_to_string<wstring>(val); } + wstring to_wstring(unsigned val) { return i_to_string<wstring>(val); } + wstring to_wstring(unsigned long val) { return i_to_string<wstring>(val); } + wstring to_wstring(unsigned long long val) { return i_to_string<wstring>(val); } ++#endif + + + string to_string (float val) { return as_string(snprintf, initial_string< string>()(), "%f", val); } + string to_string (double val) { return as_string(snprintf, initial_string< string>()(), "%f", val); } + string to_string (long double val) { return as_string(snprintf, initial_string< string>()(), "%Lf", val); } + ++#ifndef __serenity__ + wstring to_wstring(float val) { return as_string(get_swprintf(), initial_string<wstring>()(), L"%f", val); } + wstring to_wstring(double val) { return as_string(get_swprintf(), initial_string<wstring>()(), L"%f", val); } + wstring to_wstring(long double val) { return as_string(get_swprintf(), initial_string<wstring>()(), L"%Lf", val); } ++#endif + + _LIBCPP_END_NAMESPACE_STD +diff --git a/libunwind/src/AddressSpace.hpp b/libunwind/src/AddressSpace.hpp +index 171318ff6..aa7345a19 100644 +--- a/libunwind/src/AddressSpace.hpp ++++ b/libunwind/src/AddressSpace.hpp +@@ -24,7 +24,7 @@ + #include "Registers.hpp" + + #ifndef _LIBUNWIND_USE_DLADDR +- #if !defined(_LIBUNWIND_IS_BAREMETAL) && !defined(_WIN32) ++ #if !defined(_LIBUNWIND_IS_BAREMETAL) && !defined(_WIN32) && !defined(__serenity__) + #define _LIBUNWIND_USE_DLADDR 1 + #else + #define _LIBUNWIND_USE_DLADDR 0 +diff --git a/llvm/include/llvm/ADT/Triple.h b/llvm/include/llvm/ADT/Triple.h +index eed315c92..11e53b8ac 100644 +--- a/llvm/include/llvm/ADT/Triple.h ++++ b/llvm/include/llvm/ADT/Triple.h +@@ -198,7 +198,8 @@ public: + Hurd, // GNU/Hurd + WASI, // Experimental WebAssembly OS + Emscripten, +- LastOSType = Emscripten ++ Serenity, ++ LastOSType = Serenity + }; + enum EnvironmentType { + UnknownEnvironment, +@@ -626,6 +627,11 @@ public: + return getOS() == Triple::AIX; + } + ++ /// Tests whether the OS is SerenityOS ++ bool isOSSerenity() const { ++ return getOS() == Triple::Serenity; ++ } ++ + /// Tests whether the OS uses the ELF binary format. + bool isOSBinFormatELF() const { + return getObjectFormat() == Triple::ELF; +diff --git a/llvm/include/llvm/Support/SwapByteOrder.h b/llvm/include/llvm/Support/SwapByteOrder.h +index e8612ba66..f0109f4b3 100644 +--- a/llvm/include/llvm/Support/SwapByteOrder.h ++++ b/llvm/include/llvm/Support/SwapByteOrder.h +@@ -22,7 +22,7 @@ + #endif + + #if defined(__linux__) || defined(__GNU__) || defined(__HAIKU__) || \ +- defined(__Fuchsia__) || defined(__EMSCRIPTEN__) ++ defined(__Fuchsia__) || defined(__EMSCRIPTEN__) || defined(__serenity__) + #include <endian.h> + #elif defined(_AIX) + #include <sys/machine.h> +diff --git a/llvm/lib/Support/Triple.cpp b/llvm/lib/Support/Triple.cpp +index 4f483c965..bdbda22c9 100644 +--- a/llvm/lib/Support/Triple.cpp ++++ b/llvm/lib/Support/Triple.cpp +@@ -215,6 +215,7 @@ StringRef Triple::getOSTypeName(OSType Kind) { + case PS4: return "ps4"; + case RTEMS: return "rtems"; + case Solaris: return "solaris"; ++ case Serenity: return "serenity"; + case TvOS: return "tvos"; + case WASI: return "wasi"; + case WatchOS: return "watchos"; +@@ -531,6 +532,7 @@ static Triple::OSType parseOS(StringRef OSName) { + .StartsWith("hurd", Triple::Hurd) + .StartsWith("wasi", Triple::WASI) + .StartsWith("emscripten", Triple::Emscripten) ++ .StartsWith("serenity", Triple::Serenity) + .Default(Triple::UnknownOS); + } + |