blob: 8c1345c30557e3262fd359a0bd82428cc5d1219d [file] [log] [blame]
Jiyong Parka128bad2024-09-20 16:53:57 +09001#!/bin/bash
2
3# This is a script to build a Debian image that can run in a VM created via AVF.
4# TODOs:
Jiyong Parka128bad2024-09-20 16:53:57 +09005# - Add Android-specific packages via a new class
6# - Use a stable release from debian-cloud-images
7
Saswat Padhi323a8562025-01-27 13:54:07 -08008SCRIPT_DIR="$(cd -P -- "$(dirname -- "${BASH_SOURCE[0]}")" && pwd -P)"
9
Jiyong Parka128bad2024-09-20 16:53:57 +090010show_help() {
maciek swiech0fdd0512024-10-11 15:12:44 +000011 echo "Usage: sudo $0 [OPTION]... [FILE]"
12 echo "Builds a debian image and save it to FILE. [sudo is required]"
13 echo "Options:"
14 echo "-h Print usage and this help message and exit."
Saswat Padhibf8a9cb2024-12-04 02:32:33 +000015 echo "-a ARCH Architecture of the image [default is host arch: $(uname -m)]"
Saswat Padhi8ba4a532025-02-19 12:37:03 -080016 echo "-g Use Debian generic kernel [default is our custom kernel]"
Jeongik Cha06f4ac52024-11-12 15:56:05 +090017 echo "-r Release mode build"
Saswat Padhibf8a9cb2024-12-04 02:32:33 +000018 echo "-w Save temp work directory [for debugging]"
Jiyong Parka128bad2024-09-20 16:53:57 +090019}
20
21check_sudo() {
22 if [ "$EUID" -ne 0 ]; then
Saswat Padhibf8a9cb2024-12-04 02:32:33 +000023 echo "Please run as root." ; exit 1
Jiyong Parka128bad2024-09-20 16:53:57 +090024 fi
25}
26
27parse_options() {
Saswat Padhi8ba4a532025-02-19 12:37:03 -080028 while getopts "a:ghrw" option; do
Jiyong Parka128bad2024-09-20 16:53:57 +090029 case ${option} in
30 h)
Saswat Padhibf8a9cb2024-12-04 02:32:33 +000031 show_help ; exit
32 ;;
maciek swiech0fdd0512024-10-11 15:12:44 +000033 a)
maciek swiech0fdd0512024-10-11 15:12:44 +000034 arch="$OPTARG"
maciek swiech0fdd0512024-10-11 15:12:44 +000035 ;;
Saswat Padhi8ba4a532025-02-19 12:37:03 -080036 g)
37 use_generic_kernel=1
Saswat Padhiac62a8b2024-12-06 00:48:07 +000038 ;;
Jeongik Cha06f4ac52024-11-12 15:56:05 +090039 r)
40 mode=release
41 ;;
Saswat Padhi26c4ef32024-11-29 19:46:53 +000042 w)
43 save_workdir=1
44 ;;
maciek swiech0fdd0512024-10-11 15:12:44 +000045 *)
Saswat Padhibf8a9cb2024-12-04 02:32:33 +000046 echo "Invalid option: $OPTARG" ; exit 1
maciek swiech0fdd0512024-10-11 15:12:44 +000047 ;;
Jiyong Parka128bad2024-09-20 16:53:57 +090048 esac
49 done
Saswat Padhibf8a9cb2024-12-04 02:32:33 +000050 case "$arch" in
51 aarch64)
52 debian_arch="arm64"
53 ;;
54 x86_64)
55 debian_arch="amd64"
56 ;;
57 *)
58 echo "Invalid architecture: $arch" ; exit 1
59 ;;
60 esac
maciek swiech0fdd0512024-10-11 15:12:44 +000061 if [[ "${*:$OPTIND:1}" ]]; then
Jiyong Park0b3a2ba2024-12-23 12:54:21 +090062 output="${*:$OPTIND:1}"
Jiyong Parka128bad2024-09-20 16:53:57 +090063 fi
64}
65
Jiyong Park879ee4a2024-11-29 14:00:47 +090066prepare_build_id() {
Jiyong Park879ee4a2024-11-29 14:00:47 +090067 if [ -z "${KOKORO_BUILD_NUMBER}" ]; then
Jiyong Parkec71e582024-12-23 14:21:57 +090068 echo eng-$(hostname)-$(date --utc)
Jiyong Park879ee4a2024-11-29 14:00:47 +090069 else
Jiyong Parkec71e582024-12-23 14:21:57 +090070 echo ${KOKORO_BUILD_NUMBER}
Jiyong Park879ee4a2024-11-29 14:00:47 +090071 fi
Jiyong Park879ee4a2024-11-29 14:00:47 +090072}
73
Jiyong Parka128bad2024-09-20 16:53:57 +090074install_prerequisites() {
Jiyong Park0e565ed2024-09-24 12:39:53 +090075 apt update
maciek swiech0fdd0512024-10-11 15:12:44 +000076 packages=(
Jeongik Cha1542d152024-12-04 13:22:09 +090077 apt-utils
Jeongik Cha7e7f19d2024-10-31 20:50:24 +090078 automake
maciek swiech0fdd0512024-10-11 15:12:44 +000079 binfmt-support
80 build-essential
81 ca-certificates
Jeongik Cha7e7f19d2024-10-31 20:50:24 +090082 cmake
maciek swiech0fdd0512024-10-11 15:12:44 +000083 curl
84 debsums
85 dosfstools
86 fai-server
87 fai-setup-storage
88 fdisk
Jeongik Cha7e7f19d2024-10-31 20:50:24 +090089 git
90 libjson-c-dev
91 libtool
92 libwebsockets-dev
maciek swiech0fdd0512024-10-11 15:12:44 +000093 make
Jeongik Chace3a3962024-10-12 03:47:23 +090094 protobuf-compiler
maciek swiech0fdd0512024-10-11 15:12:44 +000095 python3
96 python3-libcloud
97 python3-marshmallow
98 python3-pytest
99 python3-yaml
100 qemu-user-static
101 qemu-utils
102 sudo
103 udev
104 )
105 if [[ "$arch" == "aarch64" ]]; then
106 packages+=(
107 gcc-aarch64-linux-gnu
108 libc6-dev-arm64-cross
109 qemu-system-arm
110 )
111 else
112 packages+=(
Jeongik Cha904d9622024-10-21 11:16:37 +0900113 qemu-system
Jeongik Cha8e711982024-10-20 12:45:35 +0900114 )
115 fi
116
Saswat Padhi8ba4a532025-02-19 12:37:03 -0800117 if [[ "$use_generic_kernel" != 1 ]]; then
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000118 packages+=(
119 bc
120 bison
121 debhelper
122 dh-exec
123 flex
124 gcc-12
125 kernel-wedge
126 libelf-dev
127 libpci-dev
128 lz4
129 pahole
130 python3-jinja2
131 python3-docutils
132 quilt
133 rsync
134 )
135 if [[ "$arch" == "aarch64" ]]; then
136 packages+=(
137 gcc-arm-linux-gnueabihf
138 gcc-12-aarch64-linux-gnu
139 )
140 fi
141 fi
142
Jiyong Park44dd28f2024-09-20 18:47:40 +0900143 DEBIAN_FRONTEND=noninteractive \
maciek swiech0fdd0512024-10-11 15:12:44 +0000144 apt install --no-install-recommends --assume-yes "${packages[@]}"
Jeongik Chab137a5f2024-10-02 12:53:05 +0900145
maciek swiech0fdd0512024-10-11 15:12:44 +0000146 if [ ! -f $"HOME"/.cargo/bin/cargo ]; then
Seungjae Yoo198a0fb2024-10-04 16:29:12 +0900147 curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- -y
148 fi
149
maciek swiech0fdd0512024-10-11 15:12:44 +0000150 source "$HOME"/.cargo/env
151 rustup target add "${arch}"-unknown-linux-gnu
Jeongik Cha139ddfd2024-11-01 23:16:44 +0900152 cargo install cargo-license
Jiyong Parkff0dc0d2025-01-06 17:38:21 +0900153 cargo install cargo-deb
Jiyong Parka128bad2024-09-20 16:53:57 +0900154}
155
156download_debian_cloud_image() {
Jeongik Chab5d11772025-02-04 21:18:00 +0900157 local ver=38da93fe
Jiyong Parka128bad2024-09-20 16:53:57 +0900158 local prj=debian-cloud-images
maciek swiech0fdd0512024-10-11 15:12:44 +0000159 local url="https://salsa.debian.org/cloud-team/${prj}/-/archive/${ver}/${prj}-${ver}.tar.gz"
160 local outdir="${debian_cloud_image}"
Jiyong Parka128bad2024-09-20 16:53:57 +0900161
maciek swiech0fdd0512024-10-11 15:12:44 +0000162 mkdir -p "${outdir}"
163 wget -O - "${url}" | tar xz -C "${outdir}" --strip-components=1
Jiyong Parka128bad2024-09-20 16:53:57 +0900164}
165
Jiyong Parkff0dc0d2025-01-06 17:38:21 +0900166build_rust_as_deb() {
Saswat Padhi323a8562025-01-27 13:54:07 -0800167 pushd "$SCRIPT_DIR/../../guest/$1" > /dev/null
Jiyong Parkff0dc0d2025-01-06 17:38:21 +0900168 cargo deb \
Seungjae Yoo1cfcb582024-10-17 14:06:58 +0900169 --target "${arch}-unknown-linux-gnu" \
Jiyong Parkff0dc0d2025-01-06 17:38:21 +0900170 --output "${debian_cloud_image}/localdebs"
Seungjae Yoo1cfcb582024-10-17 14:06:58 +0900171 popd > /dev/null
172}
173
Jeongik Cha7e7f19d2024-10-31 20:50:24 +0900174build_ttyd() {
175 local ttyd_version=1.7.7
176 local url="https://github.com/tsl0922/ttyd/archive/refs/tags/${ttyd_version}.tar.gz"
Saswat Padhi323a8562025-01-27 13:54:07 -0800177 cp -r "$SCRIPT_DIR/ttyd" "${workdir}/ttyd"
Jeongik Cha7e7f19d2024-10-31 20:50:24 +0900178
179 pushd "${workdir}" > /dev/null
180 wget "${url}" -O - | tar xz
181 cp ttyd/* ttyd-${ttyd_version}/scripts
182 pushd "$workdir/ttyd-${ttyd_version}" > /dev/null
183 bash -c "env BUILD_TARGET=${arch} ./scripts/cross-build.sh"
184 mkdir -p "${dst}/files/usr/local/bin/ttyd"
maciek swieche17e59f2024-11-25 20:13:23 +0000185 cp "/tmp/stage/${arch}-linux-musl/bin/ttyd" "${dst}/files/usr/local/bin/ttyd/AVF"
Jeongik Cha7e7f19d2024-10-31 20:50:24 +0900186 chmod 777 "${dst}/files/usr/local/bin/ttyd/AVF"
Jeongik Cha139ddfd2024-11-01 23:16:44 +0900187 mkdir -p "${dst}/files/usr/share/doc/ttyd"
188 cp LICENSE "${dst}/files/usr/share/doc/ttyd/copyright"
Jeongik Cha7e7f19d2024-10-31 20:50:24 +0900189 popd > /dev/null
190 popd > /dev/null
191}
192
Jiyong Park44dd28f2024-09-20 18:47:40 +0900193copy_android_config() {
maciek swieche17e59f2024-11-25 20:13:23 +0000194 local src
195 local dst
Saswat Padhi323a8562025-01-27 13:54:07 -0800196 src="$SCRIPT_DIR/fai_config"
maciek swieche17e59f2024-11-25 20:13:23 +0000197 dst="${config_space}"
Jiyong Park44dd28f2024-09-20 18:47:40 +0900198
maciek swiech0fdd0512024-10-11 15:12:44 +0000199 cp -R "${src}"/* "${dst}"
Saswat Padhi323a8562025-01-27 13:54:07 -0800200 cp "$SCRIPT_DIR/image.yaml" "${resources_dir}"
Jeongik Cha50952062024-09-23 18:13:38 +0900201
Saswat Padhi323a8562025-01-27 13:54:07 -0800202 cp -R "$SCRIPT_DIR/localdebs/" "${debian_cloud_image}/"
Jeongik Cha7e7f19d2024-10-31 20:50:24 +0900203 build_ttyd
Jiyong Parkff0dc0d2025-01-06 17:38:21 +0900204 build_rust_as_deb forwarder_guest
205 build_rust_as_deb forwarder_guest_launcher
206 build_rust_as_deb shutdown_runner
Keiichi Watanabe4d75b332025-03-05 12:18:32 +0000207 build_rust_as_deb storage_balloon_agent
Jiyong Park44dd28f2024-09-20 18:47:40 +0900208}
209
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000210package_custom_kernel() {
Saswat Padhi8ba4a532025-02-19 12:37:03 -0800211 if [[ "$use_generic_kernel" == 1 ]]; then
212 # NOTE: For bpfcc-tools, install generic headers for the generic kernel.
Saswat Padhif3e52572025-02-06 14:30:08 -0800213 cat > "${config_space}/package_config/LAST" <<EOF
214PACKAGES install
215linux-headers-generic
216EOF
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000217 return
218 fi
219
Saswat Padhi8ba4a532025-02-19 12:37:03 -0800220 # NOTE: Prevent FAI from installing a default Debian kernel, by removing
221 # linux-image meta package names from arch-specific class files.
222 sed -i "/linux-image.*-${debian_arch}/d" \
223 "${config_space}/package_config/${debian_arch^^}"
224
Saswat Padhif28d2172025-01-16 14:28:10 -0800225 local deb_base_url="https://deb.debian.org/debian"
226 local deb_security_base_url="https://security.debian.org/debian-security"
227
228 local pool_dir="pool/main/l/linux"
229 local ksrc_base_url="${deb_base_url}/${pool_dir}"
230 local ksrc_security_base_url="${deb_security_base_url}/${pool_dir}"
231
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000232 # NOTE: 6.1 is the latest LTS kernel for which Debian's kernel build scripts
233 # work on Python 3.10, the default version on our Ubuntu 22.04 builders.
Saswat Padhi6c1aaf32025-01-22 12:47:20 -0800234 #
235 # We track the latest Debian stable kernel version for the 6.1 branch,
236 # which can be found at:
237 # https://packages.debian.org/stable/linux-source-6.1
238 local debian_kver="6.1.123-1"
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000239
Saswat Padhif28d2172025-01-16 14:28:10 -0800240 local dsc_file="linux_${debian_kver}.dsc"
241 local orig_ksrc_file="linux_${debian_kver%-*}.orig.tar.xz"
242 local debian_ksrc_file="linux_${debian_kver}.debian.tar.xz"
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000243
Saswat Padhi74b3ac92024-12-11 13:41:59 -0800244 # 0. Grab the kernel sources, and the latest debian keyrings
245 mkdir -p "${workdir}/kernel"
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000246 pushd "${workdir}/kernel" > /dev/null
Saswat Padhif28d2172025-01-16 14:28:10 -0800247
248 wget "${ksrc_security_base_url}/${dsc_file}" || \
249 wget "${ksrc_base_url}/${dsc_file}"
250
251 wget "${ksrc_security_base_url}/${orig_ksrc_file}" || \
252 wget "${ksrc_base_url}/${orig_ksrc_file}"
253
254 wget "${ksrc_security_base_url}/${debian_ksrc_file}" || \
255 wget "${ksrc_base_url}/${debian_ksrc_file}"
256
Saswat Padhi74b3ac92024-12-11 13:41:59 -0800257 rsync -az --progress keyring.debian.org::keyrings/keyrings/ /usr/share/keyrings/
258
259 # 1. Verify, extract and merge patches into the original kernel sources
260 dpkg-source --require-strong-checksums \
261 --require-valid-signature \
Saswat Padhif28d2172025-01-16 14:28:10 -0800262 --extract "${dsc_file}"
Saswat Padhi74b3ac92024-12-11 13:41:59 -0800263 pushd "linux-${debian_kver%-*}" > /dev/null
Saswat Padhi61fbaed2025-01-16 15:58:00 -0800264
Saswat Padhi58333e72025-02-12 18:39:39 -0800265 local kpatches_src="$SCRIPT_DIR/kernel/patches"
Saswat Padhi61fbaed2025-01-16 15:58:00 -0800266 cp -r "${kpatches_src}/avf" debian/patches/
267 cat "${kpatches_src}/series" >> debian/patches/series
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000268 ./debian/rules orig
269
Saswat Padhif28d2172025-01-16 14:28:10 -0800270 local custom_flavour="avf"
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000271 local debarch_flavour="${custom_flavour}-${debian_arch}"
Saswat Padhif28d2172025-01-16 14:28:10 -0800272
273 local abi_kver="$(sed -nE 's;Package: linux-support-(.*);\1;p' debian/control)"
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000274 local abi_flavour="${abi_kver}-${debarch_flavour}"
275
276 # 2. Define our custom flavour and regenerate control file
277 # NOTE: Our flavour extends Debian's `cloud` config on the `none` featureset.
Saswat Padhi58333e72025-02-12 18:39:39 -0800278 cp "$SCRIPT_DIR/kernel/config" \
279 debian/config/${debian_arch}/config.${debarch_flavour}
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000280
281 sed -z "s;\[base\]\nflavours:;[base]\nflavours:\n ${debarch_flavour};" \
282 -i debian/config/${debian_arch}/none/defines
283 cat >> debian/config/${debian_arch}/none/defines <<EOF
284[${debarch_flavour}_image]
285configs:
286 config.cloud
287 ${debian_arch}/config.${debarch_flavour}
288EOF
289 cat >> debian/config/${debian_arch}/defines <<EOF
290[${debarch_flavour}_description]
291hardware: ${arch} AVF
292hardware-long: ${arch} Android Virtualization Framework
293EOF
294 ./debian/rules debian/control || true
295
296 # 3. Build the kernel and generate Debian packages
297 ./debian/rules source
298 [[ "$arch" == "$(uname -m)" ]] || export $(dpkg-architecture -a $debian_arch)
299 make -j$(nproc) -f debian/rules.gen \
300 "binary-arch_${debian_arch}_none_${debarch_flavour}"
301
302 # 4. Copy the packages to localdebs and add their names to package_config/AVF
303 popd > /dev/null
304 cp "linux-headers-${abi_flavour}_${debian_kver}_${debian_arch}.deb" \
305 "linux-image-${abi_flavour}-unsigned_${debian_kver}_${debian_arch}.deb" \
306 "${debian_cloud_image}/localdebs/"
307 popd > /dev/null
308 cat >> "${config_space}/package_config/AVF" <<EOF
309linux-headers-${abi_flavour}
310linux-image-${abi_flavour}-unsigned
311EOF
312}
313
Jiyong Parka128bad2024-09-20 16:53:57 +0900314run_fai() {
Jiyong Park0b3a2ba2024-12-23 12:54:21 +0900315 local out="${raw_disk_image}"
maciek swiech0fdd0512024-10-11 15:12:44 +0000316 make -C "${debian_cloud_image}" "image_bookworm_nocloud_${debian_arch}"
317 mv "${debian_cloud_image}/image_bookworm_nocloud_${debian_arch}.raw" "${out}"
Jiyong Parka128bad2024-09-20 16:53:57 +0900318}
319
Jiyong Parkdfec0202024-12-23 13:24:01 +0900320generate_output_package() {
321 fdisk -l "${raw_disk_image}"
Saswat Padhi87282312025-02-26 17:46:19 -0800322 local vm_config="$SCRIPT_DIR/vm_config.json"
Jiyong Park159b8f42024-12-26 18:05:22 +0900323 local root_partition_num=1
Jiyong Park159b8f42024-12-26 18:05:22 +0900324 local efi_partition_num=15
Mu-Le Lee955b6582024-11-01 15:40:58 +0800325
Jiyong Parkec71e582024-12-23 14:21:57 +0900326 pushd ${workdir} > /dev/null
327
328 echo ${build_id} > build_id
329
Jiyong Park0b3a2ba2024-12-23 12:54:21 +0900330 loop=$(losetup -f --show --partscan $raw_disk_image)
maciek swieche17e59f2024-11-25 20:13:23 +0000331 dd if="${loop}p$root_partition_num" of=root_part
maciek swieche17e59f2024-11-25 20:13:23 +0000332 dd if="${loop}p$efi_partition_num" of=efi_part
333 losetup -d "${loop}"
Mu-Le Lee955b6582024-11-01 15:40:58 +0800334
Jiyong Parkec71e582024-12-23 14:21:57 +0900335 cp ${vm_config} vm_config.json
Saswat Padhi8ba4a532025-02-19 12:37:03 -0800336 # TODO(b/363985291): remove this when ballooning is supported on generic kernel
337 if [[ "$use_generic_kernel" == 1 ]] && [[ "$arch" == "aarch64" ]]; then
338 sed -i 's/"auto_memory_balloon": true/"auto_memory_balloon": false/g' vm_config.json
339 fi
340
Jiyong Park0b3a2ba2024-12-23 12:54:21 +0900341 sed -i "s/{root_part_guid}/$(sfdisk --part-uuid $raw_disk_image $root_partition_num)/g" vm_config.json
Jiyong Park0b3a2ba2024-12-23 12:54:21 +0900342 sed -i "s/{efi_part_guid}/$(sfdisk --part-uuid $raw_disk_image $efi_partition_num)/g" vm_config.json
Jiyong Parkdfec0202024-12-23 13:24:01 +0900343
Jiyong Parkec71e582024-12-23 14:21:57 +0900344 popd > /dev/null
345
Saswat Padhi80e72872025-02-25 21:20:21 -0800346 contents=(
347 build_id
348 root_part
349 efi_part
350 vm_config.json
351 )
352
Jiyong Parkdfec0202024-12-23 13:24:01 +0900353 # --sparse option isn't supported in apache-commons-compress
Saswat Padhi80e72872025-02-25 21:20:21 -0800354 tar czv -f ${output} -C ${workdir} "${contents[@]}"
Mu-Le Lee955b6582024-11-01 15:40:58 +0800355}
356
Jiyong Parka128bad2024-09-20 16:53:57 +0900357clean_up() {
Saswat Padhibf8a9cb2024-12-04 02:32:33 +0000358 [ "$save_workdir" -eq 1 ] || rm -rf "${workdir}"
Jiyong Parka128bad2024-09-20 16:53:57 +0900359}
360
361set -e
362trap clean_up EXIT
363
Jiyong Park0b3a2ba2024-12-23 12:54:21 +0900364output=images.tar.gz
Jiyong Parka128bad2024-09-20 16:53:57 +0900365workdir=$(mktemp -d)
Jiyong Parkec71e582024-12-23 14:21:57 +0900366raw_disk_image=${workdir}/image.raw
Jiyong Park879ee4a2024-11-29 14:00:47 +0900367build_id=$(prepare_build_id)
Jiyong Parka128bad2024-09-20 16:53:57 +0900368debian_cloud_image=${workdir}/debian_cloud_image
Jiyong Park44dd28f2024-09-20 18:47:40 +0900369debian_version=bookworm
370config_space=${debian_cloud_image}/config_space/${debian_version}
Jeongik Cha37047c32024-09-20 23:09:16 +0900371resources_dir=${debian_cloud_image}/src/debian_cloud_images/resources
Saswat Padhibf8a9cb2024-12-04 02:32:33 +0000372arch="$(uname -m)"
Jeongik Cha06f4ac52024-11-12 15:56:05 +0900373mode=debug
Saswat Padhi26c4ef32024-11-29 19:46:53 +0000374save_workdir=0
Saswat Padhi8ba4a532025-02-19 12:37:03 -0800375use_generic_kernel=0
Jiyong Park879ee4a2024-11-29 14:00:47 +0900376
maciek swiech0fdd0512024-10-11 15:12:44 +0000377parse_options "$@"
Jiyong Parka128bad2024-09-20 16:53:57 +0900378check_sudo
Jiyong Parka128bad2024-09-20 16:53:57 +0900379install_prerequisites
380download_debian_cloud_image
Jiyong Park44dd28f2024-09-20 18:47:40 +0900381copy_android_config
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000382package_custom_kernel
Jiyong Park0e565ed2024-09-24 12:39:53 +0900383run_fai
Jiyong Parkdfec0202024-12-23 13:24:01 +0900384generate_output_package