pkg-python.mk 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305
  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 distutils-based packages
  34. PKG_PYTHON_DISTUTILS_ENV = \
  35. PATH=$(BR_PATH) \
  36. $(TARGET_CONFIGURE_OPTS) \
  37. LDSHARED="$(TARGET_CROSS)gcc -shared" \
  38. PYTHONPATH="$(if $(BR2_PACKAGE_PYTHON3),$(PYTHON3_PATH),$(PYTHON_PATH))" \
  39. PYTHONNOUSERSITE=1 \
  40. _PYTHON_HOST_PLATFORM="$(PKG_PYTHON_HOST_PLATFORM)" \
  41. _PYTHON_PROJECT_BASE="$(if $(BR2_PACKAGE_PYTHON3),$(PYTHON3_DIR),$(PYTHON_DIR))" \
  42. _PYTHON_SYSCONFIGDATA_NAME="$(PKG_PYTHON_SYSCONFIGDATA_NAME)" \
  43. _python_sysroot=$(STAGING_DIR) \
  44. _python_prefix=/usr \
  45. _python_exec_prefix=/usr
  46. PKG_PYTHON_DISTUTILS_BUILD_OPTS = \
  47. --executable=/usr/bin/python
  48. PKG_PYTHON_DISTUTILS_INSTALL_TARGET_OPTS = \
  49. --prefix=/usr \
  50. --root=$(TARGET_DIR)
  51. PKG_PYTHON_DISTUTILS_INSTALL_STAGING_OPTS = \
  52. --prefix=/usr \
  53. --root=$(STAGING_DIR)
  54. # Host distutils-based packages
  55. HOST_PKG_PYTHON_DISTUTILS_ENV = \
  56. PATH=$(BR_PATH) \
  57. PYTHONNOUSERSITE=1 \
  58. $(HOST_CONFIGURE_OPTS)
  59. HOST_PKG_PYTHON_DISTUTILS_INSTALL_OPTS = \
  60. --prefix=$(HOST_DIR)
  61. # Target setuptools-based packages
  62. PKG_PYTHON_SETUPTOOLS_ENV = \
  63. _PYTHON_HOST_PLATFORM="$(PKG_PYTHON_HOST_PLATFORM)" \
  64. _PYTHON_PROJECT_BASE="$(if $(BR2_PACKAGE_PYTHON3),$(PYTHON3_DIR),$(PYTHON_DIR))" \
  65. _PYTHON_SYSCONFIGDATA_NAME="$(PKG_PYTHON_SYSCONFIGDATA_NAME)" \
  66. PATH=$(BR_PATH) \
  67. $(TARGET_CONFIGURE_OPTS) \
  68. PYTHONPATH="$(if $(BR2_PACKAGE_PYTHON3),$(PYTHON3_PATH),$(PYTHON_PATH))" \
  69. PYTHONNOUSERSITE=1 \
  70. _python_sysroot=$(STAGING_DIR) \
  71. _python_prefix=/usr \
  72. _python_exec_prefix=/usr
  73. PKG_PYTHON_SETUPTOOLS_INSTALL_TARGET_OPTS = \
  74. --prefix=/usr \
  75. --executable=/usr/bin/python \
  76. --single-version-externally-managed \
  77. --root=$(TARGET_DIR)
  78. PKG_PYTHON_SETUPTOOLS_INSTALL_STAGING_OPTS = \
  79. --prefix=/usr \
  80. --executable=/usr/bin/python \
  81. --single-version-externally-managed \
  82. --root=$(STAGING_DIR)
  83. # Host setuptools-based packages
  84. HOST_PKG_PYTHON_SETUPTOOLS_ENV = \
  85. PATH=$(BR_PATH) \
  86. PYTHONNOUSERSITE=1 \
  87. $(HOST_CONFIGURE_OPTS)
  88. HOST_PKG_PYTHON_SETUPTOOLS_INSTALL_OPTS = \
  89. --prefix=$(HOST_DIR) \
  90. --root=/ \
  91. --single-version-externally-managed
  92. # Target pep517-based packages
  93. PKG_PYTHON_PEP517_ENV = \
  94. _PYTHON_SYSCONFIGDATA_NAME="$(PKG_PYTHON_SYSCONFIGDATA_NAME)" \
  95. PATH=$(BR_PATH) \
  96. $(TARGET_CONFIGURE_OPTS) \
  97. PYTHONPATH="$(PYTHON3_PATH)" \
  98. PYTHONNOUSERSITE=1 \
  99. _python_sysroot=$(STAGING_DIR) \
  100. _python_prefix=/usr \
  101. _python_exec_prefix=/usr
  102. PKG_PYTHON_PEP517_INSTALL_TARGET_OPTS = \
  103. --interpreter=/usr/bin/python \
  104. --script-kind=posix \
  105. --purelib=$(TARGET_DIR)/lib/python$(PYTHON3_VERSION_MAJOR)/site-packages \
  106. --headers=$(TARGET_DIR)/usr/include/python$(PYTHON3_VERSION_MAJOR) \
  107. --scripts=$(TARGET_DIR)/usr/bin \
  108. --data=$(TARGET_DIR)/usr
  109. PKG_PYTHON_PEP517_INSTALL_STAGING_OPTS = \
  110. --interpreter=/usr/bin/python \
  111. --script-kind=posix \
  112. --purelib=$(STAGING_DIR)/lib/python$(PYTHON3_VERSION_MAJOR)/site-packages \
  113. --headers=$(STAGING_DIR)/usr/include/python$(PYTHON3_VERSION_MAJOR) \
  114. --scripts=$(STAGING_DIR)/usr/bin \
  115. --data=$(STAGING_DIR)/usr
  116. # Host pep517-based packages
  117. HOST_PKG_PYTHON_PEP517_ENV = \
  118. PATH=$(BR_PATH) \
  119. PYTHONNOUSERSITE=1 \
  120. $(HOST_CONFIGURE_OPTS)
  121. HOST_PKG_PYTHON_PEP517_INSTALL_OPTS = \
  122. --interpreter=/usr/bin/python \
  123. --script-kind=posix \
  124. --purelib=$(HOST_DIR)/lib/python$(PYTHON3_VERSION_MAJOR)/site-packages \
  125. --headers=$(HOST_DIR)/usr/include/python$(PYTHON3_VERSION_MAJOR) \
  126. --scripts=$(HOST_DIR)/usr/bin \
  127. --data=$(HOST_DIR)/usr
  128. ################################################################################
  129. # inner-python-package -- defines how the configuration, compilation
  130. # and installation of a Python package should be done, implements a
  131. # few hooks to tune the build process and calls the generic package
  132. # infrastructure to generate the necessary make targets
  133. #
  134. # argument 1 is the lowercase package name
  135. # argument 2 is the uppercase package name, including a HOST_ prefix
  136. # for host packages
  137. # argument 3 is the uppercase package name, without the HOST_ prefix
  138. # for host packages
  139. # argument 4 is the type (target or host)
  140. ################################################################################
  141. define inner-python-package
  142. ifndef $(2)_SETUP_TYPE
  143. ifdef $(3)_SETUP_TYPE
  144. $(2)_SETUP_TYPE = $$($(3)_SETUP_TYPE)
  145. else
  146. $$(error "$(2)_SETUP_TYPE must be set")
  147. endif
  148. endif
  149. # Distutils
  150. ifeq ($$($(2)_SETUP_TYPE),distutils)
  151. ifeq ($(4),target)
  152. $(2)_BASE_ENV = $$(PKG_PYTHON_DISTUTILS_ENV)
  153. $(2)_BASE_BUILD_CMD = setup.py build
  154. $(2)_BASE_BUILD_OPTS = $$(PKG_PYTHON_DISTUTILS_BUILD_OPTS)
  155. $(2)_BASE_INSTALL_TARGET_CMD = setup.py install --no-compile $$(PKG_PYTHON_DISTUTILS_INSTALL_TARGET_OPTS)
  156. $(2)_BASE_INSTALL_STAGING_CMD = setup.py install $$(PKG_PYTHON_DISTUTILS_INSTALL_STAGING_OPTS)
  157. else
  158. $(2)_BASE_ENV = $$(HOST_PKG_PYTHON_DISTUTILS_ENV)
  159. $(2)_BASE_BUILD_CMD = setup.py build
  160. $(2)_BASE_INSTALL_CMD = setup.py install $$(HOST_PKG_PYTHON_DISTUTILS_INSTALL_OPTS)
  161. endif
  162. # Setuptools
  163. else ifeq ($$($(2)_SETUP_TYPE),setuptools)
  164. ifeq ($(4),target)
  165. $(2)_BASE_ENV = $$(PKG_PYTHON_SETUPTOOLS_ENV)
  166. $(2)_BASE_BUILD_CMD = setup.py build
  167. $(2)_BASE_INSTALL_TARGET_CMD = setup.py install --no-compile $$(PKG_PYTHON_SETUPTOOLS_INSTALL_TARGET_OPTS)
  168. $(2)_BASE_INSTALL_STAGING_CMD = setup.py install $$(PKG_PYTHON_SETUPTOOLS_INSTALL_STAGING_OPTS)
  169. else
  170. $(2)_BASE_ENV = $$(HOST_PKG_PYTHON_SETUPTOOLS_ENV)
  171. $(2)_BASE_BUILD_CMD = setup.py build
  172. $(2)_BASE_INSTALL_CMD = setup.py install $$(HOST_PKG_PYTHON_SETUPTOOLS_INSTALL_OPTS)
  173. endif
  174. else ifneq ($$(filter flit pep517,$$($(2)_SETUP_TYPE)),)
  175. ifeq ($(4),target)
  176. $(2)_BASE_ENV = $$(PKG_PYTHON_PEP517_ENV)
  177. $(2)_BASE_BUILD_CMD = -m build -n -w
  178. $(2)_BASE_INSTALL_TARGET_CMD = $(TOPDIR)/support/scripts/pyinstaller.py dist/* $$(PKG_PYTHON_PEP517_INSTALL_TARGET_OPTS)
  179. $(2)_BASE_INSTALL_STAGING_CMD = $(TOPDIR)/support/scripts/pyinstaller.py dist/* $$(PKG_PYTHON_PEP517_INSTALL_STAGING_OPTS)
  180. else
  181. $(2)_BASE_ENV = $$(HOST_PKG_PYTHON_PEP517_ENV)
  182. $(2)_BASE_BUILD_CMD = -m build -n -w
  183. $(2)_BASE_INSTALL_CMD = $(TOPDIR)/support/scripts/pyinstaller.py dist/* $$(HOST_PKG_PYTHON_PEP517_INSTALL_OPTS)
  184. endif
  185. else
  186. $$(error "Invalid $(2)_SETUP_TYPE. Valid options are 'distutils', 'setuptools', 'pep517' or 'flit'.")
  187. endif
  188. # Target packages need both the python interpreter on the target (for
  189. # runtime) and the python interpreter on the host (for
  190. # compilation). However, host packages only need the python
  191. # interpreter on the host.
  192. #
  193. ifeq ($(4),target)
  194. $(2)_DEPENDENCIES += host-python3 python3
  195. else
  196. $(2)_DEPENDENCIES += host-python3
  197. endif # ($(4),target)
  198. # Setuptools based packages will need setuptools for the host Python
  199. # interpreter (both host and target).
  200. #
  201. ifeq ($$($(2)_SETUP_TYPE),setuptools)
  202. $(2)_DEPENDENCIES += $$(if $$(filter host-python-setuptools,$(1)),,host-python-setuptools)
  203. else ifneq ($$(filter flit pep517,$$($(2)_SETUP_TYPE)),)
  204. $(2)_DEPENDENCIES += host-python-pypa-build host-python-installer
  205. ifeq ($$($(2)_SETUP_TYPE),flit)
  206. $(2)_DEPENDENCIES += host-python-flit-core
  207. endif
  208. endif # SETUP_TYPE
  209. # Python interpreter to use for building the package.
  210. #
  211. $(2)_PYTHON_INTERPRETER = $$(HOST_DIR)/bin/python
  212. #
  213. # Build step. Only define it if not already defined by the package .mk
  214. # file.
  215. #
  216. ifndef $(2)_BUILD_CMDS
  217. define $(2)_BUILD_CMDS
  218. (cd $$($$(PKG)_BUILDDIR)/; \
  219. $$($$(PKG)_BASE_ENV) $$($$(PKG)_ENV) \
  220. $$($(2)_PYTHON_INTERPRETER) \
  221. $$($$(PKG)_BASE_BUILD_CMD) \
  222. $$($$(PKG)_BASE_BUILD_OPTS) $$($$(PKG)_BUILD_OPTS))
  223. endef
  224. endif
  225. #
  226. # Host installation step. Only define it if not already defined by the
  227. # package .mk file.
  228. #
  229. ifndef $(2)_INSTALL_CMDS
  230. define $(2)_INSTALL_CMDS
  231. (cd $$($$(PKG)_BUILDDIR)/; \
  232. $$($$(PKG)_BASE_ENV) $$($$(PKG)_ENV) \
  233. $$($(2)_PYTHON_INTERPRETER) \
  234. $$($$(PKG)_BASE_INSTALL_CMD) \
  235. $$($$(PKG)_INSTALL_OPTS))
  236. endef
  237. endif
  238. #
  239. # Target installation step. Only define it if not already defined by
  240. # the package .mk file.
  241. #
  242. ifndef $(2)_INSTALL_TARGET_CMDS
  243. define $(2)_INSTALL_TARGET_CMDS
  244. (cd $$($$(PKG)_BUILDDIR)/; \
  245. $$($$(PKG)_BASE_ENV) $$($$(PKG)_ENV) \
  246. $$($(2)_PYTHON_INTERPRETER) \
  247. $$($$(PKG)_BASE_INSTALL_TARGET_CMD) \
  248. $$($$(PKG)_INSTALL_TARGET_OPTS))
  249. endef
  250. endif
  251. #
  252. # Staging installation step. Only define it if not already defined by
  253. # the package .mk file.
  254. #
  255. ifndef $(2)_INSTALL_STAGING_CMDS
  256. define $(2)_INSTALL_STAGING_CMDS
  257. (cd $$($$(PKG)_BUILDDIR)/; \
  258. $$($$(PKG)_BASE_ENV) $$($$(PKG)_ENV) \
  259. $$($(2)_PYTHON_INTERPRETER) \
  260. $$($$(PKG)_BASE_INSTALL_STAGING_CMD) \
  261. $$($$(PKG)_INSTALL_STAGING_OPTS))
  262. endef
  263. endif
  264. # Call the generic package infrastructure to generate the necessary
  265. # make targets
  266. $(call inner-generic-package,$(1),$(2),$(3),$(4))
  267. endef
  268. ################################################################################
  269. # python-package -- the target generator macro for Python packages
  270. ################################################################################
  271. python-package = $(call inner-python-package,$(pkgname),$(call UPPERCASE,$(pkgname)),$(call UPPERCASE,$(pkgname)),target)
  272. host-python-package = $(call inner-python-package,host-$(pkgname),$(call UPPERCASE,host-$(pkgname)),$(call UPPERCASE,$(pkgname)),host)