meta-ampere: mtmitchell: support Host firmware update
Support Host firmware update for HostFW, EEPROM and FRU via BMC
console and Redfish.
Tested:
1. Flash firmwares via BMC console.
2. Flash firmware via Redfish
3. Get MB CPLD firmware revision
$ ampere_firmware_version.sh mb_cpld
MB CPLD
CPLD Version: 22060268
CPLD DeviceID: 612BE043
Signed-off-by: Chanh Nguyen <chanh@os.amperecomputing.com>
Change-Id: I1cfbe4832b7d57168e2ea1df0059e23ada1b3b6d
diff --git a/meta-ampere/meta-mitchell/recipes-ampere/platform/ampere-utils.bbappend b/meta-ampere/meta-mitchell/recipes-ampere/platform/ampere-utils.bbappend
index f325395..341ca13 100644
--- a/meta-ampere/meta-mitchell/recipes-ampere/platform/ampere-utils.bbappend
+++ b/meta-ampere/meta-mitchell/recipes-ampere/platform/ampere-utils.bbappend
@@ -4,11 +4,17 @@
SRC_URI:append = " \
file://ampere_power_util.sh \
+ file://ampere_firmware_upgrade.sh \
+ file://ampere_flash_bios.sh \
file://ampere_power_on_driver_binder.sh \
+ file://ampere_firmware_version.sh \
"
do_install:append() {
install -d ${D}/usr/sbin
install -m 0755 ${WORKDIR}/ampere_power_util.sh ${D}/${sbindir}/
+ install -m 0755 ${WORKDIR}/ampere_firmware_upgrade.sh ${D}/${sbindir}/
+ install -m 0755 ${WORKDIR}/ampere_flash_bios.sh ${D}/${sbindir}/
install -m 0755 ${WORKDIR}/ampere_power_on_driver_binder.sh ${D}/${sbindir}/
+ install -m 0755 ${WORKDIR}/ampere_firmware_version.sh ${D}/${sbindir}/
}
diff --git a/meta-ampere/meta-mitchell/recipes-ampere/platform/ampere-utils/ampere_firmware_upgrade.sh b/meta-ampere/meta-mitchell/recipes-ampere/platform/ampere-utils/ampere_firmware_upgrade.sh
new file mode 100755
index 0000000..dce6e6c
--- /dev/null
+++ b/meta-ampere/meta-mitchell/recipes-ampere/platform/ampere-utils/ampere_firmware_upgrade.sh
@@ -0,0 +1,201 @@
+#!/bin/bash
+
+# Helper script to flash FRU and Boot EEPROM devices.
+#
+# Syntax for FRU:
+# ampere_firmware_upgrade.sh fru <image> [<dev>]
+# dev: 1 for MB FRU (default), 2 for BMC FRU.
+#
+# Syntax for EEPROM:
+# ampere_firmware_upgrade.sh eeprom <image> [<dev>]
+# dev: 1 for main Boot EEPROM (default), 2 for secondary Boot EEPROM (if supported)
+#
+# Syntax for Mainboard CPLD:
+# ampere_firmware_upgrade.sh main_cpld <image>
+#
+# Syntax for BMC CPLD:
+# ampere_firmware_upgrade.sh bmc_cpld <image>
+#
+# Syntax for Backplane CPLD:
+# ampere_firmware_upgrade.sh bp_cpld <image> [<target>]
+# target: 1 for Front Backplane 1
+# 2 for Front Backplane 2
+# 3 for Front Backplane 3
+# 4 for Rear Backplane 1
+# 5 for Rear Backplane 2
+#
+
+# shellcheck disable=SC2046
+
+do_eeprom_flash() {
+ FIRMWARE_IMAGE=$IMAGE
+ BACKUP_SEL=$2
+
+ # Turn off the Host if it is currently ON
+ chassisstate=$(obmcutil chassisstate | awk -F. '{print $NF}')
+ echo "Current Chassis State: $chassisstate"
+ if [ "$chassisstate" == 'On' ];
+ then
+ echo "Turning the Chassis off"
+ obmcutil chassisoff
+ sleep 15
+ # Check if HOST was OFF
+ chassisstate_off=$(obmcutil chassisstate | awk -F. '{print $NF}')
+ if [ "$chassisstate_off" == 'On' ];
+ then
+ echo "Error : Failed turning the Chassis off"
+ exit 1
+ fi
+ fi
+
+ # Switch EEPROM control to BMC AST2600 I2C
+ # BMC_GPIOW6_SPI0_PROGRAM_SEL
+ gpioset $(gpiofind spi0-program-sel)=1
+
+ # BMC_GPIOX0_I2C_BACKUP_SEL (GPIO 184)
+ if [[ $BACKUP_SEL == 1 ]]; then
+ echo "Run update Primary EEPROM"
+ gpioset $(gpiofind i2c-backup-sel)=0
+ elif [[ $BACKUP_SEL == 2 ]]; then
+ echo "Run update Failover EEPROM"
+ gpioset $(gpiofind i2c-backup-sel)=1
+ else
+ echo "Please choose Primary EEPROM (1) or Failover EEPROM (2)"
+ exit 0
+ fi
+
+ # The EEPROM (AT24C64WI) with address 0x50 at BMC_I2C11 bus
+ # Write Firmware to EEPROM and read back for validation
+ ampere_eeprom_prog -b 10 -s 0x50 -p -f "$FIRMWARE_IMAGE"
+
+ # Switch to primary EEPROM
+ gpioset $(gpiofind i2c-backup-sel)=0
+
+ # Switch EEPROM control to CPU HOST
+ gpioset $(gpiofind spi0-program-sel)=0
+
+ if [ "$chassisstate" == 'On' ];
+ then
+ sleep 5
+ echo "Turn on the Host"
+ obmcutil poweron
+ fi
+}
+
+do_fru_flash() {
+ FRU_IMAGE=$1
+ FRU_DEV=$2
+
+ if [[ $FRU_DEV == 1 ]]; then
+ if [ -f /sys/bus/i2c/devices/4-0050/eeprom ]; then
+ FRU_DEVICE="/sys/bus/i2c/devices/4-0050/eeprom"
+ else
+ FRU_DEVICE="/sys/bus/i2c/devices/3-0050/eeprom"
+ fi
+ echo "Flash MB FRU with image $IMAGE at $FRU_DEVICE"
+ elif [[ $FRU_DEV == 2 ]]; then
+ FRU_DEVICE="/sys/bus/i2c/devices/14-0050/eeprom"
+ echo "Flash BMC FRU with image $IMAGE at $FRU_DEVICE"
+ else
+ echo "Please select MB FRU (1) or BMC FRU (2)"
+ exit 0
+ fi
+
+ ampere_fru_upgrade -d "$FRU_DEVICE" -f "$FRU_IMAGE"
+
+ systemctl restart xyz.openbmc_project.FruDevice.service
+ systemctl restart phosphor-ipmi-host.service
+
+ echo "Done"
+}
+
+do_mb_cpld_flash() {
+ MB_CPLD_IMAGE=$1
+ echo "Flashing MB CPLD"
+ gpioset $(gpiofind hpm-fw-recovery)=1
+ gpioset $(gpiofind jtag-program-sel)=1
+ sleep 2
+ ampere_cpldupdate_jtag -t 1 -p "$MB_CPLD_IMAGE"
+ gpioset $(gpiofind hpm-fw-recovery)=0
+ echo "Done"
+}
+
+do_bmc_cpld_flash() {
+ BMC_CPLD_IMAGE=$1
+ echo "Flashing BMC CPLD"
+ gpioset $(gpiofind jtag-program-sel)=0
+ sleep 2
+ ampere_cpldupdate_jtag -t 1 -p "$BMC_CPLD_IMAGE"
+ echo "Done"
+}
+
+do_bp_cpld_flash() {
+ BP_CPLD_IMAGE=$1
+ BP_TARGET=$2
+ if [[ $BP_TARGET == 1 ]]; then
+ echo "Flashing Front Backplane 1 CPLD"
+ ampere_cpldupdate_i2c -b 101 -s 0x40 -t 2 -p "$BP_CPLD_IMAGE"
+ elif [[ $BP_TARGET == 2 ]]; then
+ echo "Flashing Front Backplane 2 CPLD"
+ ampere_cpldupdate_i2c -b 102 -s 0x40 -t 2 -p "$BP_CPLD_IMAGE"
+ elif [[ $BP_TARGET == 3 ]]; then
+ echo "Flashing Front Backplane 3 CPLD"
+ ampere_cpldupdate_i2c -b 100 -s 0x40 -t 2 -p "$BP_CPLD_IMAGE"
+ elif [[ $BP_TARGET == 4 ]]; then
+ echo "Flashing Rear Backplane 1 CPLD"
+ ampere_cpldupdate_i2c -b 103 -s 0x40 -t 2 -p "$BP_CPLD_IMAGE"
+ elif [[ $BP_TARGET == 5 ]]; then
+ echo "Flashing Rear Backplane 2 CPLD"
+ ampere_cpldupdate_i2c -b 104 -s 0x40 -t 2 -p "$BP_CPLD_IMAGE"
+ fi
+
+ echo "Done"
+}
+
+if [ $# -eq 0 ]; then
+ echo "Usage:"
+ echo " - Flash Boot EEPROM"
+ echo " $(basename "$0") eeprom <Image file>"
+ echo " - Flash FRU"
+ echo " $(basename "$0") fru <Image file> [dev]"
+ echo " Where:"
+ echo " dev: 1 - MB FRU, 2 - BMC FRU"
+ echo " - Flash Mainboard CPLD"
+ echo " $(basename "$0") mb_cpld <Image file>"
+ echo " - Flash BMC CPLD (only for DC-SCM BMC board)"
+ echo " $(basename "$0") bmc_cpld <Image file>"
+ echo " - Flash Backplane CPLD"
+ echo " $(basename "$0") bp_cpld <Image file> <Target> "
+ echo " Where:"
+ echo " Target: 1 - FrontBP1, 2 - FrontBP2, 3 - FrontBP3"
+ echo " 4 - RearBP1, 5 - RearBP2"
+ exit 0
+fi
+
+TYPE=$1
+IMAGE=$2
+TARGET=$3
+if [ -z "$3" ]; then
+ BACKUP_SEL=1
+else
+ BACKUP_SEL=$3
+fi
+
+if [[ $TYPE == "eeprom" ]]; then
+ # Run EEPROM update: write/read/validation with CRC32 checksum
+ do_eeprom_flash "$IMAGE" "$BACKUP_SEL"
+elif [[ $TYPE == "fru" ]]; then
+ # Run FRU update
+ do_fru_flash "$IMAGE" "$BACKUP_SEL"
+elif [[ $TYPE == "mb_cpld" ]]; then
+ # Run Mainboard CPLD update
+ do_mb_cpld_flash "$IMAGE"
+elif [[ $TYPE == "bmc_cpld" ]]; then
+ # Run CPLD BMC update
+ do_bmc_cpld_flash "$IMAGE"
+elif [[ $TYPE == "bp_cpld" ]]; then
+ # Run Backplane CPLD update
+ do_bp_cpld_flash "$IMAGE" "$TARGET"
+fi
+
+exit 0
diff --git a/meta-ampere/meta-mitchell/recipes-ampere/platform/ampere-utils/ampere_firmware_version.sh b/meta-ampere/meta-mitchell/recipes-ampere/platform/ampere-utils/ampere_firmware_version.sh
new file mode 100755
index 0000000..081e303
--- /dev/null
+++ b/meta-ampere/meta-mitchell/recipes-ampere/platform/ampere-utils/ampere_firmware_version.sh
@@ -0,0 +1,99 @@
+#!/bin/bash
+
+# Helper script to report firmware version for components on the system (MB CPLD, Backplane CPLD, …)
+# Author : Hieu Huynh (hieu.huynh@amperecomputing.com)
+#
+# Get MB CPLD firmware revision:
+# ampere_firmware_version.sh mb_cpld
+#
+# Get BMC CPLD firmware revision:
+# ampere_firmware_version.sh bmc_cpld
+#
+# Get Backplane CPLD firmware revision:
+# ampere_firmware_version.sh bp_cpld <id>
+# <id>: 1 for Front Backplane 1
+# 2 for Front Backplane 2
+# 3 for Front Backplane 3
+# 4 for Rear Backplane 1
+# 5 for Rear Backplane 2
+
+# shellcheck disable=SC2046
+
+do_mb_cpld_firmware_report() {
+ echo "MB CPLD"
+ gpioset $(gpiofind hpm-fw-recovery)=1
+ gpioset $(gpiofind jtag-program-sel)=1
+ sleep 1
+ ampere_cpldupdate_jtag -v
+ ampere_cpldupdate_jtag -i
+}
+
+do_bmc_cpld_firmware_report() {
+ echo "BMC CPLD (Only for DC-SCM board)"
+ gpioset $(gpiofind jtag-program-sel)=0
+ sleep 1
+ ampere_cpldupdate_jtag -v
+ ampere_cpldupdate_jtag -i
+}
+
+do_bp_cpld_firmware_report() {
+ BP_ID=$1
+ if [[ $BP_ID == 1 ]]; then
+ echo "Front Backplane 1 CPLD"
+ ampere_cpldupdate_i2c -b 101 -s 0x40 -t 2 -v
+ ampere_cpldupdate_i2c -b 101 -s 0x40 -t 2 -i
+ elif [[ $BP_ID == 2 ]]; then
+ echo "Front Backplane 2 CPLD"
+ ampere_cpldupdate_i2c -b 102 -s 0x40 -t 2 -v
+ ampere_cpldupdate_i2c -b 102 -s 0x40 -t 2 -i
+ elif [[ $BP_ID == 3 ]]; then
+ echo "Front Backplane 3 CPLD"
+ ampere_cpldupdate_i2c -b 100 -s 0x40 -t 2 -v
+ ampere_cpldupdate_i2c -b 100 -s 0x40 -t 2 -i
+ elif [[ $BP_ID == 4 ]]; then
+ echo "Rear Backplane 1 CPLD"
+ ampere_cpldupdate_i2c -b 103 -s 0x40 -t 2 -v
+ ampere_cpldupdate_i2c -b 103 -s 0x40 -t 2 -i
+ elif [[ $BP_ID == 5 ]]; then
+ echo "Rear Backplane 2 CPLD"
+ ampere_cpldupdate_i2c -b 104 -s 0x40 -t 2 -v
+ ampere_cpldupdate_i2c -b 104 -s 0x40 -t 2 -i
+ fi
+}
+
+if [ $# -eq 0 ]; then
+ echo "Usage:"
+ echo " - Get MB CPLD firmware revision"
+ echo " $(basename "$0") mb_cpld"
+ echo " - Get BMC CPLD firmware revision"
+ echo " $(basename "$0") bmc_cpld"
+ echo " - Get Backplane CPLD firmware revision"
+ echo " $(basename "$0") bp_cpld <id>"
+ echo " <id>:"
+ echo " 1 - FrontBP1"
+ echo " 2 - FrontBP2"
+ echo " 3 - FrontBP3"
+ echo " 4 - RearBP1"
+ echo " 5 - RearBP2"
+ exit 0
+fi
+
+TYPE=$1
+ID=$2
+
+if [[ $TYPE == "mb_cpld" ]]; then
+ do_mb_cpld_firmware_report
+elif [[ $TYPE == "bmc_cpld" ]]; then
+ do_bmc_cpld_firmware_report
+elif [[ $TYPE == "bp_cpld" ]]; then
+ if [ -z "$ID" ]; then
+ echo "Please choose backplanes id: 1 - FrontBP1, 2 - FrontBP2, 3 - FrontBP3, 4 - FrontBP4, 5 - FrontBP5"
+ exit 0
+ elif [[ "$ID" -ge "1" ]] && [[ "$ID" -le "5" ]]; then
+ do_bp_cpld_firmware_report "$ID"
+ else
+ echo "Backplanes id invalid"
+ fi
+fi
+
+exit 0
diff --git a/meta-ampere/meta-mitchell/recipes-ampere/platform/ampere-utils/ampere_flash_bios.sh b/meta-ampere/meta-mitchell/recipes-ampere/platform/ampere-utils/ampere_flash_bios.sh
new file mode 100755
index 0000000..0e0c18c
--- /dev/null
+++ b/meta-ampere/meta-mitchell/recipes-ampere/platform/ampere-utils/ampere_flash_bios.sh
@@ -0,0 +1,126 @@
+#!/bin/bash
+
+# This script is used to flash the UEFI/EDKII
+
+# Syntax: ampere_flash_bios.sh $image_file $device_sellect
+# Where:
+# $image_file : the image binary file
+# $device_sellect : 1 - Host Main SPI Nor
+# 2 - Host Second SPI Nor
+
+# Author : Chanh Nguyen (chnguyen@amperecomputing.com)
+
+# Note:
+# BMC_GPIOW6_SPI0_PROGRAM_SEL (GPIO 182): 1 => BMC owns SPI bus for upgrading
+# 0 => HOST owns SPI bus for upgrading
+
+# BMC_GPIOW7_SPI0_BACKUP_SEL (GPIO 183) : 1 => to switch SPI_CS0_L to primary SPI Nor device
+# 0 => to switch SPI_CS0_L to second SPI Nor device
+
+# shellcheck disable=SC2046
+
+do_flash () {
+ # Check the HNOR partition available
+ HOST_MTD=$(< /proc/mtd grep "pnor" | sed -n 's/^\(.*\):.*/\1/p')
+ if [ -z "$HOST_MTD" ];
+ then
+ # Check the ASpeed SMC driver binded before
+ HOST_SPI=/sys/bus/platform/drivers/spi-aspeed-smc/1e630000.spi
+ if [ -d "$HOST_SPI" ]; then
+ echo "Unbind the ASpeed SMC driver"
+ echo 1e630000.spi > /sys/bus/platform/drivers/spi-aspeed-smc/unbind
+ sleep 2
+ fi
+
+ # If the HNOR partition is not available, then bind again driver
+ echo "--- Bind the ASpeed SMC driver"
+ echo 1e630000.spi > /sys/bus/platform/drivers/spi-aspeed-smc/bind
+ sleep 2
+
+ HOST_MTD=$(< /proc/mtd grep "pnor" | sed -n 's/^\(.*\):.*/\1/p')
+ if [ -z "$HOST_MTD" ];
+ then
+ echo "Fail to probe Host SPI-NOR device"
+ exit 1
+ fi
+ fi
+
+ echo "--- Flashing firmware image $IMAGE to @/dev/$HOST_MTD"
+ flashcp -v "$IMAGE" /dev/"$HOST_MTD"
+}
+
+
+if [ $# -eq 0 ]; then
+ echo "Usage: $(basename "$0") <UEFI/EDKII image file>"
+ exit 0
+fi
+
+IMAGE="$1"
+if [ ! -f "$IMAGE" ]; then
+ echo "The image file $IMAGE does not exist"
+ exit 1
+fi
+
+if [ -z "$2" ]; then
+ DEV_SEL="1" # by default, select primary device
+else
+ DEV_SEL="$2"
+fi
+
+# Turn off the Host if it is currently ON
+chassisstate=$(obmcutil chassisstate | awk -F. '{print $NF}')
+echo "--- Current Chassis State: $chassisstate"
+if [ "$chassisstate" == 'On' ];
+then
+ echo "--- Turning the Chassis off"
+ obmcutil chassisoff
+ sleep 10
+ # Check if HOST was OFF
+ chassisstate_off=$(obmcutil chassisstate | awk -F. '{print $NF}')
+ if [ "$chassisstate_off" == 'On' ];
+ then
+ echo "--- Error : Failed turning the Chassis off"
+ exit 1
+ fi
+fi
+
+# Switch the host SPI bus to BMC"
+echo "--- Switch the host SPI bus to BMC."
+if ! gpioset $(gpiofind spi0-program-sel)=1; then
+ echo "ERROR: Switch the host SPI bus to BMC. Please check gpio state"
+ exit 1
+fi
+
+# Switch the host SPI bus (between primary and secondary)
+# 183 is BMC_GPIOW7_SPI0_BACKUP_SEL
+if [[ $DEV_SEL == 1 ]]; then
+ echo "Run update Primary Host SPI-NOR"
+ gpioset $(gpiofind spi0-backup-sel)=1 # Primary SPI
+elif [[ $DEV_SEL == 2 ]]; then
+ echo "Run update Second Host SPI-NOR"
+ gpioset $(gpiofind spi0-backup-sel)=0 # Second SPI
+else
+ echo "Please choose primary SPI (1) or second SPI (2)"
+ exit 0
+fi
+
+# Flash the firmware
+do_flash
+
+# Switch the SPI bus to the primary spi device
+echo "Switch to the Primary Host SPI-NOR"
+gpioset $(gpiofind spi0-backup-sel)=1 # Primary SPI
+
+# Switch the host SPI bus to HOST."
+echo "--- Switch the host SPI bus to HOST."
+if ! gpioset $(gpiofind spi0-program-sel)=0; then
+ echo "ERROR: Switch the host SPI bus to HOST. Please check gpio state"
+ exit 1
+fi
+
+if [ "$chassisstate" == 'On' ];
+then
+ sleep 5
+ echo "Turn on the Host"
+ obmcutil poweron
+fi