blob: 9c4d4b1b586bee8528e05f9c029e205ae5b5fdf2 [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
Jiyong Park44dd28f2024-09-20 18:47:40 +0900207}
208
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000209package_custom_kernel() {
Saswat Padhi8ba4a532025-02-19 12:37:03 -0800210 if [[ "$use_generic_kernel" == 1 ]]; then
211 # NOTE: For bpfcc-tools, install generic headers for the generic kernel.
Saswat Padhif3e52572025-02-06 14:30:08 -0800212 cat > "${config_space}/package_config/LAST" <<EOF
213PACKAGES install
214linux-headers-generic
215EOF
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000216 return
217 fi
218
Saswat Padhi8ba4a532025-02-19 12:37:03 -0800219 # NOTE: Prevent FAI from installing a default Debian kernel, by removing
220 # linux-image meta package names from arch-specific class files.
221 sed -i "/linux-image.*-${debian_arch}/d" \
222 "${config_space}/package_config/${debian_arch^^}"
223
Saswat Padhif28d2172025-01-16 14:28:10 -0800224 local deb_base_url="https://deb.debian.org/debian"
225 local deb_security_base_url="https://security.debian.org/debian-security"
226
227 local pool_dir="pool/main/l/linux"
228 local ksrc_base_url="${deb_base_url}/${pool_dir}"
229 local ksrc_security_base_url="${deb_security_base_url}/${pool_dir}"
230
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000231 # NOTE: 6.1 is the latest LTS kernel for which Debian's kernel build scripts
232 # work on Python 3.10, the default version on our Ubuntu 22.04 builders.
Saswat Padhi6c1aaf32025-01-22 12:47:20 -0800233 #
234 # We track the latest Debian stable kernel version for the 6.1 branch,
235 # which can be found at:
236 # https://packages.debian.org/stable/linux-source-6.1
237 local debian_kver="6.1.123-1"
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000238
Saswat Padhif28d2172025-01-16 14:28:10 -0800239 local dsc_file="linux_${debian_kver}.dsc"
240 local orig_ksrc_file="linux_${debian_kver%-*}.orig.tar.xz"
241 local debian_ksrc_file="linux_${debian_kver}.debian.tar.xz"
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000242
Saswat Padhi74b3ac92024-12-11 13:41:59 -0800243 # 0. Grab the kernel sources, and the latest debian keyrings
244 mkdir -p "${workdir}/kernel"
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000245 pushd "${workdir}/kernel" > /dev/null
Saswat Padhif28d2172025-01-16 14:28:10 -0800246
247 wget "${ksrc_security_base_url}/${dsc_file}" || \
248 wget "${ksrc_base_url}/${dsc_file}"
249
250 wget "${ksrc_security_base_url}/${orig_ksrc_file}" || \
251 wget "${ksrc_base_url}/${orig_ksrc_file}"
252
253 wget "${ksrc_security_base_url}/${debian_ksrc_file}" || \
254 wget "${ksrc_base_url}/${debian_ksrc_file}"
255
Saswat Padhi74b3ac92024-12-11 13:41:59 -0800256 rsync -az --progress keyring.debian.org::keyrings/keyrings/ /usr/share/keyrings/
257
258 # 1. Verify, extract and merge patches into the original kernel sources
259 dpkg-source --require-strong-checksums \
260 --require-valid-signature \
Saswat Padhif28d2172025-01-16 14:28:10 -0800261 --extract "${dsc_file}"
Saswat Padhi74b3ac92024-12-11 13:41:59 -0800262 pushd "linux-${debian_kver%-*}" > /dev/null
Saswat Padhi61fbaed2025-01-16 15:58:00 -0800263
Saswat Padhi58333e72025-02-12 18:39:39 -0800264 local kpatches_src="$SCRIPT_DIR/kernel/patches"
Saswat Padhi61fbaed2025-01-16 15:58:00 -0800265 cp -r "${kpatches_src}/avf" debian/patches/
266 cat "${kpatches_src}/series" >> debian/patches/series
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000267 ./debian/rules orig
268
Saswat Padhif28d2172025-01-16 14:28:10 -0800269 local custom_flavour="avf"
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000270 local debarch_flavour="${custom_flavour}-${debian_arch}"
Saswat Padhif28d2172025-01-16 14:28:10 -0800271
272 local abi_kver="$(sed -nE 's;Package: linux-support-(.*);\1;p' debian/control)"
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000273 local abi_flavour="${abi_kver}-${debarch_flavour}"
274
275 # 2. Define our custom flavour and regenerate control file
276 # NOTE: Our flavour extends Debian's `cloud` config on the `none` featureset.
Saswat Padhi58333e72025-02-12 18:39:39 -0800277 cp "$SCRIPT_DIR/kernel/config" \
278 debian/config/${debian_arch}/config.${debarch_flavour}
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000279
280 sed -z "s;\[base\]\nflavours:;[base]\nflavours:\n ${debarch_flavour};" \
281 -i debian/config/${debian_arch}/none/defines
282 cat >> debian/config/${debian_arch}/none/defines <<EOF
283[${debarch_flavour}_image]
284configs:
285 config.cloud
286 ${debian_arch}/config.${debarch_flavour}
287EOF
288 cat >> debian/config/${debian_arch}/defines <<EOF
289[${debarch_flavour}_description]
290hardware: ${arch} AVF
291hardware-long: ${arch} Android Virtualization Framework
292EOF
293 ./debian/rules debian/control || true
294
295 # 3. Build the kernel and generate Debian packages
296 ./debian/rules source
297 [[ "$arch" == "$(uname -m)" ]] || export $(dpkg-architecture -a $debian_arch)
298 make -j$(nproc) -f debian/rules.gen \
299 "binary-arch_${debian_arch}_none_${debarch_flavour}"
300
301 # 4. Copy the packages to localdebs and add their names to package_config/AVF
302 popd > /dev/null
303 cp "linux-headers-${abi_flavour}_${debian_kver}_${debian_arch}.deb" \
304 "linux-image-${abi_flavour}-unsigned_${debian_kver}_${debian_arch}.deb" \
305 "${debian_cloud_image}/localdebs/"
306 popd > /dev/null
307 cat >> "${config_space}/package_config/AVF" <<EOF
308linux-headers-${abi_flavour}
309linux-image-${abi_flavour}-unsigned
310EOF
311}
312
Jiyong Parka128bad2024-09-20 16:53:57 +0900313run_fai() {
Jiyong Park0b3a2ba2024-12-23 12:54:21 +0900314 local out="${raw_disk_image}"
maciek swiech0fdd0512024-10-11 15:12:44 +0000315 make -C "${debian_cloud_image}" "image_bookworm_nocloud_${debian_arch}"
316 mv "${debian_cloud_image}/image_bookworm_nocloud_${debian_arch}.raw" "${out}"
Jiyong Parka128bad2024-09-20 16:53:57 +0900317}
318
Jiyong Parkdfec0202024-12-23 13:24:01 +0900319generate_output_package() {
320 fdisk -l "${raw_disk_image}"
Saswat Padhi87282312025-02-26 17:46:19 -0800321 local vm_config="$SCRIPT_DIR/vm_config.json"
Jiyong Park159b8f42024-12-26 18:05:22 +0900322 local root_partition_num=1
Jiyong Park159b8f42024-12-26 18:05:22 +0900323 local efi_partition_num=15
Mu-Le Lee955b6582024-11-01 15:40:58 +0800324
Jiyong Parkec71e582024-12-23 14:21:57 +0900325 pushd ${workdir} > /dev/null
326
327 echo ${build_id} > build_id
328
Jiyong Park0b3a2ba2024-12-23 12:54:21 +0900329 loop=$(losetup -f --show --partscan $raw_disk_image)
maciek swieche17e59f2024-11-25 20:13:23 +0000330 dd if="${loop}p$root_partition_num" of=root_part
maciek swieche17e59f2024-11-25 20:13:23 +0000331 dd if="${loop}p$efi_partition_num" of=efi_part
332 losetup -d "${loop}"
Mu-Le Lee955b6582024-11-01 15:40:58 +0800333
Jiyong Parkec71e582024-12-23 14:21:57 +0900334 cp ${vm_config} vm_config.json
Saswat Padhi8ba4a532025-02-19 12:37:03 -0800335 # TODO(b/363985291): remove this when ballooning is supported on generic kernel
336 if [[ "$use_generic_kernel" == 1 ]] && [[ "$arch" == "aarch64" ]]; then
337 sed -i 's/"auto_memory_balloon": true/"auto_memory_balloon": false/g' vm_config.json
338 fi
339
Jiyong Park0b3a2ba2024-12-23 12:54:21 +0900340 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 +0900341 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 +0900342
Jiyong Parkec71e582024-12-23 14:21:57 +0900343 popd > /dev/null
344
Saswat Padhi80e72872025-02-25 21:20:21 -0800345 contents=(
346 build_id
347 root_part
348 efi_part
349 vm_config.json
350 )
351
Jiyong Parkdfec0202024-12-23 13:24:01 +0900352 # --sparse option isn't supported in apache-commons-compress
Saswat Padhi80e72872025-02-25 21:20:21 -0800353 tar czv -f ${output} -C ${workdir} "${contents[@]}"
Mu-Le Lee955b6582024-11-01 15:40:58 +0800354}
355
Jiyong Parka128bad2024-09-20 16:53:57 +0900356clean_up() {
Saswat Padhibf8a9cb2024-12-04 02:32:33 +0000357 [ "$save_workdir" -eq 1 ] || rm -rf "${workdir}"
Jiyong Parka128bad2024-09-20 16:53:57 +0900358}
359
360set -e
361trap clean_up EXIT
362
Jiyong Park0b3a2ba2024-12-23 12:54:21 +0900363output=images.tar.gz
Jiyong Parka128bad2024-09-20 16:53:57 +0900364workdir=$(mktemp -d)
Jiyong Parkec71e582024-12-23 14:21:57 +0900365raw_disk_image=${workdir}/image.raw
Jiyong Park879ee4a2024-11-29 14:00:47 +0900366build_id=$(prepare_build_id)
Jiyong Parka128bad2024-09-20 16:53:57 +0900367debian_cloud_image=${workdir}/debian_cloud_image
Jiyong Park44dd28f2024-09-20 18:47:40 +0900368debian_version=bookworm
369config_space=${debian_cloud_image}/config_space/${debian_version}
Jeongik Cha37047c32024-09-20 23:09:16 +0900370resources_dir=${debian_cloud_image}/src/debian_cloud_images/resources
Saswat Padhibf8a9cb2024-12-04 02:32:33 +0000371arch="$(uname -m)"
Jeongik Cha06f4ac52024-11-12 15:56:05 +0900372mode=debug
Saswat Padhi26c4ef32024-11-29 19:46:53 +0000373save_workdir=0
Saswat Padhi8ba4a532025-02-19 12:37:03 -0800374use_generic_kernel=0
Jiyong Park879ee4a2024-11-29 14:00:47 +0900375
maciek swiech0fdd0512024-10-11 15:12:44 +0000376parse_options "$@"
Jiyong Parka128bad2024-09-20 16:53:57 +0900377check_sudo
Jiyong Parka128bad2024-09-20 16:53:57 +0900378install_prerequisites
379download_debian_cloud_image
Jiyong Park44dd28f2024-09-20 18:47:40 +0900380copy_android_config
Saswat Padhiac62a8b2024-12-06 00:48:07 +0000381package_custom_kernel
Jiyong Park0e565ed2024-09-24 12:39:53 +0900382run_fai
Jiyong Parkdfec0202024-12-23 13:24:01 +0900383generate_output_package