Adriana Kobylak | 9cbfa2e | 2019-04-25 14:02:37 -0500 | [diff] [blame] | 1 | #!/bin/bash |
| 2 | set -eo pipefail |
| 3 | |
| 4 | # Get the root mtd device number (mtdX) from "/dev/ubiblockX_Y on /" |
| 5 | findrootmtd() { |
| 6 | rootmatch=" on / " |
| 7 | m="$(mount | grep "${rootmatch}" | grep "ubiblock")" |
| 8 | m="${m##*ubiblock}" |
| 9 | m="${m%_*}" |
| 10 | if [ -z "${m}" ]; then |
| 11 | # default to bmc mtd (0) |
| 12 | m=0 |
| 13 | fi |
| 14 | echo "mtd${m}" |
| 15 | } |
| 16 | |
| 17 | findrootubi() { |
| 18 | rootmatch=" on / " |
| 19 | m="$(mount | grep "${rootmatch}")" |
| 20 | m="${m##*ubiblock}" |
| 21 | m="${m% on*}" |
| 22 | echo "ubi${m}" |
| 23 | } |
| 24 | |
| 25 | # Get the mtd device number (mtdX) |
| 26 | findmtd() { |
| 27 | m="$(grep -xl "$1" /sys/class/mtd/*/name)" |
| 28 | m="${m%/name}" |
| 29 | m="${m##*/}" |
| 30 | echo "${m}" |
| 31 | } |
| 32 | |
| 33 | # Get the mtd device number only (return X of mtdX) |
| 34 | findmtdnum() { |
| 35 | m="$(findmtd "$1")" |
| 36 | m="${m##mtd}" |
| 37 | echo "${m}" |
| 38 | } |
| 39 | |
| 40 | # Get the ubi device number (ubiX_Y) |
| 41 | findubi() { |
| 42 | u="$(grep -xl "$1" /sys/class/ubi/ubi?/subsystem/ubi*/name)" |
| 43 | u="${u%/name}" |
| 44 | u="${u##*/}" |
| 45 | echo "${u}" |
| 46 | } |
| 47 | |
| 48 | # Get the ubi device number (ubiX_Y) on a specific mtd |
| 49 | findubi_onmtd() { |
| 50 | u="$(grep -xl "$1" /sys/class/ubi/ubi"$2"/subsystem/ubi"$2"*/name)" |
| 51 | u="${u%/name}" |
| 52 | u="${u##*/}" |
| 53 | echo "${u}" |
| 54 | } |
| 55 | |
| 56 | # Get all ubi device names on a specific mtd that match requested string |
| 57 | findubiname_onmtd() { |
| 58 | u="$(grep -h "$1" /sys/class/ubi/ubi"$2"/subsystem/ubi"$2"*/name)" |
| 59 | u="${u%/name}" |
| 60 | u="${u##*/}" |
| 61 | echo "${u}" |
| 62 | } |
| 63 | |
| 64 | # Get the name from the requested ubiX_Y volume |
| 65 | findname() { |
| 66 | n="$(cat /sys/class/ubi/$1/name)" |
| 67 | echo "${n}" |
| 68 | } |
| 69 | |
| 70 | # Set the u-boot envs that perform a side switch on failure to boot |
| 71 | set_wdt2bite() { |
| 72 | if ! fw_printenv wdt2bite 2>/dev/null; then |
| 73 | fw_setenv wdt2bite "mw.l 0x1e785024 0xa 1; mw.b 0x1e78502c 0xb3 1" |
| 74 | fw_setenv bootalt "run wdt2bite" |
| 75 | fw_setenv obmc_bootcmd "ubi part obmc-ubi; run do_rwreset; ubi read \ |
| 76 | \${loadaddr} \${kernelname}; bootm \${loadaddr} || run bootalt" |
| 77 | fi |
| 78 | } |
| 79 | |
| 80 | # Make space on flash before creating new volumes. This can be enhanced |
| 81 | # determine current flash usage. For now only keep a "keepmax" number of them |
| 82 | ubi_remove_volumes() |
| 83 | { |
| 84 | rootubi="$(findrootubi)" |
| 85 | rootname="$(findname "${rootubi}")" |
| 86 | rootversion="${rootname##*-}" |
| 87 | rootkernel="kernel-${rootversion}" |
| 88 | |
| 89 | # Just keep max number of volumes before updating, don't delete the version |
| 90 | # the BMC is booted from, and when a version is identified to be deleted, |
| 91 | # delete both the rofs and kernel volumes for that version. |
| 92 | rmnames="$(findubiname_onmtd "${name%-*}-" "${ro}")" |
| 93 | rmnames=(${rmnames}) |
| 94 | ubicount="${#rmnames[@]}" |
| 95 | while [ ${ubicount} -ge ${keepmax} ]; do |
| 96 | # Loop through existing volumes and skip currently active ones |
| 97 | for (( index=0; index<${#rmnames[@]}; index++ )); do |
| 98 | rmname="${rmnames[${index}]}" |
| 99 | rmversion="${rmname##*-}" |
| 100 | [ "${rmversion}" == "${version}" ] && continue |
| 101 | rmubi="$(findubi_onmtd "rofs-${rmversion}" "${ro}")" |
| 102 | if [[ ( "${rmubi}" != "${rootubi}" ) && |
| 103 | ( "${rmname}" != "${rootkernel}" ) ]]; then |
| 104 | ubi_remove "rofs-${rmversion}" "${ro}" |
| 105 | ubi_remove "kernel-${rmversion}" "${ro}" |
| 106 | # Remove priority value |
| 107 | fw_setenv "${rmversion}" |
| 108 | break |
| 109 | fi |
| 110 | done |
| 111 | # Decrease count regardless to avoid an infinite loop |
| 112 | (( ubicount-- )) |
| 113 | done |
| 114 | } |
| 115 | |
| 116 | ubi_rw() { |
| 117 | rwmtd="$(findmtd "${reqmtd}")" |
| 118 | rw="${rwmtd#mtd}" |
| 119 | ubidev="/dev/ubi${rw}" |
| 120 | |
| 121 | # Update rwfs_size, check imgsize was specified, otherwise it'd clear the var |
| 122 | if [ ! -z "$imgsize" ]; then |
| 123 | rwsize="$(fw_printenv -n rwfs_size 2>/dev/null)" || true |
| 124 | if [[ "${imgsize}" != "${rwsize}" ]]; then |
| 125 | fw_setenv rwfs_size "${imgsize}" |
| 126 | fi |
| 127 | fi |
| 128 | |
| 129 | vol="$(findubi "${name}")" |
| 130 | if [ -z "${vol}" ]; then |
| 131 | ubimkvol "${ubidev}" -N "${name}" -s "${imgsize}" |
| 132 | fi |
| 133 | } |
| 134 | |
| 135 | ubi_ro() { |
| 136 | keepmax=2 # Default 2 volumes per mtd |
| 137 | romtd="$(findmtd "${reqmtd}")" |
| 138 | romtd2="$(findmtd "${reqmtd2}")" |
| 139 | |
| 140 | if [ ! "${romtd}" == "${romtd2}" ]; then |
| 141 | # Request to use alternate mtd device, choose the non-root one |
| 142 | keepmax=1 # 1 volume on each of the requested mtds |
| 143 | rootmtd="$(findrootmtd)" |
| 144 | if [ "${rootmtd}" == "${romtd}" ]; then |
| 145 | romtd="${romtd2}" |
| 146 | fi |
| 147 | fi |
| 148 | ro="${romtd#mtd}" |
| 149 | ubidev="/dev/ubi${ro}" |
| 150 | |
| 151 | ubi_remove_volumes |
| 152 | |
| 153 | if [ -z "${imgfile}" ]; then |
| 154 | echo "Unable to create read-only volume. Image file not specified." |
| 155 | return 1 |
| 156 | fi |
| 157 | |
| 158 | # Create a ubi volume, dynamically sized to fit BMC image if size unspecified |
| 159 | img="/tmp/images/${version}/${imgfile}" |
| 160 | imgsize="$(stat -c '%s' ${img})" |
| 161 | |
| 162 | vol="$(findubi "${name}")" |
| 163 | if [ ! -z "${vol}" ]; then |
| 164 | # Allow a duplicate kernel volume on the alt mtd |
| 165 | if [[ "${name}" =~ "kernel" ]]; then |
| 166 | vol="$(findubi_onmtd "${name}" "${ro}")" |
| 167 | fi |
| 168 | fi |
| 169 | if [ -z "${vol}" ]; then |
| 170 | ubimkvol "${ubidev}" -N "${name}" -s "${imgsize}" --type=static |
| 171 | vol="$(findubi "${name}")" |
| 172 | fi |
| 173 | } |
| 174 | |
| 175 | # Squashfs images need a ubi block |
| 176 | ubi_block() { |
| 177 | vol="$(findubi "${name}")" |
| 178 | ubidevid="${vol#ubi}" |
| 179 | block="/dev/ubiblock${ubidevid}" |
| 180 | if [ ! -e "$block" ]; then |
| 181 | ubiblock --create "/dev/ubi${ubidevid}" |
| 182 | fi |
| 183 | } |
| 184 | |
| 185 | ubi_updatevol() { |
| 186 | vol="$(findubi "${name}")" |
| 187 | ubidevid="${vol#ubi}" |
| 188 | img="/tmp/images/${version}/${imgfile}" |
| 189 | ubiupdatevol "/dev/ubi${ubidevid}" "${img}" |
| 190 | } |
| 191 | |
| 192 | ubi_remove() { |
| 193 | rmname="$1" |
| 194 | rmmtd="$2" |
| 195 | if [ ! -z "${rmmtd}" ]; then |
| 196 | vol="$(findubi_onmtd "${rmname}" "${rmmtd}")" |
| 197 | else |
| 198 | vol="$(findubi "${rmname}")" |
| 199 | fi |
| 200 | |
| 201 | if [ ! -z "$vol" ]; then |
| 202 | vol="${vol%_*}" |
| 203 | |
| 204 | if grep -q "$rmname" /proc/mounts; then |
| 205 | mountdir=$(grep "$rmname" /proc/mounts | cut -d " " -f 2) |
| 206 | umount "$mountdir" |
| 207 | rm -r "$mountdir" |
| 208 | fi |
| 209 | |
| 210 | ubirmvol "/dev/${vol}" -N "$rmname" |
| 211 | fi |
| 212 | } |
| 213 | |
| 214 | ubi_cleanup() { |
| 215 | # When ubi_cleanup is run, it expects one or no active version. |
| 216 | activeVersion=$(busctl --list --no-pager tree \ |
| 217 | xyz.openbmc_project.Software.BMC.Updater | \ |
| 218 | grep /xyz/openbmc_project/software/ | tail -c 9) |
| 219 | |
| 220 | if [[ -z "$activeVersion" ]]; then |
| 221 | vols=$(ubinfo -a | grep "rofs-" | cut -c 14-) |
| 222 | vols=(${vols}) |
| 223 | else |
| 224 | vols=$(ubinfo -a | grep "rofs-" | \ |
| 225 | grep -v "$activeVersion" | cut -c 14-) |
| 226 | vols=(${vols}) |
| 227 | fi |
| 228 | |
| 229 | for (( index=0; index<${#vols[@]}; index++ )); do |
| 230 | ubi_remove ${vols[index]} |
| 231 | done |
| 232 | } |
| 233 | |
| 234 | mount_alt_rwfs() { |
| 235 | altNum="$(findmtdnum "alt-bmc")" |
| 236 | if [ ! -z "${altNum}" ]; then |
| 237 | altRwfs=$(ubinfo -a -d ${altNum} | grep -w "rwfs") || true |
| 238 | if [ ! -z "${altRwfs}" ]; then |
| 239 | altVarMount="/media/alt/var" |
| 240 | mkdir -p "${altVarMount}" |
| 241 | if mount ubi"${altNum}":rwfs "${altVarMount}" -t ubifs -o defaults; then |
| 242 | mkdir -p "${altVarMount}"/persist/etc |
| 243 | fi |
| 244 | fi |
| 245 | fi |
| 246 | } |
| 247 | |
| 248 | remount_ubi() { |
| 249 | bmcmtd="$(findmtd "bmc")" |
| 250 | altbmcmtd="$(findmtd "alt-bmc")" |
| 251 | mtds="${bmcmtd: -1}","${altbmcmtd: -1}" |
| 252 | |
| 253 | IFS=',' read -r -a mtds <<< "$mtds" |
| 254 | mtds=($(echo "${mtds[@]}" | tr ' ' '\n' | sort -u | tr '\n' ' ')) |
| 255 | for mtd in ${mtds[@]}; do |
| 256 | # Get information on all ubi volumes |
| 257 | ubinfo=$(ubinfo -d ${mtd}) |
| 258 | presentVolumes=${ubinfo##*:} |
| 259 | IFS=', ' read -r -a array <<< "$presentVolumes" |
| 260 | for element in ${array[@]}; do |
| 261 | elementProperties=$(ubinfo -d $mtd -n $element) |
| 262 | # Get ubi volume name by getting rid of additional properties |
| 263 | name=${elementProperties#*Name:} |
| 264 | name="${name%Character*}" |
| 265 | name="$(echo -e "${name}" | tr -d '[:space:]')" |
| 266 | |
| 267 | if [[ ${name} == rofs-* ]]; then |
| 268 | mountdir="/media/${name}" |
| 269 | |
| 270 | if [ ! -d ${mountdir} ]; then |
| 271 | mkdir -p "${mountdir}" |
| 272 | # U-Boot will create the ubiblock for the running version, but not |
| 273 | # for the version on the other chip |
| 274 | if [ ! -e "/dev/ubiblock${mtd}_${element}" ]; then |
| 275 | ubiblock --create /dev/ubi${mtd}_${element} |
| 276 | fi |
| 277 | mount -t squashfs -o ro "/dev/ubiblock${mtd}_${element}" "${mountdir}" |
| 278 | fi |
| 279 | fi |
| 280 | done |
| 281 | done |
| 282 | |
| 283 | set_wdt2bite |
| 284 | } |
| 285 | |
| 286 | # Read the current env variable and set it on the alternate boot env |
| 287 | copy_env_var_to_alt() { |
| 288 | varName=$1 |
| 289 | value="$(fw_printenv -n "${varName}")" |
| 290 | fw_setenv -c /etc/alt_fw_env.config "${varName}" "${value}" |
| 291 | } |
| 292 | |
| 293 | # When the alternate bmc chip boots, u-boot thinks its the primary mtdX. |
| 294 | # Therefore need to swap the chip numbers when copying the ubiblock and root to |
| 295 | # alternate bmc u-boot environment. |
| 296 | copy_ubiblock_to_alt() { |
| 297 | value="$(fw_printenv -n ubiblock)" |
| 298 | bmcNum="$(findmtdnum "bmc")" |
| 299 | altNum="$(findmtdnum "alt-bmc")" |
| 300 | replaceAlt="${value/${altNum},/${bmcNum},}" |
| 301 | |
| 302 | if [[ "${value}" == "${replaceAlt}" ]]; then |
| 303 | replaceBmc="${value/${bmcNum},/${altNum},}" |
| 304 | value=${replaceBmc} |
| 305 | else |
| 306 | value=${replaceAlt} |
| 307 | fi |
| 308 | |
| 309 | fw_setenv -c /etc/alt_fw_env.config ubiblock "${value}" |
| 310 | } |
| 311 | |
| 312 | copy_root_to_alt() { |
| 313 | value="$(fw_printenv -n root)" |
| 314 | bmcNum="$(findmtdnum "bmc")" |
| 315 | altNum="$(findmtdnum "alt-bmc")" |
| 316 | replaceAlt="${value/${altNum}_/${bmcNum}_}" |
| 317 | |
| 318 | if [[ "${value}" == "${replaceAlt}" ]]; then |
| 319 | replaceBmc="${value/${bmcNum}_/${altNum}_}" |
| 320 | value=${replaceBmc} |
| 321 | else |
| 322 | value=${replaceAlt} |
| 323 | fi |
| 324 | |
| 325 | fw_setenv -c /etc/alt_fw_env.config root "${value}" |
| 326 | } |
| 327 | |
| 328 | ubi_setenv() { |
| 329 | # The U-Boot environment maintains two banks of environment variables. |
| 330 | # The banks need to be consistent with each other to ensure that these |
| 331 | # variables can reliably be read from file. In order to guarantee that the |
| 332 | # banks are both correct, we need to run fw_setenv twice. |
| 333 | variable=$1 |
| 334 | if [[ "$variable" == *"="* ]]; then |
| 335 | varName="${variable%=*}" |
| 336 | value="${variable##*=}" |
| 337 | # Write only if var is not set already to the requested value |
| 338 | currentValue="$(fw_printenv -n "${varName}" 2>/dev/null)" || true |
| 339 | if [[ "${currenValue}" != "${value}" ]]; then |
| 340 | fw_setenv "$varName" "$value" |
| 341 | fw_setenv "$varName" "$value" |
| 342 | fi |
| 343 | else |
| 344 | fw_setenv "$variable" |
| 345 | fw_setenv "$variable" |
| 346 | fi |
| 347 | } |
| 348 | |
| 349 | mtd_write() { |
| 350 | flashmtd="$(findmtd "${reqmtd}")" |
| 351 | img="/tmp/images/${version}/${imgfile}" |
| 352 | flashcp -v ${img} /dev/${flashmtd} |
| 353 | } |
| 354 | |
| 355 | backup_env_vars() { |
| 356 | copy_env_var_to_alt kernelname |
| 357 | copy_ubiblock_to_alt |
| 358 | copy_root_to_alt |
| 359 | } |
| 360 | |
| 361 | update_env_vars() { |
| 362 | vol="$(findubi rofs-"${version}")" |
| 363 | if [ -z "${vol}" ]; then |
| 364 | return 1 |
| 365 | fi |
| 366 | ubidevid="${vol#ubi}" |
| 367 | block="/dev/ubiblock${ubidevid}" |
| 368 | if [ ! -e "${block}" ]; then |
| 369 | return 1 |
| 370 | fi |
| 371 | ubi_setenv "kernelname=kernel-${version}" |
| 372 | ubi_setenv "ubiblock=$(echo "${ubidevid}" | sed 's/_/,/')" |
| 373 | ubi_setenv "root=${block}" |
| 374 | } |
| 375 | |
| 376 | #TODO: Replace the implementation with systemd-inhibitors lock |
| 377 | # once systemd/systemd#949 is resolved |
| 378 | rebootguardenable() { |
| 379 | dir="/run/systemd/system/" |
| 380 | file="reboot-guard.conf" |
| 381 | units=("reboot" "poweroff" "halt") |
| 382 | |
| 383 | for unit in "${units[@]}"; do |
| 384 | mkdir -p ${dir}${unit}.target.d |
| 385 | echo -e "[Unit]\nRefuseManualStart=yes" >> ${dir}${unit}.target.d/${file} |
| 386 | done |
| 387 | } |
| 388 | |
| 389 | #TODO: Replace the implementation with systemd-inhibitors lock |
| 390 | # once systemd/systemd#949 is resolved |
| 391 | rebootguarddisable() { |
| 392 | dir="/run/systemd/system/" |
| 393 | file="reboot-guard.conf" |
| 394 | units=("reboot" "poweroff" "halt") |
| 395 | |
| 396 | for unit in "${units[@]}"; do |
| 397 | rm -rf ${dir}${unit}.target.d/${file} |
| 398 | done |
| 399 | } |
| 400 | |
| 401 | # Create a copy in the alt mtd |
| 402 | create_vol_in_alt() { |
| 403 | alt="alt-bmc" |
| 404 | altmtd="$(findmtd "${alt}")" |
| 405 | if [ ! -z "${altmtd}" ]; then |
| 406 | reqmtd="${alt}" |
| 407 | reqmtd2="${alt}" |
| 408 | ubi_ro |
| 409 | ubi_updatevol |
| 410 | fi |
| 411 | } |
| 412 | |
| 413 | # Copy contents of one MTD device to another |
| 414 | mtd_copy() { |
| 415 | in=$1 |
| 416 | out=$2 |
| 417 | |
| 418 | # Must erase MTD first to prevent corruption |
| 419 | flash_eraseall "${out}" |
| 420 | dd if="${in}" of="${out}" |
| 421 | } |
| 422 | |
| 423 | mirroruboot() { |
| 424 | bmc="$(findmtd "u-boot")" |
| 425 | bmcdev="/dev/${bmc}" |
| 426 | alt="$(findmtd "alt-u-boot")" |
| 427 | altdev="/dev/${alt}" |
| 428 | |
| 429 | checksum_bmc="$(md5sum "${bmcdev}")" |
| 430 | checksum_bmc="${checksum_bmc% *}" |
| 431 | checksum_alt="$(md5sum "${altdev}")" |
| 432 | checksum_alt="${checksum_alt% *}" |
| 433 | |
| 434 | if [[ "${checksum_bmc}" != "${checksum_alt}" ]]; then |
| 435 | bmcenv="$(findmtd "u-boot-env")" |
| 436 | bmcenvdev="/dev/${bmcenv}" |
| 437 | altenv="$(findmtd "alt-u-boot-env")" |
| 438 | altenvdev="/dev/${altenv}" |
| 439 | |
| 440 | echo "Mirroring U-boot to alt chip" |
| 441 | mtd_copy "${bmcdev}" "${altdev}" |
| 442 | mtd_copy "${bmcenvdev}" "${altenvdev}" |
| 443 | |
| 444 | copy_ubiblock_to_alt |
| 445 | copy_root_to_alt |
| 446 | fi |
| 447 | } |
| 448 | |
Adriana Kobylak | 62f3820 | 2020-09-29 13:04:25 -0500 | [diff] [blame^] | 449 | # Compare the device where u-boot resides with an image file. Specify the full |
| 450 | # path to the device and image file to use for the compare. Print a value of |
| 451 | # "0" if identical, "1" otherwise. |
| 452 | cmp_uboot() { |
| 453 | device="$1" |
| 454 | image="$2" |
| 455 | |
| 456 | # Since the image file can be smaller than the device, copy the device to a |
| 457 | # tmp file and write the image file on top, then compare the sum of each. |
| 458 | # Use cat / redirection since busybox does not have the conv=notrunc option. |
| 459 | tmpFile="$(mktemp ubootdev.XXXXXX)" |
| 460 | dd if="${device}" of="${tmpFile}" |
| 461 | devSum="$(sha256sum ${tmpFile})" |
| 462 | cat < "${image}" 1<> "${tmpFile}" |
| 463 | imgSum="$(sha256sum ${tmpFile})" |
| 464 | rm -f "${tmpFile}" |
| 465 | |
| 466 | if [ "${imgSum}" == "${devSum}" ]; then |
| 467 | echo "0"; |
| 468 | else |
| 469 | echo "1"; |
| 470 | fi |
| 471 | } |
| 472 | |
Adriana Kobylak | 70f5bc0 | 2020-05-13 14:08:14 -0500 | [diff] [blame] | 473 | # The eMMC partition labels for the kernel and rootfs are boot-a/b and rofs-a/b. |
Adriana Kobylak | 3412435 | 2020-05-22 09:40:40 -0500 | [diff] [blame] | 474 | # Return the label (a or b) for the running partition. |
| 475 | mmc_get_primary_label() { |
Adriana Kobylak | 5312d85 | 2020-07-18 09:49:34 -0500 | [diff] [blame] | 476 | # Get root device /dev/mmcblkpX |
Adriana Kobylak | 3412435 | 2020-05-22 09:40:40 -0500 | [diff] [blame] | 477 | rootmatch=" on / " |
Adriana Kobylak | 5312d85 | 2020-07-18 09:49:34 -0500 | [diff] [blame] | 478 | root="$(mount | grep "${rootmatch}")" |
| 479 | root="${root%${rootmatch}*}" |
| 480 | |
| 481 | # Find the device label |
| 482 | if [[ $(readlink -f /dev/disk/by-partlabel/rofs-a) == "${root}" ]]; then |
Adriana Kobylak | 3412435 | 2020-05-22 09:40:40 -0500 | [diff] [blame] | 483 | echo "a" |
Adriana Kobylak | 5312d85 | 2020-07-18 09:49:34 -0500 | [diff] [blame] | 484 | elif [[ $(readlink -f /dev/disk/by-partlabel/rofs-b) == "${root}" ]]; then |
Adriana Kobylak | 3412435 | 2020-05-22 09:40:40 -0500 | [diff] [blame] | 485 | echo "b" |
Adriana Kobylak | 5312d85 | 2020-07-18 09:49:34 -0500 | [diff] [blame] | 486 | else |
| 487 | echo "" |
Adriana Kobylak | 3412435 | 2020-05-22 09:40:40 -0500 | [diff] [blame] | 488 | fi |
| 489 | } |
| 490 | |
| 491 | # The eMMC partition labels for the kernel and rootfs are boot-a/b and rofs-a/b. |
Adriana Kobylak | 70f5bc0 | 2020-05-13 14:08:14 -0500 | [diff] [blame] | 492 | # Return the label (a or b) for the non-running partition. |
| 493 | mmc_get_secondary_label() { |
Adriana Kobylak | 5312d85 | 2020-07-18 09:49:34 -0500 | [diff] [blame] | 494 | root="$(mmc_get_primary_label)" |
| 495 | if [[ "${root}" == "a" ]]; then |
Adriana Kobylak | 70f5bc0 | 2020-05-13 14:08:14 -0500 | [diff] [blame] | 496 | echo "b" |
Adriana Kobylak | 5312d85 | 2020-07-18 09:49:34 -0500 | [diff] [blame] | 497 | elif [[ "${root}" == "b" ]]; then |
Adriana Kobylak | 70f5bc0 | 2020-05-13 14:08:14 -0500 | [diff] [blame] | 498 | echo "a" |
Adriana Kobylak | 5312d85 | 2020-07-18 09:49:34 -0500 | [diff] [blame] | 499 | else |
| 500 | echo "" |
Adriana Kobylak | 70f5bc0 | 2020-05-13 14:08:14 -0500 | [diff] [blame] | 501 | fi |
| 502 | } |
| 503 | |
| 504 | mmc_update() { |
Adriana Kobylak | 62f3820 | 2020-09-29 13:04:25 -0500 | [diff] [blame^] | 505 | # Update u-boot if needed |
| 506 | bootPartition="mmcblk0boot0" |
| 507 | devUBoot="/dev/${bootPartition}" |
| 508 | imgUBoot="${imgpath}/${version}/image-u-boot" |
| 509 | if [ "$(cmp_uboot "${devUBoot}" "${imgUBoot}")" != "0" ]; then |
| 510 | echo 0 > "/sys/block/${bootPartition}/force_ro" |
| 511 | dd if="${imgUBoot}" of="${devUBoot}" |
| 512 | echo 1 > "/sys/block/${bootPartition}/force_ro" |
| 513 | fi |
| 514 | |
Adriana Kobylak | 70f5bc0 | 2020-05-13 14:08:14 -0500 | [diff] [blame] | 515 | # Update the secondary (non-running) boot and rofs partitions. |
| 516 | label="$(mmc_get_secondary_label)" |
| 517 | |
| 518 | # Update the boot and rootfs partitions, restore their labels after the update |
| 519 | # by getting the partition number mmcblk0pX from their label. |
| 520 | zstd -d -c ${imgpath}/${version}/image-kernel | dd of="/dev/disk/by-partlabel/boot-${label}" |
| 521 | number="$(readlink -f /dev/disk/by-partlabel/boot-${label})" |
| 522 | number="${number##*mmcblk0p}" |
| 523 | sgdisk --change-name=${number}:boot-${label} /dev/mmcblk0 1>/dev/null |
| 524 | |
| 525 | zstd -d -c ${imgpath}/${version}/image-rofs | dd of="/dev/disk/by-partlabel/rofs-${label}" |
| 526 | number="$(readlink -f /dev/disk/by-partlabel/rofs-${label})" |
| 527 | number="${number##*mmcblk0p}" |
| 528 | sgdisk --change-name=${number}:rofs-${label} /dev/mmcblk0 1>/dev/null |
| 529 | |
| 530 | # Run this after sgdisk for labels to take effect. |
| 531 | partprobe |
| 532 | |
Adriana Kobylak | 8c5209d | 2020-07-12 13:35:47 -0500 | [diff] [blame] | 533 | # Update hostfw |
| 534 | if [ -f ${imgpath}/${version}/image-hostfw ]; then |
Adriana Kobylak | d148b4f | 2020-08-27 10:18:49 -0500 | [diff] [blame] | 535 | # Remove patches |
| 536 | patchdir="/usr/local/share/hostfw/alternate" |
| 537 | if [ -d "${patchdir}" ]; then |
| 538 | rm -rf "${patchdir}"/* |
| 539 | fi |
Adriana Kobylak | 8c5209d | 2020-07-12 13:35:47 -0500 | [diff] [blame] | 540 | hostfwdir=$(grep "hostfw " /proc/mounts | cut -d " " -f 2) |
| 541 | cp ${imgpath}/${version}/image-hostfw ${hostfwdir}/hostfw-${label} |
| 542 | mkdir -p ${hostfwdir}/alternate |
| 543 | mount ${hostfwdir}/hostfw-${label} ${hostfwdir}/alternate -o ro |
| 544 | fi |
| 545 | |
Adriana Kobylak | 70f5bc0 | 2020-05-13 14:08:14 -0500 | [diff] [blame] | 546 | # Store the label where the other properties like purpose and priority are |
| 547 | # preserved via the storePriority() function in the serialize files, so that |
| 548 | # it can be used for the remove function. |
| 549 | label_dir="/var/lib/phosphor-bmc-code-mgmt/${version}" |
| 550 | label_file="${label_dir}/partlabel" |
| 551 | mkdir -p "${label_dir}" |
| 552 | echo "${label}" > "${label_file}" |
| 553 | } |
| 554 | |
| 555 | mmc_remove() { |
| 556 | # Render the filesystem unbootable by wiping out the first 1MB, this |
| 557 | # invalidates the filesystem header. |
| 558 | # If the label property does not exist, assume it's the secondary |
| 559 | # (non-running) one since the running device should not be erased. |
| 560 | label="" |
| 561 | label_file="/var/lib/phosphor-bmc-code-mgmt/${version}/partlabel" |
| 562 | if [ -f "${label_file}" ]; then |
| 563 | label="$(cat "${label_file}")" |
| 564 | else |
| 565 | label="$(mmc_get_secondary_label)" |
| 566 | fi |
| 567 | dd if=/dev/zero of=/dev/disk/by-partlabel/boot-${label} count=2048 |
| 568 | dd if=/dev/zero of=/dev/disk/by-partlabel/rofs-${label} count=2048 |
Adriana Kobylak | 8c5209d | 2020-07-12 13:35:47 -0500 | [diff] [blame] | 569 | |
| 570 | hostfw_alt="hostfw/alternate" |
| 571 | if grep -q "${hostfw_alt}" /proc/mounts; then |
| 572 | hostfw_alt=$(grep "${hostfw_alt}" /proc/mounts | cut -d " " -f 2) |
| 573 | umount "${hostfw_alt}" |
| 574 | fi |
| 575 | hostfw_base="hostfw " |
| 576 | if grep -q "${hostfw_base}" /proc/mounts; then |
| 577 | hostfw_base=$(grep "${hostfw_base}" /proc/mounts | cut -d " " -f 2) |
| 578 | rm -f ${hostfw_base}/hostfw-${label} |
| 579 | fi |
Adriana Kobylak | 70f5bc0 | 2020-05-13 14:08:14 -0500 | [diff] [blame] | 580 | } |
| 581 | |
Adriana Kobylak | 3412435 | 2020-05-22 09:40:40 -0500 | [diff] [blame] | 582 | # Set the requested version as primary for the BMC to boot from upon reboot. |
| 583 | mmc_setprimary() { |
| 584 | # Point root to the label of the requested BMC rootfs. If the label property |
| 585 | # does not exist, determine if the requested version is functional or not. |
| 586 | label="" |
| 587 | label_file="/var/lib/phosphor-bmc-code-mgmt/${version}/partlabel" |
| 588 | if [ -f "${label_file}" ]; then |
| 589 | label="$(cat "${label_file}")" |
| 590 | else |
| 591 | functional="$(busctl call xyz.openbmc_project.ObjectMapper \ |
| 592 | /xyz/openbmc_project/software/functional \ |
| 593 | org.freedesktop.DBus.Properties Get ss \ |
| 594 | xyz.openbmc_project.Association endpoints)" |
| 595 | if [[ "${functional}" =~ "${version}" ]]; then |
| 596 | label="$(mmc_get_primary_label)" |
| 597 | else |
| 598 | label="$(mmc_get_secondary_label)" |
| 599 | fi |
| 600 | fi |
| 601 | fw_setenv bootside "${label}" |
| 602 | } |
| 603 | |
Adriana Kobylak | 9cbfa2e | 2019-04-25 14:02:37 -0500 | [diff] [blame] | 604 | case "$1" in |
| 605 | mtduboot) |
| 606 | reqmtd="$2" |
| 607 | version="$3" |
| 608 | imgfile="image-u-boot" |
| 609 | mtd_write |
| 610 | ;; |
| 611 | ubirw) |
| 612 | reqmtd="$2" |
| 613 | name="$3" |
| 614 | imgsize="$4" |
| 615 | ubi_rw |
| 616 | ;; |
| 617 | ubiro) |
| 618 | reqmtd="$(echo "$2" | cut -d "+" -f 1)" |
| 619 | reqmtd2="$(echo "$2" | cut -d "+" -f 2)" |
| 620 | name="$3" |
| 621 | version="$4" |
| 622 | imgfile="image-rofs" |
| 623 | ubi_ro |
| 624 | ubi_updatevol |
| 625 | ubi_block |
| 626 | ;; |
| 627 | ubikernel) |
| 628 | reqmtd="$(echo "$2" | cut -d "+" -f 1)" |
| 629 | reqmtd2="$(echo "$2" | cut -d "+" -f 2)" |
| 630 | name="$3" |
| 631 | version="$4" |
| 632 | imgfile="image-kernel" |
| 633 | ubi_ro |
| 634 | ubi_updatevol |
| 635 | create_vol_in_alt |
| 636 | ;; |
| 637 | ubiremove) |
| 638 | name="$2" |
| 639 | ubi_remove "${name}" |
| 640 | ;; |
| 641 | ubicleanup) |
| 642 | ubi_cleanup |
| 643 | ;; |
| 644 | ubisetenv) |
| 645 | ubi_setenv "$2" |
| 646 | ;; |
| 647 | ubiremount) |
| 648 | remount_ubi |
| 649 | mount_alt_rwfs |
| 650 | ;; |
| 651 | createenvbackup) |
| 652 | backup_env_vars |
| 653 | ;; |
| 654 | updateubootvars) |
| 655 | version="$2" |
| 656 | update_env_vars |
| 657 | ;; |
| 658 | rebootguardenable) |
| 659 | rebootguardenable |
| 660 | ;; |
| 661 | rebootguarddisable) |
| 662 | rebootguarddisable |
| 663 | ;; |
| 664 | mirroruboot) |
| 665 | mirroruboot |
| 666 | ;; |
Adriana Kobylak | 70f5bc0 | 2020-05-13 14:08:14 -0500 | [diff] [blame] | 667 | mmc) |
| 668 | version="$2" |
| 669 | imgpath="$3" |
| 670 | mmc_update |
| 671 | ;; |
| 672 | mmc-remove) |
| 673 | version="$2" |
| 674 | mmc_remove |
| 675 | ;; |
Adriana Kobylak | 3412435 | 2020-05-22 09:40:40 -0500 | [diff] [blame] | 676 | mmc-setprimary) |
| 677 | version="$2" |
| 678 | mmc_setprimary |
| 679 | ;; |
Adriana Kobylak | 9cbfa2e | 2019-04-25 14:02:37 -0500 | [diff] [blame] | 680 | *) |
| 681 | echo "Invalid argument" |
| 682 | exit 1 |
| 683 | ;; |
| 684 | esac |