Commit b31e59bc authored by Ohad Sharabi's avatar Ohad Sharabi Committed by Oded Gabbay

habanalabs: load linux image to device

Implementing dynamic linux image load to the device.
This patch also implements the FW communication steps during the
boot-fit.
This patch also enables the dynamic protocol based on the compatibility
flag.
Signed-off-by: default avatarOhad Sharabi <osharabi@habana.ai>
Reviewed-by: default avatarOded Gabbay <ogabbay@kernel.org>
Signed-off-by: default avatarOded Gabbay <ogabbay@kernel.org>
parent 8a43c83f
...@@ -14,13 +14,7 @@ ...@@ -14,13 +14,7 @@
#define FW_FILE_MAX_SIZE 0x1400000 /* maximum size of 20MB */ #define FW_FILE_MAX_SIZE 0x1400000 /* maximum size of 20MB */
/* #define FW_CPU_STATUS_POLL_INTERVAL_USEC 10000
* when copying image to FW we assume that PCI bar should not be re-set
* (refers mainly to DRAM in which we do such it to access arbitrary region's
* memory address) and we limit the BAR offset to 1G which should be more than
* reasonable for image copy purposes.
*/
#define FW_IMAGE_MAX_BAR_OFFSET (1024 * 1024 * 1024)
static int hl_request_fw(struct hl_device *hdev, static int hl_request_fw(struct hl_device *hdev,
const struct firmware **firmware_p, const struct firmware **firmware_p,
...@@ -898,7 +892,7 @@ static int hl_fw_read_preboot_caps(struct hl_device *hdev, ...@@ -898,7 +892,7 @@ static int hl_fw_read_preboot_caps(struct hl_device *hdev,
(status == CPU_BOOT_STATUS_READY_TO_BOOT) || (status == CPU_BOOT_STATUS_READY_TO_BOOT) ||
(status == CPU_BOOT_STATUS_SRAM_AVAIL) || (status == CPU_BOOT_STATUS_SRAM_AVAIL) ||
(status == CPU_BOOT_STATUS_WAITING_FOR_BOOT_FIT), (status == CPU_BOOT_STATUS_WAITING_FOR_BOOT_FIT),
10000, FW_CPU_STATUS_POLL_INTERVAL_USEC,
timeout); timeout);
if (rc) { if (rc) {
...@@ -916,14 +910,8 @@ static int hl_fw_read_preboot_caps(struct hl_device *hdev, ...@@ -916,14 +910,8 @@ static int hl_fw_read_preboot_caps(struct hl_device *hdev,
prop->fw_preboot_caps_map = RREG32(cpu_boot_caps_reg); prop->fw_preboot_caps_map = RREG32(cpu_boot_caps_reg);
/* prop->dynamic_fw_load = !!(prop->fw_preboot_caps_map &
* For now- force dynamic_fw_load to false as LKD does not yet CPU_BOOT_DEV_STS0_FW_LD_COM_EN);
* implements all necessary parts of it.
* TODO: once dynamic load is ready set to:
* prop->dynamic_fw_load = !!(prop->fw_preboot_caps_map &
* CPU_BOOT_DEV_STS0_FW_LD_COM_EN)
*/
prop->dynamic_fw_load = 0;
/* initialize FW loader once we know what load protocol is used */ /* initialize FW loader once we know what load protocol is used */
hdev->asic_funcs->init_firmware_loader(hdev); hdev->asic_funcs->init_firmware_loader(hdev);
...@@ -1070,13 +1058,13 @@ int hl_fw_read_preboot_status(struct hl_device *hdev, u32 cpu_boot_status_reg, ...@@ -1070,13 +1058,13 @@ int hl_fw_read_preboot_status(struct hl_device *hdev, u32 cpu_boot_status_reg,
if (rc) if (rc)
return rc; return rc;
if (!hdev->asic_prop.dynamic_fw_load) /* no need to read preboot status in dynamic load */
return hl_fw_static_read_preboot_status(hdev, cpu_boot_status_reg, if (hdev->asic_prop.dynamic_fw_load)
return 0;
return hl_fw_static_read_preboot_status(hdev, cpu_boot_status_reg,
cpu_boot_caps_reg, boot_err0_reg, cpu_boot_caps_reg, boot_err0_reg,
timeout); timeout);
dev_err(hdev->dev, "Dynamic FW load is not supported\n");
return -EINVAL;
} }
/* associate string with COMM status */ /* associate string with COMM status */
...@@ -1202,7 +1190,7 @@ static int hl_fw_dynamic_wait_for_status(struct hl_device *hdev, ...@@ -1202,7 +1190,7 @@ static int hl_fw_dynamic_wait_for_status(struct hl_device *hdev,
le32_to_cpu(dyn_regs->cpu_cmd_status_to_host), le32_to_cpu(dyn_regs->cpu_cmd_status_to_host),
status, status,
FIELD_GET(COMMS_STATUS_STATUS_MASK, status) == expected_status, FIELD_GET(COMMS_STATUS_STATUS_MASK, status) == expected_status,
10000, FW_CPU_STATUS_POLL_INTERVAL_USEC,
timeout); timeout);
if (rc) { if (rc) {
...@@ -1373,7 +1361,7 @@ static int hl_fw_dynamic_validate_memory_bound(struct hl_device *hdev, ...@@ -1373,7 +1361,7 @@ static int hl_fw_dynamic_validate_memory_bound(struct hl_device *hdev,
* memory transfers) * memory transfers)
*/ */
if (end_addr >= region->region_base - region->offset_in_bar + if (end_addr >= region->region_base - region->offset_in_bar +
FW_IMAGE_MAX_BAR_OFFSET) { region->bar_size) {
dev_err(hdev->dev, dev_err(hdev->dev,
"FW image beyond PCI BAR bounds\n"); "FW image beyond PCI BAR bounds\n");
return -EIO; return -EIO;
...@@ -1617,21 +1605,76 @@ static int hl_fw_dynamic_copy_image(struct hl_device *hdev, ...@@ -1617,21 +1605,76 @@ static int hl_fw_dynamic_copy_image(struct hl_device *hdev,
} }
/** /**
* hl_fw_dynamic_load_boot_fit - load boot fit using dynamic protocol * hl_fw_boot_fit_update_state - update internal data structures after boot-fit
* is loaded
*
* @hdev: pointer to the habanalabs device structure
* @cpu_security_boot_status_reg: register holding security status props
*
* @return 0 on success, otherwise non-zero error code
*/
static void hl_fw_boot_fit_update_state(struct hl_device *hdev,
u32 cpu_security_boot_status_reg)
{
struct asic_fixed_properties *prop = &hdev->asic_prop;
/* Clear reset status since we need to read it again from boot CPU */
prop->hard_reset_done_by_fw = false;
/* Read boot_cpu security bits */
if (prop->fw_security_status_valid) {
prop->fw_boot_cpu_security_map =
RREG32(cpu_security_boot_status_reg);
if (prop->fw_boot_cpu_security_map &
CPU_BOOT_DEV_STS0_FW_HARD_RST_EN)
prop->hard_reset_done_by_fw = true;
dev_dbg(hdev->dev,
"Firmware boot CPU security status %#x\n",
prop->fw_boot_cpu_security_map);
}
dev_dbg(hdev->dev, "Firmware boot CPU hard-reset is %s\n",
prop->hard_reset_done_by_fw ? "enabled" : "disabled");
}
/**
* hl_fw_dynamic_load_image - load FW image using dynamic protocol
* *
* @hdev: pointer to the habanalabs device structure * @hdev: pointer to the habanalabs device structure
* @fw_loader: managing structure for loading device's FW * @fw_loader: managing structure for loading device's FW
* @load_fwc: the FW component to be loaded
* @img_ld_timeout: image load timeout
* *
* @return 0 on success, otherwise non-zero error code * @return 0 on success, otherwise non-zero error code
*/ */
static int hl_fw_dynamic_load_boot_fit(struct hl_device *hdev, static int hl_fw_dynamic_load_image(struct hl_device *hdev,
struct fw_load_mgr *fw_loader) struct fw_load_mgr *fw_loader,
enum hl_fw_component load_fwc,
u32 img_ld_timeout)
{ {
enum hl_fw_component cur_fwc;
const struct firmware *fw; const struct firmware *fw;
char *fw_name;
int rc = 0; int rc = 0;
/*
* when loading image we have one of 2 scenarios:
* 1. current FW component is preboot and we want to load boot-fit
* 2. current FW component is boot-fit and we want to load linux
*/
if (load_fwc == FW_COMP_BOOT_FIT) {
cur_fwc = FW_COMP_PREBOOT;
fw_name = fw_loader->boot_fit_img.image_name;
} else {
cur_fwc = FW_COMP_BOOT_FIT;
fw_name = fw_loader->linux_img.image_name;
}
/* request FW in order to communicate to FW the size to be allocated */ /* request FW in order to communicate to FW the size to be allocated */
rc = hl_request_fw(hdev, &fw, fw_loader->boot_fit_img.image_name); rc = hl_request_fw(hdev, &fw, fw_name);
if (rc) if (rc)
return rc; return rc;
...@@ -1643,11 +1686,21 @@ static int hl_fw_dynamic_load_boot_fit(struct hl_device *hdev, ...@@ -1643,11 +1686,21 @@ static int hl_fw_dynamic_load_boot_fit(struct hl_device *hdev,
goto release_fw; goto release_fw;
/* read preboot version */ /* read preboot version */
hl_fw_dynamic_read_device_fw_version(hdev, FW_COMP_PREBOOT, hl_fw_dynamic_read_device_fw_version(hdev, cur_fwc,
fw_loader->dynamic_loader.comm_desc.cur_fw_ver); fw_loader->dynamic_loader.comm_desc.cur_fw_ver);
/* update state during preboot handshake */
hl_fw_preboot_update_state(hdev); /* update state according to boot stage */
if (cur_fwc == FW_COMP_BOOT_FIT) {
struct cpu_dyn_regs *dyn_regs;
dyn_regs = &fw_loader->dynamic_loader.comm_desc.cpu_dyn_regs;
hl_fw_boot_fit_update_state(hdev,
le32_to_cpu(dyn_regs->cpu_boot_status));
} else {
/* update state during preboot handshake */
hl_fw_preboot_update_state(hdev);
}
/* copy boot fit to space allocated by FW */ /* copy boot fit to space allocated by FW */
rc = hl_fw_dynamic_copy_image(hdev, fw, fw_loader); rc = hl_fw_dynamic_copy_image(hdev, fw, fw_loader);
...@@ -1662,13 +1715,104 @@ static int hl_fw_dynamic_load_boot_fit(struct hl_device *hdev, ...@@ -1662,13 +1715,104 @@ static int hl_fw_dynamic_load_boot_fit(struct hl_device *hdev,
rc = hl_fw_dynamic_send_protocol_cmd(hdev, fw_loader, COMMS_EXEC, rc = hl_fw_dynamic_send_protocol_cmd(hdev, fw_loader, COMMS_EXEC,
0, false, 0, false,
fw_loader->boot_fit_timeout); img_ld_timeout);
release_fw: release_fw:
hl_release_firmware(fw); hl_release_firmware(fw);
return rc; return rc;
} }
static int hl_fw_dynamic_wait_for_boot_fit_active(struct hl_device *hdev,
struct fw_load_mgr *fw_loader)
{
struct dynamic_fw_load_mgr *dyn_loader;
u32 status;
int rc;
dyn_loader = &fw_loader->dynamic_loader;
/* Make sure CPU boot-loader is running */
rc = hl_poll_timeout(
hdev,
le32_to_cpu(dyn_loader->comm_desc.cpu_dyn_regs.cpu_boot_status),
status,
(status == CPU_BOOT_STATUS_NIC_FW_RDY) ||
(status == CPU_BOOT_STATUS_READY_TO_BOOT),
FW_CPU_STATUS_POLL_INTERVAL_USEC,
dyn_loader->wait_for_bl_timeout);
if (rc) {
dev_err(hdev->dev, "failed to wait for boot\n");
return rc;
}
dev_dbg(hdev->dev, "uboot status = %d\n", status);
return 0;
}
static int hl_fw_dynamic_wait_for_linux_active(struct hl_device *hdev,
struct fw_load_mgr *fw_loader)
{
struct dynamic_fw_load_mgr *dyn_loader;
u32 status;
int rc;
dyn_loader = &fw_loader->dynamic_loader;
/* Make sure CPU boot-loader is running */
rc = hl_poll_timeout(
hdev,
le32_to_cpu(dyn_loader->comm_desc.cpu_dyn_regs.cpu_boot_status),
status,
(status == CPU_BOOT_STATUS_SRAM_AVAIL),
FW_CPU_STATUS_POLL_INTERVAL_USEC,
fw_loader->cpu_timeout);
if (rc) {
dev_err(hdev->dev, "failed to wait for Linux\n");
return rc;
}
dev_dbg(hdev->dev, "Boot status = %d\n", status);
return 0;
}
/**
* hl_fw_linux_update_state - update internal data structures after loading
* Linux
*
*
* @hdev: pointer to the habanalabs device structure
*
* @return 0 on success, otherwise non-zero error code
*/
static void hl_fw_linux_update_state(struct hl_device *hdev,
u32 cpu_boot_status_reg)
{
struct asic_fixed_properties *prop = &hdev->asic_prop;
/* Clear reset status since we need to read again from app */
prop->hard_reset_done_by_fw = false;
/* Read FW application security bits */
if (prop->fw_security_status_valid) {
prop->fw_app_security_map =
RREG32(cpu_boot_status_reg);
if (prop->fw_app_security_map &
CPU_BOOT_DEV_STS0_FW_HARD_RST_EN)
prop->hard_reset_done_by_fw = true;
dev_dbg(hdev->dev,
"Firmware application CPU security status %#x\n",
prop->fw_app_security_map);
}
dev_dbg(hdev->dev, "Firmware application CPU hard-reset is %s\n",
prop->hard_reset_done_by_fw ? "enabled" : "disabled");
dev_info(hdev->dev, "Successfully loaded firmware to device\n");
}
/** /**
* hl_fw_dynamic_init_cpu - initialize the device CPU using dynamic protocol * hl_fw_dynamic_init_cpu - initialize the device CPU using dynamic protocol
* *
...@@ -1693,6 +1837,8 @@ static int hl_fw_dynamic_init_cpu(struct hl_device *hdev, ...@@ -1693,6 +1837,8 @@ static int hl_fw_dynamic_init_cpu(struct hl_device *hdev,
struct cpu_dyn_regs *dyn_regs; struct cpu_dyn_regs *dyn_regs;
int rc; int rc;
dyn_regs = &fw_loader->dynamic_loader.comm_desc.cpu_dyn_regs;
rc = hl_fw_dynamic_send_protocol_cmd(hdev, fw_loader, COMMS_RST_STATE, rc = hl_fw_dynamic_send_protocol_cmd(hdev, fw_loader, COMMS_RST_STATE,
0, true, 0, true,
fw_loader->cpu_timeout); fw_loader->cpu_timeout);
...@@ -1700,6 +1846,9 @@ static int hl_fw_dynamic_init_cpu(struct hl_device *hdev, ...@@ -1700,6 +1846,9 @@ static int hl_fw_dynamic_init_cpu(struct hl_device *hdev,
goto protocol_err; goto protocol_err;
if (!(hdev->fw_components & FW_TYPE_BOOT_CPU)) { if (!(hdev->fw_components & FW_TYPE_BOOT_CPU)) {
/* update the preboot state */
hl_fw_preboot_update_state(hdev);
rc = hl_fw_dynamic_request_descriptor(hdev, fw_loader, 0); rc = hl_fw_dynamic_request_descriptor(hdev, fw_loader, 0);
if (rc) if (rc)
goto protocol_err; goto protocol_err;
...@@ -1711,14 +1860,50 @@ static int hl_fw_dynamic_init_cpu(struct hl_device *hdev, ...@@ -1711,14 +1860,50 @@ static int hl_fw_dynamic_init_cpu(struct hl_device *hdev,
} }
/* load boot fit to FW */ /* load boot fit to FW */
rc = hl_fw_dynamic_load_boot_fit(hdev, fw_loader); rc = hl_fw_dynamic_load_image(hdev, fw_loader, FW_COMP_BOOT_FIT,
fw_loader->boot_fit_timeout);
if (rc) {
dev_err(hdev->dev, "failed to load boot fit\n");
goto protocol_err;
}
rc = hl_fw_dynamic_wait_for_boot_fit_active(hdev, fw_loader);
if (rc) if (rc)
goto protocol_err; goto protocol_err;
if (!(hdev->fw_components & FW_TYPE_LINUX)) {
dev_info(hdev->dev, "Skip loading Linux F/W\n");
return 0;
}
if (fw_loader->skip_bmc) {
rc = hl_fw_dynamic_send_protocol_cmd(hdev, fw_loader,
COMMS_SKIP_BMC, 0,
true,
fw_loader->cpu_timeout);
if (rc) {
dev_err(hdev->dev, "failed to load boot fit\n");
goto protocol_err;
}
}
/* load Linux image to FW */
rc = hl_fw_dynamic_load_image(hdev, fw_loader, FW_COMP_LINUX,
fw_loader->cpu_timeout);
if (rc) {
dev_err(hdev->dev, "failed to load Linux\n");
goto protocol_err;
}
rc = hl_fw_dynamic_wait_for_linux_active(hdev, fw_loader);
if (rc)
goto protocol_err;
hl_fw_linux_update_state(hdev, le32_to_cpu(dyn_regs->cpu_boot_status));
return 0; return 0;
protocol_err: protocol_err:
dyn_regs = &fw_loader->dynamic_loader.comm_desc.cpu_dyn_regs;
fw_read_errors(hdev, le32_to_cpu(dyn_regs->cpu_boot_err0), fw_read_errors(hdev, le32_to_cpu(dyn_regs->cpu_boot_err0),
le32_to_cpu(dyn_regs->cpu_boot_status)); le32_to_cpu(dyn_regs->cpu_boot_status));
return rc; return rc;
...@@ -1737,7 +1922,6 @@ static int hl_fw_static_init_cpu(struct hl_device *hdev, ...@@ -1737,7 +1922,6 @@ static int hl_fw_static_init_cpu(struct hl_device *hdev,
{ {
u32 cpu_msg_status_reg, cpu_timeout, msg_to_cpu_reg, status; u32 cpu_msg_status_reg, cpu_timeout, msg_to_cpu_reg, status;
u32 cpu_boot_status_reg, cpu_security_boot_status_reg; u32 cpu_boot_status_reg, cpu_security_boot_status_reg;
struct asic_fixed_properties *prop = &hdev->asic_prop;
struct static_fw_load_mgr *static_loader; struct static_fw_load_mgr *static_loader;
int rc; int rc;
...@@ -1763,7 +1947,7 @@ static int hl_fw_static_init_cpu(struct hl_device *hdev, ...@@ -1763,7 +1947,7 @@ static int hl_fw_static_init_cpu(struct hl_device *hdev,
cpu_boot_status_reg, cpu_boot_status_reg,
status, status,
status == CPU_BOOT_STATUS_WAITING_FOR_BOOT_FIT, status == CPU_BOOT_STATUS_WAITING_FOR_BOOT_FIT,
10000, FW_CPU_STATUS_POLL_INTERVAL_USEC,
fw_loader->boot_fit_timeout); fw_loader->boot_fit_timeout);
if (rc) { if (rc) {
...@@ -1786,7 +1970,7 @@ static int hl_fw_static_init_cpu(struct hl_device *hdev, ...@@ -1786,7 +1970,7 @@ static int hl_fw_static_init_cpu(struct hl_device *hdev,
cpu_msg_status_reg, cpu_msg_status_reg,
status, status,
status == CPU_MSG_OK, status == CPU_MSG_OK,
10000, FW_CPU_STATUS_POLL_INTERVAL_USEC,
fw_loader->boot_fit_timeout); fw_loader->boot_fit_timeout);
if (rc) { if (rc) {
...@@ -1808,7 +1992,7 @@ static int hl_fw_static_init_cpu(struct hl_device *hdev, ...@@ -1808,7 +1992,7 @@ static int hl_fw_static_init_cpu(struct hl_device *hdev,
(status == CPU_BOOT_STATUS_NIC_FW_RDY) || (status == CPU_BOOT_STATUS_NIC_FW_RDY) ||
(status == CPU_BOOT_STATUS_READY_TO_BOOT) || (status == CPU_BOOT_STATUS_READY_TO_BOOT) ||
(status == CPU_BOOT_STATUS_SRAM_AVAIL), (status == CPU_BOOT_STATUS_SRAM_AVAIL),
10000, FW_CPU_STATUS_POLL_INTERVAL_USEC,
cpu_timeout); cpu_timeout);
dev_dbg(hdev->dev, "uboot status = %d\n", status); dev_dbg(hdev->dev, "uboot status = %d\n", status);
...@@ -1816,25 +2000,8 @@ static int hl_fw_static_init_cpu(struct hl_device *hdev, ...@@ -1816,25 +2000,8 @@ static int hl_fw_static_init_cpu(struct hl_device *hdev,
/* Read U-Boot version now in case we will later fail */ /* Read U-Boot version now in case we will later fail */
hl_fw_static_read_device_fw_version(hdev, FW_COMP_BOOT_FIT); hl_fw_static_read_device_fw_version(hdev, FW_COMP_BOOT_FIT);
/* Clear reset status since we need to read it again from boot CPU */ /* update state according to boot stage */
prop->hard_reset_done_by_fw = false; hl_fw_boot_fit_update_state(hdev, cpu_security_boot_status_reg);
/* Read boot_cpu security bits */
if (prop->fw_security_status_valid) {
prop->fw_boot_cpu_security_map =
RREG32(cpu_security_boot_status_reg);
if (prop->fw_boot_cpu_security_map &
CPU_BOOT_DEV_STS0_FW_HARD_RST_EN)
prop->hard_reset_done_by_fw = true;
dev_dbg(hdev->dev,
"Firmware boot CPU security status %#x\n",
prop->fw_boot_cpu_security_map);
}
dev_dbg(hdev->dev, "Firmware boot CPU hard-reset is %s\n",
prop->hard_reset_done_by_fw ? "enabled" : "disabled");
if (rc) { if (rc) {
detect_cpu_boot_status(hdev, status); detect_cpu_boot_status(hdev, status);
...@@ -1865,7 +2032,7 @@ static int hl_fw_static_init_cpu(struct hl_device *hdev, ...@@ -1865,7 +2032,7 @@ static int hl_fw_static_init_cpu(struct hl_device *hdev,
cpu_boot_status_reg, cpu_boot_status_reg,
status, status,
(status == CPU_BOOT_STATUS_BMC_WAITING_SKIPPED), (status == CPU_BOOT_STATUS_BMC_WAITING_SKIPPED),
10000, FW_CPU_STATUS_POLL_INTERVAL_USEC,
cpu_timeout); cpu_timeout);
if (rc) { if (rc) {
...@@ -1885,7 +2052,7 @@ static int hl_fw_static_init_cpu(struct hl_device *hdev, ...@@ -1885,7 +2052,7 @@ static int hl_fw_static_init_cpu(struct hl_device *hdev,
cpu_boot_status_reg, cpu_boot_status_reg,
status, status,
(status == CPU_BOOT_STATUS_SRAM_AVAIL), (status == CPU_BOOT_STATUS_SRAM_AVAIL),
10000, FW_CPU_STATUS_POLL_INTERVAL_USEC,
cpu_timeout); cpu_timeout);
/* Clear message */ /* Clear message */
...@@ -1909,27 +2076,7 @@ static int hl_fw_static_init_cpu(struct hl_device *hdev, ...@@ -1909,27 +2076,7 @@ static int hl_fw_static_init_cpu(struct hl_device *hdev,
if (rc) if (rc)
return rc; return rc;
/* Clear reset status since we need to read again from app */ hl_fw_linux_update_state(hdev, cpu_security_boot_status_reg);
prop->hard_reset_done_by_fw = false;
/* Read FW application security bits */
if (prop->fw_security_status_valid) {
prop->fw_app_security_map =
RREG32(cpu_security_boot_status_reg);
if (prop->fw_app_security_map &
CPU_BOOT_DEV_STS0_FW_HARD_RST_EN)
prop->hard_reset_done_by_fw = true;
dev_dbg(hdev->dev,
"Firmware application CPU security status %#x\n",
prop->fw_app_security_map);
}
dev_dbg(hdev->dev, "Firmware application CPU hard-reset is %s\n",
prop->hard_reset_done_by_fw ? "enabled" : "disabled");
dev_info(hdev->dev, "Successfully loaded firmware to device\n");
return 0; return 0;
......
...@@ -180,10 +180,12 @@ enum hl_pci_match_mode { ...@@ -180,10 +180,12 @@ enum hl_pci_match_mode {
* enum hl_fw_component - F/W components to read version through registers. * enum hl_fw_component - F/W components to read version through registers.
* @FW_COMP_BOOT_FIT: boot fit. * @FW_COMP_BOOT_FIT: boot fit.
* @FW_COMP_PREBOOT: preboot. * @FW_COMP_PREBOOT: preboot.
* @FW_COMP_LINUX: linux.
*/ */
enum hl_fw_component { enum hl_fw_component {
FW_COMP_BOOT_FIT, FW_COMP_BOOT_FIT,
FW_COMP_PREBOOT FW_COMP_PREBOOT,
FW_COMP_LINUX,
}; };
/** /**
...@@ -830,6 +832,7 @@ enum pci_region { ...@@ -830,6 +832,7 @@ enum pci_region {
* struct pci_mem_region - describe memory region in a PCI bar * struct pci_mem_region - describe memory region in a PCI bar
* @region_base: region base address * @region_base: region base address
* @region_size: region size * @region_size: region size
* @bar_size: size of the BAR
* @offset_in_bar: region offset into the bar * @offset_in_bar: region offset into the bar
* @bar_id: bar ID of the region * @bar_id: bar ID of the region
* @used: if used 1, otherwise 0 * @used: if used 1, otherwise 0
...@@ -837,6 +840,7 @@ enum pci_region { ...@@ -837,6 +840,7 @@ enum pci_region {
struct pci_mem_region { struct pci_mem_region {
u64 region_base; u64 region_base;
u64 region_size; u64 region_size;
u64 bar_size;
u32 offset_in_bar; u32 offset_in_bar;
u8 bar_id; u8 bar_id;
u8 used; u8 used;
...@@ -885,13 +889,15 @@ struct fw_response { ...@@ -885,13 +889,15 @@ struct fw_response {
* @response: FW to LKD response * @response: FW to LKD response
* @comm_desc: the communication descriptor with FW * @comm_desc: the communication descriptor with FW
* @image_region: region to copy the FW image to * @image_region: region to copy the FW image to
* @fw_image_size: FW image size * @fw_image_size: size of FW image to load
* @wait_for_bl_timeout: timeout for waiting for boot loader to respond
*/ */
struct dynamic_fw_load_mgr { struct dynamic_fw_load_mgr {
struct fw_response response; struct fw_response response;
struct lkd_fw_comms_desc comm_desc; struct lkd_fw_comms_desc comm_desc;
struct pci_mem_region *image_region; struct pci_mem_region *image_region;
size_t fw_image_size; size_t fw_image_size;
u32 wait_for_bl_timeout;
}; };
/** /**
......
...@@ -78,6 +78,7 @@ ...@@ -78,6 +78,7 @@
#define GAUDI_PLDM_TPC_KERNEL_WAIT_USEC (HL_DEVICE_TIMEOUT_USEC * 30) #define GAUDI_PLDM_TPC_KERNEL_WAIT_USEC (HL_DEVICE_TIMEOUT_USEC * 30)
#define GAUDI_BOOT_FIT_REQ_TIMEOUT_USEC 1000000 /* 1s */ #define GAUDI_BOOT_FIT_REQ_TIMEOUT_USEC 1000000 /* 1s */
#define GAUDI_MSG_TO_CPU_TIMEOUT_USEC 4000000 /* 4s */ #define GAUDI_MSG_TO_CPU_TIMEOUT_USEC 4000000 /* 4s */
#define GAUDI_WAIT_FOR_BL_TIMEOUT_USEC 15000000 /* 15s */
#define GAUDI_QMAN0_FENCE_VAL 0x72E91AB9 #define GAUDI_QMAN0_FENCE_VAL 0x72E91AB9
...@@ -1592,6 +1593,7 @@ static int gaudi_alloc_internal_qmans_pq_mem(struct hl_device *hdev) ...@@ -1592,6 +1593,7 @@ static int gaudi_alloc_internal_qmans_pq_mem(struct hl_device *hdev)
static void gaudi_set_pci_memory_regions(struct hl_device *hdev) static void gaudi_set_pci_memory_regions(struct hl_device *hdev)
{ {
struct asic_fixed_properties *prop = &hdev->asic_prop;
struct pci_mem_region *region; struct pci_mem_region *region;
/* CFG */ /* CFG */
...@@ -1599,6 +1601,7 @@ static void gaudi_set_pci_memory_regions(struct hl_device *hdev) ...@@ -1599,6 +1601,7 @@ static void gaudi_set_pci_memory_regions(struct hl_device *hdev)
region->region_base = CFG_BASE; region->region_base = CFG_BASE;
region->region_size = CFG_SIZE; region->region_size = CFG_SIZE;
region->offset_in_bar = CFG_BASE - SPI_FLASH_BASE_ADDR; region->offset_in_bar = CFG_BASE - SPI_FLASH_BASE_ADDR;
region->bar_size = CFG_BAR_SIZE;
region->bar_id = CFG_BAR_ID; region->bar_id = CFG_BAR_ID;
region->used = 1; region->used = 1;
...@@ -1607,6 +1610,7 @@ static void gaudi_set_pci_memory_regions(struct hl_device *hdev) ...@@ -1607,6 +1610,7 @@ static void gaudi_set_pci_memory_regions(struct hl_device *hdev)
region->region_base = SRAM_BASE_ADDR; region->region_base = SRAM_BASE_ADDR;
region->region_size = SRAM_SIZE; region->region_size = SRAM_SIZE;
region->offset_in_bar = 0; region->offset_in_bar = 0;
region->bar_size = SRAM_BAR_SIZE;
region->bar_id = SRAM_BAR_ID; region->bar_id = SRAM_BAR_ID;
region->used = 1; region->used = 1;
...@@ -1615,6 +1619,7 @@ static void gaudi_set_pci_memory_regions(struct hl_device *hdev) ...@@ -1615,6 +1619,7 @@ static void gaudi_set_pci_memory_regions(struct hl_device *hdev)
region->region_base = DRAM_PHYS_BASE; region->region_base = DRAM_PHYS_BASE;
region->region_size = hdev->asic_prop.dram_size; region->region_size = hdev->asic_prop.dram_size;
region->offset_in_bar = 0; region->offset_in_bar = 0;
region->bar_size = prop->dram_pci_bar_size;
region->bar_id = HBM_BAR_ID; region->bar_id = HBM_BAR_ID;
region->used = 1; region->used = 1;
...@@ -1623,6 +1628,7 @@ static void gaudi_set_pci_memory_regions(struct hl_device *hdev) ...@@ -1623,6 +1628,7 @@ static void gaudi_set_pci_memory_regions(struct hl_device *hdev)
region->region_base = PSOC_SCRATCHPAD_ADDR; region->region_base = PSOC_SCRATCHPAD_ADDR;
region->region_size = PSOC_SCRATCHPAD_SIZE; region->region_size = PSOC_SCRATCHPAD_SIZE;
region->offset_in_bar = PSOC_SCRATCHPAD_ADDR - SPI_FLASH_BASE_ADDR; region->offset_in_bar = PSOC_SCRATCHPAD_ADDR - SPI_FLASH_BASE_ADDR;
region->bar_size = CFG_BAR_SIZE;
region->bar_id = CFG_BAR_ID; region->bar_id = CFG_BAR_ID;
region->used = 1; region->used = 1;
} }
...@@ -3749,6 +3755,8 @@ static void gaudi_init_dynamic_firmware_loader(struct hl_device *hdev) ...@@ -3749,6 +3755,8 @@ static void gaudi_init_dynamic_firmware_loader(struct hl_device *hdev)
cpu_to_le32(mmPSOC_GLOBAL_CONF_KMD_MSG_TO_CPU); cpu_to_le32(mmPSOC_GLOBAL_CONF_KMD_MSG_TO_CPU);
dyn_regs->cpu_cmd_status_to_host = dyn_regs->cpu_cmd_status_to_host =
cpu_to_le32(mmCPU_CMD_STATUS_TO_HOST); cpu_to_le32(mmCPU_CMD_STATUS_TO_HOST);
dynamic_loader->wait_for_bl_timeout = GAUDI_WAIT_FOR_BL_TIMEOUT_USEC;
} }
static void gaudi_init_static_firmware_loader(struct hl_device *hdev) static void gaudi_init_static_firmware_loader(struct hl_device *hdev)
......
...@@ -87,6 +87,7 @@ ...@@ -87,6 +87,7 @@
#define GOYA_PLDM_QMAN0_TIMEOUT_USEC (HL_DEVICE_TIMEOUT_USEC * 30) #define GOYA_PLDM_QMAN0_TIMEOUT_USEC (HL_DEVICE_TIMEOUT_USEC * 30)
#define GOYA_BOOT_FIT_REQ_TIMEOUT_USEC 1000000 /* 1s */ #define GOYA_BOOT_FIT_REQ_TIMEOUT_USEC 1000000 /* 1s */
#define GOYA_MSG_TO_CPU_TIMEOUT_USEC 4000000 /* 4s */ #define GOYA_MSG_TO_CPU_TIMEOUT_USEC 4000000 /* 4s */
#define GOYA_WAIT_FOR_BL_TIMEOUT_USEC 15000000 /* 15s */
#define GOYA_QMAN0_FENCE_VAL 0xD169B243 #define GOYA_QMAN0_FENCE_VAL 0xD169B243
...@@ -851,6 +852,7 @@ void goya_late_fini(struct hl_device *hdev) ...@@ -851,6 +852,7 @@ void goya_late_fini(struct hl_device *hdev)
static void goya_set_pci_memory_regions(struct hl_device *hdev) static void goya_set_pci_memory_regions(struct hl_device *hdev)
{ {
struct asic_fixed_properties *prop = &hdev->asic_prop;
struct pci_mem_region *region; struct pci_mem_region *region;
/* CFG */ /* CFG */
...@@ -858,6 +860,7 @@ static void goya_set_pci_memory_regions(struct hl_device *hdev) ...@@ -858,6 +860,7 @@ static void goya_set_pci_memory_regions(struct hl_device *hdev)
region->region_base = CFG_BASE; region->region_base = CFG_BASE;
region->region_size = CFG_SIZE; region->region_size = CFG_SIZE;
region->offset_in_bar = CFG_BASE - SRAM_BASE_ADDR; region->offset_in_bar = CFG_BASE - SRAM_BASE_ADDR;
region->bar_size = CFG_BAR_SIZE;
region->bar_id = SRAM_CFG_BAR_ID; region->bar_id = SRAM_CFG_BAR_ID;
region->used = 1; region->used = 1;
...@@ -866,6 +869,7 @@ static void goya_set_pci_memory_regions(struct hl_device *hdev) ...@@ -866,6 +869,7 @@ static void goya_set_pci_memory_regions(struct hl_device *hdev)
region->region_base = SRAM_BASE_ADDR; region->region_base = SRAM_BASE_ADDR;
region->region_size = SRAM_SIZE; region->region_size = SRAM_SIZE;
region->offset_in_bar = 0; region->offset_in_bar = 0;
region->bar_size = CFG_BAR_SIZE;
region->bar_id = SRAM_CFG_BAR_ID; region->bar_id = SRAM_CFG_BAR_ID;
region->used = 1; region->used = 1;
...@@ -874,6 +878,7 @@ static void goya_set_pci_memory_regions(struct hl_device *hdev) ...@@ -874,6 +878,7 @@ static void goya_set_pci_memory_regions(struct hl_device *hdev)
region->region_base = DRAM_PHYS_BASE; region->region_base = DRAM_PHYS_BASE;
region->region_size = hdev->asic_prop.dram_size; region->region_size = hdev->asic_prop.dram_size;
region->offset_in_bar = 0; region->offset_in_bar = 0;
region->bar_size = prop->dram_pci_bar_size;
region->bar_id = DDR_BAR_ID; region->bar_id = DDR_BAR_ID;
region->used = 1; region->used = 1;
} }
...@@ -2452,6 +2457,8 @@ static void goya_init_dynamic_firmware_loader(struct hl_device *hdev) ...@@ -2452,6 +2457,8 @@ static void goya_init_dynamic_firmware_loader(struct hl_device *hdev)
cpu_to_le32(mmPSOC_GLOBAL_CONF_KMD_MSG_TO_CPU); cpu_to_le32(mmPSOC_GLOBAL_CONF_KMD_MSG_TO_CPU);
dyn_regs->cpu_cmd_status_to_host = dyn_regs->cpu_cmd_status_to_host =
cpu_to_le32(mmCPU_CMD_STATUS_TO_HOST); cpu_to_le32(mmCPU_CMD_STATUS_TO_HOST);
dynamic_loader->wait_for_bl_timeout = GOYA_WAIT_FOR_BL_TIMEOUT_USEC;
} }
static void goya_init_static_firmware_loader(struct hl_device *hdev) static void goya_init_static_firmware_loader(struct hl_device *hdev)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment