aboutsummaryrefslogtreecommitdiff
path: root/mesonbuild
diff options
context:
space:
mode:
authorKonstantin <ria.freelander@gmail.com>2022-07-15 15:46:31 +0300
committerDaniel Mensinger <daniel@mensinger-ka.de>2023-01-28 20:16:06 +0100
commit89146e84c9eab649d3847af101d61047cac45765 (patch)
treebc93358ca46ce52bcadf7af4204d8ee78f6bc236 /mesonbuild
parent01275fb09ec083aa8328ba100ed665bb9034c914 (diff)
downloadmeson-89146e84c9eab649d3847af101d61047cac45765.zip
meson-89146e84c9eab649d3847af101d61047cac45765.tar.gz
meson-89146e84c9eab649d3847af101d61047cac45765.tar.bz2
cmake: allow dynamic linking with LLVM
llvm-config is unsuitable for standard cross-compile, because we need to build llvm especially for it, which is not done is almost any distros, so, for example, standard bootstrap chroot will be unsuitable. This patch is trying to acheive feature parity between config-tool searching of LLVM and CMake-based one, which is arch-agnostic. Signed-off-by: Konstantin <ria.freelander@gmail.com>
Diffstat (limited to 'mesonbuild')
-rw-r--r--mesonbuild/dependencies/data/CMakeListsLLVM.txt150
-rw-r--r--mesonbuild/dependencies/dev.py43
2 files changed, 165 insertions, 28 deletions
diff --git a/mesonbuild/dependencies/data/CMakeListsLLVM.txt b/mesonbuild/dependencies/data/CMakeListsLLVM.txt
index da23189..f12dddc 100644
--- a/mesonbuild/dependencies/data/CMakeListsLLVM.txt
+++ b/mesonbuild/dependencies/data/CMakeListsLLVM.txt
@@ -1,8 +1,24 @@
+# fail noisily if attempt to use this file without setting:
+# cmake_minimum_required(VERSION ${CMAKE_VERSION})
+# project(... LANGUAGES ...)
+
+cmake_policy(SET CMP0000 NEW)
set(PACKAGE_FOUND FALSE)
+list(REMOVE_DUPLICATES LLVM_MESON_VERSIONS)
+
while(TRUE)
- find_package(LLVM REQUIRED CONFIG QUIET)
+ #Activate CMake version selection
+ foreach(i IN LISTS LLVM_MESON_VERSIONS)
+ find_package(LLVM ${i}
+ CONFIG
+ NAMES ${LLVM_MESON_PACKAGE_NAMES}
+ QUIET)
+ if(LLVM_FOUND)
+ break()
+ endif()
+ endforeach()
# ARCHS has to be set via the CMD interface
if(LLVM_FOUND OR "${ARCHS}" STREQUAL "")
@@ -13,9 +29,70 @@ while(TRUE)
list(REMOVE_AT ARCHS 0)
endwhile()
+function(meson_llvm_cmake_dynamic_available mod out)
+ # Check if we can only compare LLVM_DYLIB_COMPONENTS, because
+ # we do not need complex component translation logic, if all
+ # is covered by one variable
+ if(mod IN_LIST LLVM_DYLIB_COMPONENTS)
+ set(${out} TRUE PARENT_SCOPE)
+ return()
+ elseif((NOT (mod IN_LIST LLVM_DYLIB_COMPONENTS))
+ AND (NOT("${LLVM_DYLIB_COMPONENTS}" STREQUAL "all")))
+ set(${out} FALSE PARENT_SCOPE)
+ return()
+ endif()
+
+ # Complex heurisic to filter all pseudo-components and skip invalid names
+ # LLVM_DYLIB_COMPONENTS will be 'all', because in other case we returned
+ # in previous check. 'all' is also handled there.
+ set(llvm_pseudo_components "native" "backend" "engine" "all-targets")
+ is_llvm_target_specifier(${mod} mod_spec INCLUDED_TARGETS)
+ string(TOUPPER "${LLVM_AVAILABLE_LIBS}" capitalized_libs)
+ string(TOUPPER "${LLVM_TARGETS_TO_BUILD}" capitalized_tgts)
+ if(mod_spec)
+ set(${out} TRUE PARENT_SCOPE)
+ elseif(mod IN_LIST capitalized_tgts)
+ set(${out} TRUE PARENT_SCOPE)
+ elseif(mod IN_LIST llvm_pseudo_components)
+ set(${out} TRUE PARENT_SCOPE)
+ elseif(LLVM${mod} IN_LIST capitalized_libs)
+ set(${out} TRUE PARENT_SCOPE)
+ else()
+ set(${out} FALSE PARENT_SCOPE)
+ endif()
+endfunction()
+
+function(is_static target ret)
+ if(TARGET ${target})
+ get_target_property(target_type ${target} TYPE)
+ if(target_type STREQUAL "STATIC_LIBRARY")
+ set(${ret} TRUE PARENT_SCOPE)
+ return()
+ endif()
+ endif()
+ set(${ret} FALSE PARENT_SCOPE)
+endfunction()
+
+# Concatenate LLVM_MESON_REQUIRED_MODULES and LLVM_MESON_OPTIONAL_MODULES
+set(LLVM_MESON_MODULES ${LLVM_MESON_REQUIRED_MODULES} ${LLVM_MESON_OPTIONAL_MODULES})
+
+
+# Check if LLVM exists in dynamic world
+# Initialization before modules checking
if(LLVM_FOUND)
- set(PACKAGE_FOUND TRUE)
+ if(LLVM_MESON_DYLIB AND TARGET LLVM)
+ set(PACKAGE_FOUND TRUE)
+ elseif(NOT LLVM_MESON_DYLIB)
+ # Use LLVMSupport to check if static targets exist
+ set(static_tg FALSE)
+ is_static(LLVMSupport static_tg)
+ if(static_tg)
+ set(PACKAGE_FOUND TRUE)
+ endif(static_tg)
+ endif()
+endif()
+if(PACKAGE_FOUND)
foreach(mod IN LISTS LLVM_MESON_MODULES)
# Reset variables
set(out_mods)
@@ -25,23 +102,53 @@ if(LLVM_FOUND)
string(TOLOWER "${mod}" mod_L)
string(TOUPPER "${mod}" mod_U)
- # Get the mapped components
- llvm_map_components_to_libnames(out_mods ${mod} ${mod_L} ${mod_U})
- list(SORT out_mods)
- list(REMOVE_DUPLICATES out_mods)
-
- # Make sure that the modules exist
- foreach(i IN LISTS out_mods)
- if(TARGET ${i})
- list(APPEND real_mods ${i})
- endif()
- endforeach()
-
- # Set the output variables
- set(MESON_LLVM_TARGETS_${mod} ${real_mods})
- foreach(i IN LISTS real_mods)
- set(MESON_TARGET_TO_LLVM_${i} ${mod})
- endforeach()
+ # Special case - "all-targets" pseudo target
+ # Just append all targets, if pseudo-target exists
+ if("${mod}" STREQUAL "all-targets")
+ set(mod_L ${LLVM_TARGETS_TO_BUILD})
+ string(TOUPPER "${LLVM_TARGETS_TO_BUILD}" mod_U)
+ endif()
+
+ # Check if required module is linked is inside libLLVM.so.
+ # If not, skip this module
+ if(LLVM_MESON_DYLIB
+ AND DEFINED LLVM_DYLIB_COMPONENTS)
+ meson_llvm_cmake_dynamic_available(${mod} MOD_F)
+ meson_llvm_cmake_dynamic_available(${mod_L} MOD_L_F)
+ meson_llvm_cmake_dynamic_available(${mod_U} MOD_U_F)
+ if(MOD_F OR MOD_L_F OR MOD_U_F)
+ set(MESON_LLVM_TARGETS_${mod} LLVM)
+ endif()
+ elseif(LLVM_MESON_DYLIB AND (mod IN_LIST LLVM_MESON_REQUIRED_MODULES))
+ # Dynamic was requested, but no required variables set, we cannot continue
+ set(PACKAGE_FOUND FALSE)
+ break()
+ elseif(LLVM_MESON_DYLIB)
+ # Dynamic was requested, and we request optional modules only. Continue
+ continue()
+ else()
+ # CMake only do this for static components, and we
+ # replicate its behaviour
+ # Get the mapped components
+ llvm_map_components_to_libnames(out_mods ${mod} ${mod_L} ${mod_U})
+ list(SORT out_mods)
+ list(REMOVE_DUPLICATES out_mods)
+
+ # Make sure that the modules exist
+ foreach(i IN LISTS out_mods)
+ set(static_tg FALSE)
+ is_static(${i} static_tg)
+ if(static_tg)
+ list(APPEND real_mods ${i})
+ endif()
+ endforeach()
+
+ # Set the output variables
+ set(MESON_LLVM_TARGETS_${mod} ${real_mods})
+ foreach(i IN LISTS real_mods)
+ set(MESON_TARGET_TO_LLVM_${i} ${mod})
+ endforeach()
+ endif()
endforeach()
# Check the following variables:
@@ -62,7 +169,10 @@ if(LLVM_FOUND)
# LLVM_LIBRARIES
# LLVM_LIBS
set(libs)
- if(DEFINED LLVM_LIBRARIES)
+ #Hardcode LLVM, because we links with libLLVM.so when dynamic
+ if(LLVM_MESON_DYLIB)
+ get_target_property(libs LLVM IMPORTED_LOCATION)
+ elseif(DEFINED LLVM_LIBRARIES)
set(libs LLVM_LIBRARIES)
elseif(DEFINED LLVM_LIBS)
set(libs LLVM_LIBS)
diff --git a/mesonbuild/dependencies/dev.py b/mesonbuild/dependencies/dev.py
index de711e5..a9b768b 100644
--- a/mesonbuild/dependencies/dev.py
+++ b/mesonbuild/dependencies/dev.py
@@ -24,12 +24,13 @@ import pathlib
import shutil
import subprocess
import typing as T
+import functools
from mesonbuild.interpreterbase.decorators import FeatureDeprecated
from .. import mesonlib, mlog
from ..environment import get_llvm_tool_names
-from ..mesonlib import version_compare, stringlistify, extract_as_list
+from ..mesonlib import version_compare, version_compare_many, search_version, stringlistify, extract_as_list
from .base import DependencyException, DependencyMethods, detect_compiler, strip_system_libdirs, SystemDependency, ExternalDependency, DependencyTypeName
from .cmake import CMakeDependency
from .configtool import ConfigToolDependency
@@ -418,14 +419,15 @@ class LLVMDependencyCMake(CMakeDependency):
super().__init__(name, env, kwargs, language='cpp', force_use_global_compilers=True)
- # Cmake will always create a statically linked binary, so don't use
- # cmake if dynamic is required
- if not self.static:
- self.is_found = False
- mlog.warning('Ignoring LLVM CMake dependency because dynamic was requested', fatal=False)
+ if self.traceparser is None:
return
- if self.traceparser is None:
+ if not self.is_found:
+ return
+
+ #CMake will return not found due to not defined LLVM_DYLIB_COMPONENTS
+ if not self.static and version_compare(self.version, '< 7.0') and self.llvm_modules:
+ mlog.warning('Before version 7.0 cmake does not export modules for dynamic linking, cannot check required modules')
return
# Extract extra include directories and definitions
@@ -444,8 +446,33 @@ class LLVMDependencyCMake(CMakeDependency):
# Use a custom CMakeLists.txt for LLVM
return 'CMakeListsLLVM.txt'
+ # Check version in CMake to return exact version as config tool (latest allowed)
+ # It is safe to add .0 to latest argument, it will discarded if we use search_version
+ def llvm_cmake_versions(self) -> T.List[str]:
+
+ def ver_from_suf(req: str) -> str:
+ return search_version(req.strip('-')+'.0')
+
+ def version_sorter(a: str, b: str) -> int:
+ if version_compare(a, "="+b):
+ return 0
+ if version_compare(a, "<"+b):
+ return 1
+ return -1
+
+ llvm_requested_versions = [ver_from_suf(x) for x in get_llvm_tool_names('') if version_compare(ver_from_suf(x), '>=0')]
+ if self.version_reqs:
+ llvm_requested_versions = [ver_from_suf(x) for x in get_llvm_tool_names('') if version_compare_many(ver_from_suf(x), self.version_reqs)]
+ # CMake sorting before 3.18 is incorrect, sort it here instead
+ return sorted(llvm_requested_versions, key=functools.cmp_to_key(version_sorter))
+
+ # Split required and optional modules to distinguish it in CMake
def _extra_cmake_opts(self) -> T.List[str]:
- return ['-DLLVM_MESON_MODULES={}'.format(';'.join(self.llvm_modules + self.llvm_opt_modules))]
+ return ['-DLLVM_MESON_REQUIRED_MODULES={}'.format(';'.join(self.llvm_modules)),
+ '-DLLVM_MESON_OPTIONAL_MODULES={}'.format(';'.join(self.llvm_opt_modules)),
+ '-DLLVM_MESON_PACKAGE_NAMES={}'.format(';'.join(get_llvm_tool_names(self.name))),
+ '-DLLVM_MESON_VERSIONS={}'.format(';'.join(self.llvm_cmake_versions())),
+ '-DLLVM_MESON_DYLIB={}'.format('OFF' if self.static else 'ON')]
def _map_module_list(self, modules: T.List[T.Tuple[str, bool]], components: T.List[T.Tuple[str, bool]]) -> T.List[T.Tuple[str, bool]]:
res = []