hc
2024-10-12 a5969cabbb4660eab42b6ef0412cbbd1200cf14d
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
SUMMARY = "Tiny versions of many common UNIX utilities in a single small executable"
DESCRIPTION = "BusyBox combines tiny versions of many common UNIX utilities into a single small executable. It provides minimalist replacements for most of the utilities you usually find in GNU fileutils, shellutils, etc. The utilities in BusyBox generally have fewer options than their full-featured GNU cousins; however, the options that are included provide the expected functionality and behave very much like their GNU counterparts. BusyBox provides a fairly complete POSIX environment for any small or embedded system."
HOMEPAGE = "https://www.busybox.net"
BUGTRACKER = "https://bugs.busybox.net/"
 
DEPENDS += "kern-tools-native virtual/crypt"
 
# bzip2 applet in busybox is based on lightly-modified bzip2-1.0.4 source
# the GPL is version 2 only
LICENSE = "GPLv2 & bzip2-1.0.4"
LIC_FILES_CHKSUM = "file://LICENSE;md5=de10de48642ab74318e893a61105afbb \
                    file://archival/libarchive/bz/LICENSE;md5=28e3301eae987e8cfe19988e98383dae"
 
SECTION = "base"
 
# Whether to split the suid apps into a seperate binary
BUSYBOX_SPLIT_SUID ?= "1"
 
export EXTRA_CFLAGS = "${CFLAGS}"
export EXTRA_LDFLAGS = "${LDFLAGS}"
 
EXTRA_OEMAKE = "CC='${CC}' LD='${CCLD}' V=1 ARCH=${TARGET_ARCH} CROSS_COMPILE=${TARGET_PREFIX} SKIP_STRIP=y HOSTCC='${BUILD_CC}' HOSTCPP='${BUILD_CPP}'"
 
PACKAGES =+ "${PN}-httpd ${PN}-udhcpd ${PN}-udhcpc ${PN}-syslog ${PN}-mdev ${PN}-hwclock"
 
FILES:${PN}-httpd = "${sysconfdir}/init.d/busybox-httpd /srv/www"
FILES:${PN}-syslog = "${sysconfdir}/init.d/syslog* ${sysconfdir}/syslog-startup.conf* ${sysconfdir}/syslog.conf* ${systemd_system_unitdir}/syslog.service ${sysconfdir}/default/busybox-syslog"
FILES:${PN}-mdev = "${sysconfdir}/init.d/mdev ${sysconfdir}/mdev.conf ${sysconfdir}/mdev/*"
FILES:${PN}-udhcpd = "${sysconfdir}/init.d/busybox-udhcpd"
FILES:${PN}-udhcpc = "${sysconfdir}/udhcpc.d ${datadir}/udhcpc"
FILES:${PN}-hwclock = "${sysconfdir}/init.d/hwclock.sh"
 
INITSCRIPT_PACKAGES = "${PN}-httpd ${PN}-syslog ${PN}-udhcpd ${PN}-mdev ${PN}-hwclock"
 
INITSCRIPT_NAME:${PN}-httpd = "busybox-httpd"
INITSCRIPT_NAME:${PN}-hwclock = "hwclock.sh"
INITSCRIPT_NAME:${PN}-mdev = "mdev"
INITSCRIPT_PARAMS:${PN}-mdev = "start 04 S ."
INITSCRIPT_NAME:${PN}-syslog = "syslog"
INITSCRIPT_NAME:${PN}-udhcpd = "busybox-udhcpd"
 
SYSTEMD_PACKAGES = "${PN}-syslog"
SYSTEMD_SERVICE:${PN}-syslog = "${@bb.utils.contains('SRC_URI', 'file://syslog.cfg', 'busybox-syslog.service', '', d)}"
 
RDEPENDS:${PN}-syslog = "busybox"
CONFFILES:${PN}-syslog = "${sysconfdir}/syslog-startup.conf"
RCONFLICTS:${PN}-syslog = "rsyslog sysklogd syslog-ng"
 
CONFFILES:${PN}-mdev = "${sysconfdir}/mdev.conf"
 
