meta-ampere: support Altra Boot Progress handling
Handle Altra Boot Progress reported from SCP regiter map by the use of
smpro-misc driver.
Tested: check boot progress is updated in Redfish for:
1. Reboot Host.
2. IPMI chassis power reset
3. IPMI chassis power cycle
4. IPMI chassis power off, then on
5. WebUI Orderly – OS shuts down, then server reboots
6. WebUI Immediate – Server reboots without OS shutting down
Signed-off-by: Thang Q. Nguyen <thang@os.amperecomputing.com>
Change-Id: I9483328ccf5731904d0e6e71074561f7dd743bed
diff --git a/meta-ampere/meta-common/recipes-ampere/host/ac01-boot-progress.bb b/meta-ampere/meta-common/recipes-ampere/host/ac01-boot-progress.bb
new file mode 100644
index 0000000..823bea7
--- /dev/null
+++ b/meta-ampere/meta-common/recipes-ampere/host/ac01-boot-progress.bb
@@ -0,0 +1,33 @@
+SUMMARY = "Altra Boot Progress Handling Service"
+DESCRIPTION = "OpenBMC Altra Boot Progress Handling Daemon"
+
+PR = "r1"
+LICENSE = "Apache-2.0"
+LIC_FILES_CHKSUM = "file://${COREBASE}/meta/files/common-licenses/Apache-2.0;md5=89aea4e17d99a7cacdbeed46a0096b10"
+
+inherit systemd
+inherit obmc-phosphor-systemd
+
+DEPENDS += "systemd"
+RDEPENDS:${PN} += "libsystemd"
+RDEPENDS:${PN} += "bash"
+
+SRC_URI = " \
+ file://ampere_boot_progress.sh \
+ "
+
+SYSTEMD_PACKAGES = "${PN}"
+
+HOST_ON_RESET_HOSTTMPL = "ampere-boot-progress.service"
+HOST_ON_RESET_HOSTINSTMPL = "ampere-boot-progress.service"
+HOST_ON_RESET_HOSTTGTFMT = "obmc-host-already-on@{0}.target"
+HOST_ON_RESET_HOSTFMT = "../${HOST_ON_RESET_HOSTTMPL}:${HOST_ON_RESET_HOSTTGTFMT}.requires/${HOST_ON_RESET_HOSTINSTMPL}"
+SYSTEMD_LINK:${PN} += "${@compose_list_zip(d, 'HOST_ON_RESET_HOSTFMT', 'OBMC_HOST_INSTANCES')}"
+
+SYSTEMD_SERVICE:${PN} += "${HOST_ON_RESET_HOSTTMPL}"
+
+do_install () {
+ install -d ${D}${sbindir}
+ install -m 0755 ${WORKDIR}/ampere_boot_progress.sh ${D}${sbindir}/
+}
+
diff --git a/meta-ampere/meta-common/recipes-ampere/host/ac01-boot-progress/ampere-boot-progress.service b/meta-ampere/meta-common/recipes-ampere/host/ac01-boot-progress/ampere-boot-progress.service
new file mode 100644
index 0000000..267d901
--- /dev/null
+++ b/meta-ampere/meta-common/recipes-ampere/host/ac01-boot-progress/ampere-boot-progress.service
@@ -0,0 +1,13 @@
+[Unit]
+Description=Ampere Altra Boot Progress Handling
+After=ampere-host-already-on@0.target
+BindTo=obmc-host-already-on@0.target
+
+[Service]
+Restart=simple
+ExecStart=/usr/sbin/ampere_boot_progress.sh
+SyslogIdentifier=ampere-boot-progress
+RemainAfterExit=no
+
+[Install]
+WantedBy=obmc-host-already-on@0.target
diff --git a/meta-ampere/meta-common/recipes-ampere/host/ac01-boot-progress/ampere_boot_progress.sh b/meta-ampere/meta-common/recipes-ampere/host/ac01-boot-progress/ampere_boot_progress.sh
new file mode 100755
index 0000000..87ad34f
--- /dev/null
+++ b/meta-ampere/meta-common/recipes-ampere/host/ac01-boot-progress/ampere_boot_progress.sh
@@ -0,0 +1,201 @@
+#!/bin/bash
+# Initialize variables
+boot_stage=00
+boot_status=00
+uefi_code=00000000
+
+function set_postcode()
+{
+ # shellcheck disable=SC2086
+ busctl set-property xyz.openbmc_project.State.Boot.Raw \
+ /xyz/openbmc_project/state/boot/raw0 \
+ xyz.openbmc_project.State.Boot.Raw Value \(tay\) "$1" 0
+}
+
+function update_boot_progress_last_state_time()
+{
+ # Get BMC current time
+ bp_last_state_time=$(busctl get-property xyz.openbmc_project.Time.Manager \
+ /xyz/openbmc_project/time/bmc \
+ xyz.openbmc_project.Time.EpochTime \
+ Elapsed | cut -d' ' -f2)
+
+ # Update the Boot Progress LastStateTime
+ busctl set-property xyz.openbmc_project.State.Host \
+ /xyz/openbmc_project/state/host0 \
+ xyz.openbmc_project.State.Boot.Progress \
+ BootProgressLastUpdate t \
+ "$bp_last_state_time"
+}
+
+function update_boot_progress()
+{
+ bootprog=$1
+
+ busctl set-property xyz.openbmc_project.State.Host \
+ /xyz/openbmc_project/state/host0 \
+ xyz.openbmc_project.State.Boot.Progress \
+ BootProgress s \
+ "xyz.openbmc_project.State.Boot.Progress.ProgressStages.$bootprog"
+
+ # Update Boot Progress LastStateTime
+ update_boot_progress_last_state_time
+}
+
+function get_boot_stage_string()
+{
+ bootstage=$1
+ ueficode=$2
+
+ case $bootstage in
+
+ 00)
+ boot_stage_str="SMpro"
+ ;;
+
+ 01)
+ boot_stage_str="PMpro"
+ ;;
+
+ 02)
+ boot_stage_str="ATF BL1 (Code=${ueficode})"
+ ;;
+
+ 03)
+ boot_stage_str="DDR initialization (Code=${ueficode})"
+ ;;
+
+ 04)
+ boot_stage_str="DDR training progress (Code=${ueficode})"
+ ;;
+
+ 05)
+ boot_stage_str="ATF BL2 (Code=${ueficode})"
+ ;;
+
+ 06)
+ boot_stage_str="ATF BL31 (Code=${ueficode})"
+ ;;
+
+ 07)
+ boot_stage_str="ATF BL32 (Code=${ueficode})"
+ ;;
+
+ 08)
+ boot_stage_str="UEFI booting (UEFI Code=${ueficode})"
+ ;;
+ 09)
+ boot_stage_str="OS booting"
+ ;;
+
+ esac
+
+ echo "$boot_stage_str"
+}
+
+function set_boot_progress()
+{
+ boot_stage=$1
+ uefi_code=$2
+
+ case $boot_stage in
+
+ 02)
+ update_boot_progress "PrimaryProcInit"
+ ;;
+
+ 03)
+ update_boot_progress "MemoryInit"
+ ;;
+
+ 08)
+ if [[ "$uefi_code" =~ 0201* ]]; then
+ update_boot_progress "PCIInit"
+ fi
+ ;;
+ 09)
+ update_boot_progress "OSStart"
+ ;;
+
+ esac
+}
+
+function log_redfish_biosboot_ok_event()
+{
+ logger-systemd --journald << EOF
+MESSAGE=
+PRIORITY=2
+SEVERITY=
+REDFISH_MESSAGE_ID=OpenBMC.0.1.BIOSBoot.OK
+REDFISH_MESSAGE_ARGS="UEFI firmware booting done"
+EOF
+}
+
+function log_redfish_bios_panic_event()
+{
+ boot_state_str=$(get_boot_stage_string "$1" "$2")
+
+ logger-systemd --journald << EOF
+MESSAGE=
+PRIORITY=2
+SEVERITY=
+REDFISH_MESSAGE_ID=OpenBMC.0.1.BIOSFirmwarePanicReason.Warning
+REDFISH_MESSAGE_ARGS=${boot_state_str}
+EOF
+}
+
+cnt=0
+# If any reason makes SCP fail to access in 6s, break the service.
+while [ $cnt -lt 30 ];
+do
+ # Sleep 200ms
+ usleep 200000
+ if ! read -r bg <<< "$(cat /sys/bus/platform/devices/smpro-misc.2.auto/boot_progress)";
+ then
+ cnt=$((cnt + 1))
+ continue
+ fi
+ cnt=0
+
+ # Check if any update from previous check
+ if [ "$last_bg" == "$bg" ]; then
+ continue
+ fi
+ last_bg=$bg
+
+ # Check if the Host is already ON or not. If Host is already boot, update boot progress and break.
+ if [ "${boot_stage}" == "00" ] && [ "${bg[0]}" == "09" ];
+ then
+ update_boot_progress "OSRunning"
+ break
+ fi
+
+ # Update current boot progress
+ boot_stage=${bg:2:2}
+ boot_status=${bg:0:2}
+ uefi_code=${bg:4}
+ echo "Boot Progress = ${boot_stage} ${boot_status} ${uefi_code}"
+
+ # Log Boot Progress to dbus
+ if [ "${boot_status}" == "03" ]; then
+ # Log Redfish Event if failure.
+ log_redfish_bios_panic_event "$boot_stage" "$uefi_code"
+ elif [ "${boot_status}" == "01" ]; then
+ # Check and set boot progress to dbus
+ set_boot_progress "$boot_stage" "$uefi_code"
+ fi
+
+ # Log POST Code to dbus.
+ set_postcode "0x$boot_stage$boot_status$uefi_code"
+
+ # Stop the service when booting to OS
+ if [ "${boot_stage}" == "08" ] && [ "${boot_status}" == "02" ]; then
+ update_boot_progress "SystemInitComplete"
+ log_redfish_biosboot_ok_event
+ elif [ "${boot_stage}" == "09" ] && [ "${boot_status}" == "02" ];
+ then
+ update_boot_progress "OSRunning"
+ break
+ fi
+done
+