+static void set_ipmi_bootdev(struct config *config, enum ipmi_bootdev bootdev,
+ bool persistent)
+{
+ config->ipmi_bootdev = bootdev;
+ config->ipmi_bootdev_persistent = persistent;
+
+ switch (bootdev) {
+ case IPMI_BOOTDEV_NONE:
+ case IPMI_BOOTDEV_DISK:
+ case IPMI_BOOTDEV_NETWORK:
+ case IPMI_BOOTDEV_CDROM:
+ break;
+ case IPMI_BOOTDEV_SETUP:
+ config->autoboot_enabled = false;
+ break;
+ case IPMI_BOOTDEV_SAFE:
+ config->autoboot_enabled = false;
+ config->safe_mode = true;
+ break;
+ }
+}
+
+static int read_bootdev_sysparam(const char *name, uint8_t *val)
+{
+ uint8_t buf[2];
+ char path[50];
+ int fd, rc;
+
+ strcpy(path, sysparams_dir);
+ assert(strlen(name) < sizeof(path) - strlen(path));
+ strcat(path, name);
+
+ fd = open(path, O_RDONLY);
+ if (fd < 0) {
+ pb_debug("powerpc: can't access sysparam %s\n",
+ name);
+ return -1;
+ }
+
+ rc = read(fd, buf, sizeof(buf));
+
+ close(fd);
+
+ /* bootdev definitions should only be one byte in size */
+ if (rc != 1) {
+ pb_debug("powerpc: sysparam %s read returned %d\n",
+ name, rc);
+ return -1;
+ }
+
+ pb_debug("powerpc: sysparam %s: 0x%02x\n", name, buf[0]);
+
+ if (!ipmi_bootdev_is_valid(buf[0]))
+ return -1;
+
+ *val = buf[0];
+ return 0;
+}
+
+static int write_bootdev_sysparam(const char *name, uint8_t val)
+{
+ char path[50];
+ int fd, rc;
+
+ strcpy(path, sysparams_dir);
+ assert(strlen(name) < sizeof(path) - strlen(path));
+ strcat(path, name);
+
+ fd = open(path, O_WRONLY);
+ if (fd < 0) {
+ pb_debug("powerpc: can't access sysparam %s for writing\n",
+ name);
+ return -1;
+ }
+
+ for (;;) {
+ errno = 0;
+ rc = write(fd, &val, sizeof(val));
+ if (rc == sizeof(val)) {
+ rc = 0;
+ break;
+ }
+
+ if (rc <= 0 && errno != EINTR) {
+ pb_log("powerpc: error updating sysparam %s: %s",
+ name, strerror(errno));
+ rc = -1;
+ break;
+ }
+ }
+
+ close(fd);
+
+ if (!rc)
+ pb_debug("powerpc: set sysparam %s: 0x%02x\n", name, val);
+
+ return rc;
+}
+
+static int clear_ipmi_bootdev_sysparams(
+ struct platform_powerpc *platform __attribute__((unused)))
+{
+ /* invalidate next-boot-device setting */
+ write_bootdev_sysparam("next-boot-device", 0xff);
+ return 0;
+}
+
+static int get_ipmi_bootdev_sysparams(
+ struct platform_powerpc *platform __attribute__((unused)),
+ uint8_t *bootdev, bool *persistent)
+{
+ uint8_t next_bootdev, default_bootdev;
+ bool next_valid, default_valid;
+ int rc;
+
+ rc = read_bootdev_sysparam("next-boot-device", &next_bootdev);
+ next_valid = rc == 0;
+
+ rc = read_bootdev_sysparam("default-boot-device", &default_bootdev);
+ default_valid = rc == 0;
+
+ /* nothing valid? no need to change the config */
+ if (!next_valid && !default_valid)
+ return -1;
+
+ *persistent = !next_valid;
+ *bootdev = next_valid ? next_bootdev : default_bootdev;
+ return 0;
+}
+
+static int clear_ipmi_bootdev_ipmi(struct platform_powerpc *platform)
+{
+ uint16_t resp_len;
+ uint8_t resp[1];
+ uint8_t req[] = {
+ 0x05, /* parameter selector: boot flags */
+ 0x80, /* data 1: valid */
+ 0x00, /* data 2: bootdev: no override */
+ 0x00, /* data 3: system defaults */
+ 0x00, /* data 4: no request for shared mode, mux defaults */
+ 0x00, /* data 5: no instance request */
+ };
+
+ resp_len = sizeof(resp);
+
+ ipmi_transaction(platform->ipmi, IPMI_NETFN_CHASSIS,
+ IPMI_CMD_CHASSIS_SET_SYSTEM_BOOT_OPTIONS,
+ req, sizeof(req),
+ resp, &resp_len,
+ ipmi_timeout);
+ return 0;
+}
+
+static int get_ipmi_bootdev_ipmi(struct platform_powerpc *platform,
+ uint8_t *bootdev, bool *persistent)
+{
+ uint16_t resp_len;
+ uint8_t resp[8];
+ int rc;
+ uint8_t req[] = {
+ 0x05, /* parameter selector: boot flags */
+ 0x00, /* no set selector */
+ 0x00, /* no block selector */
+ };
+
+ resp_len = sizeof(resp);
+ rc = ipmi_transaction(platform->ipmi, IPMI_NETFN_CHASSIS,
+ IPMI_CMD_CHASSIS_GET_SYSTEM_BOOT_OPTIONS,
+ req, sizeof(req),
+ resp, &resp_len,
+ ipmi_timeout);
+ if (rc) {
+ pb_log("platform: error reading IPMI boot options\n");
+ return -1;
+ }
+
+ if (resp_len != sizeof(resp)) {
+ pb_log("platform: unexpected length (%d) in "
+ "boot options response\n", resp_len);
+ return -1;
+ }
+
+ if (resp[0] != 0) {
+ pb_log("platform: non-zero completion code %d from IPMI req\n",
+ resp[0]);
+ return -1;
+ }
+
+ /* check for correct parameter version */
+ if ((resp[1] & 0xf) != 0x1) {
+ pb_log("platform: unexpected version (0x%x) in "
+ "boot options response\n", resp[0]);
+ return -1;
+ }
+
+ /* check for valid paramters */
+ if (resp[2] & 0x80) {
+ pb_debug("platform: boot options are invalid/locked\n");
+ return -1;
+ }
+
+ *persistent = false;
+
+ /* check for valid flags */
+ if (!(resp[3] & 0x80)) {
+ pb_debug("platform: boot flags are invalid, ignoring\n");
+ return 0;
+ }
+
+ *persistent = resp[3] & 0x40;
+ *bootdev = (resp[4] >> 2) & 0x0f;
+ return 0;
+}
+
+