lpfc_vport.c

来自「linux 内核源代码」· C语言 代码 · 共 572 行 · 第 1/2 页

C
572
字号
/******************************************************************* * This file is part of the Emulex Linux Device Driver for         * * Fibre Channel Host Bus Adapters.                                * * Copyright (C) 2004-2006 Emulex.  All rights reserved.           * * EMULEX and SLI are trademarks of Emulex.                        * * www.emulex.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/spinlock.h>#include <scsi/scsi.h>#include <scsi/scsi_device.h>#include <scsi/scsi_host.h>#include <scsi/scsi_transport_fc.h>#include "lpfc_hw.h"#include "lpfc_sli.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;	}}static intlpfc_alloc_vpi(struct lpfc_hba *phba){	int  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);	spin_unlock_irq(&phba->hbalock);	return vpi;}static voidlpfc_free_vpi(struct lpfc_hba *phba, int vpi){	spin_lock_irq(&phba->hbalock);	clear_bit(vpi, phba->vpi_bmask);	spin_unlock_irq(&phba->hbalock);}static intlpfc_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->mb;	lpfc_read_sparam(phba, pmb, vport->vpi);	/*	 * Grab buffer pointer and clear context1 so we can use	 * lpfc_sli_issue_box_wait	 */	mp = (struct lpfc_dmabuf *) pmb->context1;	pmb->context1 = NULL;	pmb->vport = vport;	rc = lpfc_sli_issue_mbox_wait(phba, pmb, phba->fc_ratov * 2);	if (rc != MBX_SUCCESS) {		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 intlpfc_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 intlpfc_unique_wwpn(struct lpfc_hba *phba, struct lpfc_vport *new_vport){	struct lpfc_vport *vport;	unsigned long flags;	spin_lock_irqsave(&phba->hbalock, 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->hbalock, flags);			return 0;		}	}	spin_unlock_irqrestore(&phba->hbalock, flags);	return 1;}intlpfc_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;	if ((phba->sli_rev < 3) ||		!(phba->sli3_options & LPFC_SLI3_NPIV_ENABLED)) {		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;	}	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 (lpfc_vport_sparm(phba, vport)) {		lpfc_printf_vlog(vport, KERN_ERR, LOG_VPORT,				 "1813 Create VPORT failed. "				 "Cannot get sparam\n");		lpfc_free_vpi(phba, vpi);		destroy_port(vport);		rc = VPORT_NORESOURCES;		goto error_out;	}	memcpy(vport->fc_portname.u.wwn, vport->fc_sparam.portName.u.wwn, 8);	memcpy(vport->fc_nodename.u.wwn, vport->fc_sparam.nodeName.u.wwn, 8);	if (fc_vport->node_name != 0)		u64_to_wwn(fc_vport->node_name, vport->fc_nodename.u.wwn);	if (fc_vport->port_name != 0)		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");

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?