pkg-python.mk 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421
  1. ################################################################################
  2. # Python package infrastructure
  3. #
  4. # This file implements an infrastructure that eases development of
  5. # package .mk files for Python packages. It should be used for all
  6. # packages that use Python setup.py/setuptools as their build system.
  7. #
  8. # See the Buildroot documentation for details on the usage of this
  9. # infrastructure
  10. #
  11. # In terms of implementation, this Python infrastructure requires the
  12. # .mk file to only specify metadata information about the package:
  13. # name, version, download URL, etc.
  14. #
  15. # We still allow the package .mk file to override what the different
  16. # steps are doing, if needed. For example, if <PKG>_BUILD_CMDS is
  17. # already defined, it is used as the list of commands to perform to
  18. # build the package, instead of the default Python behaviour. The
  19. # package can also define some post operation hooks.
  20. #
  21. ################################################################################
  22. ifeq ($(BR2_arm)$(BR2_armeb),y)
  23. PKG_PYTHON_ARCH = arm
  24. else
  25. PKG_PYTHON_ARCH = $(ARCH)
  26. endif
  27. PKG_PYTHON_HOST_PLATFORM = linux-$(PKG_PYTHON_ARCH)
  28. # basename does not evaluate if a file exists, so we must check to ensure
  29. # the _sysconfigdata__linux_*.py file exists. The "|| true" is added to return
  30. # an empty string if the file does not exist.
  31. PKG_PYTHON_SYSCONFIGDATA_PATH = $(PYTHON3_PATH)/_sysconfigdata__linux_*.py
  32. PKG_PYTHON_SYSCONFIGDATA_NAME = `{ [ -e $(PKG_PYTHON_SYSCONFIGDATA_PATH) ] && basename $(PKG_PYTHON_SYSCONFIGDATA_PATH) .py; } || true`
  33. # Target python packages
  34. PKG_PYTHON_ENV = \
  35. _PYTHON_HOST_PLATFORM="$(PKG_PYTHON_HOST_PLATFORM)" \
  36. _PYTHON_PROJECT_BASE="$(PYTHON3_DIR)" \
  37. _PYTHON_SYSCONFIGDATA_NAME="$(PKG_PYTHON_SYSCONFIGDATA_NAME)" \
  38. PATH=$(BR_PATH) \
  39. $(TARGET_CONFIGURE_OPTS) \
  40. PYTHONPATH="$(PYTHON3_PATH)" \
  41. PYTHONNOUSERSITE=1 \
  42. _python_sysroot=$(STAGING_DIR) \
  43. _python_prefix=/usr \
  44. _python_exec_prefix=/usr
  45. # Host python packages
  46. HOST_PKG_PYTHON_ENV = \
  47. PATH=$(BR_PATH) \
  48. PYTHONNOUSERSITE=1 \
  49. $(HOST_CONFIGURE_OPTS)
  50. # Target setuptools-based packages
  51. PKG_PYTHON_SETUPTOOLS_ENV = \
  52. $(PKG_PYTHON_ENV)
  53. PKG_PYTHON_SETUPTOOLS_CMD = \
  54. $(if $(wildcard $($(PKG)_BUILDDIR)/setup.py),setup.py,-c 'from setuptools import setup;setup()')
  55. PKG_PYTHON_SETUPTOOLS_BUILD_CMD = \
  56. $(PKG_PYTHON_SETUPTOOLS_CMD) build
  57. PKG_PYTHON_SETUPTOOLS_INSTALL_OPTS = \
  58. --install-headers=/usr/include/python$(PYTHON3_VERSION_MAJOR) \
  59. --prefix=/usr \
  60. --executable=/usr/bin/python \
  61. --single-version-externally-managed
  62. PKG_PYTHON_SETUPTOOLS_INSTALL_TARGET_CMD = \
  63. $(PKG_PYTHON_SETUPTOOLS_CMD) \
  64. install --no-compile \
  65. $(PKG_PYTHON_SETUPTOOLS_INSTALL_OPTS) \
  66. --root=$(TARGET_DIR)
  67. PKG_PYTHON_SETUPTOOLS_INSTALL_STAGING_CMD = \
  68. $(PKG_PYTHON_SETUPTOOLS_CMD) \
  69. install \
  70. $(PKG_PYTHON_SETUPTOOLS_INSTALL_OPTS) \
  71. --root=$(STAGING_DIR)
  72. # Host setuptools-based packages
  73. HOST_PKG_PYTHON_SETUPTOOLS_ENV = \
  74. $(HOST_PKG_PYTHON_ENV)
  75. HOST_PKG_PYTHON_SETUPTOOLS_BUILD_CMD = \
  76. $(PKG_PYTHON_SETUPTOOLS_CMD) build
  77. HOST_PKG_PYTHON_SETUPTOOLS_INSTALL_CMD = \
  78. $(PKG_PYTHON_SETUPTOOLS_CMD) \
  79. install \
  80. --prefix=$(HOST_DIR) \
  81. --root=/ \
  82. --single-version-externally-managed
  83. # Target setuptools-rust-based packages
  84. PKG_PYTHON_SETUPTOOLS_RUST_ENV = \
  85. $(PKG_PYTHON_SETUPTOOLS_ENV) \
  86. $(PKG_CARGO_ENV) \
  87. PYO3_CROSS_LIB_DIR="$(STAGING_DIR)/usr/lib/python$(PYTHON3_VERSION_MAJOR)"
  88. PKG_PYTHON_SETUPTOOLS_RUST_BUILD_CMD = \
  89. $(PKG_PYTHON_SETUPTOOLS_BUILD_CMD)
  90. PKG_PYTHON_SETUPTOOLS_RUST_INSTALL_TARGET_CMD = \
  91. $(PKG_PYTHON_SETUPTOOLS_INSTALL_TARGET_CMD)
  92. PKG_PYTHON_SETUPTOOLS_RUST_INSTALL_STAGING_CMD = \
  93. $(PKG_PYTHON_SETUPTOOLS_INSTALL_STAGING_CMD)
  94. # Host setuptools-rust-based packages
  95. HOST_PKG_PYTHON_SETUPTOOLS_RUST_ENV = \
  96. $(HOST_PKG_PYTHON_SETUPTOOLS_ENV) \
  97. $(HOST_PKG_CARGO_ENV) \
  98. PYO3_CROSS_LIB_DIR="$(HOST_DIR)/lib/python$(PYTHON3_VERSION_MAJOR)"
  99. HOST_PKG_PYTHON_SETUPTOOLS_RUST_BUILD_CMD = \
  100. $(HOST_PKG_PYTHON_SETUPTOOLS_BUILD_CMD)
  101. HOST_PKG_PYTHON_SETUPTOOLS_RUST_INSTALL_CMD = \
  102. $(HOST_PKG_PYTHON_SETUPTOOLS_INSTALL_CMD)
  103. # Target pep517-based packages
  104. PKG_PYTHON_PEP517_ENV = \
  105. $(PKG_PYTHON_ENV)
  106. PKG_PYTHON_PEP517_BUILD_CMD = \
  107. -m build -n -w
  108. PKG_PYTHON_PEP517_INSTALL_OPTS = \
  109. --interpreter=/usr/bin/python \
  110. --script-kind=posix
  111. PKG_PYTHON_PEP517_INSTALL_TARGET_CMD = \
  112. $(TOPDIR)/support/scripts/pyinstaller.py \
  113. dist/* \
  114. $(PKG_PYTHON_PEP517_INSTALL_OPTS) \
  115. --purelib=$(TARGET_DIR)/usr/lib/python$(PYTHON3_VERSION_MAJOR)/site-packages \
  116. --headers=$(TARGET_DIR)/usr/include/python$(PYTHON3_VERSION_MAJOR) \
  117. --scripts=$(TARGET_DIR)/usr/bin \
  118. --data=$(TARGET_DIR)/usr
  119. PKG_PYTHON_PEP517_INSTALL_STAGING_CMD = \
  120. $(TOPDIR)/support/scripts/pyinstaller.py \
  121. dist/* \
  122. $(PKG_PYTHON_PEP517_INSTALL_OPTS) \
  123. --purelib=$(STAGING_DIR)/usr/lib/python$(PYTHON3_VERSION_MAJOR)/site-packages \
  124. --headers=$(STAGING_DIR)/usr/include/python$(PYTHON3_VERSION_MAJOR) \
  125. --scripts=$(STAGING_DIR)/usr/bin \
  126. --data=$(STAGING_DIR)/usr
  127. # Host pep517-based packages
  128. HOST_PKG_PYTHON_PEP517_ENV = \
  129. $(HOST_PKG_PYTHON_ENV)
  130. HOST_PKG_PYTHON_PEP517_BUILD_CMD = \
  131. -m build -n -w
  132. HOST_PKG_PYTHON_PEP517_INSTALL_CMD = \
  133. $(TOPDIR)/support/scripts/pyinstaller.py \
  134. dist/* \
  135. --interpreter=$(HOST_DIR)/bin/python \
  136. --script-kind=posix \
  137. --purelib=$(HOST_DIR)/lib/python$(PYTHON3_VERSION_MAJOR)/site-packages \
  138. --headers=$(HOST_DIR)/include/python$(PYTHON3_VERSION_MAJOR) \
  139. --scripts=$(HOST_DIR)/bin \
  140. --data=$(HOST_DIR)
  141. # Target flit packages
  142. PKG_PYTHON_FLIT_ENV = \
  143. $(PKG_PYTHON_PEP517_ENV)
  144. PKG_PYTHON_FLIT_BUILD_CMD = \
  145. $(PKG_PYTHON_PEP517_BUILD_CMD)
  146. PKG_PYTHON_FLIT_INSTALL_TARGET_CMD = \
  147. $(PKG_PYTHON_PEP517_INSTALL_TARGET_CMD)
  148. PKG_PYTHON_FLIT_INSTALL_STAGING_CMD = \
  149. $(PKG_PYTHON_PEP517_INSTALL_STAGING_CMD)
  150. # Host flit packages
  151. HOST_PKG_PYTHON_FLIT_ENV = \
  152. $(HOST_PKG_PYTHON_PEP517_ENV)
  153. HOST_PKG_PYTHON_FLIT_BUILD_CMD = \
  154. $(HOST_PKG_PYTHON_PEP517_BUILD_CMD)
  155. HOST_PKG_PYTHON_FLIT_INSTALL_CMD = \
  156. $(HOST_PKG_PYTHON_PEP517_INSTALL_CMD)
  157. # Host flit-bootstrap packages
  158. HOST_PKG_PYTHON_FLIT_BOOTSTRAP_ENV = \
  159. $(HOST_PKG_PYTHON_PEP517_ENV)
  160. HOST_PKG_PYTHON_FLIT_BOOTSTRAP_BUILD_CMD = \
  161. -m flit_core.wheel
  162. HOST_PKG_PYTHON_FLIT_BOOTSTRAP_INSTALL_CMD = \
  163. $(HOST_PKG_PYTHON_PEP517_INSTALL_CMD)
  164. # Target maturin packages
  165. PKG_PYTHON_MATURIN_ENV = \
  166. $(PKG_PYTHON_PEP517_ENV) \
  167. $(PKG_CARGO_ENV) \
  168. PYO3_CROSS_LIB_DIR="$(STAGING_DIR)/usr/lib/python$(PYTHON3_VERSION_MAJOR)"
  169. PKG_PYTHON_MATURIN_BUILD_CMD = \
  170. $(PKG_PYTHON_PEP517_BUILD_CMD)
  171. PKG_PYTHON_MATURIN_INSTALL_TARGET_CMD = \
  172. $(PKG_PYTHON_PEP517_INSTALL_TARGET_CMD)
  173. PKG_PYTHON_MATURIN_INSTALL_STAGING_CMD = \
  174. $(PKG_PYTHON_PEP517_INSTALL_STAGING_CMD)
  175. # Host maturin packages
  176. HOST_PKG_PYTHON_MATURIN_ENV = \
  177. $(HOST_PKG_PYTHON_PEP517_ENV) \
  178. $(HOST_PKG_CARGO_ENV) \
  179. PYO3_CROSS_LIB_DIR="$(HOST_DIR)/lib/python$(PYTHON3_VERSION_MAJOR)"
  180. HOST_PKG_PYTHON_MATURIN_BUILD_CMD = \
  181. $(HOST_PKG_PYTHON_PEP517_BUILD_CMD)
  182. HOST_PKG_PYTHON_MATURIN_INSTALL_CMD = \
  183. $(HOST_PKG_PYTHON_PEP517_INSTALL_CMD)
  184. ################################################################################
  185. # inner-python-package -- defines how the configuration, compilation
  186. # and installation of a Python package should be done, implements a
  187. # few hooks to tune the build process and calls the generic package
  188. # infrastructure to generate the necessary make targets
  189. #
  190. # argument 1 is the lowercase package name
  191. # argument 2 is the uppercase package name, including a HOST_ prefix
  192. # for host packages
  193. # argument 3 is the uppercase package name, without the HOST_ prefix
  194. # for host packages
  195. # argument 4 is the type (target or host)
  196. ################################################################################
  197. define inner-python-package
  198. ifndef $(2)_SETUP_TYPE
  199. ifdef $(3)_SETUP_TYPE
  200. $(2)_SETUP_TYPE = $$($(3)_SETUP_TYPE)
  201. else
  202. $$(error "$(2)_SETUP_TYPE must be set")
  203. endif
  204. endif
  205. $(2)_SETUP_TYPE_UPPER = $$(call UPPERCASE,$$($(2)_SETUP_TYPE))
  206. ifneq ($$(filter-out setuptools setuptools-rust pep517 flit flit-bootstrap maturin,$$($(2)_SETUP_TYPE)),)
  207. $$(error "Invalid $(2)_SETUP_TYPE. Valid options are 'maturin', 'setuptools', 'setuptools-rust', 'pep517' or 'flit'.")
  208. endif
  209. ifeq ($(4)-$$($(2)_SETUP_TYPE),target-flit-bootstrap)
  210. $$(error flit-bootstrap setup type only supported for host packages)
  211. endif
  212. # We need to vendor the Cargo crates at download time for pyo3 based
  213. # packages.
  214. #
  215. ifneq ($$(filter maturin setuptools-rust,$$($(2)_SETUP_TYPE)),)
  216. ifeq ($(4),target)
  217. $(2)_DL_ENV = $$(PKG_CARGO_ENV)
  218. else
  219. $(2)_DL_ENV = $$(HOST_PKG_CARGO_ENV)
  220. endif
  221. ifndef $(2)_CARGO_MANIFEST_PATH
  222. ifdef $(3)_CARGO_MANIFEST_PATH
  223. $(2)_DL_ENV += BR_CARGO_MANIFEST_PATH=$$($(3)_CARGO_MANIFEST_PATH)
  224. else
  225. ifneq ($$($(2)_SUBDIR),)
  226. $(2)_DL_ENV += BR_CARGO_MANIFEST_PATH=$$($(2)_SUBDIR)/Cargo.toml
  227. endif
  228. endif
  229. else
  230. $(2)_DL_ENV += BR_CARGO_MANIFEST_PATH=$$($(2)_CARGO_MANIFEST_PATH)
  231. endif
  232. endif
  233. # Target packages need both the python interpreter on the target (for
  234. # runtime) and the python interpreter on the host (for
  235. # compilation). However, host packages only need the python
  236. # interpreter on the host.
  237. #
  238. ifeq ($(4),target)
  239. $(2)_DEPENDENCIES += host-python3 python3
  240. else
  241. $(2)_DEPENDENCIES += host-python3
  242. endif # ($(4),target)
  243. # Setuptools based packages will need setuptools for the host Python
  244. # interpreter (both host and target).
  245. #
  246. ifneq ($$(filter setuptools setuptools-rust,$$($(2)_SETUP_TYPE)),)
  247. $(2)_DEPENDENCIES += host-python-setuptools
  248. ifeq ($$($(2)_SETUP_TYPE),setuptools-rust)
  249. $(2)_DEPENDENCIES += host-python-setuptools-rust
  250. endif
  251. else ifneq ($$(filter flit maturin pep517,$$($(2)_SETUP_TYPE)),)
  252. $(2)_DEPENDENCIES += host-python-pypa-build host-python-installer
  253. ifeq ($$($(2)_SETUP_TYPE),flit)
  254. $(2)_DEPENDENCIES += host-python-flit-core
  255. endif
  256. ifeq ($$($(2)_SETUP_TYPE),maturin)
  257. $(2)_DEPENDENCIES += host-python-maturin
  258. endif
  259. else ifeq ($$($(2)_SETUP_TYPE),flit-bootstrap)
  260. # Don't add dependency on host-python-installer for
  261. # host-python-installer itself, and its dependencies.
  262. ifeq ($$(filter host-python-flit-core host-python-installer,$(1)),)
  263. $(2)_DEPENDENCIES += host-python-installer
  264. endif
  265. endif
  266. # Pyo3 based packages(setuptools-rust and maturin) will need rust
  267. # toolchain dependencies for the host Python interpreter (both host
  268. # and target).
  269. #
  270. ifneq ($$(filter maturin setuptools-rust,$$($(2)_SETUP_TYPE)),)
  271. $(2)_DEPENDENCIES += host-rustc
  272. $(2)_DOWNLOAD_POST_PROCESS = cargo
  273. $(2)_DOWNLOAD_DEPENDENCIES = host-rustc
  274. endif # SETUP_TYPE
  275. ifeq ($(4),target)
  276. #
  277. # Build step. Only define it if not already defined by the package .mk
  278. # file.
  279. #
  280. ifndef $(2)_BUILD_CMDS
  281. define $(2)_BUILD_CMDS
  282. (cd $$($$(PKG)_BUILDDIR)/; \
  283. $$(PKG_PYTHON_$$($$(PKG)_SETUP_TYPE_UPPER)_ENV) \
  284. $$($$(PKG)_ENV) \
  285. $$(HOST_DIR)/bin/python3 \
  286. $$(PKG_PYTHON_$$($$(PKG)_SETUP_TYPE_UPPER)_BUILD_CMD) \
  287. $$($$(PKG)_BUILD_OPTS))
  288. endef
  289. endif
  290. #
  291. # Target installation step. Only define it if not already defined by
  292. # the package .mk file.
  293. #
  294. ifndef $(2)_INSTALL_TARGET_CMDS
  295. define $(2)_INSTALL_TARGET_CMDS
  296. (cd $$($$(PKG)_BUILDDIR)/; \
  297. $$(PKG_PYTHON_$$($$(PKG)_SETUP_TYPE_UPPER)_ENV) \
  298. $$($$(PKG)_ENV) \
  299. $$(HOST_DIR)/bin/python3 \
  300. $$(PKG_PYTHON_$$($$(PKG)_SETUP_TYPE_UPPER)_INSTALL_TARGET_CMD) \
  301. $$($$(PKG)_INSTALL_TARGET_OPTS))
  302. endef
  303. endif
  304. #
  305. # Staging installation step. Only define it if not already defined by
  306. # the package .mk file.
  307. #
  308. ifndef $(2)_INSTALL_STAGING_CMDS
  309. define $(2)_INSTALL_STAGING_CMDS
  310. (cd $$($$(PKG)_BUILDDIR)/; \
  311. $$(PKG_PYTHON_$$($$(PKG)_SETUP_TYPE_UPPER)_ENV) \
  312. $$($$(PKG)_ENV) \
  313. $$(HOST_DIR)/bin/python3 \
  314. $$(PKG_PYTHON_$$($$(PKG)_SETUP_TYPE_UPPER)_INSTALL_STAGING_CMD) \
  315. $$($$(PKG)_INSTALL_STAGING_OPTS))
  316. endef
  317. endif
  318. else # host
  319. #
  320. # Host build step. Only define it if not already defined by the package .mk
  321. # file.
  322. #
  323. ifndef $(2)_BUILD_CMDS
  324. define $(2)_BUILD_CMDS
  325. (cd $$($$(PKG)_BUILDDIR)/; \
  326. $$(HOST_PKG_PYTHON_$$($$(PKG)_SETUP_TYPE_UPPER)_ENV) \
  327. $$($$(PKG)_ENV) \
  328. $$(HOST_DIR)/bin/python3 \
  329. $$(HOST_PKG_PYTHON_$$($$(PKG)_SETUP_TYPE_UPPER)_BUILD_CMD) \
  330. $$($$(PKG)_BUILD_OPTS))
  331. endef
  332. endif
  333. #
  334. # Host installation step. Only define it if not already defined by the
  335. # package .mk file.
  336. #
  337. ifndef $(2)_INSTALL_CMDS
  338. define $(2)_INSTALL_CMDS
  339. (cd $$($$(PKG)_BUILDDIR)/; \
  340. $$(HOST_PKG_PYTHON_$$($$(PKG)_SETUP_TYPE_UPPER)_ENV) \
  341. $$($$(PKG)_ENV) \
  342. $$(HOST_DIR)/bin/python3 \
  343. $$(HOST_PKG_PYTHON_$$($$(PKG)_SETUP_TYPE_UPPER)_INSTALL_CMD) \
  344. $$($$(PKG)_INSTALL_OPTS))
  345. endef
  346. endif
  347. endif # host / target
  348. # Call the generic package infrastructure to generate the necessary
  349. # make targets
  350. $(call inner-generic-package,$(1),$(2),$(3),$(4))
  351. endef
  352. ################################################################################
  353. # python-package -- the target generator macro for Python packages
  354. ################################################################################
  355. python-package = $(call inner-python-package,$(pkgname),$(call UPPERCASE,$(pkgname)),$(call UPPERCASE,$(pkgname)),target)
  356. host-python-package = $(call inner-python-package,host-$(pkgname),$(call UPPERCASE,host-$(pkgname)),$(call UPPERCASE,$(pkgname)),host)