| /******************************************************************* | 
 |  * This file is part of the Emulex Linux Device Driver for         * | 
 |  * Fibre Channel Host Bus Adapters.                                * | 
 |  * Copyright (C) 2017-2019 Broadcom. All Rights Reserved. The term * | 
 |  * “Broadcom” refers to Broadcom Inc. and/or its subsidiaries.     * | 
 |  * Copyright (C) 2004-2016 Emulex.  All rights reserved.           * | 
 |  * EMULEX and SLI are trademarks of Emulex.                        * | 
 |  * www.broadcom.com                                                * | 
 |  * Portions Copyright (C) 2004-2005 Christoph Hellwig              * | 
 |  *                                                                 * | 
 |  * This program is free software; you can redistribute it and/or   * | 
 |  * modify it under the terms of version 2 of the GNU General       * | 
 |  * Public License as published by the Free Software Foundation.    * | 
 |  * This program is distributed in the hope that it will be useful. * | 
 |  * ALL EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND          * | 
 |  * WARRANTIES, INCLUDING ANY IMPLIED WARRANTY OF MERCHANTABILITY,  * | 
 |  * FITNESS FOR A PARTICULAR PURPOSE, OR NON-INFRINGEMENT, ARE      * | 
 |  * DISCLAIMED, EXCEPT TO THE EXTENT THAT SUCH DISCLAIMERS ARE HELD * | 
 |  * TO BE LEGALLY INVALID.  See the GNU General Public License for  * | 
 |  * more details, a copy of which can be found in the file COPYING  * | 
 |  * included with this package.                                     * | 
 |  *******************************************************************/ | 
 |  | 
 | #include <linux/blkdev.h> | 
 | #include <linux/delay.h> | 
 | #include <linux/dma-mapping.h> | 
 | #include <linux/idr.h> | 
 | #include <linux/interrupt.h> | 
 | #include <linux/kthread.h> | 
 | #include <linux/pci.h> | 
 | #include <linux/slab.h> | 
 | #include <linux/spinlock.h> | 
 | #include <linux/sched/signal.h> | 
 |  | 
 | #include <scsi/scsi.h> | 
 | #include <scsi/scsi_device.h> | 
 | #include <scsi/scsi_host.h> | 
 | #include <scsi/scsi_transport_fc.h> | 
 |  | 
 | #include "lpfc_hw4.h" | 
 | #include "lpfc_hw.h" | 
 | #include "lpfc_sli.h" | 
 | #include "lpfc_sli4.h" | 
 | #include "lpfc_nl.h" | 
 | #include "lpfc_disc.h" | 
 | #include "lpfc_scsi.h" | 
 | #include "lpfc.h" | 
 | #include "lpfc_logmsg.h" | 
 | #include "lpfc_crtn.h" | 
 | #include "lpfc_version.h" | 
 | #include "lpfc_vport.h" | 
 |  | 
 | inline void lpfc_vport_set_state(struct lpfc_vport *vport, | 
 | 				 enum fc_vport_state new_state) | 
 | { | 
 | 	struct fc_vport *fc_vport = vport->fc_vport; | 
 |  | 
 | 	if (fc_vport) { | 
 | 		/* | 
 | 		 * When the transport defines fc_vport_set state we will replace | 
 | 		 * this code with the following line | 
 | 		 */ | 
 | 		/* fc_vport_set_state(fc_vport, new_state); */ | 
 | 		if (new_state != FC_VPORT_INITIALIZING) | 
 | 			fc_vport->vport_last_state = fc_vport->vport_state; | 
 | 		fc_vport->vport_state = new_state; | 
 | 	} | 
 |  | 
 | 	/* for all the error states we will set the invternal state to FAILED */ | 
 | 	switch (new_state) { | 
 | 	case FC_VPORT_NO_FABRIC_SUPP: | 
 | 	case FC_VPORT_NO_FABRIC_RSCS: | 
 | 	case FC_VPORT_FABRIC_LOGOUT: | 
 | 	case FC_VPORT_FABRIC_REJ_WWN: | 
 | 	case FC_VPORT_FAILED: | 
 | 		vport->port_state = LPFC_VPORT_FAILED; | 
 | 		break; | 
 | 	case FC_VPORT_LINKDOWN: | 
 | 		vport->port_state = LPFC_VPORT_UNKNOWN; | 
 | 		break; | 
 | 	default: | 
 | 		/* do nothing */ | 
 | 		break; | 
 | 	} | 
 | } | 
 |  | 
 | int | 
 | lpfc_alloc_vpi(struct lpfc_hba *phba) | 
 | { | 
 | 	unsigned long vpi; | 
 |  | 
 | 	spin_lock_irq(&phba->hbalock); | 
 | 	/* Start at bit 1 because vpi zero is reserved for the physical port */ | 
 | 	vpi = find_next_zero_bit(phba->vpi_bmask, (phba->max_vpi + 1), 1); | 
 | 	if (vpi > phba->max_vpi) | 
 | 		vpi = 0; | 
 | 	else | 
 | 		set_bit(vpi, phba->vpi_bmask); | 
 | 	if (phba->sli_rev == LPFC_SLI_REV4) | 
 | 		phba->sli4_hba.max_cfg_param.vpi_used++; | 
 | 	spin_unlock_irq(&phba->hbalock); | 
 | 	return vpi; | 
 | } | 
 |  | 
 | static void | 
 | lpfc_free_vpi(struct lpfc_hba *phba, int vpi) | 
 | { | 
 | 	if (vpi == 0) | 
 | 		return; | 
 | 	spin_lock_irq(&phba->hbalock); | 
 | 	clear_bit(vpi, phba->vpi_bmask); | 
 | 	if (phba->sli_rev == LPFC_SLI_REV4) | 
 | 		phba->sli4_hba.max_cfg_param.vpi_used--; | 
 | 	spin_unlock_irq(&phba->hbalock); | 
 | } | 
 |  | 
 | static int | 
 | lpfc_vport_sparm(struct lpfc_hba *phba, struct lpfc_vport *vport) | 
 | { | 
 | 	LPFC_MBOXQ_t *pmb; | 
 | 	MAILBOX_t *mb; | 
 | 	struct lpfc_dmabuf *mp; | 
 | 	int  rc; | 
 |  | 
 | 	pmb = mempool_alloc(phba->mbox_mem_pool, GFP_KERNEL); | 
 | 	if (!pmb) { | 
 | 		return -ENOMEM; | 
 | 	} | 
 | 	mb = &pmb->u.mb; | 
 |  | 
 | 	rc = lpfc_read_sparam(phba, pmb, vport->vpi); | 
 | 	if (rc) { | 
 | 		mempool_free(pmb, phba->mbox_mem_pool); | 
 | 		return -ENOMEM; | 
 | 	} | 
 |  | 
 | 	/* | 
 | 	 * Grab buffer pointer and clear context1 so we can use | 
 | 	 * lpfc_sli_issue_box_wait | 
 | 	 */ | 
 | 	mp = (struct lpfc_dmabuf *)pmb->ctx_buf; | 
 | 	pmb->ctx_buf = NULL; | 
 |  | 
 | 	pmb->vport = vport; | 
 | 	rc = lpfc_sli_issue_mbox_wait(phba, pmb, phba->fc_ratov * 2); | 
 | 	if (rc != MBX_SUCCESS) { | 
 | 		if (signal_pending(current)) { | 
 | 			lpfc_printf_vlog(vport, KERN_ERR, LOG_INIT | LOG_VPORT, | 
 | 					 "1830 Signal aborted mbxCmd x%x\n", | 
 | 					 mb->mbxCommand); | 
 | 			lpfc_mbuf_free(phba, mp->virt, mp->phys); | 
 | 			kfree(mp); | 
 | 			if (rc != MBX_TIMEOUT) | 
 | 				mempool_free(pmb, phba->mbox_mem_pool); | 
 | 			return -EINTR; | 
 | 		} else { | 
 | 			lpfc_printf_vlog(vport, KERN_ERR, LOG_INIT | LOG_VPORT, | 
 | 					 "1818 VPort failed init, mbxCmd x%x " | 
 | 					 "READ_SPARM mbxStatus x%x, rc = x%x\n", | 
 | 					 mb->mbxCommand, mb->mbxStatus, rc); | 
 | 			lpfc_mbuf_free(phba, mp->virt, mp->phys); | 
 | 			kfree(mp); | 
 | 			if (rc != MBX_TIMEOUT) | 
 | 				mempool_free(pmb, phba->mbox_mem_pool); | 
 | 			return -EIO; | 
 | 		} | 
 | 	} | 
 |  | 
 | 	memcpy(&vport->fc_sparam, mp->virt, sizeof (struct serv_parm)); | 
 | 	memcpy(&vport->fc_nodename, &vport->fc_sparam.nodeName, | 
 | 	       sizeof (struct lpfc_name)); | 
 | 	memcpy(&vport->fc_portname, &vport->fc_sparam.portName, | 
 | 	       sizeof (struct lpfc_name)); | 
 |  | 
 | 	lpfc_mbuf_free(phba, mp->virt, mp->phys); | 
 | 	kfree(mp); | 
 | 	mempool_free(pmb, phba->mbox_mem_pool); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int | 
 | lpfc_valid_wwn_format(struct lpfc_hba *phba, struct lpfc_name *wwn, | 
 | 		      const char *name_type) | 
 | { | 
 | 				/* ensure that IEEE format 1 addresses | 
 | 				 * contain zeros in bits 59-48 | 
 | 				 */ | 
 | 	if (!((wwn->u.wwn[0] >> 4) == 1 && | 
 | 	      ((wwn->u.wwn[0] & 0xf) != 0 || (wwn->u.wwn[1] & 0xf) != 0))) | 
 | 		return 1; | 
 |  | 
 | 	lpfc_printf_log(phba, KERN_ERR, LOG_VPORT, | 
 | 			"1822 Invalid %s: %02x:%02x:%02x:%02x:" | 
 | 			"%02x:%02x:%02x:%02x\n", | 
 | 			name_type, | 
 | 			wwn->u.wwn[0], wwn->u.wwn[1], | 
 | 			wwn->u.wwn[2], wwn->u.wwn[3], | 
 | 			wwn->u.wwn[4], wwn->u.wwn[5], | 
 | 			wwn->u.wwn[6], wwn->u.wwn[7]); | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int | 
 | lpfc_unique_wwpn(struct lpfc_hba *phba, struct lpfc_vport *new_vport) | 
 | { | 
 | 	struct lpfc_vport *vport; | 
 | 	unsigned long flags; | 
 |  | 
 | 	spin_lock_irqsave(&phba->port_list_lock, flags); | 
 | 	list_for_each_entry(vport, &phba->port_list, listentry) { | 
 | 		if (vport == new_vport) | 
 | 			continue; | 
 | 		/* If they match, return not unique */ | 
 | 		if (memcmp(&vport->fc_sparam.portName, | 
 | 			   &new_vport->fc_sparam.portName, | 
 | 			   sizeof(struct lpfc_name)) == 0) { | 
 | 			spin_unlock_irqrestore(&phba->port_list_lock, flags); | 
 | 			return 0; | 
 | 		} | 
 | 	} | 
 | 	spin_unlock_irqrestore(&phba->port_list_lock, flags); | 
 | 	return 1; | 
 | } | 
 |  | 
 | /** | 
 |  * lpfc_discovery_wait - Wait for driver discovery to quiesce | 
 |  * @vport: The virtual port for which this call is being executed. | 
 |  * | 
 |  * This driver calls this routine specifically from lpfc_vport_delete | 
 |  * to enforce a synchronous execution of vport | 
 |  * delete relative to discovery activities.  The | 
 |  * lpfc_vport_delete routine should not return until it | 
 |  * can reasonably guarantee that discovery has quiesced. | 
 |  * Post FDISC LOGO, the driver must wait until its SAN teardown is | 
 |  * complete and all resources recovered before allowing | 
 |  * cleanup. | 
 |  * | 
 |  * This routine does not require any locks held. | 
 |  **/ | 
 | static void lpfc_discovery_wait(struct lpfc_vport *vport) | 
 | { | 
 | 	struct lpfc_hba *phba = vport->phba; | 
 | 	uint32_t wait_flags = 0; | 
 | 	unsigned long wait_time_max; | 
 | 	unsigned long start_time; | 
 |  | 
 | 	wait_flags = FC_RSCN_MODE | FC_RSCN_DISCOVERY | FC_NLP_MORE | | 
 | 		     FC_RSCN_DEFERRED | FC_NDISC_ACTIVE | FC_DISC_TMO; | 
 |  | 
 | 	/* | 
 | 	 * The time constraint on this loop is a balance between the | 
 | 	 * fabric RA_TOV value and dev_loss tmo.  The driver's | 
 | 	 * devloss_tmo is 10 giving this loop a 3x multiplier minimally. | 
 | 	 */ | 
 | 	wait_time_max = msecs_to_jiffies(((phba->fc_ratov * 3) + 3) * 1000); | 
 | 	wait_time_max += jiffies; | 
 | 	start_time = jiffies; | 
 | 	while (time_before(jiffies, wait_time_max)) { | 
 | 		if ((vport->num_disc_nodes > 0)    || | 
 | 		    (vport->fc_flag & wait_flags)  || | 
 | 		    ((vport->port_state > LPFC_VPORT_FAILED) && | 
 | 		     (vport->port_state < LPFC_VPORT_READY))) { | 
 | 			lpfc_printf_vlog(vport, KERN_INFO, LOG_VPORT, | 
 | 					"1833 Vport discovery quiesce Wait:" | 
 | 					" state x%x fc_flags x%x" | 
 | 					" num_nodes x%x, waiting 1000 msecs" | 
 | 					" total wait msecs x%x\n", | 
 | 					vport->port_state, vport->fc_flag, | 
 | 					vport->num_disc_nodes, | 
 | 					jiffies_to_msecs(jiffies - start_time)); | 
 | 			msleep(1000); | 
 | 		} else { | 
 | 			/* Base case.  Wait variants satisfied.  Break out */ | 
 | 			lpfc_printf_vlog(vport, KERN_INFO, LOG_VPORT, | 
 | 					 "1834 Vport discovery quiesced:" | 
 | 					 " state x%x fc_flags x%x" | 
 | 					 " wait msecs x%x\n", | 
 | 					 vport->port_state, vport->fc_flag, | 
 | 					 jiffies_to_msecs(jiffies | 
 | 						- start_time)); | 
 | 			break; | 
 | 		} | 
 | 	} | 
 |  | 
 | 	if (time_after(jiffies, wait_time_max)) | 
 | 		lpfc_printf_vlog(vport, KERN_ERR, LOG_VPORT, | 
 | 				"1835 Vport discovery quiesce failed:" | 
 | 				" state x%x fc_flags x%x wait msecs x%x\n", | 
 | 				vport->port_state, vport->fc_flag, | 
 | 				jiffies_to_msecs(jiffies - start_time)); | 
 | } | 
 |  | 
 | int | 
 | lpfc_vport_create(struct fc_vport *fc_vport, bool disable) | 
 | { | 
 | 	struct lpfc_nodelist *ndlp; | 
 | 	struct Scsi_Host *shost = fc_vport->shost; | 
 | 	struct lpfc_vport *pport = (struct lpfc_vport *) shost->hostdata; | 
 | 	struct lpfc_hba   *phba = pport->phba; | 
 | 	struct lpfc_vport *vport = NULL; | 
 | 	int instance; | 
 | 	int vpi; | 
 | 	int rc = VPORT_ERROR; | 
 | 	int status; | 
 |  | 
 | 	if ((phba->sli_rev < 3) || !(phba->cfg_enable_npiv)) { | 
 | 		lpfc_printf_log(phba, KERN_ERR, LOG_VPORT, | 
 | 				"1808 Create VPORT failed: " | 
 | 				"NPIV is not enabled: SLImode:%d\n", | 
 | 				phba->sli_rev); | 
 | 		rc = VPORT_INVAL; | 
 | 		goto error_out; | 
 | 	} | 
 |  | 
 | 	/* NPIV is not supported if HBA has NVME Target enabled */ | 
 | 	if (phba->nvmet_support) { | 
 | 		lpfc_printf_log(phba, KERN_ERR, LOG_VPORT, | 
 | 				"3189 Create VPORT failed: " | 
 | 				"NPIV is not supported on NVME Target\n"); | 
 | 		rc = VPORT_INVAL; | 
 | 		goto error_out; | 
 | 	} | 
 |  | 
 | 	vpi = lpfc_alloc_vpi(phba); | 
 | 	if (vpi == 0) { | 
 | 		lpfc_printf_log(phba, KERN_ERR, LOG_VPORT, | 
 | 				"1809 Create VPORT failed: " | 
 | 				"Max VPORTs (%d) exceeded\n", | 
 | 				phba->max_vpi); | 
 | 		rc = VPORT_NORESOURCES; | 
 | 		goto error_out; | 
 | 	} | 
 |  | 
 | 	/* Assign an unused board number */ | 
 | 	if ((instance = lpfc_get_instance()) < 0) { | 
 | 		lpfc_printf_log(phba, KERN_ERR, LOG_VPORT, | 
 | 				"1810 Create VPORT failed: Cannot get " | 
 | 				"instance number\n"); | 
 | 		lpfc_free_vpi(phba, vpi); | 
 | 		rc = VPORT_NORESOURCES; | 
 | 		goto error_out; | 
 | 	} | 
 |  | 
 | 	vport = lpfc_create_port(phba, instance, &fc_vport->dev); | 
 | 	if (!vport) { | 
 | 		lpfc_printf_log(phba, KERN_ERR, LOG_VPORT, | 
 | 				"1811 Create VPORT failed: vpi x%x\n", vpi); | 
 | 		lpfc_free_vpi(phba, vpi); | 
 | 		rc = VPORT_NORESOURCES; | 
 | 		goto error_out; | 
 | 	} | 
 |  | 
 | 	vport->vpi = vpi; | 
 | 	lpfc_debugfs_initialize(vport); | 
 |  | 
 | 	if ((status = lpfc_vport_sparm(phba, vport))) { | 
 | 		if (status == -EINTR) { | 
 | 			lpfc_printf_vlog(vport, KERN_ERR, LOG_VPORT, | 
 | 					 "1831 Create VPORT Interrupted.\n"); | 
 | 			rc = VPORT_ERROR; | 
 | 		} else { | 
 | 			lpfc_printf_vlog(vport, KERN_ERR, LOG_VPORT, | 
 | 					 "1813 Create VPORT failed. " | 
 | 					 "Cannot get sparam\n"); | 
 | 			rc = VPORT_NORESOURCES; | 
 | 		} | 
 | 		lpfc_free_vpi(phba, vpi); | 
 | 		destroy_port(vport); | 
 | 		goto error_out; | 
 | 	} | 
 |  | 
 | 	u64_to_wwn(fc_vport->node_name, vport->fc_nodename.u.wwn); | 
 | 	u64_to_wwn(fc_vport->port_name, vport->fc_portname.u.wwn); | 
 |  | 
 | 	memcpy(&vport->fc_sparam.portName, vport->fc_portname.u.wwn, 8); | 
 | 	memcpy(&vport->fc_sparam.nodeName, vport->fc_nodename.u.wwn, 8); | 
 |  | 
 | 	if (!lpfc_valid_wwn_format(phba, &vport->fc_sparam.nodeName, "WWNN") || | 
 | 	    !lpfc_valid_wwn_format(phba, &vport->fc_sparam.portName, "WWPN")) { | 
 | 		lpfc_printf_vlog(vport, KERN_ERR, LOG_VPORT, | 
 | 				 "1821 Create VPORT failed. " | 
 | 				 "Invalid WWN format\n"); | 
 | 		lpfc_free_vpi(phba, vpi); | 
 | 		destroy_port(vport); | 
 | 		rc = VPORT_INVAL; | 
 | 		goto error_out; | 
 | 	} | 
 |  | 
 | 	if (!lpfc_unique_wwpn(phba, vport)) { | 
 | 		lpfc_printf_vlog(vport, KERN_ERR, LOG_VPORT, | 
 | 				 "1823 Create VPORT failed. " | 
 | 				 "Duplicate WWN on HBA\n"); | 
 | 		lpfc_free_vpi(phba, vpi); | 
 | 		destroy_port(vport); | 
 | 		rc = VPORT_INVAL; | 
 | 		goto error_out; | 
 | 	} | 
 |  | 
 | 	/* Create binary sysfs attribute for vport */ | 
 | 	lpfc_alloc_sysfs_attr(vport); | 
 |  | 
 | 	/* Set the DFT_LUN_Q_DEPTH accordingly */ | 
 | 	vport->cfg_lun_queue_depth  = phba->pport->cfg_lun_queue_depth; | 
 |  | 
 | 	/* Only the physical port can support NVME for now */ | 
 | 	vport->cfg_enable_fc4_type = LPFC_ENABLE_FCP; | 
 |  | 
 | 	*(struct lpfc_vport **)fc_vport->dd_data = vport; | 
 | 	vport->fc_vport = fc_vport; | 
 |  | 
 | 	/* At this point we are fully registered with SCSI Layer.  */ | 
 | 	vport->load_flag |= FC_ALLOW_FDMI; | 
 | 	if (phba->cfg_enable_SmartSAN || | 
 | 	    (phba->cfg_fdmi_on == LPFC_FDMI_SUPPORT)) { | 
 | 		/* Setup appropriate attribute masks */ | 
 | 		vport->fdmi_hba_mask = phba->pport->fdmi_hba_mask; | 
 | 		vport->fdmi_port_mask = phba->pport->fdmi_port_mask; | 
 | 	} | 
 |  | 
 | 	/* | 
 | 	 * In SLI4, the vpi must be activated before it can be used | 
 | 	 * by the port. | 
 | 	 */ | 
 | 	if ((phba->sli_rev == LPFC_SLI_REV4) && | 
 | 	    (pport->fc_flag & FC_VFI_REGISTERED)) { | 
 | 		rc = lpfc_sli4_init_vpi(vport); | 
 | 		if (rc) { | 
 | 			lpfc_printf_log(phba, KERN_ERR, LOG_VPORT, | 
 | 					"1838 Failed to INIT_VPI on vpi %d " | 
 | 					"status %d\n", vpi, rc); | 
 | 			rc = VPORT_NORESOURCES; | 
 | 			lpfc_free_vpi(phba, vpi); | 
 | 			goto error_out; | 
 | 		} | 
 | 	} else if (phba->sli_rev == LPFC_SLI_REV4) { | 
 | 		/* | 
 | 		 * Driver cannot INIT_VPI now. Set the flags to | 
 | 		 * init_vpi when reg_vfi complete. | 
 | 		 */ | 
 | 		vport->fc_flag |= FC_VPORT_NEEDS_INIT_VPI; | 
 | 		lpfc_vport_set_state(vport, FC_VPORT_LINKDOWN); | 
 | 		rc = VPORT_OK; | 
 | 		goto out; | 
 | 	} | 
 |  | 
 | 	if ((phba->link_state < LPFC_LINK_UP) || | 
 | 	    (pport->port_state < LPFC_FABRIC_CFG_LINK) || | 
 | 	    (phba->fc_topology == LPFC_TOPOLOGY_LOOP)) { | 
 | 		lpfc_vport_set_state(vport, FC_VPORT_LINKDOWN); | 
 | 		rc = VPORT_OK; | 
 | 		goto out; | 
 | 	} | 
 |  | 
 | 	if (disable) { | 
 | 		lpfc_vport_set_state(vport, FC_VPORT_DISABLED); | 
 | 		rc = VPORT_OK; | 
 | 		goto out; | 
 | 	} | 
 |  | 
 | 	/* Use the Physical nodes Fabric NDLP to determine if the link is | 
 | 	 * up and ready to FDISC. | 
 | 	 */ | 
 | 	ndlp = lpfc_findnode_did(phba->pport, Fabric_DID); | 
 | 	if (ndlp && NLP_CHK_NODE_ACT(ndlp) && | 
 | 	    ndlp->nlp_state == NLP_STE_UNMAPPED_NODE) { | 
 | 		if (phba->link_flag & LS_NPIV_FAB_SUPPORTED) { | 
 | 			lpfc_set_disctmo(vport); | 
 | 			lpfc_initial_fdisc(vport); | 
 | 		} else { | 
 | 			lpfc_vport_set_state(vport, FC_VPORT_NO_FABRIC_SUPP); | 
 | 			lpfc_printf_vlog(vport, KERN_ERR, LOG_ELS, | 
 | 					 "0262 No NPIV Fabric support\n"); | 
 | 		} | 
 | 	} else { | 
 | 		lpfc_vport_set_state(vport, FC_VPORT_FAILED); | 
 | 	} | 
 | 	rc = VPORT_OK; | 
 |  | 
 | out: | 
 | 	lpfc_printf_vlog(vport, KERN_ERR, LOG_VPORT, | 
 | 			"1825 Vport Created.\n"); | 
 | 	lpfc_host_attrib_init(lpfc_shost_from_vport(vport)); | 
 | error_out: | 
 | 	return rc; | 
 | } | 
 |  | 
 | static int | 
 | disable_vport(struct fc_vport *fc_vport) | 
 | { | 
 | 	struct lpfc_vport *vport = *(struct lpfc_vport **)fc_vport->dd_data; | 
 | 	struct lpfc_hba   *phba = vport->phba; | 
 | 	struct lpfc_nodelist *ndlp = NULL, *next_ndlp = NULL; | 
 | 	long timeout; | 
 | 	struct Scsi_Host *shost = lpfc_shost_from_vport(vport); | 
 |  | 
 | 	ndlp = lpfc_findnode_did(vport, Fabric_DID); | 
 | 	if (ndlp && NLP_CHK_NODE_ACT(ndlp) | 
 | 	    && phba->link_state >= LPFC_LINK_UP) { | 
 | 		vport->unreg_vpi_cmpl = VPORT_INVAL; | 
 | 		timeout = msecs_to_jiffies(phba->fc_ratov * 2000); | 
 | 		if (!lpfc_issue_els_npiv_logo(vport, ndlp)) | 
 | 			while (vport->unreg_vpi_cmpl == VPORT_INVAL && timeout) | 
 | 				timeout = schedule_timeout(timeout); | 
 | 	} | 
 |  | 
 | 	lpfc_sli_host_down(vport); | 
 |  | 
 | 	/* Mark all nodes for discovery so we can remove them by | 
 | 	 * calling lpfc_cleanup_rpis(vport, 1) | 
 | 	 */ | 
 | 	list_for_each_entry_safe(ndlp, next_ndlp, &vport->fc_nodes, nlp_listp) { | 
 | 		if (!NLP_CHK_NODE_ACT(ndlp)) | 
 | 			continue; | 
 | 		if (ndlp->nlp_state == NLP_STE_UNUSED_NODE) | 
 | 			continue; | 
 | 		lpfc_disc_state_machine(vport, ndlp, NULL, | 
 | 					NLP_EVT_DEVICE_RECOVERY); | 
 | 	} | 
 | 	lpfc_cleanup_rpis(vport, 1); | 
 |  | 
 | 	lpfc_stop_vport_timers(vport); | 
 | 	lpfc_unreg_all_rpis(vport); | 
 | 	lpfc_unreg_default_rpis(vport); | 
 | 	/* | 
 | 	 * Completion of unreg_vpi (lpfc_mbx_cmpl_unreg_vpi) does the | 
 | 	 * scsi_host_put() to release the vport. | 
 | 	 */ | 
 | 	lpfc_mbx_unreg_vpi(vport); | 
 | 	if (phba->sli_rev == LPFC_SLI_REV4) { | 
 | 		spin_lock_irq(shost->host_lock); | 
 | 		vport->fc_flag |= FC_VPORT_NEEDS_INIT_VPI; | 
 | 		spin_unlock_irq(shost->host_lock); | 
 | 	} | 
 |  | 
 | 	lpfc_vport_set_state(vport, FC_VPORT_DISABLED); | 
 | 	lpfc_printf_vlog(vport, KERN_ERR, LOG_VPORT, | 
 | 			 "1826 Vport Disabled.\n"); | 
 | 	return VPORT_OK; | 
 | } | 
 |  | 
 | static int | 
 | enable_vport(struct fc_vport *fc_vport) | 
 | { | 
 | 	struct lpfc_vport *vport = *(struct lpfc_vport **)fc_vport->dd_data; | 
 | 	struct lpfc_hba   *phba = vport->phba; | 
 | 	struct lpfc_nodelist *ndlp = NULL; | 
 | 	struct Scsi_Host *shost = lpfc_shost_from_vport(vport); | 
 |  | 
 | 	if ((phba->link_state < LPFC_LINK_UP) || | 
 | 	    (phba->fc_topology == LPFC_TOPOLOGY_LOOP)) { | 
 | 		lpfc_vport_set_state(vport, FC_VPORT_LINKDOWN); | 
 | 		return VPORT_OK; | 
 | 	} | 
 |  | 
 | 	spin_lock_irq(shost->host_lock); | 
 | 	vport->load_flag |= FC_LOADING; | 
 | 	if (vport->fc_flag & FC_VPORT_NEEDS_INIT_VPI) { | 
 | 		spin_unlock_irq(shost->host_lock); | 
 | 		lpfc_issue_init_vpi(vport); | 
 | 		goto out; | 
 | 	} | 
 |  | 
 | 	vport->fc_flag |= FC_VPORT_NEEDS_REG_VPI; | 
 | 	spin_unlock_irq(shost->host_lock); | 
 |  | 
 | 	/* Use the Physical nodes Fabric NDLP to determine if the link is | 
 | 	 * up and ready to FDISC. | 
 | 	 */ | 
 | 	ndlp = lpfc_findnode_did(phba->pport, Fabric_DID); | 
 | 	if (ndlp && NLP_CHK_NODE_ACT(ndlp) | 
 | 	    && ndlp->nlp_state == NLP_STE_UNMAPPED_NODE) { | 
 | 		if (phba->link_flag & LS_NPIV_FAB_SUPPORTED) { | 
 | 			lpfc_set_disctmo(vport); | 
 | 			lpfc_initial_fdisc(vport); | 
 | 		} else { | 
 | 			lpfc_vport_set_state(vport, FC_VPORT_NO_FABRIC_SUPP); | 
 | 			lpfc_printf_vlog(vport, KERN_ERR, LOG_ELS, | 
 | 					 "0264 No NPIV Fabric support\n"); | 
 | 		} | 
 | 	} else { | 
 | 		lpfc_vport_set_state(vport, FC_VPORT_FAILED); | 
 | 	} | 
 |  | 
 | out: | 
 | 	lpfc_printf_vlog(vport, KERN_ERR, LOG_VPORT, | 
 | 			 "1827 Vport Enabled.\n"); | 
 | 	return VPORT_OK; | 
 | } | 
 |  | 
 | int | 
 | lpfc_vport_disable(struct fc_vport *fc_vport, bool disable) | 
 | { | 
 | 	if (disable) | 
 | 		return disable_vport(fc_vport); | 
 | 	else | 
 | 		return enable_vport(fc_vport); | 
 | } | 
 |  | 
 |  | 
 | int | 
 | lpfc_vport_delete(struct fc_vport *fc_vport) | 
 | { | 
 | 	struct lpfc_nodelist *ndlp = NULL; | 
 | 	struct lpfc_vport *vport = *(struct lpfc_vport **)fc_vport->dd_data; | 
 | 	struct Scsi_Host *shost = lpfc_shost_from_vport(vport); | 
 | 	struct lpfc_hba   *phba = vport->phba; | 
 | 	long timeout; | 
 | 	bool ns_ndlp_referenced = false; | 
 |  | 
 | 	if (vport->port_type == LPFC_PHYSICAL_PORT) { | 
 | 		lpfc_printf_vlog(vport, KERN_ERR, LOG_VPORT, | 
 | 				 "1812 vport_delete failed: Cannot delete " | 
 | 				 "physical host\n"); | 
 | 		return VPORT_ERROR; | 
 | 	} | 
 |  | 
 | 	/* If the vport is a static vport fail the deletion. */ | 
 | 	if ((vport->vport_flag & STATIC_VPORT) && | 
 | 		!(phba->pport->load_flag & FC_UNLOADING)) { | 
 | 		lpfc_printf_vlog(vport, KERN_ERR, LOG_VPORT, | 
 | 				 "1837 vport_delete failed: Cannot delete " | 
 | 				 "static vport.\n"); | 
 | 		return VPORT_ERROR; | 
 | 	} | 
 | 	spin_lock_irq(&phba->hbalock); | 
 | 	vport->load_flag |= FC_UNLOADING; | 
 | 	spin_unlock_irq(&phba->hbalock); | 
 | 	/* | 
 | 	 * If we are not unloading the driver then prevent the vport_delete | 
 | 	 * from happening until after this vport's discovery is finished. | 
 | 	 */ | 
 | 	if (!(phba->pport->load_flag & FC_UNLOADING)) { | 
 | 		int check_count = 0; | 
 | 		while (check_count < ((phba->fc_ratov * 3) + 3) && | 
 | 		       vport->port_state > LPFC_VPORT_FAILED && | 
 | 		       vport->port_state < LPFC_VPORT_READY) { | 
 | 			check_count++; | 
 | 			msleep(1000); | 
 | 		} | 
 | 		if (vport->port_state > LPFC_VPORT_FAILED && | 
 | 		    vport->port_state < LPFC_VPORT_READY) | 
 | 			return -EAGAIN; | 
 | 	} | 
 | 	/* | 
 | 	 * This is a bit of a mess.  We want to ensure the shost doesn't get | 
 | 	 * torn down until we're done with the embedded lpfc_vport structure. | 
 | 	 * | 
 | 	 * Beyond holding a reference for this function, we also need a | 
 | 	 * reference for outstanding I/O requests we schedule during delete | 
 | 	 * processing.  But once we scsi_remove_host() we can no longer obtain | 
 | 	 * a reference through scsi_host_get(). | 
 | 	 * | 
 | 	 * So we take two references here.  We release one reference at the | 
 | 	 * bottom of the function -- after delinking the vport.  And we | 
 | 	 * release the other at the completion of the unreg_vpi that get's | 
 | 	 * initiated after we've disposed of all other resources associated | 
 | 	 * with the port. | 
 | 	 */ | 
 | 	if (!scsi_host_get(shost)) | 
 | 		return VPORT_INVAL; | 
 | 	if (!scsi_host_get(shost)) { | 
 | 		scsi_host_put(shost); | 
 | 		return VPORT_INVAL; | 
 | 	} | 
 | 	lpfc_free_sysfs_attr(vport); | 
 |  | 
 | 	lpfc_debugfs_terminate(vport); | 
 |  | 
 | 	/* | 
 | 	 * The call to fc_remove_host might release the NameServer ndlp. Since | 
 | 	 * we might need to use the ndlp to send the DA_ID CT command, | 
 | 	 * increment the reference for the NameServer ndlp to prevent it from | 
 | 	 * being released. | 
 | 	 */ | 
 | 	ndlp = lpfc_findnode_did(vport, NameServer_DID); | 
 | 	if (ndlp && NLP_CHK_NODE_ACT(ndlp)) { | 
 | 		lpfc_nlp_get(ndlp); | 
 | 		ns_ndlp_referenced = true; | 
 | 	} | 
 |  | 
 | 	/* Remove FC host and then SCSI host with the vport */ | 
 | 	fc_remove_host(shost); | 
 | 	scsi_remove_host(shost); | 
 |  | 
 | 	ndlp = lpfc_findnode_did(phba->pport, Fabric_DID); | 
 |  | 
 | 	/* In case of driver unload, we shall not perform fabric logo as the | 
 | 	 * worker thread already stopped at this stage and, in this case, we | 
 | 	 * can safely skip the fabric logo. | 
 | 	 */ | 
 | 	if (phba->pport->load_flag & FC_UNLOADING) { | 
 | 		if (ndlp && NLP_CHK_NODE_ACT(ndlp) && | 
 | 		    ndlp->nlp_state == NLP_STE_UNMAPPED_NODE && | 
 | 		    phba->link_state >= LPFC_LINK_UP) { | 
 | 			/* First look for the Fabric ndlp */ | 
 | 			ndlp = lpfc_findnode_did(vport, Fabric_DID); | 
 | 			if (!ndlp) | 
 | 				goto skip_logo; | 
 | 			else if (!NLP_CHK_NODE_ACT(ndlp)) { | 
 | 				ndlp = lpfc_enable_node(vport, ndlp, | 
 | 							NLP_STE_UNUSED_NODE); | 
 | 				if (!ndlp) | 
 | 					goto skip_logo; | 
 | 			} | 
 | 			/* Remove ndlp from vport npld list */ | 
 | 			lpfc_dequeue_node(vport, ndlp); | 
 |  | 
 | 			/* Indicate free memory when release */ | 
 | 			spin_lock_irq(&phba->ndlp_lock); | 
 | 			NLP_SET_FREE_REQ(ndlp); | 
 | 			spin_unlock_irq(&phba->ndlp_lock); | 
 | 			/* Kick off release ndlp when it can be safely done */ | 
 | 			lpfc_nlp_put(ndlp); | 
 | 		} | 
 | 		goto skip_logo; | 
 | 	} | 
 |  | 
 | 	/* Otherwise, we will perform fabric logo as needed */ | 
 | 	if (ndlp && NLP_CHK_NODE_ACT(ndlp) && | 
 | 	    ndlp->nlp_state == NLP_STE_UNMAPPED_NODE && | 
 | 	    phba->link_state >= LPFC_LINK_UP && | 
 | 	    phba->fc_topology != LPFC_TOPOLOGY_LOOP) { | 
 | 		if (vport->cfg_enable_da_id) { | 
 | 			timeout = msecs_to_jiffies(phba->fc_ratov * 2000); | 
 | 			if (!lpfc_ns_cmd(vport, SLI_CTNS_DA_ID, 0, 0)) | 
 | 				while (vport->ct_flags && timeout) | 
 | 					timeout = schedule_timeout(timeout); | 
 | 			else | 
 | 				lpfc_printf_log(vport->phba, KERN_WARNING, | 
 | 						LOG_VPORT, | 
 | 						"1829 CT command failed to " | 
 | 						"delete objects on fabric\n"); | 
 | 		} | 
 | 		/* First look for the Fabric ndlp */ | 
 | 		ndlp = lpfc_findnode_did(vport, Fabric_DID); | 
 | 		if (!ndlp) { | 
 | 			/* Cannot find existing Fabric ndlp, allocate one */ | 
 | 			ndlp = lpfc_nlp_init(vport, Fabric_DID); | 
 | 			if (!ndlp) | 
 | 				goto skip_logo; | 
 | 			/* Indicate free memory when release */ | 
 | 			NLP_SET_FREE_REQ(ndlp); | 
 | 		} else { | 
 | 			if (!NLP_CHK_NODE_ACT(ndlp)) { | 
 | 				ndlp = lpfc_enable_node(vport, ndlp, | 
 | 						NLP_STE_UNUSED_NODE); | 
 | 				if (!ndlp) | 
 | 					goto skip_logo; | 
 | 			} | 
 |  | 
 | 			/* Remove ndlp from vport list */ | 
 | 			lpfc_dequeue_node(vport, ndlp); | 
 | 			spin_lock_irq(&phba->ndlp_lock); | 
 | 			if (!NLP_CHK_FREE_REQ(ndlp)) | 
 | 				/* Indicate free memory when release */ | 
 | 				NLP_SET_FREE_REQ(ndlp); | 
 | 			else { | 
 | 				/* Skip this if ndlp is already in free mode */ | 
 | 				spin_unlock_irq(&phba->ndlp_lock); | 
 | 				goto skip_logo; | 
 | 			} | 
 | 			spin_unlock_irq(&phba->ndlp_lock); | 
 | 		} | 
 |  | 
 | 		/* | 
 | 		 * If the vpi is not registered, then a valid FDISC doesn't | 
 | 		 * exist and there is no need for a ELS LOGO.  Just cleanup | 
 | 		 * the ndlp. | 
 | 		 */ | 
 | 		if (!(vport->vpi_state & LPFC_VPI_REGISTERED)) { | 
 | 			lpfc_nlp_put(ndlp); | 
 | 			goto skip_logo; | 
 | 		} | 
 |  | 
 | 		vport->unreg_vpi_cmpl = VPORT_INVAL; | 
 | 		timeout = msecs_to_jiffies(phba->fc_ratov * 2000); | 
 | 		if (!lpfc_issue_els_npiv_logo(vport, ndlp)) | 
 | 			while (vport->unreg_vpi_cmpl == VPORT_INVAL && timeout) | 
 | 				timeout = schedule_timeout(timeout); | 
 | 	} | 
 |  | 
 | 	if (!(phba->pport->load_flag & FC_UNLOADING)) | 
 | 		lpfc_discovery_wait(vport); | 
 |  | 
 | skip_logo: | 
 |  | 
 | 	/* | 
 | 	 * If the NameServer ndlp has been incremented to allow the DA_ID CT | 
 | 	 * command to be sent, decrement the ndlp now. | 
 | 	 */ | 
 | 	if (ns_ndlp_referenced) { | 
 | 		ndlp = lpfc_findnode_did(vport, NameServer_DID); | 
 | 		lpfc_nlp_put(ndlp); | 
 | 	} | 
 |  | 
 | 	lpfc_cleanup(vport); | 
 | 	lpfc_sli_host_down(vport); | 
 |  | 
 | 	lpfc_stop_vport_timers(vport); | 
 |  | 
 | 	if (!(phba->pport->load_flag & FC_UNLOADING)) { | 
 | 		lpfc_unreg_all_rpis(vport); | 
 | 		lpfc_unreg_default_rpis(vport); | 
 | 		/* | 
 | 		 * Completion of unreg_vpi (lpfc_mbx_cmpl_unreg_vpi) | 
 | 		 * does the scsi_host_put() to release the vport. | 
 | 		 */ | 
 | 		if (!(vport->vpi_state & LPFC_VPI_REGISTERED) || | 
 | 				lpfc_mbx_unreg_vpi(vport)) | 
 | 			scsi_host_put(shost); | 
 | 	} else | 
 | 		scsi_host_put(shost); | 
 |  | 
 | 	lpfc_free_vpi(phba, vport->vpi); | 
 | 	vport->work_port_events = 0; | 
 | 	spin_lock_irq(&phba->port_list_lock); | 
 | 	list_del_init(&vport->listentry); | 
 | 	spin_unlock_irq(&phba->port_list_lock); | 
 | 	lpfc_printf_vlog(vport, KERN_ERR, LOG_VPORT, | 
 | 			 "1828 Vport Deleted.\n"); | 
 | 	scsi_host_put(shost); | 
 | 	return VPORT_OK; | 
 | } | 
 |  | 
 | struct lpfc_vport ** | 
 | lpfc_create_vport_work_array(struct lpfc_hba *phba) | 
 | { | 
 | 	struct lpfc_vport *port_iterator; | 
 | 	struct lpfc_vport **vports; | 
 | 	int index = 0; | 
 | 	vports = kcalloc(phba->max_vports + 1, sizeof(struct lpfc_vport *), | 
 | 			 GFP_KERNEL); | 
 | 	if (vports == NULL) | 
 | 		return NULL; | 
 | 	spin_lock_irq(&phba->port_list_lock); | 
 | 	list_for_each_entry(port_iterator, &phba->port_list, listentry) { | 
 | 		if (port_iterator->load_flag & FC_UNLOADING) | 
 | 			continue; | 
 | 		if (!scsi_host_get(lpfc_shost_from_vport(port_iterator))) { | 
 | 			lpfc_printf_vlog(port_iterator, KERN_ERR, LOG_VPORT, | 
 | 					 "1801 Create vport work array FAILED: " | 
 | 					 "cannot do scsi_host_get\n"); | 
 | 			continue; | 
 | 		} | 
 | 		vports[index++] = port_iterator; | 
 | 	} | 
 | 	spin_unlock_irq(&phba->port_list_lock); | 
 | 	return vports; | 
 | } | 
 |  | 
 | void | 
 | lpfc_destroy_vport_work_array(struct lpfc_hba *phba, struct lpfc_vport **vports) | 
 | { | 
 | 	int i; | 
 | 	if (vports == NULL) | 
 | 		return; | 
 | 	for (i = 0; i <= phba->max_vports && vports[i] != NULL; i++) | 
 | 		scsi_host_put(lpfc_shost_from_vport(vports[i])); | 
 | 	kfree(vports); | 
 | } | 
 |  | 
 |  | 
 | /** | 
 |  * lpfc_vport_reset_stat_data - Reset the statistical data for the vport | 
 |  * @vport: Pointer to vport object. | 
 |  * | 
 |  * This function resets the statistical data for the vport. This function | 
 |  * is called with the host_lock held | 
 |  **/ | 
 | void | 
 | lpfc_vport_reset_stat_data(struct lpfc_vport *vport) | 
 | { | 
 | 	struct lpfc_nodelist *ndlp = NULL, *next_ndlp = NULL; | 
 |  | 
 | 	list_for_each_entry_safe(ndlp, next_ndlp, &vport->fc_nodes, nlp_listp) { | 
 | 		if (!NLP_CHK_NODE_ACT(ndlp)) | 
 | 			continue; | 
 | 		if (ndlp->lat_data) | 
 | 			memset(ndlp->lat_data, 0, LPFC_MAX_BUCKET_COUNT * | 
 | 				sizeof(struct lpfc_scsicmd_bkt)); | 
 | 	} | 
 | } | 
 |  | 
 |  | 
 | /** | 
 |  * lpfc_alloc_bucket - Allocate data buffer required for statistical data | 
 |  * @vport: Pointer to vport object. | 
 |  * | 
 |  * This function allocates data buffer required for all the FC | 
 |  * nodes of the vport to collect statistical data. | 
 |  **/ | 
 | void | 
 | lpfc_alloc_bucket(struct lpfc_vport *vport) | 
 | { | 
 | 	struct lpfc_nodelist *ndlp = NULL, *next_ndlp = NULL; | 
 |  | 
 | 	list_for_each_entry_safe(ndlp, next_ndlp, &vport->fc_nodes, nlp_listp) { | 
 | 		if (!NLP_CHK_NODE_ACT(ndlp)) | 
 | 			continue; | 
 |  | 
 | 		kfree(ndlp->lat_data); | 
 | 		ndlp->lat_data = NULL; | 
 |  | 
 | 		if (ndlp->nlp_state == NLP_STE_MAPPED_NODE) { | 
 | 			ndlp->lat_data = kcalloc(LPFC_MAX_BUCKET_COUNT, | 
 | 					 sizeof(struct lpfc_scsicmd_bkt), | 
 | 					 GFP_ATOMIC); | 
 |  | 
 | 			if (!ndlp->lat_data) | 
 | 				lpfc_printf_vlog(vport, KERN_ERR, LOG_NODE, | 
 | 					"0287 lpfc_alloc_bucket failed to " | 
 | 					"allocate statistical data buffer DID " | 
 | 					"0x%x\n", ndlp->nlp_DID); | 
 | 		} | 
 | 	} | 
 | } | 
 |  | 
 | /** | 
 |  * lpfc_free_bucket - Free data buffer required for statistical data | 
 |  * @vport: Pointer to vport object. | 
 |  * | 
 |  * Th function frees statistical data buffer of all the FC | 
 |  * nodes of the vport. | 
 |  **/ | 
 | void | 
 | lpfc_free_bucket(struct lpfc_vport *vport) | 
 | { | 
 | 	struct lpfc_nodelist *ndlp = NULL, *next_ndlp = NULL; | 
 |  | 
 | 	list_for_each_entry_safe(ndlp, next_ndlp, &vport->fc_nodes, nlp_listp) { | 
 | 		if (!NLP_CHK_NODE_ACT(ndlp)) | 
 | 			continue; | 
 |  | 
 | 		kfree(ndlp->lat_data); | 
 | 		ndlp->lat_data = NULL; | 
 | 	} | 
 | } |