diff options
author | Richard Purdie <richard.purdie@linuxfoundation.org> | 2022-08-10 14:35:29 +0100 |
---|---|---|
committer | Richard Purdie <richard.purdie@linuxfoundation.org> | 2022-08-12 15:27:17 +0100 |
commit | fd1517e2b51a170f2427122c6b95396db251d827 (patch) | |
tree | dabfe3e631339c2fc99a9ee7febb0f9c128e325e /meta/classes-recipe/systemd.bbclass | |
parent | 10317912ee319ccf7f83605d438b5cbf9663f296 (diff) | |
download | poky-fd1517e2b51a170f2427122c6b95396db251d827.tar.gz |
classes: Update classes to match new bitbake class scope functionality
Move classes to classes-global or classes-recipe as appropriate to take
advantage of new bitbake functionality to check class scope/usage.
(From OE-Core rev: f5c128008365e141082c129417eb72d2751e8045)
Signed-off-by: Richard Purdie <richard.purdie@linuxfoundation.org>
Diffstat (limited to 'meta/classes-recipe/systemd.bbclass')
-rw-r--r-- | meta/classes-recipe/systemd.bbclass | 239 |
1 files changed, 239 insertions, 0 deletions
diff --git a/meta/classes-recipe/systemd.bbclass b/meta/classes-recipe/systemd.bbclass new file mode 100644 index 0000000000..f6564c2b31 --- /dev/null +++ b/meta/classes-recipe/systemd.bbclass | |||
@@ -0,0 +1,239 @@ | |||
1 | # | ||
2 | # Copyright OpenEmbedded Contributors | ||
3 | # | ||
4 | # SPDX-License-Identifier: MIT | ||
5 | # | ||
6 | |||
7 | # The list of packages that should have systemd packaging scripts added. For | ||
8 | # each entry, optionally have a SYSTEMD_SERVICE:[package] that lists the service | ||
9 | # files in this package. If this variable isn't set, [package].service is used. | ||
10 | SYSTEMD_PACKAGES ?= "${PN}" | ||
11 | SYSTEMD_PACKAGES:class-native ?= "" | ||
12 | SYSTEMD_PACKAGES:class-nativesdk ?= "" | ||
13 | |||
14 | # Whether to enable or disable the services on installation. | ||
15 | SYSTEMD_AUTO_ENABLE ??= "enable" | ||
16 | |||
17 | # This class will be included in any recipe that supports systemd init scripts, | ||
18 | # even if systemd is not in DISTRO_FEATURES. As such don't make any changes | ||
19 | # directly but check the DISTRO_FEATURES first. | ||
20 | python __anonymous() { | ||
21 | # If the distro features have systemd but not sysvinit, inhibit update-rcd | ||
22 | # from doing any work so that pure-systemd images don't have redundant init | ||
23 | # files. | ||
24 | if bb.utils.contains('DISTRO_FEATURES', 'systemd', True, False, d): | ||
25 | d.appendVar("DEPENDS", " systemd-systemctl-native") | ||
26 | d.appendVar("PACKAGE_WRITE_DEPS", " systemd-systemctl-native") | ||
27 | if not bb.utils.contains('DISTRO_FEATURES', 'sysvinit', True, False, d): | ||
28 | d.setVar("INHIBIT_UPDATERCD_BBCLASS", "1") | ||
29 | } | ||
30 | |||
31 | systemd_postinst() { | ||
32 | if systemctl >/dev/null 2>/dev/null; then | ||
33 | OPTS="" | ||
34 | |||
35 | if [ -n "$D" ]; then | ||
36 | OPTS="--root=$D" | ||
37 | fi | ||
38 | |||
39 | if [ "${SYSTEMD_AUTO_ENABLE}" = "enable" ]; then | ||
40 | for service in ${SYSTEMD_SERVICE_ESCAPED}; do | ||
41 | systemctl ${OPTS} enable "$service" | ||
42 | done | ||
43 | fi | ||
44 | |||
45 | if [ -z "$D" ]; then | ||
46 | systemctl daemon-reload | ||
47 | systemctl preset ${SYSTEMD_SERVICE_ESCAPED} | ||
48 | |||
49 | if [ "${SYSTEMD_AUTO_ENABLE}" = "enable" ]; then | ||
50 | systemctl --no-block restart ${SYSTEMD_SERVICE_ESCAPED} | ||
51 | fi | ||
52 | fi | ||
53 | fi | ||
54 | } | ||
55 | |||
56 | systemd_prerm() { | ||
57 | if systemctl >/dev/null 2>/dev/null; then | ||
58 | if [ -z "$D" ]; then | ||
59 | systemctl stop ${SYSTEMD_SERVICE_ESCAPED} | ||
60 | |||
61 | systemctl disable ${SYSTEMD_SERVICE_ESCAPED} | ||
62 | fi | ||
63 | fi | ||
64 | } | ||
65 | |||
66 | |||
67 | systemd_populate_packages[vardeps] += "systemd_prerm systemd_postinst" | ||
68 | systemd_populate_packages[vardepsexclude] += "OVERRIDES" | ||
69 | |||
70 | |||
71 | python systemd_populate_packages() { | ||
72 | import re | ||
73 | import shlex | ||
74 | |||
75 | if not bb.utils.contains('DISTRO_FEATURES', 'systemd', True, False, d): | ||
76 | return | ||
77 | |||
78 | def get_package_var(d, var, pkg): | ||
79 | val = (d.getVar('%s:%s' % (var, pkg)) or "").strip() | ||
80 | if val == "": | ||
81 | val = (d.getVar(var) or "").strip() | ||
82 | return val | ||
83 | |||
84 | # Check if systemd-packages already included in PACKAGES | ||
85 | def systemd_check_package(pkg_systemd): | ||
86 | packages = d.getVar('PACKAGES') | ||
87 | if not pkg_systemd in packages.split(): | ||
88 | bb.error('%s does not appear in package list, please add it' % pkg_systemd) | ||
89 | |||
90 | |||
91 | def systemd_generate_package_scripts(pkg): | ||
92 | bb.debug(1, 'adding systemd calls to postinst/postrm for %s' % pkg) | ||
93 | |||
94 | paths_escaped = ' '.join(shlex.quote(s) for s in d.getVar('SYSTEMD_SERVICE:' + pkg).split()) | ||
95 | d.setVar('SYSTEMD_SERVICE_ESCAPED:' + pkg, paths_escaped) | ||
96 | |||
97 | # Add pkg to the overrides so that it finds the SYSTEMD_SERVICE:pkg | ||
98 | # variable. | ||
99 | localdata = d.createCopy() | ||
100 | localdata.prependVar("OVERRIDES", pkg + ":") | ||
101 | |||
102 | postinst = d.getVar('pkg_postinst:%s' % pkg) | ||
103 | if not postinst: | ||
104 | postinst = '#!/bin/sh\n' | ||
105 | postinst += localdata.getVar('systemd_postinst') | ||
106 | d.setVar('pkg_postinst:%s' % pkg, postinst) | ||
107 | |||
108 | prerm = d.getVar('pkg_prerm:%s' % pkg) | ||
109 | if not prerm: | ||
110 | prerm = '#!/bin/sh\n' | ||
111 | prerm += localdata.getVar('systemd_prerm') | ||
112 | d.setVar('pkg_prerm:%s' % pkg, prerm) | ||
113 | |||
114 | |||
115 | # Add files to FILES:*-systemd if existent and not already done | ||
116 | def systemd_append_file(pkg_systemd, file_append): | ||
117 | appended = False | ||
118 | if os.path.exists(oe.path.join(d.getVar("D"), file_append)): | ||
119 | var_name = "FILES:" + pkg_systemd | ||
120 | files = d.getVar(var_name, False) or "" | ||
121 | if file_append not in files.split(): | ||
122 | d.appendVar(var_name, " " + file_append) | ||
123 | appended = True | ||
124 | return appended | ||
125 | |||
126 | # Add systemd files to FILES:*-systemd, parse for Also= and follow recursive | ||
127 | def systemd_add_files_and_parse(pkg_systemd, path, service, keys): | ||
128 | # avoid infinite recursion | ||
129 | if systemd_append_file(pkg_systemd, oe.path.join(path, service)): | ||
130 | fullpath = oe.path.join(d.getVar("D"), path, service) | ||
131 | if service.find('.service') != -1: | ||
132 | # for *.service add *@.service | ||
133 | service_base = service.replace('.service', '') | ||
134 | systemd_add_files_and_parse(pkg_systemd, path, service_base + '@.service', keys) | ||
135 | if service.find('.socket') != -1: | ||
136 | # for *.socket add *.service and *@.service | ||
137 | service_base = service.replace('.socket', '') | ||
138 | systemd_add_files_and_parse(pkg_systemd, path, service_base + '.service', keys) | ||
139 | systemd_add_files_and_parse(pkg_systemd, path, service_base + '@.service', keys) | ||
140 | for key in keys.split(): | ||
141 | # recurse all dependencies found in keys ('Also';'Conflicts';..) and add to files | ||
142 | cmd = "grep %s %s | sed 's,%s=,,g' | tr ',' '\\n'" % (key, shlex.quote(fullpath), key) | ||
143 | pipe = os.popen(cmd, 'r') | ||
144 | line = pipe.readline() | ||
145 | while line: | ||
146 | line = line.replace('\n', '') | ||
147 | systemd_add_files_and_parse(pkg_systemd, path, line, keys) | ||
148 | line = pipe.readline() | ||
149 | pipe.close() | ||
150 | |||
151 | # Check service-files and call systemd_add_files_and_parse for each entry | ||
152 | def systemd_check_services(): | ||
153 | searchpaths = [oe.path.join(d.getVar("sysconfdir"), "systemd", "system"),] | ||
154 | searchpaths.append(d.getVar("systemd_system_unitdir")) | ||
155 | systemd_packages = d.getVar('SYSTEMD_PACKAGES') | ||
156 | |||
157 | keys = 'Also' | ||
158 | # scan for all in SYSTEMD_SERVICE[] | ||
159 | for pkg_systemd in systemd_packages.split(): | ||
160 | for service in get_package_var(d, 'SYSTEMD_SERVICE', pkg_systemd).split(): | ||
161 | path_found = '' | ||
162 | |||
163 | # Deal with adding, for example, 'ifplugd@eth0.service' from | ||
164 | # 'ifplugd@.service' | ||
165 | base = None | ||
166 | at = service.find('@') | ||
167 | if at != -1: | ||
168 | ext = service.rfind('.') | ||
169 | base = service[:at] + '@' + service[ext:] | ||
170 | |||
171 | for path in searchpaths: | ||
172 | if os.path.exists(oe.path.join(d.getVar("D"), path, service)): | ||
173 | path_found = path | ||
174 | break | ||
175 | elif base is not None: | ||
176 | if os.path.exists(oe.path.join(d.getVar("D"), path, base)): | ||
177 | path_found = path | ||
178 | break | ||
179 | |||
180 | if path_found != '': | ||
181 | systemd_add_files_and_parse(pkg_systemd, path_found, service, keys) | ||
182 | else: | ||
183 | bb.fatal("Didn't find service unit '{0}', specified in SYSTEMD_SERVICE:{1}. {2}".format( | ||
184 | service, pkg_systemd, "Also looked for service unit '{0}'.".format(base) if base is not None else "")) | ||
185 | |||
186 | def systemd_create_presets(pkg, action): | ||
187 | presetf = oe.path.join(d.getVar("PKGD"), d.getVar("systemd_unitdir"), "system-preset/98-%s.preset" % pkg) | ||
188 | bb.utils.mkdirhier(os.path.dirname(presetf)) | ||
189 | with open(presetf, 'a') as fd: | ||
190 | for service in d.getVar('SYSTEMD_SERVICE:%s' % pkg).split(): | ||
191 | fd.write("%s %s\n" % (action,service)) | ||
192 | d.appendVar("FILES:%s" % pkg, ' ' + oe.path.join(d.getVar("systemd_unitdir"), "system-preset/98-%s.preset" % pkg)) | ||
193 | |||
194 | # Run all modifications once when creating package | ||
195 | if os.path.exists(d.getVar("D")): | ||
196 | for pkg in d.getVar('SYSTEMD_PACKAGES').split(): | ||
197 | systemd_check_package(pkg) | ||
198 | if d.getVar('SYSTEMD_SERVICE:' + pkg): | ||
199 | systemd_generate_package_scripts(pkg) | ||
200 | action = get_package_var(d, 'SYSTEMD_AUTO_ENABLE', pkg) | ||
201 | if action in ("enable", "disable"): | ||
202 | systemd_create_presets(pkg, action) | ||
203 | elif action not in ("mask", "preset"): | ||
204 | bb.fatal("SYSTEMD_AUTO_ENABLE:%s '%s' is not 'enable', 'disable', 'mask' or 'preset'" % (pkg, action)) | ||
205 | systemd_check_services() | ||
206 | } | ||
207 | |||
208 | PACKAGESPLITFUNCS:prepend = "systemd_populate_packages " | ||
209 | |||
210 | python rm_systemd_unitdir (){ | ||
211 | import shutil | ||
212 | if not bb.utils.contains('DISTRO_FEATURES', 'systemd', True, False, d): | ||
213 | systemd_unitdir = oe.path.join(d.getVar("D"), d.getVar('systemd_unitdir')) | ||
214 | if os.path.exists(systemd_unitdir): | ||
215 | shutil.rmtree(systemd_unitdir) | ||
216 | systemd_libdir = os.path.dirname(systemd_unitdir) | ||
217 | if (os.path.exists(systemd_libdir) and not os.listdir(systemd_libdir)): | ||
218 | os.rmdir(systemd_libdir) | ||
219 | } | ||
220 | |||
221 | python rm_sysvinit_initddir (){ | ||
222 | import shutil | ||
223 | sysv_initddir = oe.path.join(d.getVar("D"), (d.getVar('INIT_D_DIR') or "/etc/init.d")) | ||
224 | |||
225 | if bb.utils.contains('DISTRO_FEATURES', 'systemd', True, False, d) and \ | ||
226 | not bb.utils.contains('DISTRO_FEATURES', 'sysvinit', True, False, d) and \ | ||
227 | os.path.exists(sysv_initddir): | ||
228 | systemd_system_unitdir = oe.path.join(d.getVar("D"), d.getVar('systemd_system_unitdir')) | ||
229 | |||
230 | # If systemd_system_unitdir contains anything, delete sysv_initddir | ||
231 | if (os.path.exists(systemd_system_unitdir) and os.listdir(systemd_system_unitdir)): | ||
232 | shutil.rmtree(sysv_initddir) | ||
233 | } | ||
234 | |||
235 | do_install[postfuncs] += "${RMINITDIR} " | ||
236 | RMINITDIR:class-target = " rm_sysvinit_initddir rm_systemd_unitdir " | ||
237 | RMINITDIR:class-nativesdk = " rm_sysvinit_initddir rm_systemd_unitdir " | ||
238 | RMINITDIR = "" | ||
239 | |||