blob: af6a8c4395087d2bea42633818bf65ba924fb4e1 [file] [log] [blame]
lh9ed821d2023-04-07 01:36:19 -07001# Path to the CMake file to process.
2OECMAKE_SOURCEPATH ??= "${S}"
3
4DEPENDS_prepend = "cmake-native "
5B = "${WORKDIR}/build"
6
7# What CMake generator to use.
8# The supported options are "Unix Makefiles" or "Ninja".
9OECMAKE_GENERATOR ?= "Ninja"
10
11python() {
12 generator = d.getVar("OECMAKE_GENERATOR")
13 if "Unix Makefiles" in generator:
14 args = "-G '" + generator + "' -DCMAKE_MAKE_PROGRAM=" + d.getVar("MAKE")
15 d.setVar("OECMAKE_GENERATOR_ARGS", args)
16 d.setVarFlag("do_compile", "progress", "percent")
17 elif "Ninja" in generator:
18 args = "-G '" + generator + "' -DCMAKE_MAKE_PROGRAM=ninja"
19 d.appendVar("DEPENDS", " ninja-native")
20 d.setVar("OECMAKE_GENERATOR_ARGS", args)
21 d.setVarFlag("do_compile", "progress", r"outof:^\[(\d+)/(\d+)\]\s+")
22 else:
23 bb.fatal("Unknown CMake Generator %s" % generator)
24
25 # C/C++ Compiler (without cpu arch/tune arguments)
26 if not d.getVar('OECMAKE_C_COMPILER'):
27 cc_list = d.getVar('CC').split()
28 if cc_list[0] == 'ccache':
29 d.setVar('OECMAKE_C_COMPILER_LAUNCHER', cc_list[0])
30 d.setVar('OECMAKE_C_COMPILER', cc_list[1])
31 else:
32 d.setVar('OECMAKE_C_COMPILER', cc_list[0])
33
34 if not d.getVar('OECMAKE_CXX_COMPILER'):
35 cxx_list = d.getVar('CXX').split()
36 if cxx_list[0] == 'ccache':
37 d.setVar('OECMAKE_CXX_COMPILER_LAUNCHER', cxx_list[0])
38 d.setVar('OECMAKE_CXX_COMPILER', cxx_list[1])
39 else:
40 d.setVar('OECMAKE_CXX_COMPILER', cxx_list[0])
41}
42OECMAKE_AR ?= "${AR}"
43
44# Compiler flags
45OECMAKE_C_FLAGS ?= "${HOST_CC_ARCH} ${TOOLCHAIN_OPTIONS} ${CFLAGS}"
46OECMAKE_CXX_FLAGS ?= "${HOST_CC_ARCH} ${TOOLCHAIN_OPTIONS} ${CXXFLAGS}"
47OECMAKE_C_FLAGS_RELEASE ?= "-DNDEBUG"
48OECMAKE_CXX_FLAGS_RELEASE ?= "-DNDEBUG"
49OECMAKE_C_LINK_FLAGS ?= "${HOST_CC_ARCH} ${TOOLCHAIN_OPTIONS} ${CPPFLAGS} ${LDFLAGS}"
50OECMAKE_CXX_LINK_FLAGS ?= "${HOST_CC_ARCH} ${TOOLCHAIN_OPTIONS} ${CXXFLAGS} ${LDFLAGS}"
51CXXFLAGS += "${HOST_CC_ARCH} ${TOOLCHAIN_OPTIONS}"
52CFLAGS += "${HOST_CC_ARCH} ${TOOLCHAIN_OPTIONS}"
53
54OECMAKE_C_COMPILER_LAUNCHER ?= ""
55OECMAKE_CXX_COMPILER_LAUNCHER ?= ""
56
57OECMAKE_RPATH ?= ""
58OECMAKE_PERLNATIVE_DIR ??= ""
59OECMAKE_EXTRA_ROOT_PATH ?= ""
60
61OECMAKE_FIND_ROOT_PATH_MODE_PROGRAM = "ONLY"
62OECMAKE_FIND_ROOT_PATH_MODE_PROGRAM_class-native = "BOTH"
63
64EXTRA_OECMAKE_append = " ${PACKAGECONFIG_CONFARGS}"
65
66export CMAKE_BUILD_PARALLEL_LEVEL
67CMAKE_BUILD_PARALLEL_LEVEL_task-compile = "${@oe.utils.parallel_make(d, False)}"
68CMAKE_BUILD_PARALLEL_LEVEL_task-install = "${@oe.utils.parallel_make(d, True)}"
69
70OECMAKE_TARGET_COMPILE ?= "all"
71OECMAKE_TARGET_INSTALL ?= "install"
72
73def map_host_os_to_system_name(host_os):
74 if host_os.startswith('mingw'):
75 return 'Windows'
76 if host_os.startswith('linux'):
77 return 'Linux'
78 return host_os
79
80# CMake expects target architectures in the format of uname(2),
81# which do not always match TARGET_ARCH, so all the necessary
82# conversions should happen here.
83def map_host_arch_to_uname_arch(host_arch):
84 if host_arch == "powerpc":
85 return "ppc"
86 if host_arch == "powerpc64":
87 return "ppc64"
88 return host_arch
89
90cmake_do_generate_toolchain_file() {
91 if [ "${BUILD_SYS}" = "${HOST_SYS}" ]; then
92 cmake_crosscompiling="set( CMAKE_CROSSCOMPILING FALSE )"
93 fi
94 cat > ${WORKDIR}/toolchain.cmake <<EOF
95# CMake system name must be something like "Linux".
96# This is important for cross-compiling.
97$cmake_crosscompiling
98set( CMAKE_SYSTEM_NAME ${@map_host_os_to_system_name(d.getVar('HOST_OS'))} )
99set( CMAKE_SYSTEM_PROCESSOR ${@map_host_arch_to_uname_arch(d.getVar('HOST_ARCH'))} )
100set( CMAKE_C_COMPILER ${OECMAKE_C_COMPILER} )
101set( CMAKE_CXX_COMPILER ${OECMAKE_CXX_COMPILER} )
102set( CMAKE_C_COMPILER_LAUNCHER ${OECMAKE_C_COMPILER_LAUNCHER} )
103set( CMAKE_CXX_COMPILER_LAUNCHER ${OECMAKE_CXX_COMPILER_LAUNCHER} )
104set( CMAKE_ASM_COMPILER ${OECMAKE_C_COMPILER} )
105find_program( CMAKE_AR ${OECMAKE_AR} DOC "Archiver" REQUIRED )
106
107set( CMAKE_C_FLAGS "${OECMAKE_C_FLAGS}" CACHE STRING "CFLAGS" )
108set( CMAKE_CXX_FLAGS "${OECMAKE_CXX_FLAGS}" CACHE STRING "CXXFLAGS" )
109set( CMAKE_ASM_FLAGS "${OECMAKE_C_FLAGS}" CACHE STRING "ASM FLAGS" )
110set( CMAKE_C_FLAGS_RELEASE "${OECMAKE_C_FLAGS_RELEASE}" CACHE STRING "Additional CFLAGS for release" )
111set( CMAKE_CXX_FLAGS_RELEASE "${OECMAKE_CXX_FLAGS_RELEASE}" CACHE STRING "Additional CXXFLAGS for release" )
112set( CMAKE_ASM_FLAGS_RELEASE "${OECMAKE_C_FLAGS_RELEASE}" CACHE STRING "Additional ASM FLAGS for release" )
113set( CMAKE_C_LINK_FLAGS "${OECMAKE_C_LINK_FLAGS}" CACHE STRING "LDFLAGS" )
114set( CMAKE_CXX_LINK_FLAGS "${OECMAKE_CXX_LINK_FLAGS}" CACHE STRING "LDFLAGS" )
115
116# only search in the paths provided so cmake doesnt pick
117# up libraries and tools from the native build machine
118set( CMAKE_FIND_ROOT_PATH ${STAGING_DIR_HOST} ${STAGING_DIR_NATIVE} ${CROSS_DIR} ${OECMAKE_PERLNATIVE_DIR} ${OECMAKE_EXTRA_ROOT_PATH} ${EXTERNAL_TOOLCHAIN} ${HOSTTOOLS_DIR})
119set( CMAKE_FIND_ROOT_PATH_MODE_PACKAGE ONLY )
120set( CMAKE_FIND_ROOT_PATH_MODE_PROGRAM ${OECMAKE_FIND_ROOT_PATH_MODE_PROGRAM} )
121set( CMAKE_FIND_ROOT_PATH_MODE_LIBRARY ONLY )
122set( CMAKE_FIND_ROOT_PATH_MODE_INCLUDE ONLY )
123set( CMAKE_PROGRAM_PATH "/" )
124
125# Use qt.conf settings
126set( ENV{QT_CONF_PATH} ${WORKDIR}/qt.conf )
127
128# We need to set the rpath to the correct directory as cmake does not provide any
129# directory as rpath by default
130set( CMAKE_INSTALL_RPATH ${OECMAKE_RPATH} )
131
132# Use RPATHs relative to build directory for reproducibility
133set( CMAKE_BUILD_RPATH_USE_ORIGIN ON )
134
135# Use our cmake modules
136list(APPEND CMAKE_MODULE_PATH "${STAGING_DATADIR}/cmake/Modules/")
137
138# add for non /usr/lib libdir, e.g. /usr/lib64
139set( CMAKE_LIBRARY_PATH ${libdir} ${base_libdir})
140
141# add include dir to implicit includes in case it differs from /usr/include
142list(APPEND CMAKE_C_IMPLICIT_INCLUDE_DIRECTORIES ${includedir})
143list(APPEND CMAKE_CXX_IMPLICIT_INCLUDE_DIRECTORIES ${includedir})
144
145EOF
146}
147
148addtask generate_toolchain_file after do_patch before do_configure
149
150CONFIGURE_FILES = "CMakeLists.txt"
151
152cmake_do_configure() {
153 if [ "${OECMAKE_BUILDPATH}" ]; then
154 bbnote "cmake.bbclass no longer uses OECMAKE_BUILDPATH. The default behaviour is now out-of-tree builds with B=WORKDIR/build."
155 fi
156
157 if [ "${S}" != "${B}" ]; then
158 rm -rf ${B}
159 mkdir -p ${B}
160 cd ${B}
161 else
162 find ${B} -name CMakeFiles -or -name Makefile -or -name cmake_install.cmake -or -name CMakeCache.txt -delete
163 fi
164
165 # Just like autotools cmake can use a site file to cache result that need generated binaries to run
166 if [ -e ${WORKDIR}/site-file.cmake ] ; then
167 oecmake_sitefile="-C ${WORKDIR}/site-file.cmake"
168 else
169 oecmake_sitefile=
170 fi
171
172 cmake \
173 ${OECMAKE_GENERATOR_ARGS} \
174 $oecmake_sitefile \
175 ${OECMAKE_SOURCEPATH} \
176 -DCMAKE_INSTALL_PREFIX:PATH=${prefix} \
177 -DCMAKE_INSTALL_BINDIR:PATH=${@os.path.relpath(d.getVar('bindir'), d.getVar('prefix') + '/')} \
178 -DCMAKE_INSTALL_SBINDIR:PATH=${@os.path.relpath(d.getVar('sbindir'), d.getVar('prefix') + '/')} \
179 -DCMAKE_INSTALL_LIBEXECDIR:PATH=${@os.path.relpath(d.getVar('libexecdir'), d.getVar('prefix') + '/')} \
180 -DCMAKE_INSTALL_SYSCONFDIR:PATH=${sysconfdir} \
181 -DCMAKE_INSTALL_SHAREDSTATEDIR:PATH=${@os.path.relpath(d.getVar('sharedstatedir'), d. getVar('prefix') + '/')} \
182 -DCMAKE_INSTALL_LOCALSTATEDIR:PATH=${localstatedir} \
183 -DCMAKE_INSTALL_LIBDIR:PATH=${@os.path.relpath(d.getVar('libdir'), d.getVar('prefix') + '/')} \
184 -DCMAKE_INSTALL_INCLUDEDIR:PATH=${@os.path.relpath(d.getVar('includedir'), d.getVar('prefix') + '/')} \
185 -DCMAKE_INSTALL_DATAROOTDIR:PATH=${@os.path.relpath(d.getVar('datadir'), d.getVar('prefix') + '/')} \
186 -DPYTHON_EXECUTABLE:PATH=${PYTHON} \
187 -DPython_EXECUTABLE:PATH=${PYTHON} \
188 -DPython3_EXECUTABLE:PATH=${PYTHON} \
189 -DLIB_SUFFIX=${@d.getVar('baselib').replace('lib', '')} \
190 -DCMAKE_INSTALL_SO_NO_EXE=0 \
191 -DCMAKE_TOOLCHAIN_FILE=${WORKDIR}/toolchain.cmake \
192 -DCMAKE_NO_SYSTEM_FROM_IMPORTED=1 \
193 ${EXTRA_OECMAKE} \
194 -Wno-dev
195}
196
197# To disable verbose cmake logs for a given recipe or globally config metadata e.g. local.conf
198# add following
199#
200# CMAKE_VERBOSE = ""
201#
202
203CMAKE_VERBOSE ??= "VERBOSE=1"
204
205# Then run do_compile again
206cmake_runcmake_build() {
207 bbnote ${DESTDIR:+DESTDIR=${DESTDIR} }${CMAKE_VERBOSE} cmake --build '${B}' "$@" -- ${EXTRA_OECMAKE_BUILD}
208 eval ${DESTDIR:+DESTDIR=${DESTDIR} }${CMAKE_VERBOSE} cmake --build '${B}' "$@" -- ${EXTRA_OECMAKE_BUILD}
209}
210
211cmake_do_compile() {
212 cmake_runcmake_build --target ${OECMAKE_TARGET_COMPILE}
213}
214
215cmake_do_install() {
216 DESTDIR='${D}' cmake_runcmake_build --target ${OECMAKE_TARGET_INSTALL}
217}
218
219EXPORT_FUNCTIONS do_configure do_compile do_install do_generate_toolchain_file