systemd.bbclass 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302
  1. #
  2. # Copyright OpenEmbedded Contributors
  3. #
  4. # SPDX-License-Identifier: MIT
  5. #
  6. # The list of packages that should have systemd packaging scripts added. For
  7. # each entry, optionally have a SYSTEMD_SERVICE:[package] that lists the service
  8. # files in this package. If this variable isn't set, [package].service is used.
  9. SYSTEMD_PACKAGES ?= "${PN}"
  10. SYSTEMD_PACKAGES:class-native ?= ""
  11. SYSTEMD_PACKAGES:class-nativesdk ?= ""
  12. # Whether to enable or disable the services on installation.
  13. SYSTEMD_AUTO_ENABLE ??= "enable"
  14. # This class will be included in any recipe that supports systemd init scripts,
  15. # even if systemd is not in DISTRO_FEATURES. As such don't make any changes
  16. # directly but check the DISTRO_FEATURES first.
  17. python __anonymous() {
  18. # If the distro features have systemd but not sysvinit, inhibit update-rcd
  19. # from doing any work so that pure-systemd images don't have redundant init
  20. # files.
  21. if bb.utils.contains('DISTRO_FEATURES', 'systemd', True, False, d):
  22. d.appendVar("DEPENDS", " systemd-systemctl-native")
  23. d.appendVar("PACKAGE_WRITE_DEPS", " systemd-systemctl-native")
  24. if not bb.utils.contains('DISTRO_FEATURES', 'sysvinit', True, False, d):
  25. d.setVar("INHIBIT_UPDATERCD_BBCLASS", "1")
  26. }
  27. systemd_postinst() {
  28. if type systemctl >/dev/null 2>/dev/null; then
  29. OPTS=""
  30. if [ -n "$D" ]; then
  31. OPTS="--root=$D"
  32. fi
  33. if [ "${SYSTEMD_AUTO_ENABLE}" = "enable" ]; then
  34. for service in ${@systemd_filter_services("${SYSTEMD_SERVICE_ESCAPED}", False, d)}; do
  35. systemctl ${OPTS} enable "$service"
  36. done
  37. for service in ${@systemd_filter_services("${SYSTEMD_SERVICE_ESCAPED}", True, d)}; do
  38. systemctl --global ${OPTS} enable "$service"
  39. done
  40. fi
  41. if [ -z "$D" ] && systemctl >/dev/null 2>/dev/null; then
  42. # Reload only system service manager
  43. # --global for daemon-reload is not supported: https://github.com/systemd/systemd/issues/19284
  44. systemctl daemon-reload
  45. [ -n "${@systemd_filter_services("${SYSTEMD_SERVICE_ESCAPED}", False, d)}" ] && \
  46. systemctl preset ${@systemd_filter_services("${SYSTEMD_SERVICE_ESCAPED}", False, d)}
  47. [ -n "${@systemd_filter_services("${SYSTEMD_SERVICE_ESCAPED}", True, d)}" ] && \
  48. systemctl --global preset ${@systemd_filter_services("${SYSTEMD_SERVICE_ESCAPED}", True, d)}
  49. if [ "${SYSTEMD_AUTO_ENABLE}" = "enable" ]; then
  50. # --global flag for restart is not supported by systemd (see above)
  51. [ -n "${@systemd_filter_services("${SYSTEMD_SERVICE_ESCAPED}", False, d)}" ] && \
  52. systemctl --no-block restart ${@systemd_filter_services("${SYSTEMD_SERVICE_ESCAPED}", False, d)}
  53. fi
  54. fi
  55. fi
  56. }
  57. systemd_prerm() {
  58. if type systemctl >/dev/null 2>/dev/null; then
  59. if [ -z "$D" ] && systemctl >/dev/null 2>/dev/null; then
  60. if [ -n "${@systemd_filter_services("${SYSTEMD_SERVICE_ESCAPED}", False, d)}" ]; then
  61. systemctl stop ${@systemd_filter_services("${SYSTEMD_SERVICE_ESCAPED}", False, d)}
  62. systemctl disable ${@systemd_filter_services("${SYSTEMD_SERVICE_ESCAPED}", False, d)}
  63. fi
  64. # same as above, --global flag is not supported for stop so do disable only
  65. if [ -n "${@systemd_filter_services("${SYSTEMD_SERVICE_ESCAPED}", True, d)}" ]; then
  66. systemctl --global disable ${@systemd_filter_services("${SYSTEMD_SERVICE_ESCAPED}", True, d)}
  67. fi
  68. fi
  69. fi
  70. }
  71. systemd_populate_packages[vardeps] += "systemd_prerm systemd_postinst"
  72. systemd_populate_packages[vardepsexclude] += "OVERRIDES"
  73. def systemd_service_path(service, searchpaths, d):
  74. path_found = ''
  75. # Deal with adding, for example, 'ifplugd@eth0.service' from
  76. # 'ifplugd@.service'
  77. base = None
  78. at = service.find('@')
  79. if at != -1:
  80. ext = service.rfind('.')
  81. base = service[:at] + '@' + service[ext:]
  82. for path in searchpaths:
  83. if os.path.lexists(oe.path.join(d.getVar("D"), path, service)):
  84. path_found = path
  85. break
  86. elif base is not None:
  87. if os.path.exists(oe.path.join(d.getVar("D"), path, base)):
  88. path_found = path
  89. break
  90. return path_found, base
  91. def systemd_service_searchpaths(user, d):
  92. if user:
  93. return [
  94. oe.path.join(d.getVar("sysconfdir"), "systemd", "user"),
  95. d.getVar("systemd_user_unitdir"),
  96. ]
  97. else:
  98. return [
  99. oe.path.join(d.getVar("sysconfdir"), "systemd", "system"),
  100. d.getVar("systemd_system_unitdir"),
  101. ]
  102. def systemd_service_exists(service, user, d):
  103. searchpaths = systemd_service_searchpaths(user, d)
  104. path, _ = systemd_service_path(service, searchpaths, d)
  105. return path != ''
  106. def systemd_filter_services(services, user, d):
  107. return ' '.join(service for service in services.split() if systemd_service_exists(service, user, d))
  108. python systemd_populate_packages() {
  109. import re
  110. import shlex
  111. if not bb.utils.contains('DISTRO_FEATURES', 'systemd', True, False, d):
  112. return
  113. def get_package_var(d, var, pkg):
  114. val = (d.getVar('%s:%s' % (var, pkg)) or "").strip()
  115. if val == "":
  116. val = (d.getVar(var) or "").strip()
  117. return val
  118. # Check if systemd-packages already included in PACKAGES
  119. def systemd_check_package(pkg_systemd):
  120. packages = d.getVar('PACKAGES')
  121. if not pkg_systemd in packages.split():
  122. bb.error('%s is marked for packaging systemd scripts, but it does not appear in package list, please add it to PACKAGES or adjust SYSTEMD_PACKAGES accordingly' % pkg_systemd)
  123. def systemd_generate_package_scripts(pkg):
  124. bb.debug(1, 'adding systemd calls to postinst/postrm for %s' % pkg)
  125. paths_escaped = ' '.join(shlex.quote(s) for s in d.getVar('SYSTEMD_SERVICE:' + pkg).split())
  126. d.setVar('SYSTEMD_SERVICE_ESCAPED:' + pkg, paths_escaped)
  127. # Add pkg to the overrides so that it finds the SYSTEMD_SERVICE:pkg
  128. # variable.
  129. localdata = d.createCopy()
  130. localdata.prependVar("OVERRIDES", pkg + ":")
  131. postinst = d.getVar('pkg_postinst:%s' % pkg)
  132. if not postinst:
  133. postinst = '#!/bin/sh\n'
  134. postinst += localdata.getVar('systemd_postinst')
  135. d.setVar('pkg_postinst:%s' % pkg, postinst)
  136. prerm = d.getVar('pkg_prerm:%s' % pkg)
  137. if not prerm:
  138. prerm = '#!/bin/sh\n'
  139. prerm += localdata.getVar('systemd_prerm')
  140. d.setVar('pkg_prerm:%s' % pkg, prerm)
  141. # Add files to FILES:*-systemd if existent and not already done
  142. def systemd_append_file(pkg_systemd, file_append):
  143. appended = False
  144. if os.path.exists(oe.path.join(d.getVar("D"), file_append)):
  145. var_name = "FILES:" + pkg_systemd
  146. files = d.getVar(var_name, False) or ""
  147. if file_append not in files.split():
  148. d.appendVar(var_name, " " + file_append)
  149. appended = True
  150. return appended
  151. # Add systemd files to FILES:*-systemd, parse for Also= and follow recursive
  152. def systemd_add_files_and_parse(pkg_systemd, path, service):
  153. # avoid infinite recursion
  154. if systemd_append_file(pkg_systemd, oe.path.join(path, service)):
  155. fullpath = oe.path.join(d.getVar("D"), path, service)
  156. if service.find('.service') != -1:
  157. # for *.service add *@.service
  158. service_base = service.replace('.service', '')
  159. systemd_add_files_and_parse(pkg_systemd, path, service_base + '@.service')
  160. # Add the socket unit which is referred by the Also= in this service file to the same package.
  161. with open(fullpath, 'r') as unit_f:
  162. for line in unit_f:
  163. if line.startswith('Also'):
  164. also_unit = line.split('=', 1)[1].strip()
  165. if also_unit.find('.socket') != -1:
  166. systemd_add_files_and_parse(pkg_systemd, path, also_unit)
  167. if service.find('.socket') != -1:
  168. # for *.socket add *.service and *@.service
  169. service_base = service.replace('.socket', '')
  170. systemd_add_files_and_parse(pkg_systemd, path, service_base + '.service')
  171. systemd_add_files_and_parse(pkg_systemd, path, service_base + '@.service')
  172. # Check service-files and call systemd_add_files_and_parse for each entry
  173. def systemd_check_services():
  174. searchpaths = systemd_service_searchpaths(False, d)
  175. searchpaths.extend(systemd_service_searchpaths(True, d))
  176. systemd_packages = d.getVar('SYSTEMD_PACKAGES')
  177. # scan for all in SYSTEMD_SERVICE[]
  178. for pkg_systemd in systemd_packages.split():
  179. for service in get_package_var(d, 'SYSTEMD_SERVICE', pkg_systemd).split():
  180. path_found, base = systemd_service_path(service, searchpaths, d)
  181. if path_found != '':
  182. systemd_add_files_and_parse(pkg_systemd, path_found, service)
  183. else:
  184. bb.fatal("Didn't find service unit '{0}', specified in SYSTEMD_SERVICE:{1}. {2}".format(
  185. service, pkg_systemd, "Also looked for service unit '{0}'.".format(base) if base is not None else ""))
  186. def systemd_create_presets(pkg, action, user):
  187. import re
  188. # Check there is at least one service of given type (system/user), don't
  189. # create empty files.
  190. needs_preset = False
  191. for service in d.getVar('SYSTEMD_SERVICE:%s' % pkg).split():
  192. if systemd_service_exists(service, user, d):
  193. needs_preset = True
  194. break
  195. if not needs_preset:
  196. return
  197. prefix = "user" if user else "system"
  198. presetf = oe.path.join(d.getVar("PKGD"), d.getVar("systemd_unitdir"), "%s-preset/98-%s.preset" % (prefix, pkg))
  199. bb.utils.mkdirhier(os.path.dirname(presetf))
  200. with open(presetf, 'a') as fd:
  201. template_services = {}
  202. for service in d.getVar('SYSTEMD_SERVICE:%s' % pkg).split():
  203. if not systemd_service_exists(service, user, d):
  204. continue
  205. if '@' in service and '@.' not in service:
  206. (servicename, instance, service_type) = re.split('[@.]', service)
  207. template_services.setdefault(servicename + '@.' + service_type, []).append(instance)
  208. else:
  209. template_services.setdefault(service, [])
  210. for template, instances in template_services.items():
  211. if instances:
  212. fd.write("%s %s %s\n" % (action, template, ' '.join(instances)))
  213. else:
  214. fd.write("%s %s\n" % (action, template))
  215. d.appendVar("FILES:%s" % pkg, ' ' + oe.path.join(d.getVar("systemd_unitdir"), "%s-preset/98-%s.preset" % (prefix, pkg)))
  216. # Run all modifications once when creating package
  217. if os.path.exists(d.getVar("D")):
  218. for pkg in d.getVar('SYSTEMD_PACKAGES').split():
  219. systemd_check_package(pkg)
  220. if d.getVar('SYSTEMD_SERVICE:' + pkg):
  221. systemd_generate_package_scripts(pkg)
  222. action = get_package_var(d, 'SYSTEMD_AUTO_ENABLE', pkg)
  223. if action in ("enable", "disable"):
  224. systemd_create_presets(pkg, action, False)
  225. systemd_create_presets(pkg, action, True)
  226. elif action not in ("mask", "preset"):
  227. bb.fatal("SYSTEMD_AUTO_ENABLE:%s '%s' is not 'enable', 'disable', 'mask' or 'preset'" % (pkg, action))
  228. systemd_check_services()
  229. }
  230. PACKAGESPLITFUNCS =+ "systemd_populate_packages"
  231. rm_systemd_unitdir() {
  232. rm -rf ${D}${systemd_unitdir}
  233. # Change into ${D} and use a relative path with rmdir -p to avoid
  234. # having it remove ${D} if it becomes empty.
  235. (cd ${D} && rmdir -p $(dirname ${systemd_unitdir#/}) 2>/dev/null || :)
  236. }
  237. rm_sysvinit_initddir() {
  238. local sysv_initddir=${INIT_D_DIR}
  239. : ${sysv_initddir:=${sysconfdir}/init.d}
  240. # If systemd_system_unitdir contains anything, delete sysv_initddir
  241. if [ "$(ls -A ${D}${systemd_system_unitdir} 2>/dev/null)" ]; then
  242. rm -rf ${D}$sysv_initddir
  243. rmdir -p $(dirname ${D}$sysv_initddir) 2>/dev/null || :
  244. fi
  245. }
  246. do_install[postfuncs] += "${RMINITDIR}"
  247. RMINITDIR = " \
  248. ${@bb.utils.contains('DISTRO_FEATURES', 'systemd', '', 'rm_systemd_unitdir', d)} \
  249. ${@'rm_sysvinit_initddir' if bb.utils.contains('DISTRO_FEATURES', 'systemd', True, False, d) and \
  250. not bb.utils.contains('DISTRO_FEATURES', 'sysvinit', True, False, d) else ''} \
  251. "
  252. RMINITDIR:class-native = ""