From 742777c7d4ec667f82985481aa5747c7694b4254 Mon Sep 17 00:00:00 2001 From: abeimler Date: Wed, 25 Jan 2023 12:55:57 +0100 Subject: [PATCH] feat: make enable_cross_compiler more open-closed (#180) --- .github/workflows/ci.cross.arm.yml | 19 +- docker-compose.yml | 17 +- docker/Dockerfile.aarch64 | 22 ++ docker/Dockerfile.arm | 4 +- docker/Dockerfile.arm-bare-metal | 27 +++ docker/Taskfile.yml | 12 ++ src/CrossCompiler.cmake | 194 +++++++++++++----- src/Vcpkg.cmake | 24 ++- ...in.cmake => aarch64-linux.toolchain.cmake} | 6 +- src/toolchains/arm-linux.toolchain.cmake | 38 ++++ src/toolchains/arm.toolchain.cmake | 8 +- ...hain.cmake => arm64-linux.toolchain.cmake} | 6 +- .../i686-w64-mingw32.toolchain.cmake | 20 +- .../x86_64-w64-mingw32.toolchain.cmake | 20 +- tests/rpi3/CMakeLists.txt | 29 ++- tests/rpi3/Taskfile.yml | 10 +- tests/rpi3/main.c | 17 +- tests/rpi4-vcpkg/Taskfile.yml | 6 + tests/rpi4/CMakeLists.txt | 21 +- tests/rpi4/Taskfile.yml | 6 + 20 files changed, 403 insertions(+), 103 deletions(-) create mode 100644 docker/Dockerfile.arm-bare-metal rename src/toolchains/{aarch64.toolchain.cmake => aarch64-linux.toolchain.cmake} (91%) create mode 100644 src/toolchains/arm-linux.toolchain.cmake rename src/toolchains/{arm64.toolchain.cmake => arm64-linux.toolchain.cmake} (91%) diff --git a/.github/workflows/ci.cross.arm.yml b/.github/workflows/ci.cross.arm.yml index a020fcc7..97050567 100644 --- a/.github/workflows/ci.cross.arm.yml +++ b/.github/workflows/ci.cross.arm.yml @@ -19,21 +19,33 @@ jobs: - true task: - rpi4:build.cross - - rpi4-vcpkg:build.cross - rpi4:build.cross.custom-toolchain + - rpi4-vcpkg:build.cross + - rpi4-vcpkg:build.cross.custom-toolchain + - rpi4:build.cross.aarch64 - rpi3:build.cross + - rpi3:build.cross.bare-metal include: - task: rpi4:build.cross install-cross-compiler: g++-aarch64-linux-gnu gcc-aarch64-linux-gnu binutils-aarch64-linux-gnu target: aarch64-linux-gnu + - task: rpi4:build.cross.custom-toolchain + install-cross-compiler: g++-aarch64-linux-gnu gcc-aarch64-linux-gnu binutils-aarch64-linux-gnu + target: aarch64-linux-gnu - task: rpi4-vcpkg:build.cross install-cross-compiler: g++-aarch64-linux-gnu gcc-aarch64-linux-gnu binutils-aarch64-linux-gnu target: aarch64-linux-gnu - - task: rpi4:build.cross.custom-toolchain + - task: rpi4-vcpkg:build.cross.custom-toolchain + install-cross-compiler: g++-aarch64-linux-gnu gcc-aarch64-linux-gnu binutils-aarch64-linux-gnu + target: aarch64-linux-gnu + - task: rpi4:build.cross.aarch64 install-cross-compiler: g++-aarch64-linux-gnu gcc-aarch64-linux-gnu binutils-aarch64-linux-gnu target: aarch64-linux-gnu - task: rpi3:build.cross - install-cross-compiler: gcc-arm-none-eabi binutils-arm-none-eabi libnewlib-arm-none-eabi + install-cross-compiler: gcc-arm-linux-gnueabi g++-arm-linux-gnueabi gcc-arm-linux-gnueabihf g++-arm-linux-gnueabihf + target: arm-linux-gnueabi + - task: rpi3:build.cross.bare-metal + install-cross-compiler: gcc-arm-none-eabi g++-arm-none-eabi binutils-arm-none-eabi libnewlib-arm-none-eabi target: arm-none-eabi steps: - uses: actions/checkout@v3 @@ -64,7 +76,6 @@ jobs: clangtidy: true task: true doxygen: true - powershell: true - name: Setup ARM (Cross) Compiler uses: awalsh128/cache-apt-pkgs-action@latest diff --git a/docker-compose.yml b/docker-compose.yml index 6b43e85c..38602127 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -109,8 +109,23 @@ services: context: . dockerfile: ./docker/Dockerfile.aarch64 target: build-vcpkg + build-rpi4-vcpkg-custom: + build: + context: . + dockerfile: ./docker/Dockerfile.aarch64 + target: build-vcpkg-custom build-rpi4-custom: build: context: . dockerfile: ./docker/Dockerfile.aarch64 - target: build-custom \ No newline at end of file + target: build-custom + build-rpi4-aarch64: + build: + context: . + dockerfile: ./docker/Dockerfile.aarch64 + target: build-aarch64 + build-rpi3-bare-metal: + build: + context: . + dockerfile: ./docker/Dockerfile.arm-bare-metal + target: build \ No newline at end of file diff --git a/docker/Dockerfile.aarch64 b/docker/Dockerfile.aarch64 index a16b09e5..f992d821 100644 --- a/docker/Dockerfile.aarch64 +++ b/docker/Dockerfile.aarch64 @@ -45,6 +45,17 @@ WORKDIR /home/project_options RUN git submodule update --init CMD ["/bin/bash", "-c", "task rpi4:build.cross.custom-toolchain"] +FROM setup AS build-vcpkg-custom +COPY . /home/project_options +WORKDIR /home/project_options +RUN git submodule update --init +CMD ["/bin/bash", "-c", "task rpi4-vcpkg:build.cross.custom-toolchain"] + +FROM setup AS build-aarch64 +COPY . /home/project_options +WORKDIR /home/project_options +RUN git submodule update --init +CMD ["/bin/bash", "-c", "task rpi4:build.cross.aarch64"] FROM setup AS test RUN apt-get update && apt-get install -y \ @@ -55,3 +66,14 @@ WORKDIR /home/project_options RUN git submodule update --init ENV QEMU_LD_PREFIX /usr/aarch64-linux-gnu CMD ["/bin/bash", "-c", "task rpi4:build.cross && qemu-aarch64 /home/project_options/tests/rpi4/build/Release/example"] + + +FROM setup AS test-aarch64 +RUN apt-get update && apt-get install -y \ + qemu-user \ + && rm -rf /var/lib/apt/lists/* +COPY . /home/project_options +WORKDIR /home/project_options +RUN git submodule update --init +ENV QEMU_LD_PREFIX /usr/aarch64-linux-gnu +CMD ["/bin/bash", "-c", "task rpi4:build.aarch64 && qemu-aarch64 /home/project_options/tests/rpi4/build/Release/example"] diff --git a/docker/Dockerfile.arm b/docker/Dockerfile.arm index 975b6ef5..28249144 100644 --- a/docker/Dockerfile.arm +++ b/docker/Dockerfile.arm @@ -14,7 +14,7 @@ FROM base AS setup RUN setup-cpp --clangtidy true --clangformat true --cmake true --ninja true --ccache true --cppcheck true --vcpkg true --conan true --task true RUN apt-get update && apt-get install -y \ - gcc-arm-none-eabi binutils-arm-none-eabi libnewlib-arm-none-eabi \ + gcc-arm-linux-gnueabi g++-arm-linux-gnueabi gcc-arm-linux-gnueabihf g++-arm-linux-gnueabihf \ && rm -rf /var/lib/apt/lists/* COPY ./docker/entrypoint.sh /docker-entrypoint.sh @@ -32,4 +32,4 @@ FROM setup AS build-debug COPY . /home/project_options WORKDIR /home/project_options RUN git submodule update --init -CMD ["/bin/bash", "-c", "task rpi3:build.cross.debug"] \ No newline at end of file +CMD ["/bin/bash", "-c", "task rpi3:build.cross.debug"] diff --git a/docker/Dockerfile.arm-bare-metal b/docker/Dockerfile.arm-bare-metal new file mode 100644 index 00000000..517d25bb --- /dev/null +++ b/docker/Dockerfile.arm-bare-metal @@ -0,0 +1,27 @@ +FROM ubuntu:22.04 AS base + +# add setup-cpp https://github.com/aminya/setup-cpp +RUN apt-get update && apt-get install -y \ + npm \ + && rm -rf /var/lib/apt/lists/* +RUN npm install -g setup-cpp + + +FROM base AS setup + +# install cmake, ninja, and ccache +RUN setup-cpp --clangtidy true --clangformat true --cmake true --ninja true --ccache true --cppcheck true --vcpkg true --conan true --task true + +RUN apt-get update && apt-get install -y \ + gcc-arm-none-eabi binutils-arm-none-eabi libnewlib-arm-none-eabi \ + && rm -rf /var/lib/apt/lists/* + +COPY ./docker/entrypoint.sh /docker-entrypoint.sh +ENTRYPOINT [ "/docker-entrypoint.sh" ] + + +FROM setup AS build +COPY . /home/project_options +WORKDIR /home/project_options +RUN git submodule update --init +CMD ["/bin/bash", "-c", "task rpi3:build.cross.bare-metal"] diff --git a/docker/Taskfile.yml b/docker/Taskfile.yml index 9a71221e..0eed9ce2 100644 --- a/docker/Taskfile.yml +++ b/docker/Taskfile.yml @@ -49,3 +49,15 @@ tasks: rpi4.custom: - docker-compose up --build build-rpi4-custom - docker-compose down + + rpi4.aarch64: + - docker-compose up --build build-rpi4-aarch64 + - docker-compose down + + rpi3.bare-metal: + - docker-compose up --build build-rpi3-bare-metal + - docker-compose down + + rpi4.vcpkg-custsom: + - docker-compose up --build build-rpi4-vcpkg-custom + - docker-compose down \ No newline at end of file diff --git a/src/CrossCompiler.cmake b/src/CrossCompiler.cmake index 8c7436c4..68fd4bd0 100644 --- a/src/CrossCompiler.cmake +++ b/src/CrossCompiler.cmake @@ -2,43 +2,102 @@ include_guard() # Enable cross-compiling macro(enable_cross_compiler) + set(options) + set(oneValueArgs + DEFAULT_TRIPLET + CC + CXX + TARGET_ARCHITECTURE + CROSS_ROOT + CROSS_TRIPLET + TOOLCHAIN_FILE) + set(multiValueArgs) + cmake_parse_arguments( + EnableCrossCompiler + "${options}" + "${oneValueArgs}" + "${multiValueArgs}" + ${ARGN}) + include("${ProjectOptions_SRC_DIR}/Utilities.cmake") detect_architecture(_arch) - if("${DEFAULT_TRIPLET}" STREQUAL "") - detect_compiler() + set(_default_triplet ${DEFAULT_TRIPLET}) + if(NOT "${EnableCrossCompiler_DEFAULT_TRIPLET}" STREQUAL "") + set(_default_triplet ${EnableCrossCompiler_DEFAULT_TRIPLET}) + else() + if("${DEFAULT_TRIPLET}" STREQUAL "") + detect_compiler() + endif() + set(_default_triplet ${CMAKE_SYSTEM_PROCESSOR}) endif() + set(_cc ${CMAKE_C_COMPILER}) set(_cxx ${CMAKE_CXX_COMPILER}) + if(NOT "${EnableCrossCompiler_CC}" STREQUAL "") + set(_cc ${EnableCrossCompiler_CC}) + endif() + if(NOT "${EnableCrossCompiler_CXX}" STREQUAL "") + set(_cxx ${EnableCrossCompiler_CXX}) + endif() + + set(_target_architecture ${TARGET_ARCHITECTURE}) + if(NOT "${EnableCrossCompiler_TARGET_ARCHITECTURE}" STREQUAL "") + set(_target_architecture ${EnableCrossCompiler_TARGET_ARCHITECTURE}) + endif() + + set(_cross_root ${CROSS_ROOT}) + if(NOT "${EnableCrossCompiler_CROSS_ROOT}" STREQUAL "") + set(_cross_root ${EnableCrossCompiler_CROSS_ROOT}) + endif() + + set(_cross_triplet ${CROSS_TRIPLET}) + if(NOT "${EnableCrossCompiler_CROSS_TRIPLET}" STREQUAL "") + set(_cross_triplet ${EnableCrossCompiler_CROSS_TRIPLET}) + endif() # detect compiler by triplet - if("${DEFAULT_TRIPLET}" STREQUAL "x64-mingw-dynamic" OR "${DEFAULT_TRIPLET}" STREQUAL "x64-mingw-static") + if("${_default_triplet}" STREQUAL "x64-mingw-dynamic" OR "${_default_triplet}" STREQUAL "x64-mingw-static") if("${_cc}" STREQUAL "") set(_cc "x86_64-w64-mingw32-gcc") endif() if("${_cxx}" STREQUAL "") set(_cxx "x86_64-w64-mingw32-g++") endif() - set(TARGET_ARCHITECTURE "x64") - elseif("${DEFAULT_TRIPLET}" STREQUAL "x86-mingw-dynamic" OR "${DEFAULT_TRIPLET}" STREQUAL "x86-mingw-static") + if("${_target_architecture}" STREQUAL "") + set(_target_architecture "x64") + endif() + elseif("${_default_triplet}" STREQUAL "x86-mingw-dynamic" OR "${_default_triplet}" STREQUAL "x86-mingw-static") if("${_cc}" STREQUAL "") set(_cc "i686-w64-mingw32-gcc") endif() if("${_cxx}" STREQUAL "") set(_cxx "i686-w64-mingw32-g++") endif() - set(TARGET_ARCHITECTURE "x86") - elseif("${DEFAULT_TRIPLET}" STREQUAL "wasm32-emscripten") - set(_cc "emcc") - set(_cxx "em++") - set(TARGET_ARCHITECTURE "wasm32-emscripten") - elseif("${DEFAULT_TRIPLET}" STREQUAL "arm-linux") - set(TARGET_ARCHITECTURE "arm-linux") - elseif("${DEFAULT_TRIPLET}" STREQUAL "arm64-linux") - set(TARGET_ARCHITECTURE "arm64-linux") + if("${_target_architecture}" STREQUAL "") + set(_target_architecture "x86") + endif() + elseif("${_default_triplet}" STREQUAL "wasm32-emscripten") + if("${_cc}" STREQUAL "") + set(_cc "emcc") + endif() + if("${_cxx}" STREQUAL "") + set(_cxx "em++") + endif() + if("${_target_architecture}" STREQUAL "") + set(_target_architecture "wasm32-emscripten") + endif() + elseif("${_default_triplet}" STREQUAL "arm64-linux") + if("${_target_architecture}" STREQUAL "") + set(_target_architecture "arm64-linux") + endif() + elseif("${_default_triplet}" STREQUAL "arm-linux") + if("${_target_architecture}" STREQUAL "") + set(_target_architecture "arm-linux") + endif() endif() - if("${TARGET_ARCHITECTURE}" STREQUAL "") + if("${_target_architecture}" STREQUAL "") if("${_cc}" STREQUAL "") set(_cc $ENV{CC}) endif() @@ -46,15 +105,15 @@ macro(enable_cross_compiler) set(_cxx $ENV{CXX}) endif() if(_cc MATCHES "x86_64(-w64)?-mingw32-[gc]..?" OR _cxx MATCHES "x86_64(-w64)?-mingw32-[gc]..?") - set(TARGET_ARCHITECTURE "x64") + set(_target_architecture "x64") elseif(_cc MATCHES "i686(-w64)?-mingw32-[gc]..?" OR _cxx MATCHES "i686(-w64)?-mingw32-[gc]..?") - set(TARGET_ARCHITECTURE "x86") + set(_target_architecture "x86") elseif(_cc MATCHES "emcc" OR _cxx MATCHES "em\\+\\+") - set(TARGET_ARCHITECTURE "wasm32-emscripten") + set(_target_architecture "wasm32-emscripten") else() # TODO: check for arm compiler message(WARNING "if you are using arm cross-compiler, please set DEFAULT_TRIPLET") - set(TARGET_ARCHITECTURE ${_arch}) + set(_target_architecture ${_arch}) endif() endif() @@ -80,56 +139,65 @@ macro(enable_cross_compiler) set(USE_CROSSCOMPILER_EMSCRIPTEN TRUE) elseif(_cc MATCHES "aarch64-linux-gnu-gcc" OR _cxx MATCHES "aarch64-linux-gnu-g\\+\\+") set(USE_CROSSCOMPILER_AARCH64 TRUE) - elseif("${DEFAULT_TRIPLET}" STREQUAL "arm-linux") - set(USE_CROSSCOMPILER_ARM TRUE) - elseif("${DEFAULT_TRIPLET}" STREQUAL "arm64-linux") - set(USE_CROSSCOMPILER_ARM64 TRUE) + elseif(_default_triplet MATCHES "arm64-linux") + set(USE_CROSSCOMPILER_ARM64_LINUX TRUE) + elseif(_default_triplet MATCHES "arm-linux") + set(USE_CROSSCOMPILER_ARM_LINUX TRUE) endif() set(LIBRARY_LINKAGE) if(BUILD_SHARED_LIBS) set(LIBRARY_LINKAGE "dynamic") - if("${TRIPLET}" STREQUAL "x64-mingw-static" OR "${TRIPLET}" STREQUAL "x86-mingw-static") + if("${_default_triplet}" STREQUAL "x64-mingw-static" OR "${_default_triplet}" STREQUAL "x86-mingw-static") message(WARNING "cross-compiler triplet is set to 'static' but BUILD_SHARED_LIBS is enabled") endif() else() - if("${TRIPLET}" STREQUAL "x64-mingw-dynamic" OR "${TRIPLET}" STREQUAL "x86-mingw-dynamic") + if("${_default_triplet}" STREQUAL "x64-mingw-dynamic" OR "${_default_triplet}" STREQUAL "x86-mingw-dynamic") set(LIBRARY_LINKAGE "dynamic") - elseif("${TRIPLET}" STREQUAL "x64-mingw-static" OR "${TRIPLET}" STREQUAL "x86-mingw-static") + elseif("${_default_triplet}" STREQUAL "x64-mingw-static" OR "${_default_triplet}" STREQUAL "x86-mingw-static") set(LIBRARY_LINKAGE "static") else() set(LIBRARY_LINKAGE "static") endif() endif() - - if(NOT DEFINED CROSS_ROOT) + + if("${_cross_root}" STREQUAL "") if(_cc MATCHES "x86_64(-w64)?-mingw32-[gc]..?" OR _cxx MATCHES "x86_64(-w64)?-mingw32-[gc]..?") - set(CROSS_ROOT "/usr/x86_64-w64-mingw32") + set(_cross_root "/usr/x86_64-w64-mingw32") elseif(_cc MATCHES "i686(-w64)?-mingw32-[gc]..?" OR _cxx MATCHES "i686(-w64)?-mingw32-[gc]..?") - set(CROSS_ROOT "/usr/i686-w64-mingw32") + set(_cross_root "/usr/i686-w64-mingw32") elseif(_cc MATCHES "(gcc-)?arm-linux-gnueabi-[gc]..?" OR _cxx MATCHES "(gcc-)?arm-linux-gnueabi-[gc]..?") - set(CROSS_ROOT "/usr/gcc-arm-linux-gnueabi") - if(NOT DEFINED CROSS_TRIPLET) - set(CROSS_TRIPLET "arm-linux-gnueabi") + set(_cross_root "/usr/gcc-arm-linux-gnueabi") + if("${_cross_triplet}" STREQUAL "") + set(_cross_triplet "arm-linux-gnueabi") endif() elseif(_cc MATCHES "(gcc-)?arm-linux-gnueabihf-[gc]..?" OR _cxx MATCHES "(gcc-)?arm-linux-gnueabihf-[gc]..?") - set(CROSS_ROOT "/usr/gcc-arm-linux-gnueabihf") - if(NOT DEFINED CROSS_TRIPLET) - set(CROSS_TRIPLET "arm-linux-gnueabihf") - endif() - elseif(_cc MATCHES "(gcc-)?arm-none-eabi-[gc]..?" OR _cxx MATCHES "(gcc-)?arm-none-eabi-[gc]..?") - set(CROSS_ROOT "/usr/gcc-arm-none-eabi") - if(NOT DEFINED CROSS_TRIPLET) - set(CROSS_TRIPLET "arm-none-eabi") + set(_cross_root "/usr/gcc-arm-linux-gnueabihf") + if("${_cross_triplet}" STREQUAL "") + set(_cross_triplet "arm-linux-gnueabihf") endif() elseif(_cc MATCHES "(gcc-)?aarch64-linux-gnu-[gc]..?" OR _cxx MATCHES "(gcc-)?aarch64-linux-gnu-[gc]..?") - set(CROSS_ROOT "/usr/gcc-aarch64-linux-gnu") - if(NOT DEFINED CROSS_TRIPLET) - set(CROSS_TRIPLET "gcc-aarch64-linux-gnu") + set(_cross_root "/usr/gcc-aarch64-linux-gnu") + if("${_cross_triplet}" STREQUAL "") + set(_cross_triplet "gcc-aarch64-linux-gnu") + endif() + elseif(_cc MATCHES "(gcc-)?arm-none-eabi-[gc]..?" OR _cxx MATCHES "(gcc-)?arm-none-eabi-[gc]..?") + set(_cross_root "/usr/gcc-arm-none-eabi") + if("${_cross_triplet}" STREQUAL "") + set(_cross_triplet "arm-none-eabi") endif() + set(USE_CROSSCOMPILER_ARM_NONE TRUE) endif() # TODO: check if path is right, check for header files or something endif() + if(NOT "${_cross_root}" STREQUAL "" AND "${_cross_triplet}" STREQUAL "") + message(WARNING "CROSS_ROOT (${_cross_root}) is set, but CROSS_TRIPLET is not") + endif() + + set(CROSS_ROOT ${_cross_root}) + set(CROSS_TRIPLET ${_cross_triplet}) + set(DEFAULT_TRIPLET ${_default_triplet}) + set(TARGET_ARCHITECTURE ${_target_architecture}) if(USE_CROSSCOMPILER_EMSCRIPTEN) if(NOT @@ -167,9 +235,15 @@ macro(enable_cross_compiler) endif() set(_toolchain_file) - get_toolchain_file(_toolchain_file) + if(${EnableCrossCompiler_TOOLCHAIN_FILE}) + set(_toolchain_file ${EnableCrossCompiler_TOOLCHAIN_FILE}) + else() + get_toolchain_file(_toolchain_file) + endif() if(NOT DEFINED CMAKE_TOOLCHAIN_FILE) set(CMAKE_TOOLCHAIN_FILE ${_toolchain_file}) + else() + set(CROSS_TOOLCHAIN_FILE ${_toolchain_file}) endif() set(CROSSCOMPILING TRUE) @@ -187,13 +261,21 @@ macro(enable_cross_compiler) #message(STATUS "EMSDK_NODE: $ENV{EMSDK_NODE}") #message(STATUS "EMSDK: $ENV{EMSDK}") message(STATUS "use emscripten cross-compiler emulator: ${CMAKE_CROSSCOMPILING_EMULATOR}") + else() + message(STATUS "use SYSROOT: ${CROSS_ROOT}") endif() message(STATUS "Target Architecture: ${TARGET_ARCHITECTURE}") - if(DEFAULT_TRIPLET) + if(NOT "${DEFAULT_TRIPLET}" STREQUAL "") message(STATUS "Default Triplet: ${DEFAULT_TRIPLET}") endif() message(STATUS "Host Triplet: ${HOST_TRIPLET}") - message(STATUS "Toolchain File: ${CMAKE_TOOLCHAIN_FILE}") + if(NOT "${CMAKE_TOOLCHAIN_FILE}" STREQUAL "") + message(STATUS "Toolchain File: ${CMAKE_TOOLCHAIN_FILE}") + else() + if(NOT DEFINED VCPKG_CHAINLOAD_TOOLCHAIN_FILE) + message(STATUS "Cross-compile Toolchain File (for vcpkg): ${CROSS_TOOLCHAIN_FILE}") + endif() + endif() endmacro() # Get the toolchain file @@ -221,19 +303,23 @@ function(get_toolchain_file value) else() message(ERROR "EMSCRIPTEN_ROOT is not set, please define EMSCRIPTEN_ROOT (emscripten repo)") endif() - elseif(USE_CROSSCOMPILER_AARCH64) + elseif(USE_CROSSCOMPILER_AARCH64_LINUX) set(${value} - ${ProjectOptions_SRC_DIR}/toolchains/aarch64.toolchain.cmake + ${ProjectOptions_SRC_DIR}/toolchains/aarch64-linux.toolchain.cmake PARENT_SCOPE) - elseif(USE_CROSSCOMPILER_ARM) + elseif(USE_CROSSCOMPILER_ARM_LINUX) set(${value} - ${ProjectOptions_SRC_DIR}/toolchains/arm.toolchain.cmake + ${ProjectOptions_SRC_DIR}/toolchains/arm-linux.toolchain.cmake + PARENT_SCOPE) + elseif(USE_CROSSCOMPILER_ARM64_LINUX) + set(${value} + ${ProjectOptions_SRC_DIR}/toolchains/arm64-linux.toolchain.cmake PARENT_SCOPE) - elseif(USE_CROSSCOMPILER_ARM64) + elseif(USE_CROSSCOMPILER_ARM_NONE) set(${value} - ${ProjectOptions_SRC_DIR}/toolchains/arm64.toolchain.cmake + ${ProjectOptions_SRC_DIR}/toolchains/arm.toolchain.cmake PARENT_SCOPE) - elseif("${DEFAULT_TRIPLET}" STREQUAL "arm-linux" OR "${DEFAULT_TRIPLET}" STREQUAL "arm64-linux") + elseif(DEFAULT_TRIPLET MATCHES "arm") message(STATUS "Don't forget to provide an cmake-toolchain file (for ${DEFAULT_TRIPLET})") endif() endfunction() diff --git a/src/Vcpkg.cmake b/src/Vcpkg.cmake index 47eeaeaf..648397bc 100644 --- a/src/Vcpkg.cmake +++ b/src/Vcpkg.cmake @@ -124,14 +124,22 @@ macro(run_vcpkg) set(VCPKG_LIBRARY_LINKAGE "${LIBRARY_LINKAGE}") endif() endif() - set(_toolchain_file) - get_toolchain_file(_toolchain_file) - if(_toolchain_file) - set(VCPKG_CHAINLOAD_TOOLCHAIN_FILE - ${_toolchain_file} - CACHE STRING "vcpkg chainload toolchain file") - message(STATUS "Setup cross-compiler for ${VCPKG_TARGET_TRIPLET}") - message(STATUS "Use cross-compiler toolchain: ${VCPKG_CHAINLOAD_TOOLCHAIN_FILE}") + + if(NOT DEFINED VCPKG_CHAINLOAD_TOOLCHAIN_FILE) + set(_toolchain_file) + if(NOT "${CROSS_TOOLCHAIN_FILE}" STREQUAL "") + set(_toolchain_file ${CROSS_TOOLCHAIN_FILE}) + else() + get_toolchain_file(_toolchain_file) + endif() + + if(${_toolchain_file}) + set(VCPKG_CHAINLOAD_TOOLCHAIN_FILE + ${_toolchain_file} + CACHE STRING "vcpkg chainload toolchain file") + message(STATUS "Setup cross-compiler for ${VCPKG_TARGET_TRIPLET}") + message(STATUS "Use cross-compiler toolchain for vcpkg: ${VCPKG_CHAINLOAD_TOOLCHAIN_FILE}") + endif() endif() endif() endmacro() diff --git a/src/toolchains/aarch64.toolchain.cmake b/src/toolchains/aarch64-linux.toolchain.cmake similarity index 91% rename from src/toolchains/aarch64.toolchain.cmake rename to src/toolchains/aarch64-linux.toolchain.cmake index 73d00afc..48c8f095 100644 --- a/src/toolchains/aarch64.toolchain.cmake +++ b/src/toolchains/aarch64-linux.toolchain.cmake @@ -3,7 +3,7 @@ cmake_minimum_required(VERSION 3.16) set(CMAKE_SYSTEM_NAME Linux) set(CMAKE_SYSTEM_PROCESSOR aarch64) -if(CROSS_ROOT) +if(NOT "${CROSS_ROOT}" STREQUAL "") set(CMAKE_SYSROOT ${CROSS_ROOT}) #set(CMAKE_FIND_ROOT_PATH ${CROSS_ROOT}) elseif("${CMAKE_SYSROOT}" STREQUAL "") @@ -11,12 +11,12 @@ elseif("${CMAKE_SYSROOT}" STREQUAL "") set(CMAKE_FIND_ROOT_PATH /usr/aarch64-linux-gnu) endif() -if(CROSS_C) +if(NOT "${CROSS_C}" STREQUAL "") set(CMAKE_C_COMPILER ${CROSS_C}) else() set(CMAKE_C_COMPILER aarch64-linux-gnu-gcc) endif() -if(CROSS_CXX) +if(NOT "${CROSS_CXX}" STREQUAL "") set(CMAKE_CXX_COMPILER ${CROSS_CXX}) else() set(CMAKE_CXX_COMPILER aarch64-linux-gnu-g++) diff --git a/src/toolchains/arm-linux.toolchain.cmake b/src/toolchains/arm-linux.toolchain.cmake new file mode 100644 index 00000000..08d19d62 --- /dev/null +++ b/src/toolchains/arm-linux.toolchain.cmake @@ -0,0 +1,38 @@ +cmake_minimum_required(VERSION 3.16) + +set(CMAKE_SYSTEM_NAME Linux) +set(CMAKE_SYSTEM_PROCESSOR arm) + +if(NOT "${CROSS_ROOT}" STREQUAL "") + set(CMAKE_SYSROOT ${CROSS_ROOT}) + #set(CMAKE_FIND_ROOT_PATH ${CROSS_ROOT}) +elseif("${CMAKE_SYSROOT}" STREQUAL "") + set(CMAKE_SYSROOT /usr/${CROSS_TRIPLET}) + #set(CMAKE_FIND_ROOT_PATH /usr/${CROSS_TRIPLET}) +endif() + +if(NOT "${CROSS_C}" STREQUAL "") + set(CMAKE_C_COMPILER ${CROSS_C}) +else() + set(CMAKE_C_COMPILER ${CROSS_TRIPLET}-gcc) +endif() +if(NOT "${CROSS_CXX}" STREQUAL "") + set(CMAKE_CXX_COMPILER ${CROSS_CXX}) +else() + set(CMAKE_CXX_COMPILER ${CROSS_TRIPLET}-g++) +endif() + +set(CMAKE_TRY_COMPILE_TARGET_TYPE STATIC_LIBRARY) + +set(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM NEVER) +set(CMAKE_FIND_ROOT_PATH_MODE_LIBRARY ONLY) +set(CMAKE_FIND_ROOT_PATH_MODE_INCLUDE ONLY) +set(CMAKE_FIND_ROOT_PATH_MODE_PACKAGE ONLY) + +# The target triple needs to match the prefix of the binutils exactly +# (e.g. CMake looks for arm-none-eabi-ar) +#set(CLANG_TARGET_TRIPLE ${CROSS_TRIPLET}) +#set(GCC_ARM_TOOLCHAIN_PREFIX ${CROSS_TRIPLET}) +#set(CMAKE_C_COMPILER_TARGET ${CROSS_TRIPLET}) +#set(CMAKE_CXX_COMPILER_TARGET ${CROSS_TRIPLET}) +#set(CMAKE_ASM_COMPILER_TARGET ${CROSS_TRIPLET}) diff --git a/src/toolchains/arm.toolchain.cmake b/src/toolchains/arm.toolchain.cmake index 99a426d7..e3f1962c 100644 --- a/src/toolchains/arm.toolchain.cmake +++ b/src/toolchains/arm.toolchain.cmake @@ -1,9 +1,9 @@ cmake_minimum_required(VERSION 3.16) -set(CMAKE_SYSTEM_NAME Linux) +set(CMAKE_SYSTEM_NAME Generic) set(CMAKE_SYSTEM_PROCESSOR arm) -if(CROSS_ROOT) +if(NOT "${CROSS_ROOT}" STREQUAL "") set(CMAKE_SYSROOT ${CROSS_ROOT}) #set(CMAKE_FIND_ROOT_PATH ${CROSS_ROOT}) elseif("${CMAKE_SYSROOT}" STREQUAL "") @@ -11,12 +11,12 @@ elseif("${CMAKE_SYSROOT}" STREQUAL "") #set(CMAKE_FIND_ROOT_PATH /usr/${CROSS_TRIPLET}) endif() -if(CROSS_C) +if(NOT "${CROSS_C}" STREQUAL "") set(CMAKE_C_COMPILER ${CROSS_C}) else() set(CMAKE_C_COMPILER ${CROSS_TRIPLET}-gcc) endif() -if(CROSS_CXX) +if(NOT "${CROSS_CXX}" STREQUAL "") set(CMAKE_CXX_COMPILER ${CROSS_CXX}) else() set(CMAKE_CXX_COMPILER ${CROSS_TRIPLET}-g++) diff --git a/src/toolchains/arm64.toolchain.cmake b/src/toolchains/arm64-linux.toolchain.cmake similarity index 91% rename from src/toolchains/arm64.toolchain.cmake rename to src/toolchains/arm64-linux.toolchain.cmake index 8caeb262..c23aef9c 100644 --- a/src/toolchains/arm64.toolchain.cmake +++ b/src/toolchains/arm64-linux.toolchain.cmake @@ -3,7 +3,7 @@ cmake_minimum_required(VERSION 3.16) set(CMAKE_SYSTEM_NAME Linux) set(CMAKE_SYSTEM_PROCESSOR arm64) -if(CROSS_ROOT) +if(NOT "${CROSS_ROOT}" STREQUAL "") set(CMAKE_SYSROOT ${CROSS_ROOT}) #set(CMAKE_FIND_ROOT_PATH ${CROSS_ROOT}) elseif("${CMAKE_SYSROOT}" STREQUAL "") @@ -11,12 +11,12 @@ elseif("${CMAKE_SYSROOT}" STREQUAL "") #set(CMAKE_FIND_ROOT_PATH /usr/${CROSS_TRIPLET}) endif() -if(CROSS_C) +if(NOT "${CROSS_C}" STREQUAL "") set(CMAKE_C_COMPILER ${CROSS_C}) else() set(CMAKE_C_COMPILER ${CROSS_TRIPLET}-gcc) endif() -if(CROSS_CXX) +if(NOT "${CROSS_CXX}" STREQUAL "") set(CMAKE_CXX_COMPILER ${CROSS_CXX}) else() set(CMAKE_CXX_COMPILER ${CROSS_TRIPLET}-g++) diff --git a/src/toolchains/i686-w64-mingw32.toolchain.cmake b/src/toolchains/i686-w64-mingw32.toolchain.cmake index 0ba729fd..9c807703 100644 --- a/src/toolchains/i686-w64-mingw32.toolchain.cmake +++ b/src/toolchains/i686-w64-mingw32.toolchain.cmake @@ -3,7 +3,7 @@ cmake_minimum_required(VERSION 3.16) set(CMAKE_SYSTEM_NAME Windows) set(CMAKE_SYSTEM_PROCESSOR "i686") -if(CROSS_ROOT) +if(NOT "${CROSS_ROOT}" STREQUAL "") set(CMAKE_SYSROOT ${CROSS_ROOT}) #set(CMAKE_FIND_ROOT_PATH ${CROSS_ROOT}) elseif("${CMAKE_SYSROOT}" STREQUAL "") @@ -11,9 +11,21 @@ elseif("${CMAKE_SYSROOT}" STREQUAL "") #set(CMAKE_FIND_ROOT_PATH /usr/i686-w64-mingw32) endif() -set(CMAKE_C_COMPILER i686-w64-mingw32-gcc) -set(CMAKE_CXX_COMPILER i686-w64-mingw32-g++) -set(CMAKE_RC_COMPILER i686-w64-mingw32-windres) +if(NOT "${CROSS_C}" STREQUAL "") + set(CMAKE_C_COMPILER ${CROSS_C}) +else() + set(CMAKE_C_COMPILER i686-w64-mingw32-gcc) +endif() +if(NOT "${CROSS_CXX}" STREQUAL "") + set(CMAKE_CXX_COMPILER ${CROSS_CXX}) +else() + set(CMAKE_CXX_COMPILER i686-w64-mingw32-g++) +endif() +if(NOT "${CROSS_RC}" STREQUAL "") + set(CMAKE_RC_COMPILER ${CROSS_RC}) +else() + set(CMAKE_RC_COMPILER i686-w64-mingw32-windres) +endif() # search for programs in the build host directories set(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM NEVER) diff --git a/src/toolchains/x86_64-w64-mingw32.toolchain.cmake b/src/toolchains/x86_64-w64-mingw32.toolchain.cmake index ad234935..243f0c96 100644 --- a/src/toolchains/x86_64-w64-mingw32.toolchain.cmake +++ b/src/toolchains/x86_64-w64-mingw32.toolchain.cmake @@ -3,7 +3,7 @@ cmake_minimum_required(VERSION 3.16) set(CMAKE_SYSTEM_NAME Windows) set(CMAKE_SYSTEM_PROCESSOR "x64") -if(CROSS_ROOT) +if(NOT "${CROSS_ROOT}" STREQUAL "") set(CMAKE_SYSROOT ${CROSS_ROOT}) #set(CMAKE_FIND_ROOT_PATH ${CROSS_ROOT}) elseif("${CMAKE_SYSROOT}" STREQUAL "") @@ -11,9 +11,21 @@ elseif("${CMAKE_SYSROOT}" STREQUAL "") #set(CMAKE_FIND_ROOT_PATH /usr/x86_64-w64-mingw32) endif() -set(CMAKE_C_COMPILER x86_64-w64-mingw32-gcc) -set(CMAKE_CXX_COMPILER x86_64-w64-mingw32-g++) -set(CMAKE_RC_COMPILER x86_64-w64-mingw32-windres) +if(NOT "${CROSS_C}" STREQUAL "") + set(CMAKE_C_COMPILER ${CROSS_C}) +else() + set(CMAKE_C_COMPILER x86_64-w64-mingw32-gcc) +endif() +if(NOT "${CROSS_CXX}" STREQUAL "") + set(CMAKE_CXX_COMPILER ${CROSS_CXX}) +else() + set(CMAKE_CXX_COMPILER x86_64-w64-mingw32-g++) +endif() +if(NOT "${CROSS_RC}" STREQUAL "") + set(CMAKE_RC_COMPILER ${CROSS_RC}) +else() + set(CMAKE_RC_COMPILER x86_64-w64-mingw32-windres) +endif() # search for programs in the build host directories set(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM NEVER) diff --git a/tests/rpi3/CMakeLists.txt b/tests/rpi3/CMakeLists.txt index 205baa02..fccdf501 100644 --- a/tests/rpi3/CMakeLists.txt +++ b/tests/rpi3/CMakeLists.txt @@ -13,9 +13,26 @@ set(CMAKE_C_STANDARD 99) include(../../src/Index.cmake) # opt-in cross-compiling -option(ENABLE_CROSS_COMPILING "Detect cross compiler and setup toolchain" OFF) -if(ENABLE_CROSS_COMPILING) - enable_cross_compiler() +if(ENABLE_BARE_METAL_CROSS_COMPILING) + # my custom arm settings + enable_cross_compiler( + CC "arm-none-eabi-gcc" + CXX "arm-none-eabi-g++" + TARGET_ARCHITECTURE "arm" + CROSS_ROOT "/usr/arm-none-eabi-gcc" + CROSS_TRIPLET "arm-none-eabi-gcc" + ) + + # some more custom compiler settings + # -Wl,--gc-sections Perform the dead code elimination. + # --specs=nano.specs Link with newlib-nano. + # --specs=nosys.specs No syscalls, provide empty implementations for the POSIX system calls. + set(CMAKE_EXE_LINKER_FLAGS "-Wl,--gc-sections --specs=nano.specs --specs=nosys.specs -mthumb" CACHE INTERNAL "Linker options") +else() + option(ENABLE_CROSS_COMPILING "Detect cross compiler and setup toolchain" OFF) + if(ENABLE_CROSS_COMPILING) + enable_cross_compiler() + endif() endif() # Set the project name to your project name, my project isn't very descriptive @@ -108,7 +125,6 @@ project_options( #target_sources(example PRIVATE main.cpp) #target_include_directories(example PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}) #target_link_libraries(example PRIVATE project_options project_warnings) -#target_link_options(example PRIVATE -Wl,--start-group -lgcc -lc -lstdc++ -lm -lrdimon -Wl,--end-group) #FIXME: linking with c++ (libs) ... /libstdc++.a(cxx11-ios_failure.o): in function `(anonymous namespace)::__io_category_instance()': add_executable(example_c) @@ -116,7 +132,9 @@ target_sources(example_c PRIVATE main.c) target_include_directories(example_c PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}) target_link_libraries(example_c PRIVATE project_options project_warnings) -if(ENABLE_CROSS_COMPILING) +if(ENABLE_BARE_METAL_CROSS_COMPILING OR ENABLE_CROSS_COMPILING) + #target_link_options(example PRIVATE -Wl,--start-group -lgcc -lc -lstdc++ -lm -lrdimon -Wl,--end-group) + # fix: ...arm-none-eabi/lib/libg.a(lib_a-exit.o): in function `exit': target_link_options( example_c @@ -125,7 +143,6 @@ if(ENABLE_CROSS_COMPILING) -lgcc -lc -lm - -lrdimon -Wl,--end-group) # custom raspberry pi 3 options diff --git a/tests/rpi3/Taskfile.yml b/tests/rpi3/Taskfile.yml index 8bdc0fd8..892a427f 100644 --- a/tests/rpi3/Taskfile.yml +++ b/tests/rpi3/Taskfile.yml @@ -14,10 +14,16 @@ tasks: cmds: - task: build.debug vars: - CMAKE_ARGS: -DENABLE_CHECKING:BOOL=ON -DENABLE_CROSS_COMPILING:BOOL=ON -DCMAKE_C_COMPILER={{.CROSS_CC | default "arm-none-eabi-gcc"}} -DCMAKE_CXX_COMPILER={{.CROSS_CXX | default "arm-none-eabi-g++"}} -DDEFAULT_TRIPLET=arm-linux + CMAKE_ARGS: -DENABLE_CHECKING:BOOL=ON -DENABLE_CROSS_COMPILING:BOOL=ON -DCMAKE_C_COMPILER={{.CROSS_CC | default "arm-linux-gnueabihf-gcc"}} -DCMAKE_CXX_COMPILER={{.CROSS_CXX | default "arm-linux-gnueabihf-g++"}} -DDEFAULT_TRIPLET=arm-linux -DCROSS_ROOT=/usr/gcc-arm-linux-gnueabihf build.cross: cmds: - task: build vars: - CMAKE_ARGS: -DENABLE_CROSS_COMPILING:BOOL=ON -DCMAKE_C_COMPILER={{.CROSS_CC | default "arm-none-eabi-gcc"}} -DCMAKE_CXX_COMPILER={{.CROSS_CXX | default "arm-none-eabi-g++"}} -DDEFAULT_TRIPLET=arm-linux + CMAKE_ARGS: -DENABLE_CROSS_COMPILING:BOOL=ON -DENABLE_CROSS_COMPILING:BOOL=ON -DCMAKE_C_COMPILER={{.CROSS_CC | default "arm-linux-gnueabihf-gcc"}} -DCMAKE_CXX_COMPILER={{.CROSS_CXX | default "arm-linux-gnueabi-g++"}} -DDEFAULT_TRIPLET=arm-linux -DCROSS_ROOT=/usr/gcc-arm-linux-gnueabihf + + build.cross.bare-metal: + cmds: + - task: build + vars: + CMAKE_ARGS: -DENABLE_BARE_METAL_CROSS_COMPILING:BOOL=ON diff --git a/tests/rpi3/main.c b/tests/rpi3/main.c index c18a0350..1b12d56d 100644 --- a/tests/rpi3/main.c +++ b/tests/rpi3/main.c @@ -1,6 +1,15 @@ -#include +// dummy functions +void uart_init() {} +void uart_send(unsigned int c) {} +char uart_getc() { return 'a'; } +void uart_puts(char *s) {} -int main() { - printf("Hello World\n"); - return 0; +void main() { + uart_init(); + + uart_puts("Hello World!\n"); + + while(1) { + uart_send(uart_getc()); + } } \ No newline at end of file diff --git a/tests/rpi4-vcpkg/Taskfile.yml b/tests/rpi4-vcpkg/Taskfile.yml index 175de702..992a7c4d 100644 --- a/tests/rpi4-vcpkg/Taskfile.yml +++ b/tests/rpi4-vcpkg/Taskfile.yml @@ -12,5 +12,11 @@ tasks: vars: CMAKE_ARGS: -DENABLE_CROSS_COMPILING:BOOL=ON -DCMAKE_C_COMPILER={{.CROSS_CC | default "aarch64-linux-gnu-gcc"}} -DCMAKE_CXX_COMPILER={{.CROSS_CXX | default "aarch64-linux-gnu-g++"}} -DDEFAULT_TRIPLET=arm64-linux + build.cross.custom-toolchain: + cmds: + - task: build + vars: + CMAKE_ARGS: -DENABLE_CROSS_COMPILING:BOOL=ON -DCMAKE_C_COMPILER={{.CROSS_CC | default "gcc-aarch64-linux-gnu-gcc"}} -DCMAKE_CXX_COMPILER={{.CROSS_CXX | default "gcc-aarch64-linux-gnu-g++"}} -DDEFAULT_TRIPLET=arm64-linux -DVCPKG_CHAINLOAD_TOOLCHAIN_FILE=./cmake/my-toolchain.cmake + lint: - ~/vcpkg/vcpkg format-manifest ./vcpkg.json \ No newline at end of file diff --git a/tests/rpi4/CMakeLists.txt b/tests/rpi4/CMakeLists.txt index 3292bb7d..b53f000e 100644 --- a/tests/rpi4/CMakeLists.txt +++ b/tests/rpi4/CMakeLists.txt @@ -13,9 +13,22 @@ set(CMAKE_C_STANDARD 99) include(../../src/Index.cmake) # opt-in cross-compiling -option(ENABLE_CROSS_COMPILING "Detect cross compiler and setup toolchain" OFF) -if(ENABLE_CROSS_COMPILING) - enable_cross_compiler() +if(ENABLE_AARCH64_CROSS_COMPILING) + # my custom aarch64 settings + enable_cross_compiler( + DEFAULT_TRIPLET "arm64-linux" + CC "aarch64-linux-gnu-gcc" + CXX "aarch64-linux-gnu-g++" + TARGET_ARCHITECTURE "arm64-linux" + CROSS_ROOT "/usr/gcc-aarch64-linux-gnu" + CROSS_TRIPLET "aarch64-linux-gnu" + #TOOLCHAIN_FILE "${CMAKE_CURRENT_SOURCE_DIR}/cmake/my-toolchain.cmake" + ) +else() + option(ENABLE_CROSS_COMPILING "Detect cross compiler and setup toolchain" OFF) + if(ENABLE_CROSS_COMPILING) + enable_cross_compiler() + endif() endif() # Set the project name to your project name, my project isn't very descriptive @@ -104,7 +117,7 @@ target_sources(example PRIVATE main.cpp) target_include_directories(example PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}) target_link_libraries(example PRIVATE project_options project_warnings) -if(ENABLE_CROSS_COMPILING) +if(ENABLE_AARCH64_CROSS_COMPILING OR ENABLE_CROSS_COMPILING) # custom raspberry pi 4 options #target_compile_definitions(project_options INTERFACE -D__ARM_NEON) target_compile_options(project_options INTERFACE -mcpu=cortex-a72) diff --git a/tests/rpi4/Taskfile.yml b/tests/rpi4/Taskfile.yml index 6c6704b5..6bad4141 100644 --- a/tests/rpi4/Taskfile.yml +++ b/tests/rpi4/Taskfile.yml @@ -27,3 +27,9 @@ tasks: - task: build vars: CMAKE_ARGS: -DENABLE_CROSS_COMPILING:BOOL=ON -DCMAKE_C_COMPILER={{.CROSS_CC | default "gcc-aarch64-linux-gnu-gcc"}} -DCMAKE_CXX_COMPILER={{.CROSS_CXX | default "gcc-aarch64-linux-gnu-g++"}} -DDEFAULT_TRIPLET=arm64-linux -DCMAKE_TOOLCHAIN_FILE=./cmake/my-toolchain.cmake + + build.cross.aarch64: + cmds: + - task: build + vars: + CMAKE_ARGS: -DENABLE_AARCH64_CROSS_COMPILING:BOOL=ON \ No newline at end of file