RRECOMMENDS:${PN} = "${PN}-udhcpc"
 
RDEPENDS:${PN} = "${@["", "busybox-inittab"][(d.getVar('VIRTUAL-RUNTIME_init_manager') == 'busybox')]}"
 
inherit cml1 systemd update-rc.d ptest
 
# busybox's unzip test case needs zip command, which busybox itself does not provide
RDEPENDS:${PN}-ptest = "zip"
 
# internal helper
def busybox_cfg(feature, tokens, cnf, rem):
    if type(tokens) == type(""):
        tokens = [tokens]
    rem.extend(['/^[# ]*' + token + '[ =]/d' for token in tokens])
    if feature:
        cnf.extend([token + '=y' for token in tokens])
    else:
        cnf.extend(['# ' + token + ' is not set' for token in tokens])
 
# Map distro features to config settings
def features_to_busybox_settings(d):
    cnf, rem = ([], [])
    busybox_cfg(bb.utils.contains('DISTRO_FEATURES', 'ipv6', True, False, d), 'CONFIG_FEATURE_IPV6', cnf, rem)
    busybox_cfg(True, 'CONFIG_LFS', cnf, rem)
    busybox_cfg(True, 'CONFIG_FDISK_SUPPORT_LARGE_DISKS', cnf, rem)
    busybox_cfg(bb.utils.contains('DISTRO_FEATURES', 'nls', True, False, d), 'CONFIG_LOCALE_SUPPORT', cnf, rem)
    busybox_cfg(bb.utils.contains('DISTRO_FEATURES', 'ipv4', True, False, d), 'CONFIG_FEATURE_IFUPDOWN_IPV4', cnf, rem)
    busybox_cfg(bb.utils.contains('DISTRO_FEATURES', 'ipv6', True, False, d), 'CONFIG_FEATURE_IFUPDOWN_IPV6', cnf, rem)
    busybox_cfg(bb.utils.contains_any('DISTRO_FEATURES', 'bluetooth wifi', True, False, d), 'CONFIG_RFKILL', cnf, rem)
    return "\n".join(cnf), "\n".join(rem)
 
# X, Y = ${@features_to_busybox_settings(d)}
# unfortunately doesn't seem to work with bitbake, workaround:
def features_to_busybox_conf(d):
    cnf, rem = features_to_busybox_settings(d)
    return cnf
def features_to_busybox_del(d):
    cnf, rem = features_to_busybox_settings(d)
    return rem
 
configmangle = '/CONFIG_EXTRA_CFLAGS/d; \
       '
OE_FEATURES := "${@features_to_busybox_conf(d)}"
OE_DEL      := "${@features_to_busybox_del(d)}"
DO_IPv4 := "${@bb.utils.contains('DISTRO_FEATURES', 'ipv4', 1, 0, d)}"
DO_IPv6 := "${@bb.utils.contains('DISTRO_FEATURES', 'ipv6', 1, 0, d)}"
 
python () {
  if "${OE_DEL}":
    d.setVar('configmangle:append', "${OE_DEL}" + "\n")
  if "${OE_FEATURES}":
    d.setVar('configmangle:append',
                   "/^### DISTRO FEATURES$/a\\\n%s\n\n" %
                   ("\\n".join((d.expand("${OE_FEATURES}").split("\n")))))
  d.setVar('configmangle:append',
                 "/^### CROSS$/a\\\n%s\n" %
                  ("\\n".join(["CONFIG_EXTRA_CFLAGS=\"${CFLAGS} ${HOST_CC_ARCH}\""
                        ])
                  ))
}
 
