blob: da2f604f2590002d4637e1b3ee93504c3057e16a [file] [log] [blame]
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -08001#!/bin/bash
2
Taylor Hutt60da6422011-06-02 13:54:43 -07003# Copyright (c) 2009-2011 The Chromium OS Authors. All rights reserved.
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -08004# Use of this source code is governed by a BSD-style license that can be
5# found in the LICENSE file.
6
7# Script to update the kernel on a live running ChromiumOS instance.
8
Brian Harringaa13ea42012-03-15 18:31:03 -07009SCRIPT_ROOT=$(dirname $(readlink -f "$0"))
David James359d3e12012-07-10 13:09:48 -070010. "${SCRIPT_ROOT}/common.sh" || exit 1
11. "${SCRIPT_ROOT}/remote_access.sh" || exit 1
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -080012
Mandeep Singh Baines2f3b5fc2011-01-14 14:20:12 -080013# Script must be run inside the chroot.
Greg Spencer798d75f2011-02-01 22:04:49 -080014restart_in_chroot_if_needed "$@"
Mandeep Singh Baines2f3b5fc2011-01-14 14:20:12 -080015
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -080016DEFINE_string board "" "Override board reported by target"
Olof Johanssonf53fa0d2011-01-26 13:06:46 -080017DEFINE_string device "" "Override boot device reported by target"
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -080018DEFINE_string partition "" "Override kernel partition reported by target"
Doug Andersonfcaed8a2014-07-09 11:34:29 -070019DEFINE_string rootoff "" "Override root offset"
Olof Johanssonf53fa0d2011-01-26 13:06:46 -080020DEFINE_string arch "" "Override architecture reported by target"
Nicolas Boichatabdf6642016-07-04 11:30:44 +080021DEFINE_boolean ignore_verity $FLAGS_FALSE "Update kernel even if system is using verity"
Olof Johansson8488f5a2011-04-20 17:27:37 -070022DEFINE_boolean reboot $FLAGS_TRUE "Reboot system after update"
Doug Anderson5a21b442012-12-07 11:47:31 -080023DEFINE_boolean vboot $FLAGS_TRUE "Update the vboot kernel"
Olof Johansson68cbfaf2013-04-23 14:06:28 -070024DEFINE_boolean syslinux $FLAGS_TRUE "Update the syslinux kernel"
Olof Johansson45225c92013-10-15 17:32:48 -070025DEFINE_boolean bootonce $FLAGS_FALSE "Mark kernel partition as boot once"
Olof Johansson4a7b2882013-10-16 11:59:58 -070026DEFINE_boolean remote_bootargs $FLAGS_FALSE "Use bootargs from running kernel on target"
Nicolas Boichata595deb2017-02-07 18:53:44 +080027DEFINE_boolean firmware $FLAGS_FALSE "Also update firmwares (/lib/firmware)"
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -080028
Doug Anderson549f3b52013-09-26 14:46:18 -070029ORIG_ARGS=("$@")
30
Mandeep Singh Baines2f3b5fc2011-01-14 14:20:12 -080031# Parse command line.
32FLAGS "$@" || exit 1
33eval set -- "${FLAGS_ARGV}"
34
35# Only now can we die on error. shflags functions leak non-zero error codes,
Brian Harring7f175a52012-03-02 05:37:00 -080036# so will die prematurely if 'switch_to_strict_mode' is specified before now.
37switch_to_strict_mode
Mandeep Singh Baines2f3b5fc2011-01-14 14:20:12 -080038
Mike Frysinger6b1abb22012-05-11 13:44:06 -040039cleanup() {
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -080040 cleanup_remote_access
41 rm -rf "${TMP}"
42}
43
Mike Frysinger6b1abb22012-05-11 13:44:06 -040044learn_device() {
Olof Johanssonf53fa0d2011-01-26 13:06:46 -080045 [ -n "${FLAGS_device}" ] && return
46 remote_sh df /mnt/stateful_partition
47 FLAGS_device=$(echo "${REMOTE_OUT}" | awk '/dev/ {print $1}' | sed s/1\$//)
48 info "Target reports root device is ${FLAGS_device}"
49}
50
Ben Zhang15fa4b12017-06-19 20:02:57 -070051# Delete the fixed numbers after R65 when we don't care about <R57 upgrades.
52load_default_partition_numbers() {
53 PARTITION_NUM_KERN_A=2
54 PARTITION_NUM_ROOT_A=3
55 PARTITION_NUM_KERN_B=4
56 PARTITION_NUM_EFI_SYSTEM=12
57}
58
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -080059# Ask the target what the kernel partition is
Mike Frysinger6b1abb22012-05-11 13:44:06 -040060learn_partition_and_ro() {
Mandeep Singh Bainese39579a2011-03-04 15:58:57 -080061 ! remote_sh rootdev
Paul Taysoma64d9db2012-09-21 13:30:43 -070062 if [ "${REMOTE_OUT%%-*}" == "/dev/dm" ]; then
63 remote_sh rootdev -s
Olof Johansson8488f5a2011-04-20 17:27:37 -070064 REMOTE_VERITY=${FLAGS_TRUE}
Nicolas Boichatabdf6642016-07-04 11:30:44 +080065 if [[ ${FLAGS_ignore_verity} -eq ${FLAGS_TRUE} ]]; then
66 warn "System is using verity: not updating firmware/modules"
67 else
68 warn "System is using verity: First remove rootfs verification using"
69 warn "/usr/share/vboot/bin/make_dev_ssd.sh --remove_rootfs_verification"
70 warn "on the DUT, or add --ignore_verity parameter to this command."
Brian Norrised751572018-03-30 13:48:43 -070071 die_notrace
Nicolas Boichatabdf6642016-07-04 11:30:44 +080072 fi
Olof Johansson8488f5a2011-04-20 17:27:37 -070073 else
74 REMOTE_VERITY=${FLAGS_FALSE}
75 info "System is not using verity: updating firmware and modules"
Mandeep Singh Bainese39579a2011-03-04 15:58:57 -080076 fi
Olof Johansson4996bfb2013-11-13 12:58:52 -080077 [ -n "${FLAGS_partition}" ] && return
Steven 'Steve' Kendall019f38f2016-06-09 12:43:28 -040078 if [ "${REMOTE_OUT}" == "${FLAGS_device}${PARTITION_NUM_ROOT_A}" ]; then
79 FLAGS_partition="${FLAGS_device}${PARTITION_NUM_KERN_A}"
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -080080 else
Steven 'Steve' Kendall019f38f2016-06-09 12:43:28 -040081 FLAGS_partition="${FLAGS_device}${PARTITION_NUM_KERN_B}"
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -080082 fi
83 if [ -z "${FLAGS_partition}" ]; then
Brian Norrised751572018-03-30 13:48:43 -070084 die_notrace "Partition required"
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -080085 fi
Doug Anderson5a21b442012-12-07 11:47:31 -080086 if [ ${REMOTE_VERITY} -eq ${FLAGS_TRUE} ]; then
87 info "Target reports kernel partition is ${FLAGS_partition}"
88 if [ ${FLAGS_vboot} -eq ${FLAGS_FALSE} ]; then
Brian Norrised751572018-03-30 13:48:43 -070089 die_notrace "Must update vboot when target is using verity"
Doug Anderson5a21b442012-12-07 11:47:31 -080090 fi
91 fi
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -080092}
93
Olof Johansson4a7b2882013-10-16 11:59:58 -070094get_bootargs() {
Mike Frysingerd4b6f952017-04-10 16:42:17 -040095 local local_config="${SRC_ROOT}/build/images/${FLAGS_board}/latest/config.txt"
96
97 # Autodetect by default. https://crbug.com/316239
98 # This isn't quite right if people use --noremote_bootargs, but that's not
99 # a scenario people do today, so we won't worry about it.
100 if [[ ${FLAGS_remote_bootargs} -eq ${FLAGS_FALSE} && \
101 ! -e "${local_config}" ]]; then
102 warn "Local kernel config does not exist: ${local_config}"
103 FLAGS_remote_bootargs=${FLAGS_TRUE}
104 fi
105
Olof Johansson4a7b2882013-10-16 11:59:58 -0700106 if [ ${FLAGS_remote_bootargs} -eq ${FLAGS_TRUE} ] ; then
107 info "Using remote bootargs"
Peter Shih0bacb332018-11-27 16:14:06 +0800108 remote_sh cat /proc/cmdline
109 # Remove multiple instances of cros_secure, https://crbug.com/907772
110 echo "${REMOTE_OUT}" | sed -E 's/\b(cros_secure )+/cros_secure /g'
Olof Johansson4a7b2882013-10-16 11:59:58 -0700111 else
Doug Andersonfcaed8a2014-07-09 11:34:29 -0700112 if [ -n "${FLAGS_rootoff}" ]; then
Mike Frysingerd4b6f952017-04-10 16:42:17 -0400113 sed "s/PARTNROFF=1/PARTNROFF=${FLAGS_rootoff}/" "${local_config}"
Doug Andersonfcaed8a2014-07-09 11:34:29 -0700114 else
Mike Frysingerd4b6f952017-04-10 16:42:17 -0400115 cat "${local_config}"
Doug Andersonfcaed8a2014-07-09 11:34:29 -0700116 fi
Olof Johansson4a7b2882013-10-16 11:59:58 -0700117 fi
118}
119
Mike Frysinger6b1abb22012-05-11 13:44:06 -0400120make_kernelimage() {
Tom Wai-Hong Tam6b50a072011-05-25 17:00:15 +0800121 local bootloader_path
122 local kernel_image
Olof Johansson4a7b2882013-10-16 11:59:58 -0700123 local config_path="$(mktemp /tmp/config.txt.XXXXX)"
Olof Johanssonf53fa0d2011-01-26 13:06:46 -0800124 if [[ "${FLAGS_arch}" == "arm" ]]; then
Taylor Hutt60da6422011-06-02 13:54:43 -0700125 name="bootloader.bin"
126 bootloader_path="${SRC_ROOT}/build/images/${FLAGS_board}/latest/${name}"
Mike Frysinger085c5d92017-08-16 17:29:12 -0400127 # If there is no local bootloader stub, create a dummy file. This matches
128 # build_kernel_image.sh. If we wanted to be super paranoid, we could copy
129 # and extract it from the remote image, if it had one.
130 if [[ ! -e "${bootloader_path}" ]]; then
131 warn "Bootloader does not exist; creating a stub: ${bootloader_path}"
132 mkdir -p "${bootloader_path%/*}"
133 truncate -s 512 "${bootloader_path}"
134 fi
Tom Wai-Hong Tam6b50a072011-05-25 17:00:15 +0800135 kernel_image="/build/${FLAGS_board}/boot/vmlinux.uimg"
Olof Johanssonf53fa0d2011-01-26 13:06:46 -0800136 else
Tom Wai-Hong Tam6b50a072011-05-25 17:00:15 +0800137 bootloader_path="/lib64/bootstub/bootstub.efi"
138 kernel_image="/build/${FLAGS_board}/boot/vmlinuz"
139 fi
Olof Johansson4a7b2882013-10-16 11:59:58 -0700140 get_bootargs > "${config_path}"
Kees Cook43a32132011-10-18 13:17:11 -0700141 vbutil_kernel --pack $TMP/new_kern.bin \
Olof Johanssonf53fa0d2011-01-26 13:06:46 -0800142 --keyblock /usr/share/vboot/devkeys/kernel.keyblock \
143 --signprivate /usr/share/vboot/devkeys/kernel_data_key.vbprivk \
144 --version 1 \
Olof Johansson4a7b2882013-10-16 11:59:58 -0700145 --config ${config_path} \
Tom Wai-Hong Tam6b50a072011-05-25 17:00:15 +0800146 --bootloader "${bootloader_path}" \
147 --vmlinuz "${kernel_image}" \
148 --arch "${FLAGS_arch}"
Olof Johansson4a7b2882013-10-16 11:59:58 -0700149 rm "${config_path}"
Olof Johanssonf53fa0d2011-01-26 13:06:46 -0800150}
151
Edward Hyunkoo Jeea979a902017-03-08 13:44:26 -0800152copy_kernelmodules() {
153 echo "copying modules"
154 local modules_dir=/build/"${FLAGS_board}"/lib/modules/
155 if [ ! -d "${modules_dir}" ]; then
156 info "No modules. Skipping."
157 return
158 fi
159 remote_send_to "${modules_dir}" /lib/modules/
160 local kernel_release
161 remote_sh "cd /lib/modules; echo *"
162 for kernel_release in "${REMOTE_OUT}"; do
163 local system_map="${modules_dir}"/"${kernel_release}"/build/System.map
164 if [ -r "${system_map}" ]; then
165 remote_sh mktemp -d /tmp/update_kernel_system_map_"${kernel_release}".XXXXXX
166 local temp_dir="${REMOTE_OUT}"
167 remote_cp_to "${system_map}" "${temp_dir}"
168 remote_sh depmod -ae -F "${temp_dir}"/System.map "${kernel_release}"
169 remote_sh rm -rf "${temp_dir}"
170 fi
171 done
172}
173
Mike Frysinger6b1abb22012-05-11 13:44:06 -0400174copy_kernelimage() {
Doug Andersonadf8a002012-12-17 11:40:34 -0800175 remote_sh dd of="${FLAGS_partition}" bs=4K < "${TMP}/new_kern.bin"
Olof Johansson8488f5a2011-04-20 17:27:37 -0700176}
177
Jonathan Kliegman775bc8e2012-08-14 12:30:49 -0400178check_kernelbuildtime() {
179 local version=$(readlink "/build/${FLAGS_board}/boot/vmlinuz" | cut -d- -f2-)
180 local build_dir="/build/${FLAGS_board}/lib/modules/${version}/build"
181 if [ "${build_dir}/Makefile" -nt "/build/${FLAGS_board}/boot/vmlinuz" ]; then
182 warn "Your build directory has been built more recently than"
183 warn "the installed kernel being updated to. Did you forget to"
184 warn "run 'cros_workon_make chromeos-kernel --install'?"
185 fi
186}
187
Olof Johansson45225c92013-10-15 17:32:48 -0700188mark_boot_once() {
189 local idx=${FLAGS_partition##*[^0-9]}
Chirantan Ekbotef25b4402014-05-06 12:42:18 -0700190 remote_sh cgpt add -i ${idx} -S 0 -T 1 -P 15 ${FLAGS_device%p}
Olof Johansson45225c92013-10-15 17:32:48 -0700191}
192
Olof Johansson68cbfaf2013-04-23 14:06:28 -0700193update_syslinux_kernel() {
194 # ARM does not have the syslinux directory, so skip it when the
Evan Greenfd9652b2018-10-01 13:36:02 -0700195 # partition is missing, the file system fails to mount, or the syslinux
196 # vmlinuz target is missing.
Olof Johansson68cbfaf2013-04-23 14:06:28 -0700197 echo "updating syslinux kernel"
Steven 'Steve' Kendall019f38f2016-06-09 12:43:28 -0400198 remote_sh grep $(echo ${FLAGS_device}${PARTITION_NUM_EFI_SYSTEM} | cut -d/ -f3) /proc/partitions
Olof Johansson68cbfaf2013-04-23 14:06:28 -0700199 if [ $(echo "$REMOTE_OUT" | wc -l) -eq 1 ]; then
Steven 'Steve' Kendall019f38f2016-06-09 12:43:28 -0400200 remote_sh mkdir -p /tmp/${PARTITION_NUM_EFI_SYSTEM}
Evan Greenfd9652b2018-10-01 13:36:02 -0700201 if remote_sh mount ${FLAGS_device}${PARTITION_NUM_EFI_SYSTEM} \
202 /tmp/${PARTITION_NUM_EFI_SYSTEM}; then
Olof Johansson68cbfaf2013-04-23 14:06:28 -0700203
Evan Greenfd9652b2018-10-01 13:36:02 -0700204 if [ "$FLAGS_partition" = "${FLAGS_device}${PARTITION_NUM_KERN_A}" ]; then
205 target="/tmp/${PARTITION_NUM_EFI_SYSTEM}/syslinux/vmlinuz.A"
206 else
207 target="/tmp/${PARTITION_NUM_EFI_SYSTEM}/syslinux/vmlinuz.B"
208 fi
209 remote_sh "test ! -f $target || cp /boot/vmlinuz $target"
210
211 remote_sh umount /tmp/${PARTITION_NUM_EFI_SYSTEM}
Olof Johansson68cbfaf2013-04-23 14:06:28 -0700212 fi
Steven 'Steve' Kendall019f38f2016-06-09 12:43:28 -0400213 remote_sh rmdir /tmp/${PARTITION_NUM_EFI_SYSTEM}
Olof Johansson68cbfaf2013-04-23 14:06:28 -0700214 fi
215}
216
Doug Anderson549f3b52013-09-26 14:46:18 -0700217multi_main() {
218 local host
219
220 IFS=","
221 for host in ${FLAGS_remote}; do
222 "$0" "${ORIG_ARGS[@]}" --remote="${host}" \
223 |& sed "s/^/${V_BOLD_YELLOW}${host}: ${V_VIDOFF}/" &
224 done
225 wait
226}
227
Mike Frysinger6b1abb22012-05-11 13:44:06 -0400228main() {
Doug Anderson549f3b52013-09-26 14:46:18 -0700229 # If there are commas in the --remote, run the script in parallel.
230 if [[ ${FLAGS_remote} == *,* ]]; then
231 multi_main
232 return $?
233 fi
234
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -0800235 trap cleanup EXIT
236
Kees Cook43a32132011-10-18 13:17:11 -0700237 TMP=$(mktemp -d /tmp/update_kernel.XXXXXX)
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -0800238
239 remote_access_init
240
Olof Johanssonf53fa0d2011-01-26 13:06:46 -0800241 learn_arch
242
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -0800243 learn_board
244
Olof Johanssonf53fa0d2011-01-26 13:06:46 -0800245 learn_device
246
Ian Coolidgec3d5d912017-03-07 14:21:28 -0800247 learn_partition_layout
Ben Zhang15fa4b12017-06-19 20:02:57 -0700248 if [[ -z "${PARTITION_NUM_KERN_A}" ]]; then
249 info "Target has no partition number info, use default instead"
250 load_default_partition_numbers
251 fi
Ian Coolidgec3d5d912017-03-07 14:21:28 -0800252
Olof Johansson8488f5a2011-04-20 17:27:37 -0700253 learn_partition_and_ro
Olof Johanssonf53fa0d2011-01-26 13:06:46 -0800254
Brian Norris9774b0d2017-06-15 13:44:26 -0700255 if ! remote_sh "test -e '${FLAGS_partition}'"; then
256 die_notrace "Could not find kernel partition on DUT; path='${FLAGS_partition}'"
257 fi
258
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -0800259 remote_sh uname -r -v
260
261 old_kernel="${REMOTE_OUT}"
262
Jonathan Kliegman775bc8e2012-08-14 12:30:49 -0400263 check_kernelbuildtime
264
Doug Anderson5a21b442012-12-07 11:47:31 -0800265 if [ ${FLAGS_vboot} -eq ${FLAGS_TRUE} ]; then
266 make_kernelimage
267 fi
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -0800268
Olof Johansson8488f5a2011-04-20 17:27:37 -0700269 if [[ ${REMOTE_VERITY} -eq ${FLAGS_FALSE} ]]; then
Olof Johansson8488f5a2011-04-20 17:27:37 -0700270 remote_sh mount -o remount,rw /
Kees Cook7d7d2ef2011-10-18 13:12:12 -0700271 echo "copying kernel"
Doug Anderson48b52002012-12-07 12:40:07 -0800272 remote_send_to /build/"${FLAGS_board}"/boot/ /boot/
Kees Cook7d7d2ef2011-10-18 13:12:12 -0700273
Olof Johansson68cbfaf2013-04-23 14:06:28 -0700274 if [ ${FLAGS_syslinux} -eq ${FLAGS_TRUE} ]; then
275 update_syslinux_kernel
Kees Cook7d7d2ef2011-10-18 13:12:12 -0700276 fi
Olof Johansson9a83e4e2012-08-17 02:45:12 -0700277
Edward Hyunkoo Jeea979a902017-03-08 13:44:26 -0800278 copy_kernelmodules
Olof Johansson9a83e4e2012-08-17 02:45:12 -0700279
Nicolas Boichata595deb2017-02-07 18:53:44 +0800280 if [[ ${FLAGS_firmware} -eq ${FLAGS_TRUE} ]]; then
281 echo "copying firmware"
282 remote_send_to /build/"${FLAGS_board}"/lib/firmware/ /lib/firmware/
283 else
284 info "Skipping update of firmware (per request)."
285 fi
Olof Johansson5a46bfb2010-12-22 12:14:21 -0800286 fi
287
Doug Anderson5a21b442012-12-07 11:47:31 -0800288 if [ ${FLAGS_vboot} -eq ${FLAGS_TRUE} ]; then
289 info "Copying vboot kernel image"
290 copy_kernelimage
291 else
292 info "Skipping update of vboot (per request)"
293 fi
Olof Johansson8488f5a2011-04-20 17:27:37 -0700294
Olof Johansson45225c92013-10-15 17:32:48 -0700295 if [ ${FLAGS_bootonce} -eq ${FLAGS_TRUE} ]; then
296 info "Marking kernel partition ${FLAGS_partition} as boot once"
297 mark_boot_once
298 fi
299
Jonathan Kliegmand6f3d072012-07-12 12:45:33 -0400300 # An early kernel panic can prevent the normal sync on reboot. Explicitly
301 # sync for safety to avoid random file system corruption.
302 remote_sh sync
303
Doug Andersonb2fe4652012-12-07 17:33:56 -0800304 if [ ${FLAGS_reboot} -eq ${FLAGS_TRUE} ]; then
Olof Johansson8488f5a2011-04-20 17:27:37 -0700305 remote_reboot
306
307 remote_sh uname -r -v
308 info "old kernel: ${old_kernel}"
309 info "new kernel: ${REMOTE_OUT}"
310 else
311 info "Not rebooting (per request)"
312 fi
Mandeep Singh Bainesa63cd2d2010-12-02 11:58:26 -0800313}
314
Olof Johanssonf53fa0d2011-01-26 13:06:46 -0800315main "$@"