blob: ce66b949df36af45d0f5c69c483bfe79e2eae722 [file] [log] [blame]
Rahul Chaturvedib5643e82010-07-09 10:46:05 +05301#!/bin/bash
2
3# Copyright (c) 2010 The Chromium OS Authors. All rights reserved.
4# Use of this source code is governed by a BSD-style license that can be
5# found in the LICENSE file.
6
Gaurav Shah550edca2014-09-25 11:13:55 -07007# Script to convert the output of build_image.sh to a QEMU image.
Rahul Chaturvedib5643e82010-07-09 10:46:05 +05308
Kees Cook84a4c7a2011-10-18 13:21:41 -07009# Helper scripts should be run from the same location as this script.
10SCRIPT_ROOT=$(dirname "$(readlink -f "$0")")
David James359d3e12012-07-10 13:09:48 -070011. "${SCRIPT_ROOT}/common.sh" || exit 1
Mike Frysingerdf7d5e62019-08-21 23:24:51 -040012. "${SCRIPT_ROOT}/build_library/ext2_sb_util.sh" || exit 1
Greg Spencer798d75f2011-02-01 22:04:49 -080013
14# Need to be inside the chroot to load chromeos-common.sh
15assert_inside_chroot
16
Mike Frysinger566c9432019-03-02 14:18:29 -050017# Default values for creating VM's.
18DEFAULT_QEMU_IMAGE="chromiumos_qemu_image.bin"
Rahul Chaturvedib5643e82010-07-09 10:46:05 +053019
Rahul Chaturvedib5643e82010-07-09 10:46:05 +053020# Flags
Liam McLoughlin12a9a842012-10-10 10:37:06 -040021DEFINE_string adjust_part "" \
22 "Adjustments to apply to the partition table"
Rahul Chaturvedib5643e82010-07-09 10:46:05 +053023DEFINE_string board "${DEFAULT_BOARD}" \
24 "Board for which the image was built"
Rahul Chaturvedib5643e82010-07-09 10:46:05 +053025DEFINE_string from "" \
26 "Directory containing rootfs.image and mbr.image"
Don Garrett7937ef82014-08-25 18:04:05 -070027DEFINE_string disk_layout "2gb-rootfs-updatable" \
Liam McLoughlinb78a7c32012-11-06 20:19:05 -050028 "The disk layout type to use for this image."
Rahul Chaturvedib5643e82010-07-09 10:46:05 +053029DEFINE_boolean test_image "${FLAGS_FALSE}" \
Jacob Kopczynski75d465c2018-07-19 12:54:17 -070030 "Acquires image from ${CHROMEOS_TEST_IMAGE_NAME} instead of " \
31 "${CHROMEOS_IMAGE_NAME}."
Rahul Chaturvedib5643e82010-07-09 10:46:05 +053032DEFINE_string to "" \
33 "Destination folder for VM output file(s)"
Rahul Chaturvedib5643e82010-07-09 10:46:05 +053034
35# Parse command line
36FLAGS "$@" || exit 1
37eval set -- "${FLAGS_ARGV}"
38
39# Die on any errors.
Brian Harring7f175a52012-03-02 05:37:00 -080040switch_to_strict_mode
Rahul Chaturvedib5643e82010-07-09 10:46:05 +053041
Mike Frysinger79e9dcd2019-07-29 00:48:36 -040042# Get the size of a regular file or a block device.
43#
44# $1 - The regular file or block device to get the size of.
45bd_safe_size() {
46 local file="$1"
47 if [[ -b "${file}" ]]; then
48 sudo blockdev --getsize64 "${file}"
49 else
50 stat -c%s "${file}"
51 fi
52}
53
Gabe Black067b3542014-09-23 01:15:14 -070054TEMP_DIR=$(mktemp -d)
55TEMP_MNT=""
56TEMP_ESP_MNT=""
57SRC_DEV=""
58DST_DEV=""
59cleanup() {
60 if [[ -n "${TEMP_MNT}" ]]; then
61 safe_umount "${TEMP_MNT}" || true
62 rmdir "${TEMP_MNT}" || true
63 fi
64 if [[ -n "${TEMP_ESP_MNT}" ]]; then
65 safe_umount "${TEMP_ESP_MNT}" || true
66 rmdir "${TEMP_ESP_MNT}" || true
67 fi
68
69 if [[ -n "${SRC_DEV}" ]]; then
70 loopback_detach "${SRC_DEV}" || true
71 fi
72 if [[ -n "${DST_DEV}" ]]; then
73 loopback_detach "${DST_DEV}" || true
74 fi
75 rm -rf "${TEMP_DIR}"
76}
Mike Frysinger9ebc8ce2015-04-06 13:15:01 -040077trap 'ret=$?; cleanup; die_err_trap ${ret}' INT TERM EXIT
Gabe Black067b3542014-09-23 01:15:14 -070078
Rahul Chaturvedib5643e82010-07-09 10:46:05 +053079# Default to the most recent image
80if [ -z "${FLAGS_from}" ] ; then
Alex Klein999443c2018-10-17 12:02:05 -060081 FLAGS_from="${IMAGES_DIR}/${FLAGS_board}/latest"
Rahul Chaturvedib5643e82010-07-09 10:46:05 +053082fi
83if [ -z "${FLAGS_to}" ] ; then
84 FLAGS_to="${FLAGS_from}"
85fi
86
Mike Frysingerb86854a2014-11-25 18:43:58 -050087# Convert args to full paths. Use echo here on the unquoted value to process all
88# shell level expansions like ~ and *.
89if ! resolved=$(readlink -f "$(echo ${FLAGS_from})"); then
90 die_notrace "image_to_vm: processing --from failed." \
91 "Verify the path exists: ${FLAGS_from}" \
92 " cwd: ${PWD}"
93fi
94FLAGS_from=${resolved}
95if ! resolved=$(readlink -f "$(echo ${FLAGS_to})"); then
96 die_notrace "image_to_vm: Processing --to failed." \
97 "Verify the path exists: ${FLAGS_to}" \
98 " cwd: ${PWD}"
99fi
100FLAGS_to=${resolved}
101
Hung-Te Lin269680c2016-05-30 14:47:37 +0800102if [ ${FLAGS_test_image} -eq ${FLAGS_TRUE} ]; then
David James7efb76c2013-01-09 15:25:58 -0800103 SRC_IMAGE="${FLAGS_from}/${CHROMEOS_TEST_IMAGE_NAME}"
Simon Glass142ca062011-02-09 13:39:43 -0800104else
105 # Use the standard image
106 SRC_IMAGE="${FLAGS_from}/${CHROMEOS_IMAGE_NAME}"
Rahul Chaturvedib5643e82010-07-09 10:46:05 +0530107fi
Mike Frysingerb86854a2014-11-25 18:43:58 -0500108if [[ ! -e ${SRC_IMAGE} ]]; then
109 die_notrace "image_to_vm: src image does not exist: ${SRC_IMAGE}" \
110 "Please verify you have selected the right input." \
111 "Note: only dev/test/factory images can be used as inputs."
112fi
Rahul Chaturvedib5643e82010-07-09 10:46:05 +0530113
Prathmesh Prabhuaa96e572017-08-31 22:42:37 -0700114if [[ -z "${FLAGS_board}" ]] && [[ -n "${FLAGS_from}" ]]; then
115 # The user may not know the board of in the input image, so infer it for them.
116 # TODO(pprabhu): This will fail if the user's chroot has a default_board set
117 # which is different from the image provided, because FLAGS_board will use
118 # that. Fix this project-wide by respecting the --board flag when provided,
119 # but preferring the board inferred from FLAGS_from over the default,
120 # everywhere.
121 FLAGS_board="$(
122 . "${BUILD_LIBRARY_DIR}/mount_gpt_util.sh"
123 get_board_from_image "${SRC_IMAGE}"
124 )"
125fi
126
127if [[ ! -d "/build/${FLAGS_board}" ]]; then
128 # Using board options and overrides requires that the board sysroot be setup
129 # (in order to read kernel and disk-image options).
130 # OTOH, we don't actually need to build any packages / update the host
131 # sysroot.
Alex Kleind65766f2019-01-11 14:57:34 -0700132 setup_board --quiet --board="${FLAGS_board}" \
133 --skip-toolchain-update --skip-chroot-upgrade --skip-board-pkg-init
Prathmesh Prabhuaa96e572017-08-31 22:42:37 -0700134fi
Bertrand SIMONNET1e77c192015-06-03 15:22:01 -0700135. "${BUILD_LIBRARY_DIR}/board_options.sh" || exit 1
136. "${SCRIPT_ROOT}/build_library/disk_layout_util.sh" || exit 1
137
Rahul Chaturvedib5643e82010-07-09 10:46:05 +0530138# Memory units are in MBs
Greg Spencer798d75f2011-02-01 22:04:49 -0800139TEMP_IMG="$(dirname "${SRC_IMAGE}")/vm_temp_image.bin"
Rahul Chaturvedib5643e82010-07-09 10:46:05 +0530140
Rahul Chaturvedib5643e82010-07-09 10:46:05 +0530141# Split apart the partitions and make some new ones
Gabe Black067b3542014-09-23 01:15:14 -0700142SRC_DEV=$(loopback_partscan "${SRC_IMAGE}")
Rahul Chaturvedib5643e82010-07-09 10:46:05 +0530143
144# Fix the kernel command line
Gabe Black067b3542014-09-23 01:15:14 -0700145SRC_STATE="${SRC_DEV}"p1
146SRC_ROOTFS="${SRC_DEV}"p3
147SRC_KERN="${SRC_DEV}"p4
148SRC_OEM="${SRC_DEV}"p8
149SRC_ESP="${SRC_DEV}"p12
Mike Frysinger629b6ad2021-01-25 23:23:31 -0500150STATEFUL_SIZE_BYTES=$(get_filesystem_size "${FLAGS_disk_layout}" 1)
151STATEFUL_SIZE_MEGABYTES=$(( STATEFUL_SIZE_BYTES / 1024 / 1024 ))
152original_image_size=$(bd_safe_size "${SRC_STATE}")
153if [ "${original_image_size}" -gt "${STATEFUL_SIZE_BYTES}" ]; then
154 if [ $(( original_image_size - STATEFUL_SIZE_BYTES )) -lt \
155 $(( 1024 * 1024 )) ]; then
156 # cgpt.py sometimes makes the stateful a tiny bit larger to
157 # counteract alignment losses.
158 # This is fine -- just keep using the slightly larger partition as it is.
159 TEMP_STATE="${SRC_STATE}"
Sam Hurstc94b7f92018-06-07 07:14:54 -0700160 else
Mike Frysinger629b6ad2021-01-25 23:23:31 -0500161 die "Cannot resize stateful image to smaller than original. Exiting."
Rahul Chaturvedie770e162010-07-15 00:35:11 +0530162 fi
Mike Frysinger629b6ad2021-01-25 23:23:31 -0500163else
164 echo "Resizing stateful partition to ${STATEFUL_SIZE_MEGABYTES}MB"
165 # Extend the original file size to the new size.
166 TEMP_STATE="${TEMP_DIR}"/stateful
167 # Create TEMP_STATE as a regular user so a regular user can delete it.
168 sudo dd if="${SRC_STATE}" bs=16M status=none > "${TEMP_STATE}"
169 sudo e2fsck -pf "${TEMP_STATE}"
170 sudo resize2fs "${TEMP_STATE}" ${STATEFUL_SIZE_MEGABYTES}M
Rahul Chaturvedib5643e82010-07-09 10:46:05 +0530171fi
Rahul Chaturvedib5643e82010-07-09 10:46:05 +0530172TEMP_PMBR="${TEMP_DIR}"/pmbr
173dd if="${SRC_IMAGE}" of="${TEMP_PMBR}" bs=512 count=1
174
Gabe Black067b3542014-09-23 01:15:14 -0700175# Set up a new partition table.
176PARTITION_SCRIPT_PATH=$(mktemp)
177write_partition_script "${FLAGS_disk_layout}" "${PARTITION_SCRIPT_PATH}"
178. "${PARTITION_SCRIPT_PATH}"
179write_partition_table "${TEMP_IMG}" "${TEMP_PMBR}"
180rm "${PARTITION_SCRIPT_PATH}"
181
182DST_DEV=$(loopback_partscan "${TEMP_IMG}")
183DST_STATE="${DST_DEV}"p1
184DST_ROOTFS="${DST_DEV}"p3
185DST_KERN="${DST_DEV}"p4
186DST_OEM="${DST_DEV}"p8
187DST_ESP="${DST_DEV}"p12
188
189# Copy into the partition parts of the file.
LaMont Jonesb24a4bd2019-06-17 09:00:34 -0600190# When copying to (or from) a non-regular file, cp ignores --sparse. Since we
191# have created a collection of empty partitions for the new image above, we can
192# use 'dd conv=sparse' to both speed up the copy, and (apparently) avoid
193# b/135292499. See also crbug.com/957712. This only works because we know that
194# the destination partition is all zeros.
195sudo dd if="${SRC_ROOTFS}" of="${DST_ROOTFS}" conv=sparse bs=2M
196sudo dd if="${TEMP_STATE}" of="${DST_STATE}" conv=sparse bs=2M
197sudo dd if="${SRC_ESP}" of="${DST_ESP}" conv=sparse bs=2M
198sudo dd if="${SRC_OEM}" of="${DST_OEM}" conv=sparse bs=2M
Allen Webb6b80db62019-06-12 10:06:01 -0700199sync
Gabe Black067b3542014-09-23 01:15:14 -0700200
Rahul Chaturvedib5643e82010-07-09 10:46:05 +0530201TEMP_MNT=$(mktemp -d)
Will Drewryefce6682010-07-23 19:43:27 -0500202TEMP_ESP_MNT=$(mktemp -d)
Rahul Chaturvedib5643e82010-07-09 10:46:05 +0530203mkdir -p "${TEMP_MNT}"
Gabe Black067b3542014-09-23 01:15:14 -0700204enable_rw_mount "${DST_ROOTFS}"
205sudo mount "${DST_ROOTFS}" "${TEMP_MNT}"
Will Drewryefce6682010-07-23 19:43:27 -0500206mkdir -p "${TEMP_ESP_MNT}"
Gabe Black067b3542014-09-23 01:15:14 -0700207sudo mount "${DST_ESP}" "${TEMP_ESP_MNT}"
Will Drewryefce6682010-07-23 19:43:27 -0500208
Will Drewry537caa92010-08-20 20:30:56 -0500209# Unmount everything prior to building a final image
Mike Frysinger9ebc8ce2015-04-06 13:15:01 -0400210trap 'die_err_trap' INT TERM EXIT
Rahul Chaturvedib5643e82010-07-09 10:46:05 +0530211cleanup
212
Gabe Blackb9827592014-09-03 21:58:11 -0700213# Make the built-image bootable.
Will Drewry537caa92010-08-20 20:30:56 -0500214# NOTE: The TEMP_IMG must live in the same image dir as the original image
215# to operate automatically below.
216${SCRIPTS_DIR}/bin/cros_make_image_bootable $(dirname "${TEMP_IMG}") \
217 $(basename "${TEMP_IMG}") \
Arkaitz Ruiz Alvarez2bf88592011-07-07 15:47:31 -0700218 --force_developer_mode
Will Drewry537caa92010-08-20 20:30:56 -0500219
Gabe Black34bbc102014-09-10 16:35:56 -0700220IMAGE_DEV=""
Gabe Blackfbdb1d52014-09-22 23:43:35 -0700221detach_loopback() {
Gabe Black34bbc102014-09-10 16:35:56 -0700222 if [ -n "${IMAGE_DEV}" ]; then
Gabe Blackfbdb1d52014-09-22 23:43:35 -0700223 loopback_detach "${IMAGE_DEV}"
Gabe Black34bbc102014-09-10 16:35:56 -0700224 fi
225}
Mike Frysinger9ebc8ce2015-04-06 13:15:01 -0400226trap 'ret=$?; detach_loopback; die_err_trap ${ret}' INT TERM EXIT
Gabe Black34bbc102014-09-10 16:35:56 -0700227
Gabe Blackb9827592014-09-03 21:58:11 -0700228# cros_make_image_bootable made the kernel in slot A recovery signed. We want
229# it to be normally signed like the one in slot B, so copy B into A.
LaMont Jonesb24a4bd2019-06-17 09:00:34 -0600230# Because cros_make_image_bootable overwrote p2 above, we cannot do a sparse
231# copy.
Gabe Blackfbdb1d52014-09-22 23:43:35 -0700232IMAGE_DEV=$(loopback_partscan "${TEMP_IMG}")
LaMont Jonesb24a4bd2019-06-17 09:00:34 -0600233sudo dd if=${IMAGE_DEV}p4 of=${IMAGE_DEV}p2 bs=2M
Allen Webb6b80db62019-06-12 10:06:01 -0700234sync
Gabe Blackb9827592014-09-03 21:58:11 -0700235
Mike Frysinger9ebc8ce2015-04-06 13:15:01 -0400236trap 'die_err_trap' INT TERM EXIT
237switch_to_strict_mode
Gabe Blackfbdb1d52014-09-22 23:43:35 -0700238loopback_detach "${IMAGE_DEV}"
Gabe Blackb9827592014-09-03 21:58:11 -0700239
Rahul Chaturvedib5643e82010-07-09 10:46:05 +0530240echo Creating final image
Gaurav Shah550edca2014-09-25 11:13:55 -0700241mv "${TEMP_IMG}" "${FLAGS_to}/${DEFAULT_QEMU_IMAGE}"
Rahul Chaturvedib5643e82010-07-09 10:46:05 +0530242
Gabe Black067b3542014-09-23 01:15:14 -0700243rm -rf "${TEMP_IMG}"
Rahul Chaturvedib5643e82010-07-09 10:46:05 +0530244
245echo "Created image at ${FLAGS_to}"
246
Nicolas Norvez82f51ec2018-01-26 10:10:37 -0800247echo "You can start the image with:"
Achuith Bhandarkar0faf2d12020-06-30 08:36:32 +0000248echo "cros_vm --start --board ${FLAGS_board} \
249--image-path ${FLAGS_to}/${DEFAULT_QEMU_IMAGE}"