do_prepare_config () {
   if [ "${BUILD_REPRODUCIBLE_BINARIES}" = "1" ]; then
       export KCONFIG_NOTIMESTAMP=1
   fi
   sed -e '/CONFIG_STATIC/d' \
       < ${WORKDIR}/defconfig > ${S}/.config
   echo "# CONFIG_STATIC is not set" >> .config
   for i in 'CROSS' 'DISTRO FEATURES'; do echo "### $i"; done >> \
       ${S}/.config
   sed -i -e '${configmangle}' ${S}/.config
   if test ${DO_IPv4} -eq 0 && test ${DO_IPv6} -eq 0; then
       # disable networking applets
       mv ${S}/.config ${S}/.config.oe-tmp
       awk 'BEGIN{net=0}
       /^# Networking Utilities/{net=1}
       /^#$/{if(net){net=net+1}}
       {if(net==2&&$0 !~ /^#/&&$1){print("# "$1" is not set")}else{print}}' \
       ${S}/.config.oe-tmp > ${S}/.config
   fi
   sed -i 's/CONFIG_IFUPDOWN_UDHCPC_CMD_OPTIONS="-R -n"/CONFIG_IFUPDOWN_UDHCPC_CMD_OPTIONS="-R -b"/' ${S}/.config
   if [ -n "${DEBUG_PREFIX_MAP}" ]; then
       sed -i 's|${DEBUG_PREFIX_MAP}||g' ${S}/.config
   fi
}
 
do_configure () {
   set -x
   do_prepare_config
   merge_config.sh -m .config ${@" ".join(find_cfgs(d))}
   cml1_do_configure
}
 
do_compile() {
   unset CFLAGS CPPFLAGS CXXFLAGS LDFLAGS
   if [ "${BUILD_REPRODUCIBLE_BINARIES}" = "1" ]; then
       export KCONFIG_NOTIMESTAMP=1
   fi
   if [ "${BUSYBOX_SPLIT_SUID}" = "1" -a x`grep "CONFIG_FEATURE_INDIVIDUAL=y" .config` = x ]; then
       # split the .config into two parts, and make two busybox binaries
       if [ -e .config.orig ]; then
           # Need to guard again an interrupted do_compile - restore any backup
           cp .config.orig .config
       fi
       cp .config .config.orig
       oe_runmake busybox.cfg.suid
       oe_runmake busybox.cfg.nosuid
 
       # workaround for suid bug 10346
       if ! grep -q "CONFIG_SH_IS_NONE" busybox.cfg.nosuid; then
           echo "CONFIG_SH_IS_NONE" >> busybox.cfg.suid
       fi
 
       for i in `cat busybox.cfg.suid busybox.cfg.nosuid`; do
           echo "# $i is not set" >> .config.disable.apps
       done
       merge_config.sh -m .config.orig .config.disable.apps
       cp .config .config.nonapps
       for s in suid nosuid; do
           cat busybox.cfg.$s | while read item; do
               grep -w "$item" .config.orig
           done > .config.app.$s
 
           # workaround for suid bug 10346
           if [ "$s" = "suid" ] ; then
               sed "s/.*CONFIG_SH_IS_NONE.*$/CONFIG_SH_IS_NONE=y/" -i .config.app.suid
           fi
 
           merge_config.sh -m .config.nonapps .config.app.$s
           oe_runmake busybox_unstripped
           mv busybox_unstripped busybox.$s
           oe_runmake busybox.links
           sort busybox.links > busybox.links.$s
           rm busybox.links
       done
 
       # hard fail if sh is being linked to the suid busybox (detects bug 10346)
       if grep -q -x "/bin/sh" busybox.links.suid; then
           bbfatal "busybox suid binary incorrectly provides /bin/sh"
       fi
 
       # copy .config.orig back to .config, because the install process may check this file
       cp .config.orig .config
       # cleanup
       rm .config.orig .config.app.suid .config.app.nosuid .config.disable.apps .config.nonapps
   else
       oe_runmake busybox_unstripped
       cp busybox_unstripped busybox
       oe_runmake busybox.links
   fi
}
 
do_install () {
   sed -i "s:^/bin/:BASE_BINDIR/:" busybox.links*
   sed -i "s:^/sbin/:BASE_SBINDIR/:" busybox.links*
   sed -i "s:^/usr/bin/:BINDIR/:" busybox.links*
   sed -i "s:^/usr/sbin/:SBINDIR/:" busybox.links*
 
   # Move arch/link to BINDIR to match coreutils
   sed -i "s:^BASE_BINDIR/arch:BINDIR/arch:" busybox.links*
   sed -i "s:^BASE_BINDIR/link:BINDIR/link:" busybox.links*
 
   sed -i "s:^BASE_BINDIR/:${base_bindir}/:" busybox.links*
   sed -i "s:^BASE_SBINDIR/:${base_sbindir}/:" busybox.links*
   sed -i "s:^BINDIR/:${bindir}/:" busybox.links*
   sed -i "s:^SBINDIR/:${sbindir}/:" busybox.links*
 
   install -d ${D}${sysconfdir}/init.d
 
   if ! grep -q "CONFIG_FEATURE_INDIVIDUAL=y" ${B}/.config; then
       # Install ${base_bindir}/busybox, and the ${base_bindir}/sh link so the postinst script
       # can run. Let update-alternatives handle the rest.
       install -d ${D}${base_bindir}
       if [ "${BUSYBOX_SPLIT_SUID}" = "1" ]; then
           install -m 4755 ${B}/busybox.suid ${D}${base_bindir}
           install -m 0755 ${B}/busybox.nosuid ${D}${base_bindir}
           install -m 0644 ${S}/busybox.links.suid ${D}${sysconfdir}
           install -m 0644 ${S}/busybox.links.nosuid ${D}${sysconfdir}
           if grep -q "CONFIG_SH_IS_ASH=y" ${B}/.config; then
               ln -sf busybox.nosuid ${D}${base_bindir}/sh
           fi
           # Keep a default busybox for people who want to invoke busybox directly.
           # This is also useful for the on device upgrade. Because we want
           # to use the busybox command in postinst.
           ln -sf busybox.nosuid ${D}${base_bindir}/busybox
       else
           if grep -q "CONFIG_FEATURE_SUID=y" ${B}/.config; then
               install -m 4755 ${B}/busybox ${D}${base_bindir}
           else
               install -m 0755 ${B}/busybox ${D}${base_bindir}
           fi
           install -m 0644 ${S}/busybox.links ${D}${sysconfdir}
           if grep -q "CONFIG_SH_IS_ASH=y" ${B}/.config; then
               ln -sf busybox ${D}${base_bindir}/sh
           fi
           # We make this symlink here to eliminate the error when upgrading together
           # with busybox-syslog. Without this symlink, the opkg may think of the
           # busybox.nosuid as obsolete and remove it, resulting in dead links like
           # ${base_bindir}/sed -> ${base_bindir}/busybox.nosuid. This will make upgrading busybox-syslog fail.
           # This symlink will be safely deleted in postinst, thus no negative effect.
           ln -sf busybox ${D}${base_bindir}/busybox.nosuid
       fi
   else
       install -d ${D}${base_bindir} ${D}${bindir} ${D}${libdir}
       cat busybox.links | while read FILE; do
           NAME=`basename "$FILE"`
           install -m 0755 "0_lib/$NAME" "${D}$FILE.${BPN}"
       done
       # add suid bit where needed
       for i in `grep -E "APPLET.*BB_SUID_((MAYBE|REQUIRE))" include/applets.h | grep -v _BB_SUID_DROP | cut -f 3 -d '(' | cut -f 1 -d ','`; do
           find ${D} -name $i.${BPN} -exec chmod a+s {} \;
       done
       install -m 0755 0_lib/libbusybox.so.${PV} ${D}${libdir}/libbusybox.so.${PV}
       ln -sf sh.${BPN} ${D}${base_bindir}/sh
       ln -sf ln.${BPN} ${D}${base_bindir}/ln
       ln -sf test.${BPN} ${D}${bindir}/test
       if [ -f ${D}/linuxrc.${BPN} ]; then
           mv ${D}/linuxrc.${BPN} ${D}/linuxrc
       fi
       install -m 0644 ${S}/busybox.links ${D}${sysconfdir}
   fi
 
   if grep -q "CONFIG_SYSLOGD=y" ${B}/.config; then
       install -m 0755 ${WORKDIR}/syslog ${D}${sysconfdir}/init.d/syslog
       install -m 644 ${WORKDIR}/syslog-startup.conf ${D}${sysconfdir}/syslog-startup.conf
       install -m 644 ${WORKDIR}/syslog.conf ${D}${sysconfdir}/syslog.conf
   fi
   if grep -q "CONFIG_CROND=y" ${B}/.config; then
       install -m 0755 ${WORKDIR}/busybox-cron ${D}${sysconfdir}/init.d/
   fi
   if grep -q "CONFIG_HTTPD=y" ${B}/.config; then
       install -m 0755 ${WORKDIR}/busybox-httpd ${D}${sysconfdir}/init.d/
       install -d ${D}/srv/www
   fi
   if grep -q "CONFIG_UDHCPD=y" ${B}/.config; then
       install -m 0755 ${WORKDIR}/busybox-udhcpd ${D}${sysconfdir}/init.d/
   fi
   if grep -q "CONFIG_HWCLOCK=y" ${B}/.config; then
       install -m 0755 ${WORKDIR}/hwclock.sh ${D}${sysconfdir}/init.d/
   fi
   if grep -q "CONFIG_UDHCPC=y" ${B}/.config; then
       install -d ${D}${sysconfdir}/udhcpc.d
       install -d ${D}${datadir}/udhcpc
       install -m 0755 ${WORKDIR}/simple.script ${D}${sysconfdir}/udhcpc.d/50default
       sed -i "s:/SBIN_DIR/:${base_sbindir}/:" ${D}${sysconfdir}/udhcpc.d/50default
       install -m 0755 ${WORKDIR}/default.script ${D}${datadir}/udhcpc/default.script
   fi
   if grep -q "CONFIG_INETD=y" ${B}/.config; then
       install -m 0755 ${WORKDIR}/inetd ${D}${sysconfdir}/init.d/inetd.${BPN}
       sed -i "s:/usr/sbin/:${sbindir}/:" ${D}${sysconfdir}/init.d/inetd.${BPN}
       install -m 0644 ${WORKDIR}/inetd.conf ${D}${sysconfdir}/
   fi
   if grep -q "CONFIG_MDEV=y" ${B}/.config; then
       install -m 0755 ${WORKDIR}/mdev ${D}${sysconfdir}/init.d/mdev
       if grep "CONFIG_FEATURE_MDEV_CONF=y" ${B}/.config; then
           install -m 644 ${WORKDIR}/mdev.conf ${D}${sysconfdir}/mdev.conf
           install -d ${D}${sysconfdir}/mdev
           install -m 0755 ${WORKDIR}/find-touchscreen.sh ${D}${sysconfdir}/mdev
           install -m 0755 ${WORKDIR}/mdev-mount.sh ${D}${sysconfdir}/mdev
       fi
   fi
   if grep -q "CONFIG_INIT=y" ${B}/.config && ${@bb.utils.contains('VIRTUAL-RUNTIME_init_manager','busybox','true','false',d)}; then
       install -D -m 0755 ${WORKDIR}/rcS ${D}${sysconfdir}/init.d/rcS
       install -D -m 0755 ${WORKDIR}/rcK ${D}${sysconfdir}/init.d/rcK
       install -D -m 0755 ${WORKDIR}/rcS.default ${D}${sysconfdir}/default/rcS
   fi
 
   if ${@bb.utils.contains('DISTRO_FEATURES','systemd','true','false',d)}; then
       if grep -q "CONFIG_KLOGD=y" ${B}/.config; then
           install -d ${D}${systemd_system_unitdir}
           sed 's,@base_sbindir@,${base_sbindir},g' < ${WORKDIR}/busybox-klogd.service.in \
           > ${D}${systemd_system_unitdir}/busybox-klogd.service
       fi
 
       if grep -q "CONFIG_SYSLOGD=y" ${B}/.config; then
           install -d ${D}${systemd_system_unitdir}
           sed 's,@base_sbindir@,${base_sbindir},g' < ${WORKDIR}/busybox-syslog.service.in \
           > ${D}${systemd_system_unitdir}/busybox-syslog.service
           if  [ ! -e ${D}${systemd_system_unitdir}/busybox-klogd.service ] ; then
               sed -i '/klog/d' ${D}${systemd_system_unitdir}/busybox-syslog.service
           fi
           if [ -f ${WORKDIR}/busybox-syslog.default ] ; then
               install -d ${D}${sysconfdir}/default
               install -m 0644 ${WORKDIR}/busybox-syslog.default ${D}${sysconfdir}/default/busybox-syslog
           fi
       fi
   fi
 
   # Remove the sysvinit specific configuration file for systemd systems to avoid confusion
   if ${@bb.utils.contains('DISTRO_FEATURES', 'sysvinit', 'false', 'true', d)}; then
       rm -f ${D}${sysconfdir}/syslog-startup.conf
   fi
}
 
PTEST_BINDIR = "1"
 
do_install_ptest () {
   cp -r ${B}/testsuite ${D}${PTEST_PATH}/
        # These access the internet which is not guaranteed to work on machines running the tests
        rm -rf ${D}${PTEST_PATH}/testsuite/wget
   sort ${B}/.config > ${D}${PTEST_PATH}/.config
   ln -s /bin/busybox   ${D}${PTEST_PATH}/busybox
}
 
inherit update-alternatives
 
ALTERNATIVE_PRIORITY = "50"
 
python do_package:prepend () {
    # We need to load the full set of busybox provides from the /etc/busybox.links
    # Use this to see the update-alternatives with the right information
 
    dvar = d.getVar('D')
    pn = d.getVar('PN')
    def set_alternative_vars(links, target):
        links = d.expand(links)
        target = d.expand(target)
        f = open('%s%s' % (dvar, links), 'r')
        for alt_link_name in f:
            alt_link_name = alt_link_name.strip()
            alt_name = os.path.basename(alt_link_name)
            # Match coreutils
            if alt_name == '[':
                alt_name = 'lbracket'
            if alt_name == 'klogd' or alt_name == 'syslogd':
                d.appendVar('ALTERNATIVE:%s-syslog' % (pn), ' ' + alt_name)
            else:
                d.appendVar('ALTERNATIVE:%s' % (pn), ' ' + alt_name)
            d.setVarFlag('ALTERNATIVE_LINK_NAME', alt_name, alt_link_name)
            if os.path.exists('%s%s' % (dvar, target)):
                d.setVarFlag('ALTERNATIVE_TARGET', alt_name, target)
        f.close()
        return
 
    if os.path.exists('%s/etc/busybox.links' % (dvar)):
        set_alternative_vars("${sysconfdir}/busybox.links", "${base_bindir}/busybox")
    else:
        set_alternative_vars("${sysconfdir}/busybox.links.nosuid", "${base_bindir}/busybox.nosuid")
        set_alternative_vars("${sysconfdir}/busybox.links.suid", "${base_bindir}/busybox.suid")
}
 
# This part of code is dedicated to the on target upgrade problem.  It's known
# that if we don't make appropriate symlinks before update-alternatives calls,
# there will be errors indicating missing commands such as 'sed'.
# These symlinks will later be updated by update-alternatives calls.
# The update-alternatives.bbclass' postinst script runs firstly before other
# postinst, but this part of code needs run firstly, so add this funtion.
python populate_packages_updatealternatives:append() {
    postinst = """
test -n 2 > /dev/null || alias test='busybox test'
if test "x$D" = "x"; then
    # Remove busybox.nosuid if it's a symlink, because this situation indicates
    # that we're installing or upgrading to a one-binary busybox.
    if test -h ${base_bindir}/busybox.nosuid; then
        rm -f ${base_bindir}/busybox.nosuid
    fi
    for suffix in "" ".nosuid" ".suid"; do
        if test -e ${sysconfdir}/busybox.links$suffix; then
            while read link; do
                if test ! -e "$link"; then
                    # we can use busybox here because even if we are using splitted busybox
                    # we've made a symlink from /bin/busybox to /bin/busybox.nosuid.
                    busybox rm -f $link
                    busybox ln -s "${base_bindir}/busybox$suffix" $link
                fi
            done < ${sysconfdir}/busybox.links$suffix
        fi
    done
fi
if grep -q "^${base_bindir}/bash$" $D${sysconfdir}/busybox.links*; then
    grep -q "^${base_bindir}/bash$" $D${sysconfdir}/shells || echo ${base_bindir}/bash >> $D${sysconfdir}/shells
fi
 
"""
    d.prependVar('pkg_postinst:%s' % pkg, postinst)
}
 
pkg_postinst:${PN}:prepend () {
        # Need path to saved utils, but they may have be removed on upgrade of busybox
        # Only use shell to get paths. Also capture if busybox was saved.
        BUSYBOX=""
        if [ "x$D" = "x" ] ; then 
           for busybox_rmdir in /tmp/busyboxrm-*; do
               if [ "$busybox_rmdir" != '/tmp/busyboxrm-*' ] ; then
                  export PATH=$busybox_rmdir:$PATH
                  if [ -e $busybox_rmdir/busybox* ] ; then
                    BUSYBOX="$busybox_rmdir/busybox*"
                  fi
               fi
           done
        fi
}
 
pkg_postinst:${PN}:append () {
        # If busybox exists in the remove directory it is because it was the only shell left.
        if [ "x$D" = "x" ] ; then
           if [ "x$BUSYBOX" != "x" ] ; then
              update-alternatives --remove sh $BUSYBOX
              rm -f $BUSYBOX
           fi
        fi
 
pkg_prerm:${PN} () {
   # This is so you can make busybox commit suicide - removing busybox with no other packages
   # providing its files, this will make update-alternatives work, but the update-rc.d part
   # for syslog, httpd and/or udhcpd will fail if there is no other package providing sh
   tmpdir=`mktemp -d /tmp/busyboxrm-XXXXXX`
   ln -s ${base_bindir}/busybox $tmpdir/[
   ln -s ${base_bindir}/busybox $tmpdir/test
   ln -s ${base_bindir}/busybox $tmpdir/head
   ln -s ${base_bindir}/busybox $tmpdir/sh
   ln -s ${base_bindir}/busybox $tmpdir/basename
   ln -s ${base_bindir}/busybox $tmpdir/echo
   ln -s ${base_bindir}/busybox $tmpdir/mv
   ln -s ${base_bindir}/busybox $tmpdir/ln
   ln -s ${base_bindir}/busybox $tmpdir/dirname
   ln -s ${base_bindir}/busybox $tmpdir/rm
   ln -s ${base_bindir}/busybox $tmpdir/sed
   ln -s ${base_bindir}/busybox $tmpdir/sort
   ln -s ${base_bindir}/busybox $tmpdir/grep
   ln -s ${base_bindir}/busybox $tmpdir/tail
   export PATH=$PATH:$tmpdir
 
        # If busybox is the shell, we need to save it since its the lowest priority shell
        # Register saved bitbake as the lowest priority shell possible as back up.
        if [ -n "$(readlink -f /bin/sh | grep busybox)" ] ; then
           BUSYBOX=$(readlink -f /bin/sh)
           cp $BUSYBOX $tmpdir/$(basename $BUSYBOX)
           update-alternatives --install /bin/sh sh $tmpdir/$(basename $BUSYBOX) 1 
        fi
}
 
pkg_postrm:${PN} () {
        # Add path to remove dir in case we removed our only grep
        if [ "x$D" = "x" ] ; then
           for busybox_rmdir in /tmp/busyboxrm-*; do
               if [ "$busybox_rmdir" != '/tmp/busyboxrm-*' ] ; then
                  export PATH=$busybox_rmdir:$PATH
               fi
           done
        fi
 
   if grep -q "^${base_bindir}/bash$" $D${sysconfdir}/busybox.links* && [ ! -e $D${base_bindir}/bash ]; then
       printf "$(grep -v "^${base_bindir}/bash$" $D${sysconfdir}/shells)\n" > $D${sysconfdir}/shells
   fi
}
 
pkg_prerm:${PN}-syslog () {
   # remove syslog
   if test "x$D" = "x"; then
       if test "$1" = "upgrade" -o "$1" = "remove"; then
           ${sysconfdir}/init.d/syslog stop || :
       fi
   fi
}
 
RPROVIDES:${PN} += "${@bb.utils.contains('DISTRO_FEATURES', 'usrmerge', '/bin/sh /bin/ash', '', d)}"