#include "xe_heci_gsc.h"
#include "xe_i2c.h"
#include "xe_mmio.h"
+#include "xe_nvm.h"
#include "xe_pcode_api.h"
#include "xe_vsec.h"
*
* - ``aux_info<n>`` : Some failures have additional debug information
*
+ * - ``fdo_mode`` : To allow recovery in scenarios where MEI itself fails, a new SPI Flash
+ * Descriptor Override (FDO) mode is added in v2 survivability breadcrumbs. This mode is enabled
+ * by PCODE and provides the ability to directly update the firmware via SPI Driver without
+ * any dependency on MEI. Xe KMD initializes the nvm aux driver if FDO mode is enabled.
+ *
* Runtime Survivability
* =====================
*
[AUX_INFO4] = "Auxiliary Info 4",
};
+#define FDO_INFO (MAX_SCRATCH_REG + 1)
+
struct xe_survivability_attribute {
struct device_attribute attr;
u8 index;
set_survivability_info(mmio, info, CAPABILITY_INFO);
reg_value = info[CAPABILITY_INFO];
+ survivability->version = REG_FIELD_GET(BREADCRUMB_VERSION, reg_value);
+ /* FDO mode is exposed only from version 2 */
+ if (survivability->version >= 2)
+ survivability->fdo_mode = REG_FIELD_GET(FDO_MODE, reg_value);
+
if (reg_value & HISTORY_TRACKING) {
set_survivability_info(mmio, info, POSTCODE_TRACE);
struct xe_survivability *survivability = &xe->survivability;
u32 *info = survivability->info;
+ if (sa->index == FDO_INFO)
+ return sysfs_emit(buff, "%s\n", str_enabled_disabled(survivability->fdo_mode));
+
return sysfs_emit(buff, "0x%x\n", info[sa->index]);
}
SURVIVABILITY_ATTR_RO(aux_info2, AUX_INFO2);
SURVIVABILITY_ATTR_RO(aux_info3, AUX_INFO3);
SURVIVABILITY_ATTR_RO(aux_info4, AUX_INFO4);
+SURVIVABILITY_ATTR_RO(fdo_mode, FDO_INFO);
static void xe_survivability_mode_fini(void *arg)
{
struct xe_device *xe = arg;
+ struct xe_survivability *survivability = &xe->survivability;
struct pci_dev *pdev = to_pci_dev(xe->drm.dev);
struct device *dev = &pdev->dev;
+ if (survivability->fdo_mode)
+ xe_nvm_fini(xe);
+
device_remove_file(dev, &dev_attr_survivability_mode);
}
struct xe_survivability *survivability = &xe->survivability;
u32 *info = survivability->info;
- if (info[idx])
+ /* FDO mode is visible only when supported */
+ if (idx >= MAX_SCRATCH_REG && survivability->version >= 2)
+ return 0400;
+ else if (info[idx])
return 0400;
return 0;
&attr_aux_info2.attr.attr,
&attr_aux_info3.attr.attr,
&attr_aux_info4.attr.attr,
+ &attr_fdo_mode.attr.attr,
NULL,
};
/* Make sure xe_heci_gsc_init() knows about survivability mode */
survivability->mode = true;
- ret = xe_heci_gsc_init(xe);
- if (ret)
- goto err;
+ xe_heci_gsc_init(xe);
xe_vsec_init(xe);
+ if (survivability->fdo_mode) {
+ ret = xe_nvm_init(xe);
+ if (ret)
+ goto err;
+ }
+
ret = xe_i2c_probe(xe);
if (ret)
goto err;
return 0;
err:
+ dev_err(dev, "Failed to enable Survivability Mode\n");
survivability->mode = false;
return ret;
}
populate_survivability_info(xe);
- /* Log breadcrumbs but do not enter survivability mode for Critical boot errors */
- if (survivability->boot_status == CRITICAL_FAILURE) {
+ /*
+ * v2 supports survivability mode for critical errors
+ */
+ if (survivability->version < 2 && survivability->boot_status == CRITICAL_FAILURE) {
log_survivability_info(pdev);
return -ENXIO;
}