firmware: raspberrypi: Add support for tryonce reboot flag
authorTim Gover <tim.gover@raspberrypi.com>
Tue, 20 Oct 2020 10:55:37 +0000 (11:55 +0100)
committerDom Cobley <popcornmix@gmail.com>
Mon, 19 Feb 2024 11:33:06 +0000 (11:33 +0000)
Define a new mailbox (SET_REBOOT_FLAGS) which may be used to
pass optional flags to the Raspberry Pi firmware that changes
the behaviour of the bootloader and firmware during a reboot.

Currently this just defines the 'tryboot' flag which causes
the firmware to load tryboot.txt instead config.txt. This
alternate configuration file can be used to specify the
path of an alternate firmware and kernels allowing a fallback
mechanism to be implemented for OS upgrades.

drivers/firmware/raspberrypi.c

index 2c8f3479fd0a5830fa316d3c57b462ed642620a7..17abde1e7d55ee5579ab8b7fa85d2f520195a446 100644 (file)
@@ -194,6 +194,7 @@ static int rpi_firmware_notify_reboot(struct notifier_block *nb,
 {
        struct rpi_firmware *fw;
        struct platform_device *pdev = g_pdev;
+       u32 reboot_flags = 0;
 
        if (!pdev)
                return 0;
@@ -202,8 +203,28 @@ static int rpi_firmware_notify_reboot(struct notifier_block *nb,
        if (!fw)
                return 0;
 
-       (void)rpi_firmware_property(fw, RPI_FIRMWARE_NOTIFY_REBOOT,
-                                   0, 0);
+       // The partition id is the first parameter followed by zero or
+       // more flags separated by spaces indicating the reason for the reboot.
+       //
+       // 'tryboot': Sets a one-shot flag which is cleared upon reboot and
+       //            causes the tryboot.txt to be loaded instead of config.txt
+       //            by the bootloader and the start.elf firmware.
+       //
+       //            This is intended to allow automatic fallback to a known
+       //            good image if an OS/FW upgrade fails.
+       //
+       // N.B. The firmware mechanism for storing reboot flags may vary
+       // on different Raspberry Pi models.
+       if (data && strstr(data, " tryboot"))
+               reboot_flags |= 0x1;
+
+       // The mailbox might have been called earlier, directly via vcmailbox
+       // so only overwrite if reboot flags are passed to the reboot command.
+       if (reboot_flags)
+               (void)rpi_firmware_property(fw, RPI_FIRMWARE_SET_REBOOT_FLAGS,
+                               &reboot_flags, sizeof(reboot_flags));
+
+       (void)rpi_firmware_property(fw, RPI_FIRMWARE_NOTIFY_REBOOT, NULL, 0);
 
        return 0;
 }