| // SPDX-License-Identifier: GPL-2.0 |
| /* Huawei HiNIC PCI Express Linux driver |
| * Copyright(c) 2017 Huawei Technologies Co., Ltd |
| * |
| * This program is free software; you can redistribute it and/or modify it |
| * under the terms and conditions of the GNU General Public License, |
| * version 2, as published by the Free Software Foundation. |
| * |
| * This program is distributed in the hope it will be useful, but WITHOUT |
| * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
| * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License |
| * for more details. |
| * |
| */ |
| #include <linux/netlink.h> |
| #include <net/devlink.h> |
| #include <linux/firmware.h> |
| |
| #include "hinic_port.h" |
| #include "hinic_devlink.h" |
| #include "hinic_hw_dev.h" |
| |
| static bool check_image_valid(struct hinic_devlink_priv *priv, const u8 *buf, |
| u32 image_size, struct host_image_st *host_image) |
| { |
| struct fw_image_st *fw_image = NULL; |
| u32 len = 0; |
| u32 i; |
| |
| fw_image = (struct fw_image_st *)buf; |
| |
| if (fw_image->fw_magic != HINIC_MAGIC_NUM) { |
| dev_err(&priv->hwdev->hwif->pdev->dev, "Wrong fw_magic read from file, fw_magic: 0x%x\n", |
| fw_image->fw_magic); |
| return false; |
| } |
| |
| if (fw_image->fw_info.fw_section_cnt > MAX_FW_TYPE_NUM) { |
| dev_err(&priv->hwdev->hwif->pdev->dev, "Wrong fw_type_num read from file, fw_type_num: 0x%x\n", |
| fw_image->fw_info.fw_section_cnt); |
| return false; |
| } |
| |
| for (i = 0; i < fw_image->fw_info.fw_section_cnt; i++) { |
| len += fw_image->fw_section_info[i].fw_section_len; |
| memcpy(&host_image->image_section_info[i], |
| &fw_image->fw_section_info[i], |
| sizeof(struct fw_section_info_st)); |
| } |
| |
| if (len != fw_image->fw_len || |
| (fw_image->fw_len + UPDATEFW_IMAGE_HEAD_SIZE) != image_size) { |
| dev_err(&priv->hwdev->hwif->pdev->dev, "Wrong data size read from file\n"); |
| return false; |
| } |
| |
| host_image->image_info.up_total_len = fw_image->fw_len; |
| host_image->image_info.fw_version = fw_image->fw_version; |
| host_image->section_type_num = fw_image->fw_info.fw_section_cnt; |
| host_image->device_id = fw_image->device_id; |
| |
| return true; |
| } |
| |
| static bool check_image_integrity(struct hinic_devlink_priv *priv, |
| struct host_image_st *host_image, |
| u32 update_type) |
| { |
| u32 collect_section_type = 0; |
| u32 i, type; |
| |
| for (i = 0; i < host_image->section_type_num; i++) { |
| type = host_image->image_section_info[i].fw_section_type; |
| if (collect_section_type & (1U << type)) { |
| dev_err(&priv->hwdev->hwif->pdev->dev, "Duplicate section type: %u\n", |
| type); |
| return false; |
| } |
| collect_section_type |= (1U << type); |
| } |
| |
| if (update_type == FW_UPDATE_COLD && |
| (((collect_section_type & _IMAGE_COLD_SUB_MODULES_MUST_IN) == |
| _IMAGE_COLD_SUB_MODULES_MUST_IN) || |
| collect_section_type == _IMAGE_CFG_SUB_MODULES_MUST_IN)) |
| return true; |
| |
| if (update_type == FW_UPDATE_HOT && |
| (collect_section_type & _IMAGE_HOT_SUB_MODULES_MUST_IN) == |
| _IMAGE_HOT_SUB_MODULES_MUST_IN) |
| return true; |
| |
| if (update_type == FW_UPDATE_COLD) |
| dev_err(&priv->hwdev->hwif->pdev->dev, "Check file integrity failed, valid: 0x%x or 0x%lx, current: 0x%x\n", |
| _IMAGE_COLD_SUB_MODULES_MUST_IN, |
| _IMAGE_CFG_SUB_MODULES_MUST_IN, collect_section_type); |
| else |
| dev_err(&priv->hwdev->hwif->pdev->dev, "Check file integrity failed, valid:0x%x, current: 0x%x\n", |
| _IMAGE_HOT_SUB_MODULES_MUST_IN, collect_section_type); |
| |
| return false; |
| } |
| |
| static int check_image_device_type(struct hinic_devlink_priv *priv, |
| u32 image_device_type) |
| { |
| struct hinic_comm_board_info board_info = {0}; |
| |
| if (hinic_get_board_info(priv->hwdev, &board_info)) { |
| dev_err(&priv->hwdev->hwif->pdev->dev, "Get board info failed\n"); |
| return false; |
| } |
| |
| if (image_device_type == board_info.info.board_type) |
| return true; |
| |
| dev_err(&priv->hwdev->hwif->pdev->dev, "The device type of upgrade file doesn't match the device type of current firmware, please check the upgrade file\n"); |
| dev_err(&priv->hwdev->hwif->pdev->dev, "The image device type: 0x%x, firmware device type: 0x%x\n", |
| image_device_type, board_info.info.board_type); |
| |
| return false; |
| } |
| |
| static int hinic_flash_fw(struct hinic_devlink_priv *priv, const u8 *data, |
| struct host_image_st *host_image) |
| { |
| u32 section_remain_send_len, send_fragment_len, send_pos, up_total_len; |
| struct hinic_cmd_update_fw *fw_update_msg = NULL; |
| u32 section_type, section_crc, section_version; |
| u32 i, len, section_len, section_offset; |
| u16 out_size = sizeof(*fw_update_msg); |
| int total_len_flag = 0; |
| int err; |
| |
| fw_update_msg = kzalloc(sizeof(*fw_update_msg), GFP_KERNEL); |
| if (!fw_update_msg) |
| return -ENOMEM; |
| |
| up_total_len = host_image->image_info.up_total_len; |
| |
| for (i = 0; i < host_image->section_type_num; i++) { |
| len = host_image->image_section_info[i].fw_section_len; |
| if (host_image->image_section_info[i].fw_section_type == |
| UP_FW_UPDATE_BOOT) { |
| up_total_len = up_total_len - len; |
| break; |
| } |
| } |
| |
| for (i = 0; i < host_image->section_type_num; i++) { |
| section_len = |
| host_image->image_section_info[i].fw_section_len; |
| section_offset = |
| host_image->image_section_info[i].fw_section_offset; |
| section_remain_send_len = section_len; |
| section_type = |
| host_image->image_section_info[i].fw_section_type; |
| section_crc = host_image->image_section_info[i].fw_section_crc; |
| section_version = |
| host_image->image_section_info[i].fw_section_version; |
| |
| if (section_type == UP_FW_UPDATE_BOOT) |
| continue; |
| |
| send_fragment_len = 0; |
| send_pos = 0; |
| |
| while (section_remain_send_len > 0) { |
| if (!total_len_flag) { |
| fw_update_msg->total_len = up_total_len; |
| total_len_flag = 1; |
| } else { |
| fw_update_msg->total_len = 0; |
| } |
| |
| memset(fw_update_msg->data, 0, MAX_FW_FRAGMENT_LEN); |
| |
| fw_update_msg->ctl_info.SF = |
| (section_remain_send_len == section_len) ? |
| true : false; |
| fw_update_msg->section_info.FW_section_CRC = section_crc; |
| fw_update_msg->fw_section_version = section_version; |
| fw_update_msg->ctl_info.flag = UP_TYPE_A; |
| |
| if (section_type <= UP_FW_UPDATE_UP_DATA_B) { |
| fw_update_msg->section_info.FW_section_type = |
| (section_type % 2) ? |
| UP_FW_UPDATE_UP_DATA : |
| UP_FW_UPDATE_UP_TEXT; |
| |
| fw_update_msg->ctl_info.flag = UP_TYPE_B; |
| if (section_type <= UP_FW_UPDATE_UP_DATA_A) |
| fw_update_msg->ctl_info.flag = UP_TYPE_A; |
| } else { |
| fw_update_msg->section_info.FW_section_type = |
| section_type - 0x2; |
| } |
| |
| fw_update_msg->setion_total_len = section_len; |
| fw_update_msg->section_offset = send_pos; |
| |
| if (section_remain_send_len <= MAX_FW_FRAGMENT_LEN) { |
| fw_update_msg->ctl_info.SL = true; |
| fw_update_msg->ctl_info.fragment_len = |
| section_remain_send_len; |
| send_fragment_len += section_remain_send_len; |
| } else { |
| fw_update_msg->ctl_info.SL = false; |
| fw_update_msg->ctl_info.fragment_len = |
| MAX_FW_FRAGMENT_LEN; |
| send_fragment_len += MAX_FW_FRAGMENT_LEN; |
| } |
| |
| memcpy(fw_update_msg->data, |
| data + UPDATEFW_IMAGE_HEAD_SIZE + |
| section_offset + send_pos, |
| fw_update_msg->ctl_info.fragment_len); |
| |
| err = hinic_port_msg_cmd(priv->hwdev, |
| HINIC_PORT_CMD_UPDATE_FW, |
| fw_update_msg, |
| sizeof(*fw_update_msg), |
| fw_update_msg, &out_size); |
| if (err || !out_size || fw_update_msg->status) { |
| dev_err(&priv->hwdev->hwif->pdev->dev, "Failed to update firmware, err: %d, status: 0x%x, out size: 0x%x\n", |
| err, fw_update_msg->status, out_size); |
| err = fw_update_msg->status ? |
| fw_update_msg->status : -EIO; |
| kfree(fw_update_msg); |
| return err; |
| } |
| |
| send_pos = send_fragment_len; |
| section_remain_send_len = section_len - |
| send_fragment_len; |
| } |
| } |
| |
| kfree(fw_update_msg); |
| |
| return 0; |
| } |
| |
| static int hinic_firmware_update(struct hinic_devlink_priv *priv, |
| const struct firmware *fw, |
| struct netlink_ext_ack *extack) |
| { |
| struct host_image_st host_image; |
| int err; |
| |
| memset(&host_image, 0, sizeof(struct host_image_st)); |
| |
| if (!check_image_valid(priv, fw->data, fw->size, &host_image) || |
| !check_image_integrity(priv, &host_image, FW_UPDATE_COLD) || |
| !check_image_device_type(priv, host_image.device_id)) { |
| NL_SET_ERR_MSG_MOD(extack, "Check image failed"); |
| return -EINVAL; |
| } |
| |
| dev_info(&priv->hwdev->hwif->pdev->dev, "Flash firmware begin\n"); |
| |
| err = hinic_flash_fw(priv, fw->data, &host_image); |
| if (err) { |
| if (err == HINIC_FW_DISMATCH_ERROR) { |
| dev_err(&priv->hwdev->hwif->pdev->dev, "Firmware image doesn't match this card, please use newer image, err: %d\n", |
| err); |
| NL_SET_ERR_MSG_MOD(extack, |
| "Firmware image doesn't match this card, please use newer image"); |
| } else { |
| dev_err(&priv->hwdev->hwif->pdev->dev, "Send firmware image data failed, err: %d\n", |
| err); |
| NL_SET_ERR_MSG_MOD(extack, "Send firmware image data failed"); |
| } |
| |
| return err; |
| } |
| |
| dev_info(&priv->hwdev->hwif->pdev->dev, "Flash firmware end\n"); |
| |
| return 0; |
| } |
| |
| static int hinic_devlink_flash_update(struct devlink *devlink, |
| struct devlink_flash_update_params *params, |
| struct netlink_ext_ack *extack) |
| { |
| struct hinic_devlink_priv *priv = devlink_priv(devlink); |
| |
| return hinic_firmware_update(priv, params->fw, extack); |
| } |
| |
| static const struct devlink_ops hinic_devlink_ops = { |
| .flash_update = hinic_devlink_flash_update, |
| }; |
| |
| struct devlink *hinic_devlink_alloc(struct device *dev) |
| { |
| return devlink_alloc(&hinic_devlink_ops, sizeof(struct hinic_dev), dev); |
| } |
| |
| void hinic_devlink_free(struct devlink *devlink) |
| { |
| devlink_free(devlink); |
| } |
| |
| int hinic_devlink_register(struct hinic_devlink_priv *priv) |
| { |
| struct devlink *devlink = priv_to_devlink(priv); |
| |
| return devlink_register(devlink); |
| } |
| |
| void hinic_devlink_unregister(struct hinic_devlink_priv *priv) |
| { |
| struct devlink *devlink = priv_to_devlink(priv); |
| |
| devlink_unregister(devlink); |
| } |
| |
| static int chip_fault_show(struct devlink_fmsg *fmsg, |
| struct hinic_fault_event *event) |
| { |
| const char * const level_str[FAULT_LEVEL_MAX + 1] = { |
| "fatal", "reset", "flr", "general", "suggestion", "Unknown"}; |
| u8 fault_level; |
| int err; |
| |
| fault_level = (event->event.chip.err_level < FAULT_LEVEL_MAX) ? |
| event->event.chip.err_level : FAULT_LEVEL_MAX; |
| if (fault_level == FAULT_LEVEL_SERIOUS_FLR) { |
| err = devlink_fmsg_u32_pair_put(fmsg, "Function level err func_id", |
| (u32)event->event.chip.func_id); |
| if (err) |
| return err; |
| } |
| |
| err = devlink_fmsg_u8_pair_put(fmsg, "module_id", event->event.chip.node_id); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_u32_pair_put(fmsg, "err_type", (u32)event->event.chip.err_type); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_string_pair_put(fmsg, "err_level", level_str[fault_level]); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_u32_pair_put(fmsg, "err_csr_addr", |
| event->event.chip.err_csr_addr); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_u32_pair_put(fmsg, "err_csr_value", |
| event->event.chip.err_csr_value); |
| if (err) |
| return err; |
| |
| return 0; |
| } |
| |
| static int fault_report_show(struct devlink_fmsg *fmsg, |
| struct hinic_fault_event *event) |
| { |
| const char * const type_str[FAULT_TYPE_MAX + 1] = { |
| "chip", "ucode", "mem rd timeout", "mem wr timeout", |
| "reg rd timeout", "reg wr timeout", "phy fault", "Unknown"}; |
| u8 fault_type; |
| int err; |
| |
| fault_type = (event->type < FAULT_TYPE_MAX) ? event->type : FAULT_TYPE_MAX; |
| |
| err = devlink_fmsg_string_pair_put(fmsg, "Fault type", type_str[fault_type]); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_binary_pair_put(fmsg, "Fault raw data", |
| event->event.val, sizeof(event->event.val)); |
| if (err) |
| return err; |
| |
| switch (event->type) { |
| case FAULT_TYPE_CHIP: |
| err = chip_fault_show(fmsg, event); |
| if (err) |
| return err; |
| break; |
| case FAULT_TYPE_UCODE: |
| err = devlink_fmsg_u8_pair_put(fmsg, "Cause_id", event->event.ucode.cause_id); |
| if (err) |
| return err; |
| err = devlink_fmsg_u8_pair_put(fmsg, "core_id", event->event.ucode.core_id); |
| if (err) |
| return err; |
| err = devlink_fmsg_u8_pair_put(fmsg, "c_id", event->event.ucode.c_id); |
| if (err) |
| return err; |
| err = devlink_fmsg_u8_pair_put(fmsg, "epc", event->event.ucode.epc); |
| if (err) |
| return err; |
| break; |
| case FAULT_TYPE_MEM_RD_TIMEOUT: |
| case FAULT_TYPE_MEM_WR_TIMEOUT: |
| err = devlink_fmsg_u32_pair_put(fmsg, "Err_csr_ctrl", |
| event->event.mem_timeout.err_csr_ctrl); |
| if (err) |
| return err; |
| err = devlink_fmsg_u32_pair_put(fmsg, "err_csr_data", |
| event->event.mem_timeout.err_csr_data); |
| if (err) |
| return err; |
| err = devlink_fmsg_u32_pair_put(fmsg, "ctrl_tab", |
| event->event.mem_timeout.ctrl_tab); |
| if (err) |
| return err; |
| err = devlink_fmsg_u32_pair_put(fmsg, "mem_index", |
| event->event.mem_timeout.mem_index); |
| if (err) |
| return err; |
| break; |
| case FAULT_TYPE_REG_RD_TIMEOUT: |
| case FAULT_TYPE_REG_WR_TIMEOUT: |
| err = devlink_fmsg_u32_pair_put(fmsg, "Err_csr", event->event.reg_timeout.err_csr); |
| if (err) |
| return err; |
| break; |
| case FAULT_TYPE_PHY_FAULT: |
| err = devlink_fmsg_u8_pair_put(fmsg, "Op_type", event->event.phy_fault.op_type); |
| if (err) |
| return err; |
| err = devlink_fmsg_u8_pair_put(fmsg, "port_id", event->event.phy_fault.port_id); |
| if (err) |
| return err; |
| err = devlink_fmsg_u8_pair_put(fmsg, "dev_ad", event->event.phy_fault.dev_ad); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_u32_pair_put(fmsg, "csr_addr", event->event.phy_fault.csr_addr); |
| if (err) |
| return err; |
| err = devlink_fmsg_u32_pair_put(fmsg, "op_data", event->event.phy_fault.op_data); |
| if (err) |
| return err; |
| break; |
| default: |
| break; |
| } |
| |
| return 0; |
| } |
| |
| static int hinic_hw_reporter_dump(struct devlink_health_reporter *reporter, |
| struct devlink_fmsg *fmsg, void *priv_ctx, |
| struct netlink_ext_ack *extack) |
| { |
| if (priv_ctx) |
| return fault_report_show(fmsg, priv_ctx); |
| |
| return 0; |
| } |
| |
| static int mgmt_watchdog_report_show(struct devlink_fmsg *fmsg, |
| struct hinic_mgmt_watchdog_info *watchdog_info) |
| { |
| int err; |
| |
| err = devlink_fmsg_u32_pair_put(fmsg, "Mgmt deadloop time_h", watchdog_info->curr_time_h); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_u32_pair_put(fmsg, "time_l", watchdog_info->curr_time_l); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_u32_pair_put(fmsg, "task_id", watchdog_info->task_id); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_u32_pair_put(fmsg, "sp", watchdog_info->sp); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_u32_pair_put(fmsg, "stack_current_used", watchdog_info->curr_used); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_u32_pair_put(fmsg, "peak_used", watchdog_info->peak_used); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_u32_pair_put(fmsg, "\n Overflow_flag", watchdog_info->is_overflow); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_u32_pair_put(fmsg, "stack_top", watchdog_info->stack_top); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_u32_pair_put(fmsg, "stack_bottom", watchdog_info->stack_bottom); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_u32_pair_put(fmsg, "mgmt_pc", watchdog_info->pc); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_u32_pair_put(fmsg, "lr", watchdog_info->lr); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_u32_pair_put(fmsg, "cpsr", watchdog_info->cpsr); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_binary_pair_put(fmsg, "Mgmt register info", |
| watchdog_info->reg, sizeof(watchdog_info->reg)); |
| if (err) |
| return err; |
| |
| err = devlink_fmsg_binary_pair_put(fmsg, "Mgmt dump stack(start from sp)", |
| watchdog_info->data, sizeof(watchdog_info->data)); |
| if (err) |
| return err; |
| |
| return 0; |
| } |
| |
| static int hinic_fw_reporter_dump(struct devlink_health_reporter *reporter, |
| struct devlink_fmsg *fmsg, void *priv_ctx, |
| struct netlink_ext_ack *extack) |
| { |
| if (priv_ctx) |
| return mgmt_watchdog_report_show(fmsg, priv_ctx); |
| |
| return 0; |
| } |
| |
| static const struct devlink_health_reporter_ops hinic_hw_fault_reporter_ops = { |
| .name = "hw", |
| .dump = hinic_hw_reporter_dump, |
| }; |
| |
| static const struct devlink_health_reporter_ops hinic_fw_fault_reporter_ops = { |
| .name = "fw", |
| .dump = hinic_fw_reporter_dump, |
| }; |
| |
| int hinic_health_reporters_create(struct hinic_devlink_priv *priv) |
| { |
| struct devlink *devlink = priv_to_devlink(priv); |
| |
| priv->hw_fault_reporter = |
| devlink_health_reporter_create(devlink, &hinic_hw_fault_reporter_ops, |
| 0, priv); |
| if (IS_ERR(priv->hw_fault_reporter)) { |
| dev_warn(&priv->hwdev->hwif->pdev->dev, "Failed to create hw fault reporter, err: %ld\n", |
| PTR_ERR(priv->hw_fault_reporter)); |
| return PTR_ERR(priv->hw_fault_reporter); |
| } |
| |
| priv->fw_fault_reporter = |
| devlink_health_reporter_create(devlink, &hinic_fw_fault_reporter_ops, |
| 0, priv); |
| if (IS_ERR(priv->fw_fault_reporter)) { |
| dev_warn(&priv->hwdev->hwif->pdev->dev, "Failed to create fw fault reporter, err: %ld\n", |
| PTR_ERR(priv->fw_fault_reporter)); |
| devlink_health_reporter_destroy(priv->hw_fault_reporter); |
| priv->hw_fault_reporter = NULL; |
| return PTR_ERR(priv->fw_fault_reporter); |
| } |
| |
| return 0; |
| } |
| |
| void hinic_health_reporters_destroy(struct hinic_devlink_priv *priv) |
| { |
| if (!IS_ERR_OR_NULL(priv->fw_fault_reporter)) { |
| devlink_health_reporter_destroy(priv->fw_fault_reporter); |
| priv->fw_fault_reporter = NULL; |
| } |
| |
| if (!IS_ERR_OR_NULL(priv->hw_fault_reporter)) { |
| devlink_health_reporter_destroy(priv->hw_fault_reporter); |
| priv->hw_fault_reporter = NULL; |
| } |
| } |