-
Notifications
You must be signed in to change notification settings - Fork 48
/
mkinitfs.in
executable file
·336 lines (294 loc) · 7.67 KB
/
mkinitfs.in
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
#!/bin/sh
VERSION=@VERSION@
sysconfdir="${SYSCONFDIR:-@sysconfdir@}"
datadir="${DATADIR:-@datadir@}"
config="$sysconfdir"/mkinitfs.conf
init="$datadir"/initramfs-init
fstab="$datadir"/fstab
passwd="$datadir"/passwd
group="$datadir"/group
startdir=$PWD
initfscomp=gzip
feature_files() {
local dir="$1"
local suffix="$2"
local glob file fdir
for f in $features; do
for fdir in $features_dirs; do
[ -f "$fdir/$f.$suffix" ] || continue
for glob in $(sed -e '/^$/d' -e '/^#/d' -e "s|^/*|$dir|" "$fdir/$f.$suffix"); do
for file in $glob; do
if [ -d $file ]; then
find $file -type f
elif [ -e "$file" ]; then
echo $file
fi
done
done
break
done
done
}
initfs_base() {
local i= dirs= glob= file=
for i in dev proc sys sbin bin run .modloop lib/modules media/cdrom \
etc/apk media/floppy media/usb newroot; do
dirs="$dirs $tmpdir/$i"
done
mkdir -p $dirs
local oldpwd="$PWD"
cd "${basedir}"
lddtree -R "$basedir" -l --no-auto-root \
$(feature_files "$basedir" files) \
\
| sed -e "s|^$basedir||" | sort -u \
| cpio --quiet -pdm "$tmpdir" || return 1
# copy init
cd "$startdir"
install -m755 "$init" "$tmpdir"/init || return 1
# copy modloop signature
if [ -n "$modloop_sig" ]; then
install -Dm644 "$modloop_sig" \
"$tmpdir"/var/cache/misc/${modloop_sig##*/}
fi
for i in "$fstab" "$passwd" "$group"; do
install -Dm644 "$i" "$tmpdir"/etc/${i##*/} || return 1
done
cd "$oldpwd"
}
find_kmod_deps() {
awk -v prepend="/lib/modules/$kernel/" -v modulesdep="${basedir}lib/modules/$kernel/modules.dep" '
function recursedeps(k, j, dep) {
if (k in visited)
return;
visited[k] = 1;
split(deps[k], dep, " ");
for (j in dep)
recursedeps(dep[j]);
print(prepend k);
}
BEGIN {
if (modulesdep == "")
modulesdep="modules.dep";
FS = ": ";
while ( (getline < modulesdep) > 0) {
if (substr($0,1,1) == "/") {
gsub(prepend, "", $1);
gsub(prepend, "", $2);
}
deps[$1] = $2;
}
}
{
mod[$0] = 1;
}
END {
for (i in mod)
recursedeps(i);
}'
}
find_kmods() {
local oldpwd="$PWD"
cd "$kerneldir" || return 1
for file in $(feature_files "${kerneldir}/" modules); do
echo ${file#${kerneldir%/}/}
done | find_kmod_deps
cd "$oldpwd"
}
initfs_kmods() {
[ -z "$nokernel" ] || return 0
local glob= file= files= dirs=
rm -rf "$tmpdir"/lib/modules
# make sure we have modules.dep
if ! [ -f "$kerneldir"/modules.dep ]; then
depmod -b "${basedir}" $kernel
fi
local oldpwd="$PWD"
cd "${basedir}"
for file in $(find_kmods); do
echo "${file#/}"
# DO NOT return with 1 if there are no custom modules for initramfs
done | sort -u | cpio --quiet -pdm "$tmpdir"
mkdir -p "$tmpdir"/lib/modules/$kernel/
for file in modules.order modules.builtin modules.builtin.modinfo; do
if [ -f "$kerneldir"/$file ]; then
cp "$kerneldir"/$file "$tmpdir"/lib/modules/$kernel/
fi
done
$MOCK depmod $kernel -b "$tmpdir"
cd "$oldpwd"
}
initfs_firmware() {
[ -z "$nokernel" ] || return 0
rm -rf "$tmpdir"/lib/firmware
mkdir -p "$tmpdir"/lib/firmware
# Verify if there are initfs modules
_modules=`find "$tmpdir"/lib/modules -type f -name "*.ko*" -print -quit`
[ -n "$_modules" ] || return 0
find "$tmpdir"/lib/modules -type f -name "*.ko*" | xargs modinfo -k $kernel -F firmware | sort -u | while read FW; do
for fname in "$FW" "$FW.zst" "$FW.xz"; do
if [ -e "${basedir}/lib/firmware/$fname" ]; then
install -pD "${basedir}/lib/firmware/$fname" "$tmpdir"/lib/firmware/$fname
break
fi
done
done
return 0
}
initfs_apk_keys() {
mkdir -p "$tmpdir"/etc/apk/keys
[ "$hostkeys" ] && cp "/etc/apk/keys/"* "$tmpdir"/etc/apk/keys/
if [ -d "${basedir}etc/apk/keys" ]; then
cp "${basedir}etc/apk/keys/"* "$tmpdir"/etc/apk/keys/
fi
}
initfs_custom_files() {
# shellcheck disable=SC2086 # word splitting is required
for file in $custom_files; do
if [ -f "$basedir"/$file ]; then
mkdir -p "$tmpdir"/${file%/*}
cp "$basedir"/$file "$tmpdir"/$file
fi
done
}
initfs_cpio() {
if [ -n "$list_sources" ]; then
(cd "$tmpdir" && find . | sort)
return
fi
rm -f $outfile
if [ -n "$SOURCE_DATE_EPOCH" ]; then
# normalize timestamps
find "$tmpdir" -exec touch -h -d "@$SOURCE_DATE_EPOCH" {} +
fi
umask 0077
(cd "$tmpdir" && find . | sort | cpio --quiet --renumber-inodes -o -H newc | $comp) > "$outfile"
}
cmd_exists() {
local cmd="$1"
if ! command -v "$cmd" >/dev/null 2>&1; then
echo "Command \"$cmd\" is not available."
exit 1
fi
}
usage() {
cat <<EOF
usage: mkinitfs [-hkKLln] [-b basedir] [-c configfile] [-F features] [-f fstab]
[-C initramfs compression] [-i initfile] [-o outfile]
[-P featuresdir] [-t tempdir] [kernelversion]
options:
-b prefix files and kernel modules with basedir
-c use configfile instead of $config
-C initramfs compression (gzip|xz|zstd|lz4|none defaults to $initfscomp)
-f use fstab instead of $fstab
-F use specified features
-h print this help
-i use initfile as init instead of $init
-k keep tempdir
-K copy also host keys to initramfs
-l only list files that would have been used
-L list available features
-n don't include kernel modules or firmware
-o set another outfile
-P prepend features.d search path
-q Quiet mode
-s Include modloop signature
-t use tempdir when creating initramfs image
EOF
}
# main
features_dirs=${features_dir:-"${basedir%/:-}/${sysconfdir#/}/features.d"}
while getopts "b:c:C:f:F:hi:kKLlno:P:qs:t:" opt; do
case "$opt" in
b) basedir="$OPTARG";;
c) config="$OPTARG";;
C) initfscomp="$OPTARG";;
F) myfeatures="$OPTARG";;
f) fstab="$OPTARG";;
h) usage; exit 0;;
i) init=$OPTARG;;
k) keeptmp=1;;
K) hostkeys=1;;
L) list_features=1;;
l) list_sources=1;;
n) nokernel=1;;
o) outfile="$OPTARG";;
P) features_dirs="$OPTARG $features_dirs";;
q) quiet=1;;
s) modloop_sig="$OPTARG";;
t) tmpdir="$OPTARG";;
*) usage >&2; exit 1;;
esac
done
shift $(( $OPTIND - 1 ))
if [ -e "$config" ]; then
. "$config"
fi
[ -n "$myfeatures" ] && features="$myfeatures"
if [ -n "$list_features" ]; then
for dir in $features_dirs; do
for i in $dir/*.files $dir/*.modules; do
[ -e "$i" ] || continue
file=${i##*/}
echo ${file%.*}
done
done | sort -u
exit 0
fi
basedir="${basedir%/}/"
[ "${basedir}" = "${basedir#/}" ] && basedir="${PWD}/${basedir}"
if [ -z "$nokernel" ]; then
[ -n "$1" ] && kernel="$1"
[ -z "$kernel" ] && kernel=$(uname -r)
kerneldir="${basedir}lib/modules/$kernel"
kflavor=${kernel##*-}
[ "$kflavor" = "$kernel" ] && kflavor=vanilla
else
kflavor=generic
fi
if [ -z "$outfile" ]; then
outfile="${basedir}boot/initramfs-${kflavor}"
fi
if [ -z "$nokernel" ] && [ ! -d "$kerneldir" ]; then
echo "$kerneldir does not exist or is not a directory"
exit 1
fi
if [ -n "$DEBUG_KMOD" ]; then
find_kmods
exit 0
fi
if [ -z "$tmpdir" ]; then
tmpdir=$(mktemp -d /tmp/mkinitfs.XXXXXX)
else
mkdir -p "$tmpdir"
fi
if [ -z "$keeptmp" ]; then
[ -d "$tmpdir" ] && rm -rf "$tmpdir"/*
fi
if [ -z "$list_sources" ] && [ -z "$quiet" ]; then
echo "==> initramfs: creating $outfile for $kernel"
fi
complevel_gzip="${complevel_gzip:--9}"
complevel_xz="${complevel_xz:--9}"
complevel_zstd="${complevel_zstd:--19}"
complevel_lz4="${complevel_lz4:--12}"
case "$initfscomp" in
gzip) comp="$(command -v pigz 2>/dev/null || echo gzip) $complevel_gzip" ;;
xz) cmd_exists xz; comp="xz -C crc32 -T0 $complevel_xz" ;;
zstd) cmd_exists zstd; comp="zstd -T0 $complevel_zstd" ;;
lz4) cmd_exists lz4; comp="lz4 --favor-decSpeed -lz $complevel_lz4" ;;
none) comp="cat";;
*) echo "Initramfs compression \"$initfscomp\" not supported!"; exit 1 ;;
esac
initfs_base \
&& initfs_kmods \
&& initfs_firmware \
&& initfs_apk_keys \
&& initfs_custom_files \
&& initfs_cpio
rc=$?
# cleanup
if [ -z "$keeptmp" ]; then
[ -d "$tmpdir" ] && rm -rf "$tmpdir"
fi
exit $rc