> On Jul 29, 2022, at 6:16 AM, Sreekanth Reddy <sreekanth.reddy@xxxxxxxxxxxx> wrote: > > Added framework to issue MPT transport commands to > controllers. > Also issued the MPT transport commands to get the > manufacturing info of sas expander device. > > Signed-off-by: Sreekanth Reddy <sreekanth.reddy@xxxxxxxxxxxx> > --- > drivers/scsi/mpi3mr/mpi3mr.h | 6 +- > drivers/scsi/mpi3mr/mpi3mr_fw.c | 17 ++ > drivers/scsi/mpi3mr/mpi3mr_os.c | 2 + > drivers/scsi/mpi3mr/mpi3mr_transport.c | 228 +++++++++++++++++++++++++ > 4 files changed, 252 insertions(+), 1 deletion(-) > > diff --git a/drivers/scsi/mpi3mr/mpi3mr.h b/drivers/scsi/mpi3mr/mpi3mr.h > index 21ea021..a6c880c 100644 > --- a/drivers/scsi/mpi3mr/mpi3mr.h > +++ b/drivers/scsi/mpi3mr/mpi3mr.h > @@ -99,9 +99,10 @@ extern atomic64_t event_counter; > #define MPI3MR_HOSTTAG_PEL_WAIT 4 > #define MPI3MR_HOSTTAG_BLK_TMS 5 > #define MPI3MR_HOSTTAG_CFG_CMDS 6 > +#define MPI3MR_HOSTTAG_TRANSPORT_CMDS 7 > > #define MPI3MR_NUM_DEVRMCMD 16 > -#define MPI3MR_HOSTTAG_DEVRMCMD_MIN (MPI3MR_HOSTTAG_BLK_TMS + 1) > +#define MPI3MR_HOSTTAG_DEVRMCMD_MIN (MPI3MR_HOSTTAG_TRANSPORT_CMDS + 1) > #define MPI3MR_HOSTTAG_DEVRMCMD_MAX (MPI3MR_HOSTTAG_DEVRMCMD_MIN + \ > MPI3MR_NUM_DEVRMCMD - 1) > > @@ -279,6 +280,7 @@ enum mpi3mr_reset_reason { > MPI3MR_RESET_FROM_SYSFS_TIMEOUT = 24, > MPI3MR_RESET_FROM_FIRMWARE = 27, > MPI3MR_RESET_FROM_CFG_REQ_TIMEOUT = 29, > + MPI3MR_RESET_FROM_SAS_TRANSPORT_TIMEOUT = 30, > }; > > /* Queue type definitions */ > @@ -1004,6 +1006,7 @@ struct scmd_priv { > * @cfg_page_sz: Default configuration page memory size > * @sas_transport_enabled: SAS transport enabled or not > * @scsi_device_channel: Channel ID for SCSI devices > + * @transport_cmds: Command tracker for SAS transport commands > * @sas_hba: SAS node for the controller > * @sas_expander_list: SAS node list of expanders > * @sas_node_lock: Lock to protect SAS node list > @@ -1188,6 +1191,7 @@ struct mpi3mr_ioc { > > u8 sas_transport_enabled; > u8 scsi_device_channel; > + struct mpi3mr_drv_cmd transport_cmds; > struct mpi3mr_sas_node sas_hba; > struct list_head sas_expander_list; > spinlock_t sas_node_lock; > diff --git a/drivers/scsi/mpi3mr/mpi3mr_fw.c b/drivers/scsi/mpi3mr/mpi3mr_fw.c > index 9155434..1bf3cae 100644 > --- a/drivers/scsi/mpi3mr/mpi3mr_fw.c > +++ b/drivers/scsi/mpi3mr/mpi3mr_fw.c > @@ -312,6 +312,8 @@ mpi3mr_get_drv_cmd(struct mpi3mr_ioc *mrioc, u16 host_tag, > return &mrioc->pel_abort_cmd; > case MPI3MR_HOSTTAG_PEL_WAIT: > return &mrioc->pel_cmds; > + case MPI3MR_HOSTTAG_TRANSPORT_CMDS: > + return &mrioc->transport_cmds; > case MPI3MR_HOSTTAG_INVALID: > if (def_reply && def_reply->function == > MPI3_FUNCTION_EVENT_NOTIFICATION) > @@ -913,6 +915,10 @@ static const struct { > { MPI3MR_RESET_FROM_SYSFS_TIMEOUT, "sysfs TM timeout" }, > { MPI3MR_RESET_FROM_FIRMWARE, "firmware asynchronous reset" }, > { MPI3MR_RESET_FROM_CFG_REQ_TIMEOUT, "configuration request timeout"}, > + { > + MPI3MR_RESET_FROM_SAS_TRANSPORT_TIMEOUT, > + "timeout of a SAS transport layer request" > + }, Fix coding style mismatch here > }; > > /** > @@ -2866,6 +2872,10 @@ static int mpi3mr_alloc_reply_sense_bufs(struct mpi3mr_ioc *mrioc) > if (!mrioc->bsg_cmds.reply) > goto out_failed; > > + mrioc->transport_cmds.reply = kzalloc(mrioc->reply_sz, GFP_KERNEL); > + if (!mrioc->transport_cmds.reply) > + goto out_failed; > + > for (i = 0; i < MPI3MR_NUM_DEVRMCMD; i++) { > mrioc->dev_rmhs_cmds[i].reply = kzalloc(mrioc->reply_sz, > GFP_KERNEL); > @@ -4072,6 +4082,8 @@ void mpi3mr_memset_buffers(struct mpi3mr_ioc *mrioc) > sizeof(*mrioc->pel_cmds.reply)); > memset(mrioc->pel_abort_cmd.reply, 0, > sizeof(*mrioc->pel_abort_cmd.reply)); > + memset(mrioc->transport_cmds.reply, 0, > + sizeof(*mrioc->transport_cmds.reply)); > for (i = 0; i < MPI3MR_NUM_DEVRMCMD; i++) > memset(mrioc->dev_rmhs_cmds[i].reply, 0, > sizeof(*mrioc->dev_rmhs_cmds[i].reply)); > @@ -4217,6 +4229,9 @@ void mpi3mr_free_mem(struct mpi3mr_ioc *mrioc) > kfree(mrioc->chain_bitmap); > mrioc->chain_bitmap = NULL; > > + kfree(mrioc->transport_cmds.reply); > + mrioc->transport_cmds.reply = NULL; > + > for (i = 0; i < MPI3MR_NUM_DEVRMCMD; i++) { > kfree(mrioc->dev_rmhs_cmds[i].reply); > mrioc->dev_rmhs_cmds[i].reply = NULL; > @@ -4417,6 +4432,8 @@ static void mpi3mr_flush_drv_cmds(struct mpi3mr_ioc *mrioc) > cmdptr = &mrioc->pel_abort_cmd; > mpi3mr_drv_cmd_comp_reset(mrioc, cmdptr); > > + cmdptr = &mrioc->transport_cmds; > + mpi3mr_drv_cmd_comp_reset(mrioc, cmdptr); > } > > /** > diff --git a/drivers/scsi/mpi3mr/mpi3mr_os.c b/drivers/scsi/mpi3mr/mpi3mr_os.c > index a486fe3..3b20096 100644 > --- a/drivers/scsi/mpi3mr/mpi3mr_os.c > +++ b/drivers/scsi/mpi3mr/mpi3mr_os.c > @@ -4840,6 +4840,8 @@ mpi3mr_probe(struct pci_dev *pdev, const struct pci_device_id *id) > mpi3mr_init_drv_cmd(&mrioc->host_tm_cmds, MPI3MR_HOSTTAG_BLK_TMS); > mpi3mr_init_drv_cmd(&mrioc->bsg_cmds, MPI3MR_HOSTTAG_BSG_CMDS); > mpi3mr_init_drv_cmd(&mrioc->cfg_cmds, MPI3MR_HOSTTAG_CFG_CMDS); > + mpi3mr_init_drv_cmd(&mrioc->transport_cmds, > + MPI3MR_HOSTTAG_TRANSPORT_CMDS); > > for (i = 0; i < MPI3MR_NUM_DEVRMCMD; i++) > mpi3mr_init_drv_cmd(&mrioc->dev_rmhs_cmds[i], > diff --git a/drivers/scsi/mpi3mr/mpi3mr_transport.c b/drivers/scsi/mpi3mr/mpi3mr_transport.c > index c449820..44a30d7 100644 > --- a/drivers/scsi/mpi3mr/mpi3mr_transport.c > +++ b/drivers/scsi/mpi3mr/mpi3mr_transport.c > @@ -9,6 +9,225 @@ > > #include "mpi3mr.h" > > +/** > + * mpi3mr_post_transport_req - Issue transport requests and wait > + * @mrioc: Adapter instance reference > + * @request: Properly populated MPI3 request > + * @request_sz: Size of the MPI3 request > + * @reply: Pointer to return MPI3 reply > + * @reply_sz: Size of the MPI3 reply buffer > + * @timeout: Timeout in seconds > + * @ioc_status: Pointer to return ioc status > + * > + * A generic function for posting MPI3 requests from the SAS > + * transport layer that uses transport command infrastructure. > + * This blocks for the completion of request for timeout seconds > + * and if the request times out this function faults the > + * controller with proper reason code. > + * > + * On successful completion of the request this function returns > + * appropriate ioc status from the firmware back to the caller. > + * > + * Return: 0 on success, non-zero on failure. > + */ > +static int mpi3mr_post_transport_req(struct mpi3mr_ioc *mrioc, void *request, > + u16 request_sz, void *reply, u16 reply_sz, int timeout, > + u16 *ioc_status) > +{ > + int retval = 0; > + > + mutex_lock(&mrioc->transport_cmds.mutex); > + if (mrioc->transport_cmds.state & MPI3MR_CMD_PENDING) { > + retval = -1; > + ioc_err(mrioc, "sending transport request failed due to command in use\n"); > + mutex_unlock(&mrioc->transport_cmds.mutex); > + goto out; > + } > + mrioc->transport_cmds.state = MPI3MR_CMD_PENDING; > + mrioc->transport_cmds.is_waiting = 1; > + mrioc->transport_cmds.callback = NULL; > + mrioc->transport_cmds.ioc_status = 0; > + mrioc->transport_cmds.ioc_loginfo = 0; > + > + init_completion(&mrioc->transport_cmds.done); > + dprint_cfg_info(mrioc, "posting transport request\n"); > + if (mrioc->logging_level & MPI3_DEBUG_TRANSPORT_INFO) > + dprint_dump(request, request_sz, "transport_req"); > + retval = mpi3mr_admin_request_post(mrioc, request, request_sz, 1); > + if (retval) { > + ioc_err(mrioc, "posting transport request failed\n"); > + goto out_unlock; > + } > + wait_for_completion_timeout(&mrioc->transport_cmds.done, > + (timeout * HZ)); > + if (!(mrioc->transport_cmds.state & MPI3MR_CMD_COMPLETE)) { > + mpi3mr_check_rh_fault_ioc(mrioc, > + MPI3MR_RESET_FROM_SAS_TRANSPORT_TIMEOUT); > + ioc_err(mrioc, "transport request timed out\n"); > + retval = -1; > + goto out_unlock; > + } > + *ioc_status = mrioc->transport_cmds.ioc_status & > + MPI3_IOCSTATUS_STATUS_MASK; > + if ((*ioc_status) != MPI3_IOCSTATUS_SUCCESS) > + dprint_transport_err(mrioc, > + "transport request returned with ioc_status(0x%04x), log_info(0x%08x)\n", > + *ioc_status, mrioc->transport_cmds.ioc_loginfo); > + > + if ((reply) && (mrioc->transport_cmds.state & MPI3MR_CMD_REPLY_VALID)) > + memcpy((u8 *)reply, mrioc->transport_cmds.reply, reply_sz); > + > +out_unlock: > + mrioc->transport_cmds.state = MPI3MR_CMD_NOTUSED; > + mutex_unlock(&mrioc->transport_cmds.mutex); > + > +out: > + return retval; > +} > + > +/* report manufacture request structure */ > +struct rep_manu_request { > + u8 smp_frame_type; > + u8 function; > + u8 reserved; > + u8 request_length; > +}; > + > +/* report manufacture reply structure */ > +struct rep_manu_reply { > + u8 smp_frame_type; /* 0x41 */ > + u8 function; /* 0x01 */ > + u8 function_result; > + u8 response_length; > + u16 expander_change_count; > + u8 reserved0[2]; > + u8 sas_format; > + u8 reserved2[3]; > + u8 vendor_id[SAS_EXPANDER_VENDOR_ID_LEN]; > + u8 product_id[SAS_EXPANDER_PRODUCT_ID_LEN]; > + u8 product_rev[SAS_EXPANDER_PRODUCT_REV_LEN]; > + u8 component_vendor_id[SAS_EXPANDER_COMPONENT_VENDOR_ID_LEN]; > + u16 component_id; > + u8 component_revision_id; > + u8 reserved3; > + u8 vendor_specific[8]; > +}; > + > +/** > + * mpi3mr_report_manufacture - obtain SMP report_manufacture > + * @mrioc: Adapter instance reference > + * @sas_address: SAS address of the expander device > + * @edev: SAS transport layer sas_expander_device object > + * @port_id: ID of the HBA port > + * > + * Fills in the sas_expander_device with manufacturing info. > + * > + * Return: 0 for success, non-zero for failure. > + */ > +static int mpi3mr_report_manufacture(struct mpi3mr_ioc *mrioc, > + u64 sas_address, struct sas_expander_device *edev, u8 port_id) > +{ > + struct mpi3_smp_passthrough_request mpi_request; > + struct mpi3_smp_passthrough_reply mpi_reply; > + struct rep_manu_reply *manufacture_reply; > + struct rep_manu_request *manufacture_request; > + int rc = 0; > + void *psge; > + void *data_out = NULL; > + dma_addr_t data_out_dma; > + dma_addr_t data_in_dma; > + size_t data_in_sz; > + size_t data_out_sz; > + u8 sgl_flags = MPI3MR_SGEFLAGS_SYSTEM_SIMPLE_END_OF_LIST; > + u16 request_sz = sizeof(struct mpi3_smp_passthrough_request); > + u16 reply_sz = sizeof(struct mpi3_smp_passthrough_reply); > + u16 ioc_status; > + > + if (mrioc->reset_in_progress) { > + ioc_err(mrioc, "%s: host reset in progress!\n", __func__); > + return -EFAULT; > + } > + > + data_out_sz = sizeof(struct rep_manu_request); > + data_in_sz = sizeof(struct rep_manu_reply); > + data_out = dma_alloc_coherent(&mrioc->pdev->dev, > + data_out_sz + data_in_sz, &data_out_dma, GFP_KERNEL); > + if (!data_out) { > + rc = -ENOMEM; > + goto out; > + } > + > + data_in_dma = data_out_dma + data_out_sz; > + manufacture_reply = data_out + data_out_sz; > + > + manufacture_request = data_out; > + manufacture_request->smp_frame_type = 0x40; > + manufacture_request->function = 1; > + manufacture_request->reserved = 0; > + manufacture_request->request_length = 0; > + > + memset(&mpi_request, 0, request_sz); > + memset(&mpi_reply, 0, reply_sz); > + mpi_request.host_tag = cpu_to_le16(MPI3MR_HOSTTAG_TRANSPORT_CMDS); > + mpi_request.function = MPI3_FUNCTION_SMP_PASSTHROUGH; > + mpi_request.io_unit_port = (u8) port_id; > + mpi_request.sas_address = cpu_to_le64(sas_address); > + > + psge = &mpi_request.request_sge; > + mpi3mr_add_sg_single(psge, sgl_flags, data_out_sz, data_out_dma); > + > + psge = &mpi_request.response_sge; > + mpi3mr_add_sg_single(psge, sgl_flags, data_in_sz, data_in_dma); > + > + dprint_transport_info(mrioc, > + "sending report manufacturer SMP request to sas_address(0x%016llx), port(%d)\n", > + (unsigned long long)sas_address, port_id); > + > + if (mpi3mr_post_transport_req(mrioc, &mpi_request, request_sz, > + &mpi_reply, reply_sz, MPI3MR_INTADMCMD_TIMEOUT, &ioc_status)) > + goto out; > + > + dprint_transport_info(mrioc, > + "report manufacturer SMP request completed with ioc_status(0x%04x)\n", > + ioc_status); > + > + if (ioc_status == MPI3_IOCSTATUS_SUCCESS) { > + u8 *tmp; > + > + dprint_transport_info(mrioc, > + "report manufacturer - reply data transfer size(%d)\n", > + le16_to_cpu(mpi_reply.response_data_length)); > + > + if (le16_to_cpu(mpi_reply.response_data_length) != > + sizeof(struct rep_manu_reply)) > + goto out; > + > + strscpy(edev->vendor_id, manufacture_reply->vendor_id, > + SAS_EXPANDER_VENDOR_ID_LEN); > + strscpy(edev->product_id, manufacture_reply->product_id, > + SAS_EXPANDER_PRODUCT_ID_LEN); > + strscpy(edev->product_rev, manufacture_reply->product_rev, > + SAS_EXPANDER_PRODUCT_REV_LEN); > + edev->level = manufacture_reply->sas_format & 1; > + if (edev->level) { > + strscpy(edev->component_vendor_id, > + manufacture_reply->component_vendor_id, > + SAS_EXPANDER_COMPONENT_VENDOR_ID_LEN); > + tmp = (u8 *)&manufacture_reply->component_id; > + edev->component_id = tmp[0] << 8 | tmp[1]; > + edev->component_revision_id = > + manufacture_reply->component_revision_id; > + } > + } > + > +out: > + if (data_out) > + dma_free_coherent(&mrioc->pdev->dev, data_out_sz + data_in_sz, > + data_out, data_out_dma); > + > + return rc; > +} > + > /** > * __mpi3mr_expander_find_by_handle - expander search by handle > * @mrioc: Adapter instance reference > @@ -1223,6 +1442,15 @@ static struct mpi3mr_sas_port *mpi3mr_sas_port_add(struct mpi3mr_ioc *mrioc, > mpi3mr_print_device_event_notice(mrioc, true); > } > > + /* fill in report manufacture */ > + if (mr_sas_port->remote_identify.device_type == > + SAS_EDGE_EXPANDER_DEVICE || > + mr_sas_port->remote_identify.device_type == > + SAS_FANOUT_EXPANDER_DEVICE) > + mpi3mr_report_manufacture(mrioc, > + mr_sas_port->remote_identify.sas_address, > + rphy_to_expander_device(rphy), hba_port->port_id); > + > return mr_sas_port; > > out_fail: > -- > 2.27.0 > With small nit fixed, You can add Reviewed-by: Himanshu Madhani <himanshu.madhani@xxxxxxxxxx> -- Himanshu Madhani Oracle Linux Engineering