mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02:00 
			
		
		
		
	[SCSI] mvsas: add support for 94xx; layout change; bug fixes
This version contains following main changes - Switch to new layout to support more types of ASIC. - SSP TMF supported and related Error Handing enhanced. - Support flash feature with delay 2*HZ when PHY changed. - Support Marvell 94xx series ASIC for 6G SAS/SATA, which has 2 88SE64xx chips but any different register description. - Support SPI flash for HBA-related configuration info. - Other patch enhanced from kernel side such as increasing PHY type [jejb: fold back in DMA_BIT_MASK changes] Signed-off-by: Ying Chu <jasonchu@marvell.com> Signed-off-by: Andy Yan <ayan@marvell.com> Signed-off-by: Ke Wei <kewei@marvell.com> Signed-off-by: Jeff Garzik <jgarzik@redhat.com> Signed-off-by: James Bottomley <James.Bottomley@HansenPartnership.com>
This commit is contained in:
		
							parent
							
								
									dd4969a892
								
							
						
					
					
						commit
						20b09c2992
					
				
					 11 changed files with 3915 additions and 1399 deletions
				
			
		| 
						 | 
				
			
			@ -1,35 +1,42 @@
 | 
			
		|||
#
 | 
			
		||||
# Kernel configuration file for 88SE64XX SAS/SATA driver.
 | 
			
		||||
# Kernel configuration file for 88SE64XX/88SE94XX SAS/SATA driver.
 | 
			
		||||
#
 | 
			
		||||
# Copyright 2007 Red Hat, Inc.
 | 
			
		||||
# Copyright 2008 Marvell. <kewei@marvell.com>
 | 
			
		||||
#
 | 
			
		||||
# This file is licensed under GPLv2.
 | 
			
		||||
#
 | 
			
		||||
# This file is part of the 88SE64XX driver.
 | 
			
		||||
# This file is part of the 88SE64XX/88SE94XX driver.
 | 
			
		||||
#
 | 
			
		||||
# The 88SE64XX driver is free software; you can redistribute
 | 
			
		||||
# The 88SE64XX/88SE94XX driver is free software; you can redistribute
 | 
			
		||||
# it and/or modify it under the terms of the GNU General Public License
 | 
			
		||||
# as published by the Free Software Foundation; version 2 of the
 | 
			
		||||
# License.
 | 
			
		||||
#
 | 
			
		||||
# The 88SE64XX driver is distributed in the hope that it will be
 | 
			
		||||
# The 88SE64XX/88SE94XX driver is distributed in the hope that 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.
 | 
			
		||||
#
 | 
			
		||||
# You should have received a copy of the GNU General Public License
 | 
			
		||||
# along with 88SE64XX Driver; if not, write to the Free Software
 | 
			
		||||
# along with 88SE64XX/88SE94XX Driver; if not, write to the Free Software
 | 
			
		||||
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 | 
			
		||||
#
 | 
			
		||||
#
 | 
			
		||||
 | 
			
		||||
config SCSI_MVSAS
 | 
			
		||||
	tristate "Marvell 88SE64XX SAS/SATA support"
 | 
			
		||||
	tristate "Marvell 88SE64XX/88SE94XX SAS/SATA support"
 | 
			
		||||
	depends on PCI
 | 
			
		||||
	select SCSI_SAS_LIBSAS
 | 
			
		||||
	select FW_LOADER
 | 
			
		||||
	help
 | 
			
		||||
		This driver supports Marvell's SAS/SATA 3Gb/s PCI-E 88SE64XX
 | 
			
		||||
		chip based host adapters.
 | 
			
		||||
		This driver supports Marvell's SAS/SATA 3Gb/s PCI-E 88SE64XX and 6Gb/s
 | 
			
		||||
		PCI-E 88SE94XX chip based host adapters.
 | 
			
		||||
 | 
			
		||||
config SCSI_MVSAS_DEBUG
 | 
			
		||||
	bool "Compile in debug mode"
 | 
			
		||||
	default y
 | 
			
		||||
	depends on SCSI_MVSAS
 | 
			
		||||
	help
 | 
			
		||||
		Compiles the 88SE64XX/88SE94XX driver in debug mode.  In debug mode,
 | 
			
		||||
		the driver prints some messages to the console.
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
#
 | 
			
		||||
# Makefile for Marvell 88SE64xx SAS/SATA driver.
 | 
			
		||||
# Makefile for Marvell 88SE64xx/88SE84xx SAS/SATA driver.
 | 
			
		||||
#
 | 
			
		||||
# Copyright 2007 Red Hat, Inc.
 | 
			
		||||
# Copyright 2008 Marvell. <kewei@marvell.com>
 | 
			
		||||
| 
						 | 
				
			
			@ -21,7 +21,12 @@
 | 
			
		|||
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
 | 
			
		||||
# USA
 | 
			
		||||
 | 
			
		||||
ifeq ($(CONFIG_SCSI_MVSAS_DEBUG),y)
 | 
			
		||||
	EXTRA_CFLAGS += -DMV_DEBUG
 | 
			
		||||
endif
 | 
			
		||||
 | 
			
		||||
obj-$(CONFIG_SCSI_MVSAS) += mvsas.o
 | 
			
		||||
mvsas-y +=  mv_init.o  \
 | 
			
		||||
           mv_sas.o   \
 | 
			
		||||
           mv_64xx.o
 | 
			
		||||
           mv_64xx.o  \
 | 
			
		||||
           mv_94xx.o
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,184 +1,793 @@
 | 
			
		|||
/*
 | 
			
		||||
	mv_64xx.c - Marvell 88SE6440 SAS/SATA support
 | 
			
		||||
 | 
			
		||||
	Copyright 2007 Red Hat, Inc.
 | 
			
		||||
	Copyright 2008 Marvell. <kewei@marvell.com>
 | 
			
		||||
 | 
			
		||||
	This program is free software; you can redistribute it and/or
 | 
			
		||||
	modify it under the terms of the GNU General Public License as
 | 
			
		||||
	published by the Free Software Foundation; either version 2,
 | 
			
		||||
	or (at your option) any later version.
 | 
			
		||||
 | 
			
		||||
	This program is distributed in the hope that 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.
 | 
			
		||||
 | 
			
		||||
	You should have received a copy of the GNU General Public
 | 
			
		||||
	License along with this program; see the file COPYING.	If not,
 | 
			
		||||
	write to the Free Software Foundation, 675 Mass Ave, Cambridge,
 | 
			
		||||
	MA 02139, USA.
 | 
			
		||||
 | 
			
		||||
 */
 | 
			
		||||
 * Marvell 88SE64xx hardware specific
 | 
			
		||||
 *
 | 
			
		||||
 * Copyright 2007 Red Hat, Inc.
 | 
			
		||||
 * Copyright 2008 Marvell. <kewei@marvell.com>
 | 
			
		||||
 *
 | 
			
		||||
 * This file is licensed under GPLv2.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is free software; you can redistribute it and/or
 | 
			
		||||
 * modify it under the terms of the GNU General Public License as
 | 
			
		||||
 * published by the Free Software Foundation; version 2 of the
 | 
			
		||||
 * License.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is distributed in the hope that 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.
 | 
			
		||||
 *
 | 
			
		||||
 * You should have received a copy of the GNU General Public License
 | 
			
		||||
 * along with this program; if not, write to the Free Software
 | 
			
		||||
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
 | 
			
		||||
 * USA
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
#include "mv_sas.h"
 | 
			
		||||
#include "mv_64xx.h"
 | 
			
		||||
#include "mv_chips.h"
 | 
			
		||||
 | 
			
		||||
void mvs_detect_porttype(struct mvs_info *mvi, int i)
 | 
			
		||||
static void mvs_64xx_detect_porttype(struct mvs_info *mvi, int i)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	u32 reg;
 | 
			
		||||
	struct mvs_phy *phy = &mvi->phy[i];
 | 
			
		||||
 | 
			
		||||
	/* TODO check & save device type */
 | 
			
		||||
	reg = mr32(GBL_PORT_TYPE);
 | 
			
		||||
 | 
			
		||||
	reg = mr32(MVS_GBL_PORT_TYPE);
 | 
			
		||||
	phy->phy_type &= ~(PORT_TYPE_SAS | PORT_TYPE_SATA);
 | 
			
		||||
	if (reg & MODE_SAS_SATA & (1 << i))
 | 
			
		||||
		phy->phy_type |= PORT_TYPE_SAS;
 | 
			
		||||
	else
 | 
			
		||||
		phy->phy_type |= PORT_TYPE_SATA;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mvs_enable_xmt(struct mvs_info *mvi, int PhyId)
 | 
			
		||||
static void __devinit mvs_64xx_enable_xmt(struct mvs_info *mvi, int phy_id)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
 | 
			
		||||
	tmp = mr32(PCS);
 | 
			
		||||
	tmp = mr32(MVS_PCS);
 | 
			
		||||
	if (mvi->chip->n_phy <= 4)
 | 
			
		||||
		tmp |= 1 << (PhyId + PCS_EN_PORT_XMT_SHIFT);
 | 
			
		||||
		tmp |= 1 << (phy_id + PCS_EN_PORT_XMT_SHIFT);
 | 
			
		||||
	else
 | 
			
		||||
		tmp |= 1 << (PhyId + PCS_EN_PORT_XMT_SHIFT2);
 | 
			
		||||
	mw32(PCS, tmp);
 | 
			
		||||
		tmp |= 1 << (phy_id + PCS_EN_PORT_XMT_SHIFT2);
 | 
			
		||||
	mw32(MVS_PCS, tmp);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void __devinit mvs_phy_hacks(struct mvs_info *mvi)
 | 
			
		||||
static void __devinit mvs_64xx_phy_hacks(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
 | 
			
		||||
	mvs_phy_hacks(mvi);
 | 
			
		||||
 | 
			
		||||
	if (!(mvi->flags & MVF_FLAG_SOC)) {
 | 
			
		||||
		/* TEST - for phy decoding error, adjust voltage levels */
 | 
			
		||||
		mw32(MVS_P0_VSR_ADDR + 0, 0x8);
 | 
			
		||||
		mw32(MVS_P0_VSR_DATA + 0, 0x2F0);
 | 
			
		||||
 | 
			
		||||
		mw32(MVS_P0_VSR_ADDR + 8, 0x8);
 | 
			
		||||
		mw32(MVS_P0_VSR_DATA + 8, 0x2F0);
 | 
			
		||||
 | 
			
		||||
		mw32(MVS_P0_VSR_ADDR + 16, 0x8);
 | 
			
		||||
		mw32(MVS_P0_VSR_DATA + 16, 0x2F0);
 | 
			
		||||
 | 
			
		||||
		mw32(MVS_P0_VSR_ADDR + 24, 0x8);
 | 
			
		||||
		mw32(MVS_P0_VSR_DATA + 24, 0x2F0);
 | 
			
		||||
	} else {
 | 
			
		||||
		int i;
 | 
			
		||||
		/* disable auto port detection */
 | 
			
		||||
		mw32(MVS_GBL_PORT_TYPE, 0);
 | 
			
		||||
		for (i = 0; i < mvi->chip->n_phy; i++) {
 | 
			
		||||
			mvs_write_port_vsr_addr(mvi, i, VSR_PHY_MODE7);
 | 
			
		||||
			mvs_write_port_vsr_data(mvi, i, 0x90000000);
 | 
			
		||||
			mvs_write_port_vsr_addr(mvi, i, VSR_PHY_MODE9);
 | 
			
		||||
			mvs_write_port_vsr_data(mvi, i, 0x50f2);
 | 
			
		||||
			mvs_write_port_vsr_addr(mvi, i, VSR_PHY_MODE11);
 | 
			
		||||
			mvs_write_port_vsr_data(mvi, i, 0x0e);
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_64xx_stp_reset(struct mvs_info *mvi, u32 phy_id)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	u32 reg, tmp;
 | 
			
		||||
 | 
			
		||||
	if (!(mvi->flags & MVF_FLAG_SOC)) {
 | 
			
		||||
		if (phy_id < 4)
 | 
			
		||||
			pci_read_config_dword(mvi->pdev, PCR_PHY_CTL, ®);
 | 
			
		||||
		else
 | 
			
		||||
			pci_read_config_dword(mvi->pdev, PCR_PHY_CTL2, ®);
 | 
			
		||||
 | 
			
		||||
	} else
 | 
			
		||||
		reg = mr32(MVS_PHY_CTL);
 | 
			
		||||
 | 
			
		||||
	tmp = reg;
 | 
			
		||||
	if (phy_id < 4)
 | 
			
		||||
		tmp |= (1U << phy_id) << PCTL_LINK_OFFS;
 | 
			
		||||
	else
 | 
			
		||||
		tmp |= (1U << (phy_id - 4)) << PCTL_LINK_OFFS;
 | 
			
		||||
 | 
			
		||||
	if (!(mvi->flags & MVF_FLAG_SOC)) {
 | 
			
		||||
		if (phy_id < 4) {
 | 
			
		||||
			pci_write_config_dword(mvi->pdev, PCR_PHY_CTL, tmp);
 | 
			
		||||
			mdelay(10);
 | 
			
		||||
			pci_write_config_dword(mvi->pdev, PCR_PHY_CTL, reg);
 | 
			
		||||
		} else {
 | 
			
		||||
			pci_write_config_dword(mvi->pdev, PCR_PHY_CTL2, tmp);
 | 
			
		||||
			mdelay(10);
 | 
			
		||||
			pci_write_config_dword(mvi->pdev, PCR_PHY_CTL2, reg);
 | 
			
		||||
		}
 | 
			
		||||
	} else {
 | 
			
		||||
		mw32(MVS_PHY_CTL, tmp);
 | 
			
		||||
		mdelay(10);
 | 
			
		||||
		mw32(MVS_PHY_CTL, reg);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_64xx_phy_reset(struct mvs_info *mvi, u32 phy_id, int hard)
 | 
			
		||||
{
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
	tmp = mvs_read_port_irq_stat(mvi, phy_id);
 | 
			
		||||
	tmp &= ~PHYEV_RDY_CH;
 | 
			
		||||
	mvs_write_port_irq_stat(mvi, phy_id, tmp);
 | 
			
		||||
	tmp = mvs_read_phy_ctl(mvi, phy_id);
 | 
			
		||||
	if (hard)
 | 
			
		||||
		tmp |= PHY_RST_HARD;
 | 
			
		||||
	else
 | 
			
		||||
		tmp |= PHY_RST;
 | 
			
		||||
	mvs_write_phy_ctl(mvi, phy_id, tmp);
 | 
			
		||||
	if (hard) {
 | 
			
		||||
		do {
 | 
			
		||||
			tmp = mvs_read_phy_ctl(mvi, phy_id);
 | 
			
		||||
		} while (tmp & PHY_RST_HARD);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int __devinit mvs_64xx_chip_reset(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
	int i;
 | 
			
		||||
 | 
			
		||||
	/* make sure interrupts are masked immediately (paranoia) */
 | 
			
		||||
	mw32(MVS_GBL_CTL, 0);
 | 
			
		||||
	tmp = mr32(MVS_GBL_CTL);
 | 
			
		||||
 | 
			
		||||
	/* Reset Controller */
 | 
			
		||||
	if (!(tmp & HBA_RST)) {
 | 
			
		||||
		if (mvi->flags & MVF_PHY_PWR_FIX) {
 | 
			
		||||
			pci_read_config_dword(mvi->pdev, PCR_PHY_CTL, &tmp);
 | 
			
		||||
			tmp &= ~PCTL_PWR_OFF;
 | 
			
		||||
			tmp |= PCTL_PHY_DSBL;
 | 
			
		||||
			pci_write_config_dword(mvi->pdev, PCR_PHY_CTL, tmp);
 | 
			
		||||
 | 
			
		||||
			pci_read_config_dword(mvi->pdev, PCR_PHY_CTL2, &tmp);
 | 
			
		||||
			tmp &= ~PCTL_PWR_OFF;
 | 
			
		||||
			tmp |= PCTL_PHY_DSBL;
 | 
			
		||||
			pci_write_config_dword(mvi->pdev, PCR_PHY_CTL2, tmp);
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* make sure interrupts are masked immediately (paranoia) */
 | 
			
		||||
	mw32(MVS_GBL_CTL, 0);
 | 
			
		||||
	tmp = mr32(MVS_GBL_CTL);
 | 
			
		||||
 | 
			
		||||
	/* Reset Controller */
 | 
			
		||||
	if (!(tmp & HBA_RST)) {
 | 
			
		||||
		/* global reset, incl. COMRESET/H_RESET_N (self-clearing) */
 | 
			
		||||
		mw32_f(MVS_GBL_CTL, HBA_RST);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* wait for reset to finish; timeout is just a guess */
 | 
			
		||||
	i = 1000;
 | 
			
		||||
	while (i-- > 0) {
 | 
			
		||||
		msleep(10);
 | 
			
		||||
 | 
			
		||||
		if (!(mr32(MVS_GBL_CTL) & HBA_RST))
 | 
			
		||||
			break;
 | 
			
		||||
	}
 | 
			
		||||
	if (mr32(MVS_GBL_CTL) & HBA_RST) {
 | 
			
		||||
		dev_printk(KERN_ERR, mvi->dev, "HBA reset failed\n");
 | 
			
		||||
		return -EBUSY;
 | 
			
		||||
	}
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_64xx_phy_disable(struct mvs_info *mvi, u32 phy_id)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
	if (!(mvi->flags & MVF_FLAG_SOC)) {
 | 
			
		||||
		u32 offs;
 | 
			
		||||
		if (phy_id < 4)
 | 
			
		||||
			offs = PCR_PHY_CTL;
 | 
			
		||||
		else {
 | 
			
		||||
			offs = PCR_PHY_CTL2;
 | 
			
		||||
			phy_id -= 4;
 | 
			
		||||
		}
 | 
			
		||||
		pci_read_config_dword(mvi->pdev, offs, &tmp);
 | 
			
		||||
		tmp |= 1U << (PCTL_PHY_DSBL_OFFS + phy_id);
 | 
			
		||||
		pci_write_config_dword(mvi->pdev, offs, tmp);
 | 
			
		||||
	} else {
 | 
			
		||||
		tmp = mr32(MVS_PHY_CTL);
 | 
			
		||||
		tmp |= 1U << (PCTL_PHY_DSBL_OFFS + phy_id);
 | 
			
		||||
		mw32(MVS_PHY_CTL, tmp);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_64xx_phy_enable(struct mvs_info *mvi, u32 phy_id)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
	if (!(mvi->flags & MVF_FLAG_SOC)) {
 | 
			
		||||
		u32 offs;
 | 
			
		||||
		if (phy_id < 4)
 | 
			
		||||
			offs = PCR_PHY_CTL;
 | 
			
		||||
		else {
 | 
			
		||||
			offs = PCR_PHY_CTL2;
 | 
			
		||||
			phy_id -= 4;
 | 
			
		||||
		}
 | 
			
		||||
		pci_read_config_dword(mvi->pdev, offs, &tmp);
 | 
			
		||||
		tmp &= ~(1U << (PCTL_PHY_DSBL_OFFS + phy_id));
 | 
			
		||||
		pci_write_config_dword(mvi->pdev, offs, tmp);
 | 
			
		||||
	} else {
 | 
			
		||||
		tmp = mr32(MVS_PHY_CTL);
 | 
			
		||||
		tmp &= ~(1U << (PCTL_PHY_DSBL_OFFS + phy_id));
 | 
			
		||||
		mw32(MVS_PHY_CTL, tmp);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int __devinit mvs_64xx_init(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	int i;
 | 
			
		||||
	u32 tmp, cctl;
 | 
			
		||||
 | 
			
		||||
	if (mvi->pdev && mvi->pdev->revision == 0)
 | 
			
		||||
		mvi->flags |= MVF_PHY_PWR_FIX;
 | 
			
		||||
	if (!(mvi->flags & MVF_FLAG_SOC)) {
 | 
			
		||||
		mvs_show_pcie_usage(mvi);
 | 
			
		||||
		tmp = mvs_64xx_chip_reset(mvi);
 | 
			
		||||
		if (tmp)
 | 
			
		||||
			return tmp;
 | 
			
		||||
	} else {
 | 
			
		||||
		tmp = mr32(MVS_PHY_CTL);
 | 
			
		||||
		tmp &= ~PCTL_PWR_OFF;
 | 
			
		||||
		tmp |= PCTL_PHY_DSBL;
 | 
			
		||||
		mw32(MVS_PHY_CTL, tmp);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* Init Chip */
 | 
			
		||||
	/* make sure RST is set; HBA_RST /should/ have done that for us */
 | 
			
		||||
	cctl = mr32(MVS_CTL) & 0xFFFF;
 | 
			
		||||
	if (cctl & CCTL_RST)
 | 
			
		||||
		cctl &= ~CCTL_RST;
 | 
			
		||||
	else
 | 
			
		||||
		mw32_f(MVS_CTL, cctl | CCTL_RST);
 | 
			
		||||
 | 
			
		||||
	if (!(mvi->flags & MVF_FLAG_SOC)) {
 | 
			
		||||
		/* write to device control _AND_ device status register */
 | 
			
		||||
		pci_read_config_dword(mvi->pdev, PCR_DEV_CTRL, &tmp);
 | 
			
		||||
		tmp &= ~PRD_REQ_MASK;
 | 
			
		||||
		tmp |= PRD_REQ_SIZE;
 | 
			
		||||
		pci_write_config_dword(mvi->pdev, PCR_DEV_CTRL, tmp);
 | 
			
		||||
 | 
			
		||||
		pci_read_config_dword(mvi->pdev, PCR_PHY_CTL, &tmp);
 | 
			
		||||
		tmp &= ~PCTL_PWR_OFF;
 | 
			
		||||
		tmp &= ~PCTL_PHY_DSBL;
 | 
			
		||||
		pci_write_config_dword(mvi->pdev, PCR_PHY_CTL, tmp);
 | 
			
		||||
 | 
			
		||||
		pci_read_config_dword(mvi->pdev, PCR_PHY_CTL2, &tmp);
 | 
			
		||||
		tmp &= PCTL_PWR_OFF;
 | 
			
		||||
		tmp &= ~PCTL_PHY_DSBL;
 | 
			
		||||
		pci_write_config_dword(mvi->pdev, PCR_PHY_CTL2, tmp);
 | 
			
		||||
	} else {
 | 
			
		||||
		tmp = mr32(MVS_PHY_CTL);
 | 
			
		||||
		tmp &= ~PCTL_PWR_OFF;
 | 
			
		||||
		tmp |= PCTL_COM_ON;
 | 
			
		||||
		tmp &= ~PCTL_PHY_DSBL;
 | 
			
		||||
		tmp |= PCTL_LINK_RST;
 | 
			
		||||
		mw32(MVS_PHY_CTL, tmp);
 | 
			
		||||
		msleep(100);
 | 
			
		||||
		tmp &= ~PCTL_LINK_RST;
 | 
			
		||||
		mw32(MVS_PHY_CTL, tmp);
 | 
			
		||||
		msleep(100);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* reset control */
 | 
			
		||||
	mw32(MVS_PCS, 0);		/* MVS_PCS */
 | 
			
		||||
	/* init phys */
 | 
			
		||||
	mvs_64xx_phy_hacks(mvi);
 | 
			
		||||
 | 
			
		||||
	/* enable auto port detection */
 | 
			
		||||
	mw32(MVS_GBL_PORT_TYPE, MODE_AUTO_DET_EN);
 | 
			
		||||
 | 
			
		||||
	mw32(MVS_CMD_LIST_LO, mvi->slot_dma);
 | 
			
		||||
	mw32(MVS_CMD_LIST_HI, (mvi->slot_dma >> 16) >> 16);
 | 
			
		||||
 | 
			
		||||
	mw32(MVS_RX_FIS_LO, mvi->rx_fis_dma);
 | 
			
		||||
	mw32(MVS_RX_FIS_HI, (mvi->rx_fis_dma >> 16) >> 16);
 | 
			
		||||
 | 
			
		||||
	mw32(MVS_TX_CFG, MVS_CHIP_SLOT_SZ);
 | 
			
		||||
	mw32(MVS_TX_LO, mvi->tx_dma);
 | 
			
		||||
	mw32(MVS_TX_HI, (mvi->tx_dma >> 16) >> 16);
 | 
			
		||||
 | 
			
		||||
	mw32(MVS_RX_CFG, MVS_RX_RING_SZ);
 | 
			
		||||
	mw32(MVS_RX_LO, mvi->rx_dma);
 | 
			
		||||
	mw32(MVS_RX_HI, (mvi->rx_dma >> 16) >> 16);
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < mvi->chip->n_phy; i++) {
 | 
			
		||||
		/* set phy local SAS address */
 | 
			
		||||
		/* should set little endian SAS address to 64xx chip */
 | 
			
		||||
		mvs_set_sas_addr(mvi, i, PHYR_ADDR_LO, PHYR_ADDR_HI,
 | 
			
		||||
				cpu_to_be64(mvi->phy[i].dev_sas_addr));
 | 
			
		||||
 | 
			
		||||
		mvs_64xx_enable_xmt(mvi, i);
 | 
			
		||||
 | 
			
		||||
		mvs_64xx_phy_reset(mvi, i, 1);
 | 
			
		||||
		msleep(500);
 | 
			
		||||
		mvs_64xx_detect_porttype(mvi, i);
 | 
			
		||||
	}
 | 
			
		||||
	if (mvi->flags & MVF_FLAG_SOC) {
 | 
			
		||||
		/* set select registers */
 | 
			
		||||
		writel(0x0E008000, regs + 0x000);
 | 
			
		||||
		writel(0x59000008, regs + 0x004);
 | 
			
		||||
		writel(0x20, regs + 0x008);
 | 
			
		||||
		writel(0x20, regs + 0x00c);
 | 
			
		||||
		writel(0x20, regs + 0x010);
 | 
			
		||||
		writel(0x20, regs + 0x014);
 | 
			
		||||
		writel(0x20, regs + 0x018);
 | 
			
		||||
		writel(0x20, regs + 0x01c);
 | 
			
		||||
	}
 | 
			
		||||
	for (i = 0; i < mvi->chip->n_phy; i++) {
 | 
			
		||||
		/* clear phy int status */
 | 
			
		||||
		tmp = mvs_read_port_irq_stat(mvi, i);
 | 
			
		||||
		tmp &= ~PHYEV_SIG_FIS;
 | 
			
		||||
		mvs_write_port_irq_stat(mvi, i, tmp);
 | 
			
		||||
 | 
			
		||||
		/* set phy int mask */
 | 
			
		||||
		tmp = PHYEV_RDY_CH | PHYEV_BROAD_CH | PHYEV_UNASSOC_FIS |
 | 
			
		||||
			PHYEV_ID_DONE | PHYEV_DCDR_ERR | PHYEV_CRC_ERR |
 | 
			
		||||
			PHYEV_DEC_ERR;
 | 
			
		||||
		mvs_write_port_irq_mask(mvi, i, tmp);
 | 
			
		||||
 | 
			
		||||
		msleep(100);
 | 
			
		||||
		mvs_update_phyinfo(mvi, i, 1);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* FIXME: update wide port bitmaps */
 | 
			
		||||
 | 
			
		||||
	/* little endian for open address and command table, etc. */
 | 
			
		||||
	/*
 | 
			
		||||
	 * it seems that ( from the spec ) turning on big-endian won't
 | 
			
		||||
	 * do us any good on big-endian machines, need further confirmation
 | 
			
		||||
	 */
 | 
			
		||||
	cctl = mr32(MVS_CTL);
 | 
			
		||||
	cctl |= CCTL_ENDIAN_CMD;
 | 
			
		||||
	cctl |= CCTL_ENDIAN_DATA;
 | 
			
		||||
	cctl &= ~CCTL_ENDIAN_OPEN;
 | 
			
		||||
	cctl |= CCTL_ENDIAN_RSP;
 | 
			
		||||
	mw32_f(MVS_CTL, cctl);
 | 
			
		||||
 | 
			
		||||
	/* reset CMD queue */
 | 
			
		||||
	tmp = mr32(MVS_PCS);
 | 
			
		||||
	tmp |= PCS_CMD_RST;
 | 
			
		||||
	mw32(MVS_PCS, tmp);
 | 
			
		||||
	/* interrupt coalescing may cause missing HW interrput in some case,
 | 
			
		||||
	 * and the max count is 0x1ff, while our max slot is 0x200,
 | 
			
		||||
	 * it will make count 0.
 | 
			
		||||
	 */
 | 
			
		||||
	tmp = 0;
 | 
			
		||||
	mw32(MVS_INT_COAL, tmp);
 | 
			
		||||
 | 
			
		||||
	tmp = 0x100;
 | 
			
		||||
	mw32(MVS_INT_COAL_TMOUT, tmp);
 | 
			
		||||
 | 
			
		||||
	/* ladies and gentlemen, start your engines */
 | 
			
		||||
	mw32(MVS_TX_CFG, 0);
 | 
			
		||||
	mw32(MVS_TX_CFG, MVS_CHIP_SLOT_SZ | TX_EN);
 | 
			
		||||
	mw32(MVS_RX_CFG, MVS_RX_RING_SZ | RX_EN);
 | 
			
		||||
	/* enable CMD/CMPL_Q/RESP mode */
 | 
			
		||||
	mw32(MVS_PCS, PCS_SATA_RETRY | PCS_FIS_RX_EN |
 | 
			
		||||
		PCS_CMD_EN | PCS_CMD_STOP_ERR);
 | 
			
		||||
 | 
			
		||||
	/* enable completion queue interrupt */
 | 
			
		||||
	tmp = (CINT_PORT_MASK | CINT_DONE | CINT_MEM | CINT_SRS | CINT_CI_STOP |
 | 
			
		||||
		CINT_DMA_PCIE);
 | 
			
		||||
 | 
			
		||||
	mw32(MVS_INT_MASK, tmp);
 | 
			
		||||
 | 
			
		||||
	/* Enable SRS interrupt */
 | 
			
		||||
	mw32(MVS_INT_MASK_SRS_0, 0xFFFF);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int mvs_64xx_ioremap(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	if (!mvs_ioremap(mvi, 4, 2))
 | 
			
		||||
		return 0;
 | 
			
		||||
	return -1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_64xx_iounmap(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	mvs_iounmap(mvi->regs);
 | 
			
		||||
	mvs_iounmap(mvi->regs_ex);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_64xx_interrupt_enable(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
 | 
			
		||||
	/* workaround for SATA R-ERR, to ignore phy glitch */
 | 
			
		||||
	tmp = mvs_cr32(regs, CMD_PHY_TIMER);
 | 
			
		||||
	tmp &= ~(1 << 9);
 | 
			
		||||
	tmp |= (1 << 10);
 | 
			
		||||
	mvs_cw32(regs, CMD_PHY_TIMER, tmp);
 | 
			
		||||
 | 
			
		||||
	/* enable retry 127 times */
 | 
			
		||||
	mvs_cw32(regs, CMD_SAS_CTL1, 0x7f7f);
 | 
			
		||||
 | 
			
		||||
	/* extend open frame timeout to max */
 | 
			
		||||
	tmp = mvs_cr32(regs, CMD_SAS_CTL0);
 | 
			
		||||
	tmp &= ~0xffff;
 | 
			
		||||
	tmp |= 0x3fff;
 | 
			
		||||
	mvs_cw32(regs, CMD_SAS_CTL0, tmp);
 | 
			
		||||
 | 
			
		||||
	/* workaround for WDTIMEOUT , set to 550 ms */
 | 
			
		||||
	mvs_cw32(regs, CMD_WD_TIMER, 0x86470);
 | 
			
		||||
 | 
			
		||||
	/* not to halt for different port op during wideport link change */
 | 
			
		||||
	mvs_cw32(regs, CMD_APP_ERR_CONFIG, 0xffefbf7d);
 | 
			
		||||
 | 
			
		||||
	/* workaround for Seagate disk not-found OOB sequence, recv
 | 
			
		||||
	 * COMINIT before sending out COMWAKE */
 | 
			
		||||
	tmp = mvs_cr32(regs, CMD_PHY_MODE_21);
 | 
			
		||||
	tmp &= 0x0000ffff;
 | 
			
		||||
	tmp |= 0x00fa0000;
 | 
			
		||||
	mvs_cw32(regs, CMD_PHY_MODE_21, tmp);
 | 
			
		||||
 | 
			
		||||
	tmp = mvs_cr32(regs, CMD_PHY_TIMER);
 | 
			
		||||
	tmp &= 0x1fffffff;
 | 
			
		||||
	tmp |= (2U << 29);	/* 8 ms retry */
 | 
			
		||||
	mvs_cw32(regs, CMD_PHY_TIMER, tmp);
 | 
			
		||||
 | 
			
		||||
	/* TEST - for phy decoding error, adjust voltage levels */
 | 
			
		||||
	mw32(P0_VSR_ADDR + 0, 0x8);
 | 
			
		||||
	mw32(P0_VSR_DATA + 0, 0x2F0);
 | 
			
		||||
 | 
			
		||||
	mw32(P0_VSR_ADDR + 8, 0x8);
 | 
			
		||||
	mw32(P0_VSR_DATA + 8, 0x2F0);
 | 
			
		||||
 | 
			
		||||
	mw32(P0_VSR_ADDR + 16, 0x8);
 | 
			
		||||
	mw32(P0_VSR_DATA + 16, 0x2F0);
 | 
			
		||||
 | 
			
		||||
	mw32(P0_VSR_ADDR + 24, 0x8);
 | 
			
		||||
	mw32(P0_VSR_DATA + 24, 0x2F0);
 | 
			
		||||
 | 
			
		||||
	tmp = mr32(MVS_GBL_CTL);
 | 
			
		||||
	mw32(MVS_GBL_CTL, tmp | INT_EN);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mvs_hba_interrupt_enable(struct mvs_info *mvi)
 | 
			
		||||
static void mvs_64xx_interrupt_disable(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
 | 
			
		||||
	tmp = mr32(GBL_CTL);
 | 
			
		||||
 | 
			
		||||
	mw32(GBL_CTL, tmp | INT_EN);
 | 
			
		||||
	tmp = mr32(MVS_GBL_CTL);
 | 
			
		||||
	mw32(MVS_GBL_CTL, tmp & ~INT_EN);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mvs_hba_interrupt_disable(struct mvs_info *mvi)
 | 
			
		||||
static u32 mvs_64xx_isr_status(struct mvs_info *mvi, int irq)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	u32 stat;
 | 
			
		||||
 | 
			
		||||
	if (!(mvi->flags & MVF_FLAG_SOC)) {
 | 
			
		||||
		stat = mr32(MVS_GBL_INT_STAT);
 | 
			
		||||
 | 
			
		||||
		if (stat == 0 || stat == 0xffffffff)
 | 
			
		||||
			return 0;
 | 
			
		||||
	} else
 | 
			
		||||
		stat = 1;
 | 
			
		||||
	return stat;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static irqreturn_t mvs_64xx_isr(struct mvs_info *mvi, int irq, u32 stat)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
 | 
			
		||||
	/* clear CMD_CMPLT ASAP */
 | 
			
		||||
	mw32_f(MVS_INT_STAT, CINT_DONE);
 | 
			
		||||
#ifndef MVS_USE_TASKLET
 | 
			
		||||
	spin_lock(&mvi->lock);
 | 
			
		||||
#endif
 | 
			
		||||
	mvs_int_full(mvi);
 | 
			
		||||
#ifndef MVS_USE_TASKLET
 | 
			
		||||
	spin_unlock(&mvi->lock);
 | 
			
		||||
#endif
 | 
			
		||||
	return IRQ_HANDLED;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_64xx_command_active(struct mvs_info *mvi, u32 slot_idx)
 | 
			
		||||
{
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
	mvs_cw32(mvi, 0x40 + (slot_idx >> 3), 1 << (slot_idx % 32));
 | 
			
		||||
	mvs_cw32(mvi, 0x00 + (slot_idx >> 3), 1 << (slot_idx % 32));
 | 
			
		||||
	do {
 | 
			
		||||
		tmp = mvs_cr32(mvi, 0x00 + (slot_idx >> 3));
 | 
			
		||||
	} while (tmp & 1 << (slot_idx % 32));
 | 
			
		||||
	do {
 | 
			
		||||
		tmp = mvs_cr32(mvi, 0x40 + (slot_idx >> 3));
 | 
			
		||||
	} while (tmp & 1 << (slot_idx % 32));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_64xx_issue_stop(struct mvs_info *mvi, enum mvs_port_type type,
 | 
			
		||||
				u32 tfs)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
 | 
			
		||||
	tmp = mr32(GBL_CTL);
 | 
			
		||||
 | 
			
		||||
	mw32(GBL_CTL, tmp & ~INT_EN);
 | 
			
		||||
	if (type == PORT_TYPE_SATA) {
 | 
			
		||||
		tmp = mr32(MVS_INT_STAT_SRS_0) | (1U << tfs);
 | 
			
		||||
		mw32(MVS_INT_STAT_SRS_0, tmp);
 | 
			
		||||
	}
 | 
			
		||||
	mw32(MVS_INT_STAT, CINT_CI_STOP);
 | 
			
		||||
	tmp = mr32(MVS_PCS) | 0xFF00;
 | 
			
		||||
	mw32(MVS_PCS, tmp);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mvs_free_reg_set(struct mvs_info *mvi, struct mvs_port *port)
 | 
			
		||||
static void mvs_64xx_free_reg_set(struct mvs_info *mvi, u8 *tfs)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	u32 tmp, offs;
 | 
			
		||||
	u8 *tfs = &port->taskfileset;
 | 
			
		||||
 | 
			
		||||
	if (*tfs == MVS_ID_NOT_MAPPED)
 | 
			
		||||
		return;
 | 
			
		||||
 | 
			
		||||
	offs = 1U << ((*tfs & 0x0f) + PCS_EN_SATA_REG_SHIFT);
 | 
			
		||||
	if (*tfs < 16) {
 | 
			
		||||
		tmp = mr32(PCS);
 | 
			
		||||
		mw32(PCS, tmp & ~offs);
 | 
			
		||||
		tmp = mr32(MVS_PCS);
 | 
			
		||||
		mw32(MVS_PCS, tmp & ~offs);
 | 
			
		||||
	} else {
 | 
			
		||||
		tmp = mr32(CTL);
 | 
			
		||||
		mw32(CTL, tmp & ~offs);
 | 
			
		||||
		tmp = mr32(MVS_CTL);
 | 
			
		||||
		mw32(MVS_CTL, tmp & ~offs);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	tmp = mr32(INT_STAT_SRS) & (1U << *tfs);
 | 
			
		||||
	tmp = mr32(MVS_INT_STAT_SRS_0) & (1U << *tfs);
 | 
			
		||||
	if (tmp)
 | 
			
		||||
		mw32(INT_STAT_SRS, tmp);
 | 
			
		||||
		mw32(MVS_INT_STAT_SRS_0, tmp);
 | 
			
		||||
 | 
			
		||||
	*tfs = MVS_ID_NOT_MAPPED;
 | 
			
		||||
	return;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
u8 mvs_assign_reg_set(struct mvs_info *mvi, struct mvs_port *port)
 | 
			
		||||
static u8 mvs_64xx_assign_reg_set(struct mvs_info *mvi, u8 *tfs)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
	u32 tmp, offs;
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
 | 
			
		||||
	if (port->taskfileset != MVS_ID_NOT_MAPPED)
 | 
			
		||||
	if (*tfs != MVS_ID_NOT_MAPPED)
 | 
			
		||||
		return 0;
 | 
			
		||||
 | 
			
		||||
	tmp = mr32(PCS);
 | 
			
		||||
	tmp = mr32(MVS_PCS);
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < mvi->chip->srs_sz; i++) {
 | 
			
		||||
		if (i == 16)
 | 
			
		||||
			tmp = mr32(CTL);
 | 
			
		||||
			tmp = mr32(MVS_CTL);
 | 
			
		||||
		offs = 1U << ((i & 0x0f) + PCS_EN_SATA_REG_SHIFT);
 | 
			
		||||
		if (!(tmp & offs)) {
 | 
			
		||||
			port->taskfileset = i;
 | 
			
		||||
			*tfs = i;
 | 
			
		||||
 | 
			
		||||
			if (i < 16)
 | 
			
		||||
				mw32(PCS, tmp | offs);
 | 
			
		||||
				mw32(MVS_PCS, tmp | offs);
 | 
			
		||||
			else
 | 
			
		||||
				mw32(CTL, tmp | offs);
 | 
			
		||||
			tmp = mr32(INT_STAT_SRS) & (1U << i);
 | 
			
		||||
				mw32(MVS_CTL, tmp | offs);
 | 
			
		||||
			tmp = mr32(MVS_INT_STAT_SRS_0) & (1U << i);
 | 
			
		||||
			if (tmp)
 | 
			
		||||
				mw32(INT_STAT_SRS, tmp);
 | 
			
		||||
				mw32(MVS_INT_STAT_SRS_0, tmp);
 | 
			
		||||
			return 0;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	return MVS_ID_NOT_MAPPED;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mvs_64xx_make_prd(struct scatterlist *scatter, int nr, void *prd)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
	struct scatterlist *sg;
 | 
			
		||||
	struct mvs_prd *buf_prd = prd;
 | 
			
		||||
	for_each_sg(scatter, sg, nr, i) {
 | 
			
		||||
		buf_prd->addr = cpu_to_le64(sg_dma_address(sg));
 | 
			
		||||
		buf_prd->len = cpu_to_le32(sg_dma_len(sg));
 | 
			
		||||
		buf_prd++;
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int mvs_64xx_oob_done(struct mvs_info *mvi, int i)
 | 
			
		||||
{
 | 
			
		||||
	u32 phy_st;
 | 
			
		||||
	mvs_write_port_cfg_addr(mvi, i,
 | 
			
		||||
			PHYR_PHY_STAT);
 | 
			
		||||
	phy_st = mvs_read_port_cfg_data(mvi, i);
 | 
			
		||||
	if (phy_st & PHY_OOB_DTCTD)
 | 
			
		||||
		return 1;
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_64xx_fix_phy_info(struct mvs_info *mvi, int i,
 | 
			
		||||
				struct sas_identify_frame *id)
 | 
			
		||||
 | 
			
		||||
{
 | 
			
		||||
	struct mvs_phy *phy = &mvi->phy[i];
 | 
			
		||||
	struct asd_sas_phy *sas_phy = &phy->sas_phy;
 | 
			
		||||
 | 
			
		||||
	sas_phy->linkrate =
 | 
			
		||||
		(phy->phy_status & PHY_NEG_SPP_PHYS_LINK_RATE_MASK) >>
 | 
			
		||||
			PHY_NEG_SPP_PHYS_LINK_RATE_MASK_OFFSET;
 | 
			
		||||
 | 
			
		||||
	phy->minimum_linkrate =
 | 
			
		||||
		(phy->phy_status &
 | 
			
		||||
			PHY_MIN_SPP_PHYS_LINK_RATE_MASK) >> 8;
 | 
			
		||||
	phy->maximum_linkrate =
 | 
			
		||||
		(phy->phy_status &
 | 
			
		||||
			PHY_MAX_SPP_PHYS_LINK_RATE_MASK) >> 12;
 | 
			
		||||
 | 
			
		||||
	mvs_write_port_cfg_addr(mvi, i, PHYR_IDENTIFY);
 | 
			
		||||
	phy->dev_info = mvs_read_port_cfg_data(mvi, i);
 | 
			
		||||
 | 
			
		||||
	mvs_write_port_cfg_addr(mvi, i, PHYR_ATT_DEV_INFO);
 | 
			
		||||
	phy->att_dev_info = mvs_read_port_cfg_data(mvi, i);
 | 
			
		||||
 | 
			
		||||
	mvs_write_port_cfg_addr(mvi, i, PHYR_ATT_ADDR_HI);
 | 
			
		||||
	phy->att_dev_sas_addr =
 | 
			
		||||
	     (u64) mvs_read_port_cfg_data(mvi, i) << 32;
 | 
			
		||||
	mvs_write_port_cfg_addr(mvi, i, PHYR_ATT_ADDR_LO);
 | 
			
		||||
	phy->att_dev_sas_addr |= mvs_read_port_cfg_data(mvi, i);
 | 
			
		||||
	phy->att_dev_sas_addr = SAS_ADDR(&phy->att_dev_sas_addr);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_64xx_phy_work_around(struct mvs_info *mvi, int i)
 | 
			
		||||
{
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
	struct mvs_phy *phy = &mvi->phy[i];
 | 
			
		||||
	/* workaround for HW phy decoding error on 1.5g disk drive */
 | 
			
		||||
	mvs_write_port_vsr_addr(mvi, i, VSR_PHY_MODE6);
 | 
			
		||||
	tmp = mvs_read_port_vsr_data(mvi, i);
 | 
			
		||||
	if (((phy->phy_status & PHY_NEG_SPP_PHYS_LINK_RATE_MASK) >>
 | 
			
		||||
	     PHY_NEG_SPP_PHYS_LINK_RATE_MASK_OFFSET) ==
 | 
			
		||||
		SAS_LINK_RATE_1_5_GBPS)
 | 
			
		||||
		tmp &= ~PHY_MODE6_LATECLK;
 | 
			
		||||
	else
 | 
			
		||||
		tmp |= PHY_MODE6_LATECLK;
 | 
			
		||||
	mvs_write_port_vsr_data(mvi, i, tmp);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mvs_64xx_phy_set_link_rate(struct mvs_info *mvi, u32 phy_id,
 | 
			
		||||
			struct sas_phy_linkrates *rates)
 | 
			
		||||
{
 | 
			
		||||
	u32 lrmin = 0, lrmax = 0;
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
 | 
			
		||||
	tmp = mvs_read_phy_ctl(mvi, phy_id);
 | 
			
		||||
	lrmin = (rates->minimum_linkrate << 8);
 | 
			
		||||
	lrmax = (rates->maximum_linkrate << 12);
 | 
			
		||||
 | 
			
		||||
	if (lrmin) {
 | 
			
		||||
		tmp &= ~(0xf << 8);
 | 
			
		||||
		tmp |= lrmin;
 | 
			
		||||
	}
 | 
			
		||||
	if (lrmax) {
 | 
			
		||||
		tmp &= ~(0xf << 12);
 | 
			
		||||
		tmp |= lrmax;
 | 
			
		||||
	}
 | 
			
		||||
	mvs_write_phy_ctl(mvi, phy_id, tmp);
 | 
			
		||||
	mvs_64xx_phy_reset(mvi, phy_id, 1);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_64xx_clear_active_cmds(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	tmp = mr32(MVS_PCS);
 | 
			
		||||
	mw32(MVS_PCS, tmp & 0xFFFF);
 | 
			
		||||
	mw32(MVS_PCS, tmp);
 | 
			
		||||
	tmp = mr32(MVS_CTL);
 | 
			
		||||
	mw32(MVS_CTL, tmp & 0xFFFF);
 | 
			
		||||
	mw32(MVS_CTL, tmp);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
u32 mvs_64xx_spi_read_data(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs_ex;
 | 
			
		||||
	return ior32(SPI_DATA_REG_64XX);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mvs_64xx_spi_write_data(struct mvs_info *mvi, u32 data)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs_ex;
 | 
			
		||||
	 iow32(SPI_DATA_REG_64XX, data);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
int mvs_64xx_spi_buildcmd(struct mvs_info *mvi,
 | 
			
		||||
			u32      *dwCmd,
 | 
			
		||||
			u8       cmd,
 | 
			
		||||
			u8       read,
 | 
			
		||||
			u8       length,
 | 
			
		||||
			u32      addr
 | 
			
		||||
			)
 | 
			
		||||
{
 | 
			
		||||
	u32  dwTmp;
 | 
			
		||||
 | 
			
		||||
	dwTmp = ((u32)cmd << 24) | ((u32)length << 19);
 | 
			
		||||
	if (read)
 | 
			
		||||
		dwTmp |= 1U<<23;
 | 
			
		||||
 | 
			
		||||
	if (addr != MV_MAX_U32) {
 | 
			
		||||
		dwTmp |= 1U<<22;
 | 
			
		||||
		dwTmp |= (addr & 0x0003FFFF);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	*dwCmd = dwTmp;
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
int mvs_64xx_spi_issuecmd(struct mvs_info *mvi, u32 cmd)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs_ex;
 | 
			
		||||
	int     retry;
 | 
			
		||||
 | 
			
		||||
	for (retry = 0; retry < 1; retry++) {
 | 
			
		||||
		iow32(SPI_CTRL_REG_64XX, SPI_CTRL_VENDOR_ENABLE);
 | 
			
		||||
		iow32(SPI_CMD_REG_64XX, cmd);
 | 
			
		||||
		iow32(SPI_CTRL_REG_64XX,
 | 
			
		||||
			SPI_CTRL_VENDOR_ENABLE | SPI_CTRL_SPISTART);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int mvs_64xx_spi_waitdataready(struct mvs_info *mvi, u32 timeout)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs_ex;
 | 
			
		||||
	u32 i, dwTmp;
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < timeout; i++) {
 | 
			
		||||
		dwTmp = ior32(SPI_CTRL_REG_64XX);
 | 
			
		||||
		if (!(dwTmp & SPI_CTRL_SPISTART))
 | 
			
		||||
			return 0;
 | 
			
		||||
		msleep(10);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return -1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#ifndef DISABLE_HOTPLUG_DMA_FIX
 | 
			
		||||
void mvs_64xx_fix_dma(dma_addr_t buf_dma, int buf_len, int from, void *prd)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
	struct mvs_prd *buf_prd = prd;
 | 
			
		||||
	buf_prd	+= from;
 | 
			
		||||
	for (i = 0; i < MAX_SG_ENTRY - from; i++) {
 | 
			
		||||
		buf_prd->addr = cpu_to_le64(buf_dma);
 | 
			
		||||
		buf_prd->len = cpu_to_le32(buf_len);
 | 
			
		||||
		++buf_prd;
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
const struct mvs_dispatch mvs_64xx_dispatch = {
 | 
			
		||||
	"mv64xx",
 | 
			
		||||
	mvs_64xx_init,
 | 
			
		||||
	NULL,
 | 
			
		||||
	mvs_64xx_ioremap,
 | 
			
		||||
	mvs_64xx_iounmap,
 | 
			
		||||
	mvs_64xx_isr,
 | 
			
		||||
	mvs_64xx_isr_status,
 | 
			
		||||
	mvs_64xx_interrupt_enable,
 | 
			
		||||
	mvs_64xx_interrupt_disable,
 | 
			
		||||
	mvs_read_phy_ctl,
 | 
			
		||||
	mvs_write_phy_ctl,
 | 
			
		||||
	mvs_read_port_cfg_data,
 | 
			
		||||
	mvs_write_port_cfg_data,
 | 
			
		||||
	mvs_write_port_cfg_addr,
 | 
			
		||||
	mvs_read_port_vsr_data,
 | 
			
		||||
	mvs_write_port_vsr_data,
 | 
			
		||||
	mvs_write_port_vsr_addr,
 | 
			
		||||
	mvs_read_port_irq_stat,
 | 
			
		||||
	mvs_write_port_irq_stat,
 | 
			
		||||
	mvs_read_port_irq_mask,
 | 
			
		||||
	mvs_write_port_irq_mask,
 | 
			
		||||
	mvs_get_sas_addr,
 | 
			
		||||
	mvs_64xx_command_active,
 | 
			
		||||
	mvs_64xx_issue_stop,
 | 
			
		||||
	mvs_start_delivery,
 | 
			
		||||
	mvs_rx_update,
 | 
			
		||||
	mvs_int_full,
 | 
			
		||||
	mvs_64xx_assign_reg_set,
 | 
			
		||||
	mvs_64xx_free_reg_set,
 | 
			
		||||
	mvs_get_prd_size,
 | 
			
		||||
	mvs_get_prd_count,
 | 
			
		||||
	mvs_64xx_make_prd,
 | 
			
		||||
	mvs_64xx_detect_porttype,
 | 
			
		||||
	mvs_64xx_oob_done,
 | 
			
		||||
	mvs_64xx_fix_phy_info,
 | 
			
		||||
	mvs_64xx_phy_work_around,
 | 
			
		||||
	mvs_64xx_phy_set_link_rate,
 | 
			
		||||
	mvs_hw_max_link_rate,
 | 
			
		||||
	mvs_64xx_phy_disable,
 | 
			
		||||
	mvs_64xx_phy_enable,
 | 
			
		||||
	mvs_64xx_phy_reset,
 | 
			
		||||
	mvs_64xx_stp_reset,
 | 
			
		||||
	mvs_64xx_clear_active_cmds,
 | 
			
		||||
	mvs_64xx_spi_read_data,
 | 
			
		||||
	mvs_64xx_spi_write_data,
 | 
			
		||||
	mvs_64xx_spi_buildcmd,
 | 
			
		||||
	mvs_64xx_spi_issuecmd,
 | 
			
		||||
	mvs_64xx_spi_waitdataready,
 | 
			
		||||
#ifndef DISABLE_HOTPLUG_DMA_FIX
 | 
			
		||||
	mvs_64xx_fix_dma,
 | 
			
		||||
#endif
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,11 +1,43 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Marvell 88SE64xx hardware specific head file
 | 
			
		||||
 *
 | 
			
		||||
 * Copyright 2007 Red Hat, Inc.
 | 
			
		||||
 * Copyright 2008 Marvell. <kewei@marvell.com>
 | 
			
		||||
 *
 | 
			
		||||
 * This file is licensed under GPLv2.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is free software; you can redistribute it and/or
 | 
			
		||||
 * modify it under the terms of the GNU General Public License as
 | 
			
		||||
 * published by the Free Software Foundation; version 2 of the
 | 
			
		||||
 * License.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is distributed in the hope that 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.
 | 
			
		||||
 *
 | 
			
		||||
 * You should have received a copy of the GNU General Public License
 | 
			
		||||
 * along with this program; if not, write to the Free Software
 | 
			
		||||
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
 | 
			
		||||
 * USA
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
#ifndef _MVS64XX_REG_H_
 | 
			
		||||
#define _MVS64XX_REG_H_
 | 
			
		||||
 | 
			
		||||
#include <linux/types.h>
 | 
			
		||||
 | 
			
		||||
#define MAX_LINK_RATE		SAS_LINK_RATE_3_0_GBPS
 | 
			
		||||
 | 
			
		||||
/* enhanced mode registers (BAR4) */
 | 
			
		||||
enum hw_registers {
 | 
			
		||||
	MVS_GBL_CTL		= 0x04,  /* global control */
 | 
			
		||||
	MVS_GBL_INT_STAT	= 0x08,  /* global irq status */
 | 
			
		||||
	MVS_GBL_PI		= 0x0C,  /* ports implemented bitmask */
 | 
			
		||||
 | 
			
		||||
	MVS_PHY_CTL		= 0x40,  /* SOC PHY Control */
 | 
			
		||||
	MVS_PORTS_IMP		= 0x9C,  /* SOC Port Implemented */
 | 
			
		||||
 | 
			
		||||
	MVS_GBL_PORT_TYPE	= 0xa0,  /* port type */
 | 
			
		||||
 | 
			
		||||
	MVS_CTL			= 0x100, /* SAS/SATA port configuration */
 | 
			
		||||
| 
						 | 
				
			
			@ -30,17 +62,19 @@ enum hw_registers {
 | 
			
		|||
	MVS_INT_COAL_TMOUT	= 0x14C, /* Int coalescing timeout */
 | 
			
		||||
	MVS_INT_STAT		= 0x150, /* Central int status */
 | 
			
		||||
	MVS_INT_MASK		= 0x154, /* Central int enable */
 | 
			
		||||
	MVS_INT_STAT_SRS	= 0x158, /* SATA register set status */
 | 
			
		||||
	MVS_INT_MASK_SRS	= 0x15C,
 | 
			
		||||
	MVS_INT_STAT_SRS_0	= 0x158, /* SATA register set status */
 | 
			
		||||
	MVS_INT_MASK_SRS_0	= 0x15C,
 | 
			
		||||
 | 
			
		||||
					 /* ports 1-3 follow after this */
 | 
			
		||||
	MVS_P0_INT_STAT		= 0x160, /* port0 interrupt status */
 | 
			
		||||
	MVS_P0_INT_MASK		= 0x164, /* port0 interrupt mask */
 | 
			
		||||
	MVS_P4_INT_STAT		= 0x200, /* Port 4 interrupt status */
 | 
			
		||||
	MVS_P4_INT_MASK		= 0x204, /* Port 4 interrupt enable mask */
 | 
			
		||||
					 /* ports 5-7 follow after this */
 | 
			
		||||
	MVS_P4_INT_STAT		= 0x200, /* Port4 interrupt status */
 | 
			
		||||
	MVS_P4_INT_MASK		= 0x204, /* Port4 interrupt enable mask */
 | 
			
		||||
 | 
			
		||||
					 /* ports 1-3 follow after this */
 | 
			
		||||
	MVS_P0_SER_CTLSTAT	= 0x180, /* port0 serial control/status */
 | 
			
		||||
					 /* ports 5-7 follow after this */
 | 
			
		||||
	MVS_P4_SER_CTLSTAT	= 0x220, /* port4 serial control/status */
 | 
			
		||||
 | 
			
		||||
	MVS_CMD_ADDR		= 0x1B8, /* Command register port (addr) */
 | 
			
		||||
| 
						 | 
				
			
			@ -49,20 +83,23 @@ enum hw_registers {
 | 
			
		|||
					 /* ports 1-3 follow after this */
 | 
			
		||||
	MVS_P0_CFG_ADDR		= 0x1C0, /* port0 phy register address */
 | 
			
		||||
	MVS_P0_CFG_DATA		= 0x1C4, /* port0 phy register data */
 | 
			
		||||
	MVS_P4_CFG_ADDR		= 0x230, /* Port 4 config address */
 | 
			
		||||
	MVS_P4_CFG_DATA		= 0x234, /* Port 4 config data */
 | 
			
		||||
					 /* ports 5-7 follow after this */
 | 
			
		||||
	MVS_P4_CFG_ADDR		= 0x230, /* Port4 config address */
 | 
			
		||||
	MVS_P4_CFG_DATA		= 0x234, /* Port4 config data */
 | 
			
		||||
 | 
			
		||||
					 /* ports 1-3 follow after this */
 | 
			
		||||
	MVS_P0_VSR_ADDR		= 0x1E0, /* port0 VSR address */
 | 
			
		||||
	MVS_P0_VSR_DATA		= 0x1E4, /* port0 VSR data */
 | 
			
		||||
	MVS_P4_VSR_ADDR		= 0x250, /* port 4 VSR addr */
 | 
			
		||||
	MVS_P4_VSR_DATA		= 0x254, /* port 4 VSR data */
 | 
			
		||||
					 /* ports 5-7 follow after this */
 | 
			
		||||
	MVS_P4_VSR_ADDR		= 0x250, /* port4 VSR addr */
 | 
			
		||||
	MVS_P4_VSR_DATA		= 0x254, /* port4 VSR data */
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum pci_cfg_registers {
 | 
			
		||||
	PCR_PHY_CTL		= 0x40,
 | 
			
		||||
	PCR_PHY_CTL2		= 0x90,
 | 
			
		||||
	PCR_DEV_CTRL		= 0xE8,
 | 
			
		||||
	PCR_LINK_STAT		= 0xF2,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/*  SAS/SATA Vendor Specific Port Registers */
 | 
			
		||||
| 
						 | 
				
			
			@ -83,10 +120,32 @@ enum sas_sata_vsp_regs {
 | 
			
		|||
	VSR_PHY_VS1		= 0x0D, /* Vednor Specific 1 */
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum chip_register_bits {
 | 
			
		||||
	PHY_MIN_SPP_PHYS_LINK_RATE_MASK = (0xF << 8),
 | 
			
		||||
	PHY_MAX_SPP_PHYS_LINK_RATE_MASK = (0xF << 12),
 | 
			
		||||
	PHY_NEG_SPP_PHYS_LINK_RATE_MASK_OFFSET = (16),
 | 
			
		||||
	PHY_NEG_SPP_PHYS_LINK_RATE_MASK =
 | 
			
		||||
			(0xF << PHY_NEG_SPP_PHYS_LINK_RATE_MASK_OFFSET),
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
#define MAX_SG_ENTRY		64
 | 
			
		||||
 | 
			
		||||
struct mvs_prd {
 | 
			
		||||
	__le64			addr;		/* 64-bit buffer address */
 | 
			
		||||
	__le32			reserved;
 | 
			
		||||
	__le32			len;		/* 16-bit length */
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
#define SPI_CTRL_REG				0xc0
 | 
			
		||||
#define SPI_CTRL_VENDOR_ENABLE		(1U<<29)
 | 
			
		||||
#define SPI_CTRL_SPIRDY         		(1U<<22)
 | 
			
		||||
#define SPI_CTRL_SPISTART			(1U<<20)
 | 
			
		||||
 | 
			
		||||
#define SPI_CMD_REG		0xc4
 | 
			
		||||
#define SPI_DATA_REG		0xc8
 | 
			
		||||
 | 
			
		||||
#define SPI_CTRL_REG_64XX		0x10
 | 
			
		||||
#define SPI_CMD_REG_64XX		0x14
 | 
			
		||||
#define SPI_DATA_REG_64XX		0x18
 | 
			
		||||
 | 
			
		||||
#endif
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										672
									
								
								drivers/scsi/mvsas/mv_94xx.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										672
									
								
								drivers/scsi/mvsas/mv_94xx.c
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,672 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Marvell 88SE94xx hardware specific
 | 
			
		||||
 *
 | 
			
		||||
 * Copyright 2007 Red Hat, Inc.
 | 
			
		||||
 * Copyright 2008 Marvell. <kewei@marvell.com>
 | 
			
		||||
 *
 | 
			
		||||
 * This file is licensed under GPLv2.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is free software; you can redistribute it and/or
 | 
			
		||||
 * modify it under the terms of the GNU General Public License as
 | 
			
		||||
 * published by the Free Software Foundation; version 2 of the
 | 
			
		||||
 * License.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is distributed in the hope that 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.
 | 
			
		||||
 *
 | 
			
		||||
 * You should have received a copy of the GNU General Public License
 | 
			
		||||
 * along with this program; if not, write to the Free Software
 | 
			
		||||
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
 | 
			
		||||
 * USA
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
#include "mv_sas.h"
 | 
			
		||||
#include "mv_94xx.h"
 | 
			
		||||
#include "mv_chips.h"
 | 
			
		||||
 | 
			
		||||
static void mvs_94xx_detect_porttype(struct mvs_info *mvi, int i)
 | 
			
		||||
{
 | 
			
		||||
	u32 reg;
 | 
			
		||||
	struct mvs_phy *phy = &mvi->phy[i];
 | 
			
		||||
	u32 phy_status;
 | 
			
		||||
 | 
			
		||||
	mvs_write_port_vsr_addr(mvi, i, VSR_PHY_MODE3);
 | 
			
		||||
	reg = mvs_read_port_vsr_data(mvi, i);
 | 
			
		||||
	phy_status = ((reg & 0x3f0000) >> 16) & 0xff;
 | 
			
		||||
	phy->phy_type &= ~(PORT_TYPE_SAS | PORT_TYPE_SATA);
 | 
			
		||||
	switch (phy_status) {
 | 
			
		||||
	case 0x10:
 | 
			
		||||
		phy->phy_type |= PORT_TYPE_SAS;
 | 
			
		||||
		break;
 | 
			
		||||
	case 0x1d:
 | 
			
		||||
	default:
 | 
			
		||||
		phy->phy_type |= PORT_TYPE_SATA;
 | 
			
		||||
		break;
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void __devinit mvs_94xx_enable_xmt(struct mvs_info *mvi, int phy_id)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
 | 
			
		||||
	tmp = mr32(MVS_PCS);
 | 
			
		||||
	tmp |= 1 << (phy_id + PCS_EN_PORT_XMT_SHIFT2);
 | 
			
		||||
	mw32(MVS_PCS, tmp);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_94xx_phy_reset(struct mvs_info *mvi, u32 phy_id, int hard)
 | 
			
		||||
{
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
 | 
			
		||||
	tmp = mvs_read_port_irq_stat(mvi, phy_id);
 | 
			
		||||
	tmp &= ~PHYEV_RDY_CH;
 | 
			
		||||
	mvs_write_port_irq_stat(mvi, phy_id, tmp);
 | 
			
		||||
	if (hard) {
 | 
			
		||||
		tmp = mvs_read_phy_ctl(mvi, phy_id);
 | 
			
		||||
		tmp |= PHY_RST_HARD;
 | 
			
		||||
		mvs_write_phy_ctl(mvi, phy_id, tmp);
 | 
			
		||||
		do {
 | 
			
		||||
			tmp = mvs_read_phy_ctl(mvi, phy_id);
 | 
			
		||||
		} while (tmp & PHY_RST_HARD);
 | 
			
		||||
	} else {
 | 
			
		||||
		mvs_write_port_vsr_addr(mvi, phy_id, VSR_PHY_STAT);
 | 
			
		||||
		tmp = mvs_read_port_vsr_data(mvi, phy_id);
 | 
			
		||||
		tmp |= PHY_RST;
 | 
			
		||||
		mvs_write_port_vsr_data(mvi, phy_id, tmp);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_94xx_phy_disable(struct mvs_info *mvi, u32 phy_id)
 | 
			
		||||
{
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
	mvs_write_port_vsr_addr(mvi, phy_id, VSR_PHY_MODE2);
 | 
			
		||||
	tmp = mvs_read_port_vsr_data(mvi, phy_id);
 | 
			
		||||
	mvs_write_port_vsr_data(mvi, phy_id, tmp | 0x00800000);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_94xx_phy_enable(struct mvs_info *mvi, u32 phy_id)
 | 
			
		||||
{
 | 
			
		||||
	mvs_write_port_vsr_addr(mvi, phy_id, 0x1B4);
 | 
			
		||||
	mvs_write_port_vsr_data(mvi, phy_id, 0x8300ffc1);
 | 
			
		||||
	mvs_write_port_vsr_addr(mvi, phy_id, 0x104);
 | 
			
		||||
	mvs_write_port_vsr_data(mvi, phy_id, 0x00018080);
 | 
			
		||||
	mvs_write_port_vsr_addr(mvi, phy_id, VSR_PHY_MODE2);
 | 
			
		||||
	mvs_write_port_vsr_data(mvi, phy_id, 0x00207fff);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int __devinit mvs_94xx_init(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	int i;
 | 
			
		||||
	u32 tmp, cctl;
 | 
			
		||||
 | 
			
		||||
	mvs_show_pcie_usage(mvi);
 | 
			
		||||
	if (mvi->flags & MVF_FLAG_SOC) {
 | 
			
		||||
		tmp = mr32(MVS_PHY_CTL);
 | 
			
		||||
		tmp &= ~PCTL_PWR_OFF;
 | 
			
		||||
		tmp |= PCTL_PHY_DSBL;
 | 
			
		||||
		mw32(MVS_PHY_CTL, tmp);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* Init Chip */
 | 
			
		||||
	/* make sure RST is set; HBA_RST /should/ have done that for us */
 | 
			
		||||
	cctl = mr32(MVS_CTL) & 0xFFFF;
 | 
			
		||||
	if (cctl & CCTL_RST)
 | 
			
		||||
		cctl &= ~CCTL_RST;
 | 
			
		||||
	else
 | 
			
		||||
		mw32_f(MVS_CTL, cctl | CCTL_RST);
 | 
			
		||||
 | 
			
		||||
	if (mvi->flags & MVF_FLAG_SOC) {
 | 
			
		||||
		tmp = mr32(MVS_PHY_CTL);
 | 
			
		||||
		tmp &= ~PCTL_PWR_OFF;
 | 
			
		||||
		tmp |= PCTL_COM_ON;
 | 
			
		||||
		tmp &= ~PCTL_PHY_DSBL;
 | 
			
		||||
		tmp |= PCTL_LINK_RST;
 | 
			
		||||
		mw32(MVS_PHY_CTL, tmp);
 | 
			
		||||
		msleep(100);
 | 
			
		||||
		tmp &= ~PCTL_LINK_RST;
 | 
			
		||||
		mw32(MVS_PHY_CTL, tmp);
 | 
			
		||||
		msleep(100);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* reset control */
 | 
			
		||||
	mw32(MVS_PCS, 0);		/* MVS_PCS */
 | 
			
		||||
	mw32(MVS_STP_REG_SET_0, 0);
 | 
			
		||||
	mw32(MVS_STP_REG_SET_1, 0);
 | 
			
		||||
 | 
			
		||||
	/* init phys */
 | 
			
		||||
	mvs_phy_hacks(mvi);
 | 
			
		||||
 | 
			
		||||
	/* disable Multiplexing, enable phy implemented */
 | 
			
		||||
	mw32(MVS_PORTS_IMP, 0xFF);
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	mw32(MVS_PA_VSR_ADDR, 0x00000104);
 | 
			
		||||
	mw32(MVS_PA_VSR_PORT, 0x00018080);
 | 
			
		||||
	mw32(MVS_PA_VSR_ADDR, VSR_PHY_MODE8);
 | 
			
		||||
	mw32(MVS_PA_VSR_PORT, 0x0084ffff);
 | 
			
		||||
 | 
			
		||||
	/* set LED blink when IO*/
 | 
			
		||||
	mw32(MVS_PA_VSR_ADDR, 0x00000030);
 | 
			
		||||
	tmp = mr32(MVS_PA_VSR_PORT);
 | 
			
		||||
	tmp &= 0xFFFF00FF;
 | 
			
		||||
	tmp |= 0x00003300;
 | 
			
		||||
	mw32(MVS_PA_VSR_PORT, tmp);
 | 
			
		||||
 | 
			
		||||
	mw32(MVS_CMD_LIST_LO, mvi->slot_dma);
 | 
			
		||||
	mw32(MVS_CMD_LIST_HI, (mvi->slot_dma >> 16) >> 16);
 | 
			
		||||
 | 
			
		||||
	mw32(MVS_RX_FIS_LO, mvi->rx_fis_dma);
 | 
			
		||||
	mw32(MVS_RX_FIS_HI, (mvi->rx_fis_dma >> 16) >> 16);
 | 
			
		||||
 | 
			
		||||
	mw32(MVS_TX_CFG, MVS_CHIP_SLOT_SZ);
 | 
			
		||||
	mw32(MVS_TX_LO, mvi->tx_dma);
 | 
			
		||||
	mw32(MVS_TX_HI, (mvi->tx_dma >> 16) >> 16);
 | 
			
		||||
 | 
			
		||||
	mw32(MVS_RX_CFG, MVS_RX_RING_SZ);
 | 
			
		||||
	mw32(MVS_RX_LO, mvi->rx_dma);
 | 
			
		||||
	mw32(MVS_RX_HI, (mvi->rx_dma >> 16) >> 16);
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < mvi->chip->n_phy; i++) {
 | 
			
		||||
		mvs_94xx_phy_disable(mvi, i);
 | 
			
		||||
		/* set phy local SAS address */
 | 
			
		||||
		mvs_set_sas_addr(mvi, i, CONFIG_ID_FRAME3, CONFIG_ID_FRAME4,
 | 
			
		||||
						(mvi->phy[i].dev_sas_addr));
 | 
			
		||||
 | 
			
		||||
		mvs_94xx_enable_xmt(mvi, i);
 | 
			
		||||
		mvs_94xx_phy_enable(mvi, i);
 | 
			
		||||
 | 
			
		||||
		mvs_94xx_phy_reset(mvi, i, 1);
 | 
			
		||||
		msleep(500);
 | 
			
		||||
		mvs_94xx_detect_porttype(mvi, i);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (mvi->flags & MVF_FLAG_SOC) {
 | 
			
		||||
		/* set select registers */
 | 
			
		||||
		writel(0x0E008000, regs + 0x000);
 | 
			
		||||
		writel(0x59000008, regs + 0x004);
 | 
			
		||||
		writel(0x20, regs + 0x008);
 | 
			
		||||
		writel(0x20, regs + 0x00c);
 | 
			
		||||
		writel(0x20, regs + 0x010);
 | 
			
		||||
		writel(0x20, regs + 0x014);
 | 
			
		||||
		writel(0x20, regs + 0x018);
 | 
			
		||||
		writel(0x20, regs + 0x01c);
 | 
			
		||||
	}
 | 
			
		||||
	for (i = 0; i < mvi->chip->n_phy; i++) {
 | 
			
		||||
		/* clear phy int status */
 | 
			
		||||
		tmp = mvs_read_port_irq_stat(mvi, i);
 | 
			
		||||
		tmp &= ~PHYEV_SIG_FIS;
 | 
			
		||||
		mvs_write_port_irq_stat(mvi, i, tmp);
 | 
			
		||||
 | 
			
		||||
		/* set phy int mask */
 | 
			
		||||
		tmp = PHYEV_RDY_CH | PHYEV_BROAD_CH |
 | 
			
		||||
			PHYEV_ID_DONE  | PHYEV_DCDR_ERR | PHYEV_CRC_ERR ;
 | 
			
		||||
		mvs_write_port_irq_mask(mvi, i, tmp);
 | 
			
		||||
 | 
			
		||||
		msleep(100);
 | 
			
		||||
		mvs_update_phyinfo(mvi, i, 1);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* FIXME: update wide port bitmaps */
 | 
			
		||||
 | 
			
		||||
	/* little endian for open address and command table, etc. */
 | 
			
		||||
	/*
 | 
			
		||||
	 * it seems that ( from the spec ) turning on big-endian won't
 | 
			
		||||
	 * do us any good on big-endian machines, need further confirmation
 | 
			
		||||
	 */
 | 
			
		||||
	cctl = mr32(MVS_CTL);
 | 
			
		||||
	cctl |= CCTL_ENDIAN_CMD;
 | 
			
		||||
	cctl |= CCTL_ENDIAN_DATA;
 | 
			
		||||
	cctl &= ~CCTL_ENDIAN_OPEN;
 | 
			
		||||
	cctl |= CCTL_ENDIAN_RSP;
 | 
			
		||||
	mw32_f(MVS_CTL, cctl);
 | 
			
		||||
 | 
			
		||||
	/* reset CMD queue */
 | 
			
		||||
	tmp = mr32(MVS_PCS);
 | 
			
		||||
	tmp |= PCS_CMD_RST;
 | 
			
		||||
	mw32(MVS_PCS, tmp);
 | 
			
		||||
	/* interrupt coalescing may cause missing HW interrput in some case,
 | 
			
		||||
	 * and the max count is 0x1ff, while our max slot is 0x200,
 | 
			
		||||
	 * it will make count 0.
 | 
			
		||||
	 */
 | 
			
		||||
	tmp = 0;
 | 
			
		||||
	mw32(MVS_INT_COAL, tmp);
 | 
			
		||||
 | 
			
		||||
	tmp = 0x100;
 | 
			
		||||
	mw32(MVS_INT_COAL_TMOUT, tmp);
 | 
			
		||||
 | 
			
		||||
	/* ladies and gentlemen, start your engines */
 | 
			
		||||
	mw32(MVS_TX_CFG, 0);
 | 
			
		||||
	mw32(MVS_TX_CFG, MVS_CHIP_SLOT_SZ | TX_EN);
 | 
			
		||||
	mw32(MVS_RX_CFG, MVS_RX_RING_SZ | RX_EN);
 | 
			
		||||
	/* enable CMD/CMPL_Q/RESP mode */
 | 
			
		||||
	mw32(MVS_PCS, PCS_SATA_RETRY_2 | PCS_FIS_RX_EN |
 | 
			
		||||
		PCS_CMD_EN | PCS_CMD_STOP_ERR);
 | 
			
		||||
 | 
			
		||||
	/* enable completion queue interrupt */
 | 
			
		||||
	tmp = (CINT_PORT_MASK | CINT_DONE | CINT_MEM | CINT_SRS | CINT_CI_STOP |
 | 
			
		||||
		CINT_DMA_PCIE);
 | 
			
		||||
	tmp |= CINT_PHY_MASK;
 | 
			
		||||
	mw32(MVS_INT_MASK, tmp);
 | 
			
		||||
 | 
			
		||||
	/* Enable SRS interrupt */
 | 
			
		||||
	mw32(MVS_INT_MASK_SRS_0, 0xFFFF);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int mvs_94xx_ioremap(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	if (!mvs_ioremap(mvi, 2, -1)) {
 | 
			
		||||
		mvi->regs_ex = mvi->regs + 0x10200;
 | 
			
		||||
		mvi->regs += 0x20000;
 | 
			
		||||
		if (mvi->id == 1)
 | 
			
		||||
			mvi->regs += 0x4000;
 | 
			
		||||
		return 0;
 | 
			
		||||
	}
 | 
			
		||||
	return -1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_94xx_iounmap(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	if (mvi->regs) {
 | 
			
		||||
		mvi->regs -= 0x20000;
 | 
			
		||||
		if (mvi->id == 1)
 | 
			
		||||
			mvi->regs -= 0x4000;
 | 
			
		||||
		mvs_iounmap(mvi->regs);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_94xx_interrupt_enable(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs_ex;
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
 | 
			
		||||
	tmp = mr32(MVS_GBL_CTL);
 | 
			
		||||
	tmp |= (IRQ_SAS_A | IRQ_SAS_B);
 | 
			
		||||
	mw32(MVS_GBL_INT_STAT, tmp);
 | 
			
		||||
	writel(tmp, regs + 0x0C);
 | 
			
		||||
	writel(tmp, regs + 0x10);
 | 
			
		||||
	writel(tmp, regs + 0x14);
 | 
			
		||||
	writel(tmp, regs + 0x18);
 | 
			
		||||
	mw32(MVS_GBL_CTL, tmp);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_94xx_interrupt_disable(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs_ex;
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
 | 
			
		||||
	tmp = mr32(MVS_GBL_CTL);
 | 
			
		||||
 | 
			
		||||
	tmp &= ~(IRQ_SAS_A | IRQ_SAS_B);
 | 
			
		||||
	mw32(MVS_GBL_INT_STAT, tmp);
 | 
			
		||||
	writel(tmp, regs + 0x0C);
 | 
			
		||||
	writel(tmp, regs + 0x10);
 | 
			
		||||
	writel(tmp, regs + 0x14);
 | 
			
		||||
	writel(tmp, regs + 0x18);
 | 
			
		||||
	mw32(MVS_GBL_CTL, tmp);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static u32 mvs_94xx_isr_status(struct mvs_info *mvi, int irq)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs_ex;
 | 
			
		||||
	u32 stat = 0;
 | 
			
		||||
	if (!(mvi->flags & MVF_FLAG_SOC)) {
 | 
			
		||||
		stat = mr32(MVS_GBL_INT_STAT);
 | 
			
		||||
 | 
			
		||||
		if (!(stat & (IRQ_SAS_A | IRQ_SAS_B)))
 | 
			
		||||
			return 0;
 | 
			
		||||
	}
 | 
			
		||||
	return stat;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static irqreturn_t mvs_94xx_isr(struct mvs_info *mvi, int irq, u32 stat)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
 | 
			
		||||
	if (((stat & IRQ_SAS_A) && mvi->id == 0) ||
 | 
			
		||||
			((stat & IRQ_SAS_B) && mvi->id == 1)) {
 | 
			
		||||
		mw32_f(MVS_INT_STAT, CINT_DONE);
 | 
			
		||||
	#ifndef MVS_USE_TASKLET
 | 
			
		||||
		spin_lock(&mvi->lock);
 | 
			
		||||
	#endif
 | 
			
		||||
		mvs_int_full(mvi);
 | 
			
		||||
	#ifndef MVS_USE_TASKLET
 | 
			
		||||
		spin_unlock(&mvi->lock);
 | 
			
		||||
	#endif
 | 
			
		||||
	}
 | 
			
		||||
	return IRQ_HANDLED;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_94xx_command_active(struct mvs_info *mvi, u32 slot_idx)
 | 
			
		||||
{
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
	mvs_cw32(mvi, 0x300 + (slot_idx >> 3), 1 << (slot_idx % 32));
 | 
			
		||||
	do {
 | 
			
		||||
		tmp = mvs_cr32(mvi, 0x300 + (slot_idx >> 3));
 | 
			
		||||
	} while (tmp & 1 << (slot_idx % 32));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_94xx_issue_stop(struct mvs_info *mvi, enum mvs_port_type type,
 | 
			
		||||
				u32 tfs)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
 | 
			
		||||
	if (type == PORT_TYPE_SATA) {
 | 
			
		||||
		tmp = mr32(MVS_INT_STAT_SRS_0) | (1U << tfs);
 | 
			
		||||
		mw32(MVS_INT_STAT_SRS_0, tmp);
 | 
			
		||||
	}
 | 
			
		||||
	mw32(MVS_INT_STAT, CINT_CI_STOP);
 | 
			
		||||
	tmp = mr32(MVS_PCS) | 0xFF00;
 | 
			
		||||
	mw32(MVS_PCS, tmp);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_94xx_free_reg_set(struct mvs_info *mvi, u8 *tfs)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
	u8 reg_set = *tfs;
 | 
			
		||||
 | 
			
		||||
	if (*tfs == MVS_ID_NOT_MAPPED)
 | 
			
		||||
		return;
 | 
			
		||||
 | 
			
		||||
	mvi->sata_reg_set &= ~bit(reg_set);
 | 
			
		||||
	if (reg_set < 32) {
 | 
			
		||||
		w_reg_set_enable(reg_set, (u32)mvi->sata_reg_set);
 | 
			
		||||
		tmp = mr32(MVS_INT_STAT_SRS_0) & (u32)mvi->sata_reg_set;
 | 
			
		||||
		if (tmp)
 | 
			
		||||
			mw32(MVS_INT_STAT_SRS_0, tmp);
 | 
			
		||||
	} else {
 | 
			
		||||
		w_reg_set_enable(reg_set, mvi->sata_reg_set);
 | 
			
		||||
		tmp = mr32(MVS_INT_STAT_SRS_1) & mvi->sata_reg_set;
 | 
			
		||||
		if (tmp)
 | 
			
		||||
			mw32(MVS_INT_STAT_SRS_1, tmp);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	*tfs = MVS_ID_NOT_MAPPED;
 | 
			
		||||
 | 
			
		||||
	return;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static u8 mvs_94xx_assign_reg_set(struct mvs_info *mvi, u8 *tfs)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
 | 
			
		||||
	if (*tfs != MVS_ID_NOT_MAPPED)
 | 
			
		||||
		return 0;
 | 
			
		||||
 | 
			
		||||
	i = mv_ffc64(mvi->sata_reg_set);
 | 
			
		||||
	if (i > 32) {
 | 
			
		||||
		mvi->sata_reg_set |= bit(i);
 | 
			
		||||
		w_reg_set_enable(i, (u32)(mvi->sata_reg_set >> 32));
 | 
			
		||||
		*tfs = i;
 | 
			
		||||
		return 0;
 | 
			
		||||
	} else if (i >= 0) {
 | 
			
		||||
		mvi->sata_reg_set |= bit(i);
 | 
			
		||||
		w_reg_set_enable(i, (u32)mvi->sata_reg_set);
 | 
			
		||||
		*tfs = i;
 | 
			
		||||
		return 0;
 | 
			
		||||
	}
 | 
			
		||||
	return MVS_ID_NOT_MAPPED;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_94xx_make_prd(struct scatterlist *scatter, int nr, void *prd)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
	struct scatterlist *sg;
 | 
			
		||||
	struct mvs_prd *buf_prd = prd;
 | 
			
		||||
	for_each_sg(scatter, sg, nr, i) {
 | 
			
		||||
		buf_prd->addr = cpu_to_le64(sg_dma_address(sg));
 | 
			
		||||
		buf_prd->im_len.len = cpu_to_le32(sg_dma_len(sg));
 | 
			
		||||
		buf_prd++;
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int mvs_94xx_oob_done(struct mvs_info *mvi, int i)
 | 
			
		||||
{
 | 
			
		||||
	u32 phy_st;
 | 
			
		||||
	phy_st = mvs_read_phy_ctl(mvi, i);
 | 
			
		||||
	if (phy_st & PHY_READY_MASK)	/* phy ready */
 | 
			
		||||
		return 1;
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_94xx_get_dev_identify_frame(struct mvs_info *mvi, int port_id,
 | 
			
		||||
					struct sas_identify_frame *id)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
	u32 id_frame[7];
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < 7; i++) {
 | 
			
		||||
		mvs_write_port_cfg_addr(mvi, port_id,
 | 
			
		||||
					CONFIG_ID_FRAME0 + i * 4);
 | 
			
		||||
		id_frame[i] = mvs_read_port_cfg_data(mvi, port_id);
 | 
			
		||||
	}
 | 
			
		||||
	memcpy(id, id_frame, 28);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_94xx_get_att_identify_frame(struct mvs_info *mvi, int port_id,
 | 
			
		||||
					struct sas_identify_frame *id)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
	u32 id_frame[7];
 | 
			
		||||
 | 
			
		||||
	/* mvs_hexdump(28, (u8 *)id_frame, 0); */
 | 
			
		||||
	for (i = 0; i < 7; i++) {
 | 
			
		||||
		mvs_write_port_cfg_addr(mvi, port_id,
 | 
			
		||||
					CONFIG_ATT_ID_FRAME0 + i * 4);
 | 
			
		||||
		id_frame[i] = mvs_read_port_cfg_data(mvi, port_id);
 | 
			
		||||
		mv_dprintk("94xx phy %d atta frame %d %x.\n",
 | 
			
		||||
			port_id + mvi->id * mvi->chip->n_phy, i, id_frame[i]);
 | 
			
		||||
	}
 | 
			
		||||
	/* mvs_hexdump(28, (u8 *)id_frame, 0); */
 | 
			
		||||
	memcpy(id, id_frame, 28);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static u32 mvs_94xx_make_dev_info(struct sas_identify_frame *id)
 | 
			
		||||
{
 | 
			
		||||
	u32 att_dev_info = 0;
 | 
			
		||||
 | 
			
		||||
	att_dev_info |= id->dev_type;
 | 
			
		||||
	if (id->stp_iport)
 | 
			
		||||
		att_dev_info |= PORT_DEV_STP_INIT;
 | 
			
		||||
	if (id->smp_iport)
 | 
			
		||||
		att_dev_info |= PORT_DEV_SMP_INIT;
 | 
			
		||||
	if (id->ssp_iport)
 | 
			
		||||
		att_dev_info |= PORT_DEV_SSP_INIT;
 | 
			
		||||
	if (id->stp_tport)
 | 
			
		||||
		att_dev_info |= PORT_DEV_STP_TRGT;
 | 
			
		||||
	if (id->smp_tport)
 | 
			
		||||
		att_dev_info |= PORT_DEV_SMP_TRGT;
 | 
			
		||||
	if (id->ssp_tport)
 | 
			
		||||
		att_dev_info |= PORT_DEV_SSP_TRGT;
 | 
			
		||||
 | 
			
		||||
	att_dev_info |= (u32)id->phy_id<<24;
 | 
			
		||||
	return att_dev_info;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static u32 mvs_94xx_make_att_info(struct sas_identify_frame *id)
 | 
			
		||||
{
 | 
			
		||||
	return mvs_94xx_make_dev_info(id);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_94xx_fix_phy_info(struct mvs_info *mvi, int i,
 | 
			
		||||
				struct sas_identify_frame *id)
 | 
			
		||||
{
 | 
			
		||||
	struct mvs_phy *phy = &mvi->phy[i];
 | 
			
		||||
	struct asd_sas_phy *sas_phy = &phy->sas_phy;
 | 
			
		||||
	mv_dprintk("get all reg link rate is 0x%x\n", phy->phy_status);
 | 
			
		||||
	sas_phy->linkrate =
 | 
			
		||||
		(phy->phy_status & PHY_NEG_SPP_PHYS_LINK_RATE_MASK) >>
 | 
			
		||||
			PHY_NEG_SPP_PHYS_LINK_RATE_MASK_OFFSET;
 | 
			
		||||
	sas_phy->linkrate += 0x8;
 | 
			
		||||
	mv_dprintk("get link rate is %d\n", sas_phy->linkrate);
 | 
			
		||||
	phy->minimum_linkrate = SAS_LINK_RATE_1_5_GBPS;
 | 
			
		||||
	phy->maximum_linkrate = SAS_LINK_RATE_6_0_GBPS;
 | 
			
		||||
	mvs_94xx_get_dev_identify_frame(mvi, i, id);
 | 
			
		||||
	phy->dev_info = mvs_94xx_make_dev_info(id);
 | 
			
		||||
 | 
			
		||||
	if (phy->phy_type & PORT_TYPE_SAS) {
 | 
			
		||||
		mvs_94xx_get_att_identify_frame(mvi, i, id);
 | 
			
		||||
		phy->att_dev_info = mvs_94xx_make_att_info(id);
 | 
			
		||||
		phy->att_dev_sas_addr = *(u64 *)id->sas_addr;
 | 
			
		||||
	} else {
 | 
			
		||||
		phy->att_dev_info = PORT_DEV_STP_TRGT | 1;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mvs_94xx_phy_set_link_rate(struct mvs_info *mvi, u32 phy_id,
 | 
			
		||||
			struct sas_phy_linkrates *rates)
 | 
			
		||||
{
 | 
			
		||||
	/* TODO */
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_94xx_clear_active_cmds(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	tmp = mr32(MVS_STP_REG_SET_0);
 | 
			
		||||
	mw32(MVS_STP_REG_SET_0, 0);
 | 
			
		||||
	mw32(MVS_STP_REG_SET_0, tmp);
 | 
			
		||||
	tmp = mr32(MVS_STP_REG_SET_1);
 | 
			
		||||
	mw32(MVS_STP_REG_SET_1, 0);
 | 
			
		||||
	mw32(MVS_STP_REG_SET_1, tmp);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
u32 mvs_94xx_spi_read_data(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs_ex - 0x10200;
 | 
			
		||||
	return mr32(SPI_RD_DATA_REG_94XX);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mvs_94xx_spi_write_data(struct mvs_info *mvi, u32 data)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs_ex - 0x10200;
 | 
			
		||||
	 mw32(SPI_RD_DATA_REG_94XX, data);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
int mvs_94xx_spi_buildcmd(struct mvs_info *mvi,
 | 
			
		||||
				u32      *dwCmd,
 | 
			
		||||
				u8       cmd,
 | 
			
		||||
				u8       read,
 | 
			
		||||
				u8       length,
 | 
			
		||||
				u32      addr
 | 
			
		||||
				)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs_ex - 0x10200;
 | 
			
		||||
	u32  dwTmp;
 | 
			
		||||
 | 
			
		||||
	dwTmp = ((u32)cmd << 8) | ((u32)length << 4);
 | 
			
		||||
	if (read)
 | 
			
		||||
		dwTmp |= SPI_CTRL_READ_94XX;
 | 
			
		||||
 | 
			
		||||
	if (addr != MV_MAX_U32) {
 | 
			
		||||
		mw32(SPI_ADDR_REG_94XX, (addr & 0x0003FFFFL));
 | 
			
		||||
		dwTmp |= SPI_ADDR_VLD_94XX;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	*dwCmd = dwTmp;
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
int mvs_94xx_spi_issuecmd(struct mvs_info *mvi, u32 cmd)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs_ex - 0x10200;
 | 
			
		||||
	mw32(SPI_CTRL_REG_94XX, cmd | SPI_CTRL_SpiStart_94XX);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int mvs_94xx_spi_waitdataready(struct mvs_info *mvi, u32 timeout)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs_ex - 0x10200;
 | 
			
		||||
	u32   i, dwTmp;
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < timeout; i++) {
 | 
			
		||||
		dwTmp = mr32(SPI_CTRL_REG_94XX);
 | 
			
		||||
		if (!(dwTmp & SPI_CTRL_SpiStart_94XX))
 | 
			
		||||
			return 0;
 | 
			
		||||
		msleep(10);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return -1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#ifndef DISABLE_HOTPLUG_DMA_FIX
 | 
			
		||||
void mvs_94xx_fix_dma(dma_addr_t buf_dma, int buf_len, int from, void *prd)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
	struct mvs_prd *buf_prd = prd;
 | 
			
		||||
	buf_prd += from;
 | 
			
		||||
	for (i = 0; i < MAX_SG_ENTRY - from; i++) {
 | 
			
		||||
		buf_prd->addr = cpu_to_le64(buf_dma);
 | 
			
		||||
		buf_prd->im_len.len = cpu_to_le32(buf_len);
 | 
			
		||||
		++buf_prd;
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
const struct mvs_dispatch mvs_94xx_dispatch = {
 | 
			
		||||
	"mv94xx",
 | 
			
		||||
	mvs_94xx_init,
 | 
			
		||||
	NULL,
 | 
			
		||||
	mvs_94xx_ioremap,
 | 
			
		||||
	mvs_94xx_iounmap,
 | 
			
		||||
	mvs_94xx_isr,
 | 
			
		||||
	mvs_94xx_isr_status,
 | 
			
		||||
	mvs_94xx_interrupt_enable,
 | 
			
		||||
	mvs_94xx_interrupt_disable,
 | 
			
		||||
	mvs_read_phy_ctl,
 | 
			
		||||
	mvs_write_phy_ctl,
 | 
			
		||||
	mvs_read_port_cfg_data,
 | 
			
		||||
	mvs_write_port_cfg_data,
 | 
			
		||||
	mvs_write_port_cfg_addr,
 | 
			
		||||
	mvs_read_port_vsr_data,
 | 
			
		||||
	mvs_write_port_vsr_data,
 | 
			
		||||
	mvs_write_port_vsr_addr,
 | 
			
		||||
	mvs_read_port_irq_stat,
 | 
			
		||||
	mvs_write_port_irq_stat,
 | 
			
		||||
	mvs_read_port_irq_mask,
 | 
			
		||||
	mvs_write_port_irq_mask,
 | 
			
		||||
	mvs_get_sas_addr,
 | 
			
		||||
	mvs_94xx_command_active,
 | 
			
		||||
	mvs_94xx_issue_stop,
 | 
			
		||||
	mvs_start_delivery,
 | 
			
		||||
	mvs_rx_update,
 | 
			
		||||
	mvs_int_full,
 | 
			
		||||
	mvs_94xx_assign_reg_set,
 | 
			
		||||
	mvs_94xx_free_reg_set,
 | 
			
		||||
	mvs_get_prd_size,
 | 
			
		||||
	mvs_get_prd_count,
 | 
			
		||||
	mvs_94xx_make_prd,
 | 
			
		||||
	mvs_94xx_detect_porttype,
 | 
			
		||||
	mvs_94xx_oob_done,
 | 
			
		||||
	mvs_94xx_fix_phy_info,
 | 
			
		||||
	NULL,
 | 
			
		||||
	mvs_94xx_phy_set_link_rate,
 | 
			
		||||
	mvs_hw_max_link_rate,
 | 
			
		||||
	mvs_94xx_phy_disable,
 | 
			
		||||
	mvs_94xx_phy_enable,
 | 
			
		||||
	mvs_94xx_phy_reset,
 | 
			
		||||
	NULL,
 | 
			
		||||
	mvs_94xx_clear_active_cmds,
 | 
			
		||||
	mvs_94xx_spi_read_data,
 | 
			
		||||
	mvs_94xx_spi_write_data,
 | 
			
		||||
	mvs_94xx_spi_buildcmd,
 | 
			
		||||
	mvs_94xx_spi_issuecmd,
 | 
			
		||||
	mvs_94xx_spi_waitdataready,
 | 
			
		||||
#ifndef DISABLE_HOTPLUG_DMA_FIX
 | 
			
		||||
	mvs_94xx_fix_dma,
 | 
			
		||||
#endif
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										222
									
								
								drivers/scsi/mvsas/mv_94xx.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										222
									
								
								drivers/scsi/mvsas/mv_94xx.h
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,222 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Marvell 88SE94xx hardware specific head file
 | 
			
		||||
 *
 | 
			
		||||
 * Copyright 2007 Red Hat, Inc.
 | 
			
		||||
 * Copyright 2008 Marvell. <kewei@marvell.com>
 | 
			
		||||
 *
 | 
			
		||||
 * This file is licensed under GPLv2.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is free software; you can redistribute it and/or
 | 
			
		||||
 * modify it under the terms of the GNU General Public License as
 | 
			
		||||
 * published by the Free Software Foundation; version 2 of the
 | 
			
		||||
 * License.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is distributed in the hope that 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.
 | 
			
		||||
 *
 | 
			
		||||
 * You should have received a copy of the GNU General Public License
 | 
			
		||||
 * along with this program; if not, write to the Free Software
 | 
			
		||||
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
 | 
			
		||||
 * USA
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
#ifndef _MVS94XX_REG_H_
 | 
			
		||||
#define _MVS94XX_REG_H_
 | 
			
		||||
 | 
			
		||||
#include <linux/types.h>
 | 
			
		||||
 | 
			
		||||
#define MAX_LINK_RATE		SAS_LINK_RATE_6_0_GBPS
 | 
			
		||||
 | 
			
		||||
enum hw_registers {
 | 
			
		||||
	MVS_GBL_CTL		= 0x04,  /* global control */
 | 
			
		||||
	MVS_GBL_INT_STAT	= 0x00,  /* global irq status */
 | 
			
		||||
	MVS_GBL_PI		= 0x0C,  /* ports implemented bitmask */
 | 
			
		||||
 | 
			
		||||
	MVS_PHY_CTL		= 0x40,  /* SOC PHY Control */
 | 
			
		||||
	MVS_PORTS_IMP		= 0x9C,  /* SOC Port Implemented */
 | 
			
		||||
 | 
			
		||||
	MVS_GBL_PORT_TYPE	= 0xa0,  /* port type */
 | 
			
		||||
 | 
			
		||||
	MVS_CTL			= 0x100, /* SAS/SATA port configuration */
 | 
			
		||||
	MVS_PCS			= 0x104, /* SAS/SATA port control/status */
 | 
			
		||||
	MVS_CMD_LIST_LO		= 0x108, /* cmd list addr */
 | 
			
		||||
	MVS_CMD_LIST_HI		= 0x10C,
 | 
			
		||||
	MVS_RX_FIS_LO		= 0x110, /* RX FIS list addr */
 | 
			
		||||
	MVS_RX_FIS_HI		= 0x114,
 | 
			
		||||
	MVS_STP_REG_SET_0	= 0x118, /* STP/SATA Register Set Enable */
 | 
			
		||||
	MVS_STP_REG_SET_1	= 0x11C,
 | 
			
		||||
	MVS_TX_CFG		= 0x120, /* TX configuration */
 | 
			
		||||
	MVS_TX_LO		= 0x124, /* TX (delivery) ring addr */
 | 
			
		||||
	MVS_TX_HI		= 0x128,
 | 
			
		||||
 | 
			
		||||
	MVS_TX_PROD_IDX		= 0x12C, /* TX producer pointer */
 | 
			
		||||
	MVS_TX_CONS_IDX		= 0x130, /* TX consumer pointer (RO) */
 | 
			
		||||
	MVS_RX_CFG		= 0x134, /* RX configuration */
 | 
			
		||||
	MVS_RX_LO		= 0x138, /* RX (completion) ring addr */
 | 
			
		||||
	MVS_RX_HI		= 0x13C,
 | 
			
		||||
	MVS_RX_CONS_IDX		= 0x140, /* RX consumer pointer (RO) */
 | 
			
		||||
 | 
			
		||||
	MVS_INT_COAL		= 0x148, /* Int coalescing config */
 | 
			
		||||
	MVS_INT_COAL_TMOUT	= 0x14C, /* Int coalescing timeout */
 | 
			
		||||
	MVS_INT_STAT		= 0x150, /* Central int status */
 | 
			
		||||
	MVS_INT_MASK		= 0x154, /* Central int enable */
 | 
			
		||||
	MVS_INT_STAT_SRS_0	= 0x158, /* SATA register set status */
 | 
			
		||||
	MVS_INT_MASK_SRS_0	= 0x15C,
 | 
			
		||||
	MVS_INT_STAT_SRS_1	= 0x160,
 | 
			
		||||
	MVS_INT_MASK_SRS_1	= 0x164,
 | 
			
		||||
	MVS_NON_NCQ_ERR_0	= 0x168, /* SRS Non-specific NCQ Error */
 | 
			
		||||
	MVS_NON_NCQ_ERR_1	= 0x16C,
 | 
			
		||||
	MVS_CMD_ADDR		= 0x170, /* Command register port (addr) */
 | 
			
		||||
	MVS_CMD_DATA		= 0x174, /* Command register port (data) */
 | 
			
		||||
	MVS_MEM_PARITY_ERR	= 0x178, /* Memory parity error */
 | 
			
		||||
 | 
			
		||||
					 /* ports 1-3 follow after this */
 | 
			
		||||
	MVS_P0_INT_STAT		= 0x180, /* port0 interrupt status */
 | 
			
		||||
	MVS_P0_INT_MASK		= 0x184, /* port0 interrupt mask */
 | 
			
		||||
					 /* ports 5-7 follow after this */
 | 
			
		||||
	MVS_P4_INT_STAT		= 0x1A0, /* Port4 interrupt status */
 | 
			
		||||
	MVS_P4_INT_MASK		= 0x1A4, /* Port4 interrupt enable mask */
 | 
			
		||||
 | 
			
		||||
					 /* ports 1-3 follow after this */
 | 
			
		||||
	MVS_P0_SER_CTLSTAT	= 0x1D0, /* port0 serial control/status */
 | 
			
		||||
					 /* ports 5-7 follow after this */
 | 
			
		||||
	MVS_P4_SER_CTLSTAT	= 0x1E0, /* port4 serial control/status */
 | 
			
		||||
 | 
			
		||||
					 /* ports 1-3 follow after this */
 | 
			
		||||
	MVS_P0_CFG_ADDR		= 0x200, /* port0 phy register address */
 | 
			
		||||
	MVS_P0_CFG_DATA		= 0x204, /* port0 phy register data */
 | 
			
		||||
					 /* ports 5-7 follow after this */
 | 
			
		||||
	MVS_P4_CFG_ADDR		= 0x220, /* Port4 config address */
 | 
			
		||||
	MVS_P4_CFG_DATA		= 0x224, /* Port4 config data */
 | 
			
		||||
 | 
			
		||||
					 /* phys 1-3 follow after this */
 | 
			
		||||
	MVS_P0_VSR_ADDR		= 0x250, /* phy0 VSR address */
 | 
			
		||||
	MVS_P0_VSR_DATA		= 0x254, /* phy0 VSR data */
 | 
			
		||||
					 /* phys 1-3 follow after this */
 | 
			
		||||
					 /* multiplexing */
 | 
			
		||||
	MVS_P4_VSR_ADDR 	= 0x250, /* phy4 VSR address */
 | 
			
		||||
	MVS_P4_VSR_DATA 	= 0x254, /* phy4 VSR data */
 | 
			
		||||
	MVS_PA_VSR_ADDR		= 0x290, /* All port VSR addr */
 | 
			
		||||
	MVS_PA_VSR_PORT		= 0x294, /* All port VSR data */
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum pci_cfg_registers {
 | 
			
		||||
	PCR_PHY_CTL		= 0x40,
 | 
			
		||||
	PCR_PHY_CTL2		= 0x90,
 | 
			
		||||
	PCR_DEV_CTRL		= 0x78,
 | 
			
		||||
	PCR_LINK_STAT		= 0x82,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/*  SAS/SATA Vendor Specific Port Registers */
 | 
			
		||||
enum sas_sata_vsp_regs {
 | 
			
		||||
	VSR_PHY_STAT		= 0x00 * 4, /* Phy Status */
 | 
			
		||||
	VSR_PHY_MODE1		= 0x01 * 4, /* phy tx */
 | 
			
		||||
	VSR_PHY_MODE2		= 0x02 * 4, /* tx scc */
 | 
			
		||||
	VSR_PHY_MODE3		= 0x03 * 4, /* pll */
 | 
			
		||||
	VSR_PHY_MODE4		= 0x04 * 4, /* VCO */
 | 
			
		||||
	VSR_PHY_MODE5		= 0x05 * 4, /* Rx */
 | 
			
		||||
	VSR_PHY_MODE6		= 0x06 * 4, /* CDR */
 | 
			
		||||
	VSR_PHY_MODE7		= 0x07 * 4, /* Impedance */
 | 
			
		||||
	VSR_PHY_MODE8		= 0x08 * 4, /* Voltage */
 | 
			
		||||
	VSR_PHY_MODE9		= 0x09 * 4, /* Test */
 | 
			
		||||
	VSR_PHY_MODE10		= 0x0A * 4, /* Power */
 | 
			
		||||
	VSR_PHY_MODE11		= 0x0B * 4, /* Phy Mode */
 | 
			
		||||
	VSR_PHY_VS0		= 0x0C * 4, /* Vednor Specific 0 */
 | 
			
		||||
	VSR_PHY_VS1		= 0x0D * 4, /* Vednor Specific 1 */
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum chip_register_bits {
 | 
			
		||||
	PHY_MIN_SPP_PHYS_LINK_RATE_MASK = (0x7 << 8),
 | 
			
		||||
	PHY_MAX_SPP_PHYS_LINK_RATE_MASK = (0x7 << 8),
 | 
			
		||||
	PHY_NEG_SPP_PHYS_LINK_RATE_MASK_OFFSET = (12),
 | 
			
		||||
	PHY_NEG_SPP_PHYS_LINK_RATE_MASK =
 | 
			
		||||
			(0x3 << PHY_NEG_SPP_PHYS_LINK_RATE_MASK_OFFSET),
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum pci_interrupt_cause {
 | 
			
		||||
	/*  MAIN_IRQ_CAUSE (R10200) Bits*/
 | 
			
		||||
	IRQ_COM_IN_I2O_IOP0            = (1 << 0),
 | 
			
		||||
	IRQ_COM_IN_I2O_IOP1            = (1 << 1),
 | 
			
		||||
	IRQ_COM_IN_I2O_IOP2            = (1 << 2),
 | 
			
		||||
	IRQ_COM_IN_I2O_IOP3            = (1 << 3),
 | 
			
		||||
	IRQ_COM_OUT_I2O_HOS0           = (1 << 4),
 | 
			
		||||
	IRQ_COM_OUT_I2O_HOS1           = (1 << 5),
 | 
			
		||||
	IRQ_COM_OUT_I2O_HOS2           = (1 << 6),
 | 
			
		||||
	IRQ_COM_OUT_I2O_HOS3           = (1 << 7),
 | 
			
		||||
	IRQ_PCIF_TO_CPU_DRBL0          = (1 << 8),
 | 
			
		||||
	IRQ_PCIF_TO_CPU_DRBL1          = (1 << 9),
 | 
			
		||||
	IRQ_PCIF_TO_CPU_DRBL2          = (1 << 10),
 | 
			
		||||
	IRQ_PCIF_TO_CPU_DRBL3          = (1 << 11),
 | 
			
		||||
	IRQ_PCIF_DRBL0                 = (1 << 12),
 | 
			
		||||
	IRQ_PCIF_DRBL1                 = (1 << 13),
 | 
			
		||||
	IRQ_PCIF_DRBL2                 = (1 << 14),
 | 
			
		||||
	IRQ_PCIF_DRBL3                 = (1 << 15),
 | 
			
		||||
	IRQ_XOR_A                      = (1 << 16),
 | 
			
		||||
	IRQ_XOR_B                      = (1 << 17),
 | 
			
		||||
	IRQ_SAS_A                      = (1 << 18),
 | 
			
		||||
	IRQ_SAS_B                      = (1 << 19),
 | 
			
		||||
	IRQ_CPU_CNTRL                  = (1 << 20),
 | 
			
		||||
	IRQ_GPIO                       = (1 << 21),
 | 
			
		||||
	IRQ_UART                       = (1 << 22),
 | 
			
		||||
	IRQ_SPI                        = (1 << 23),
 | 
			
		||||
	IRQ_I2C                        = (1 << 24),
 | 
			
		||||
	IRQ_SGPIO                      = (1 << 25),
 | 
			
		||||
	IRQ_COM_ERR                    = (1 << 29),
 | 
			
		||||
	IRQ_I2O_ERR                    = (1 << 30),
 | 
			
		||||
	IRQ_PCIE_ERR                   = (1 << 31),
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
#define MAX_SG_ENTRY		255
 | 
			
		||||
 | 
			
		||||
struct mvs_prd_imt {
 | 
			
		||||
	__le32			len:22;
 | 
			
		||||
	u8			_r_a:2;
 | 
			
		||||
	u8			misc_ctl:4;
 | 
			
		||||
	u8			inter_sel:4;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct mvs_prd {
 | 
			
		||||
	/* 64-bit buffer address */
 | 
			
		||||
	__le64			addr;
 | 
			
		||||
	/* 22-bit length */
 | 
			
		||||
	struct mvs_prd_imt	im_len;
 | 
			
		||||
} __attribute__ ((packed));
 | 
			
		||||
 | 
			
		||||
#define SPI_CTRL_REG_94XX           	0xc800
 | 
			
		||||
#define SPI_ADDR_REG_94XX            	0xc804
 | 
			
		||||
#define SPI_WR_DATA_REG_94XX         0xc808
 | 
			
		||||
#define SPI_RD_DATA_REG_94XX         	0xc80c
 | 
			
		||||
#define SPI_CTRL_READ_94XX         	(1U << 2)
 | 
			
		||||
#define SPI_ADDR_VLD_94XX         	(1U << 1)
 | 
			
		||||
#define SPI_CTRL_SpiStart_94XX     	(1U << 0)
 | 
			
		||||
 | 
			
		||||
#define mv_ffc(x)   ffz(x)
 | 
			
		||||
 | 
			
		||||
static inline int
 | 
			
		||||
mv_ffc64(u64 v)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
	i = mv_ffc((u32)v);
 | 
			
		||||
	if (i >= 0)
 | 
			
		||||
		return i;
 | 
			
		||||
	i = mv_ffc((u32)(v>>32));
 | 
			
		||||
 | 
			
		||||
	if (i != 0)
 | 
			
		||||
		return 32 + i;
 | 
			
		||||
 | 
			
		||||
	return -1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#define r_reg_set_enable(i) \
 | 
			
		||||
	(((i) > 31) ? mr32(MVS_STP_REG_SET_1) : \
 | 
			
		||||
	mr32(MVS_STP_REG_SET_0))
 | 
			
		||||
 | 
			
		||||
#define w_reg_set_enable(i, tmp) \
 | 
			
		||||
	(((i) > 31) ? mw32(MVS_STP_REG_SET_1, tmp) : \
 | 
			
		||||
	mw32(MVS_STP_REG_SET_0, tmp))
 | 
			
		||||
 | 
			
		||||
extern const struct mvs_dispatch mvs_94xx_dispatch;
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -1,46 +1,81 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Marvell 88SE64xx/88SE94xx register IO interface
 | 
			
		||||
 *
 | 
			
		||||
 * Copyright 2007 Red Hat, Inc.
 | 
			
		||||
 * Copyright 2008 Marvell. <kewei@marvell.com>
 | 
			
		||||
 *
 | 
			
		||||
 * This file is licensed under GPLv2.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is free software; you can redistribute it and/or
 | 
			
		||||
 * modify it under the terms of the GNU General Public License as
 | 
			
		||||
 * published by the Free Software Foundation; version 2 of the
 | 
			
		||||
 * License.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is distributed in the hope that 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.
 | 
			
		||||
 *
 | 
			
		||||
 * You should have received a copy of the GNU General Public License
 | 
			
		||||
 * along with this program; if not, write to the Free Software
 | 
			
		||||
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
 | 
			
		||||
 * USA
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#ifndef _MV_CHIPS_H_
 | 
			
		||||
#define _MV_CHIPS_H_
 | 
			
		||||
 | 
			
		||||
#define mr32(reg)	readl(regs + MVS_##reg)
 | 
			
		||||
#define mw32(reg,val)	writel((val), regs + MVS_##reg)
 | 
			
		||||
#define mw32_f(reg,val)	do {			\
 | 
			
		||||
	writel((val), regs + MVS_##reg);	\
 | 
			
		||||
	readl(regs + MVS_##reg);		\
 | 
			
		||||
	} while (0)
 | 
			
		||||
#define mr32(reg)	readl(regs + reg)
 | 
			
		||||
#define mw32(reg, val)	writel((val), regs + reg)
 | 
			
		||||
#define mw32_f(reg, val)	do {			\
 | 
			
		||||
				mw32(reg, val);	\
 | 
			
		||||
				mr32(reg);	\
 | 
			
		||||
			} while (0)
 | 
			
		||||
 | 
			
		||||
static inline u32 mvs_cr32(void __iomem *regs, u32 addr)
 | 
			
		||||
#define iow32(reg, val) 	outl(val, (unsigned long)(regs + reg))
 | 
			
		||||
#define ior32(reg) 		inl((unsigned long)(regs + reg))
 | 
			
		||||
#define iow16(reg, val) 	outw((unsigned long)(val, regs + reg))
 | 
			
		||||
#define ior16(reg) 		inw((unsigned long)(regs + reg))
 | 
			
		||||
#define iow8(reg, val) 		outb((unsigned long)(val, regs + reg))
 | 
			
		||||
#define ior8(reg) 		inb((unsigned long)(regs + reg))
 | 
			
		||||
 | 
			
		||||
static inline u32 mvs_cr32(struct mvs_info *mvi, u32 addr)
 | 
			
		||||
{
 | 
			
		||||
	mw32(CMD_ADDR, addr);
 | 
			
		||||
	return mr32(CMD_DATA);
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	mw32(MVS_CMD_ADDR, addr);
 | 
			
		||||
	return mr32(MVS_CMD_DATA);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline void mvs_cw32(void __iomem *regs, u32 addr, u32 val)
 | 
			
		||||
static inline void mvs_cw32(struct mvs_info *mvi, u32 addr, u32 val)
 | 
			
		||||
{
 | 
			
		||||
	mw32(CMD_ADDR, addr);
 | 
			
		||||
	mw32(CMD_DATA, val);
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	mw32(MVS_CMD_ADDR, addr);
 | 
			
		||||
	mw32(MVS_CMD_DATA, val);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline u32 mvs_read_phy_ctl(struct mvs_info *mvi, u32 port)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	return (port < 4)?mr32(P0_SER_CTLSTAT + port * 4):
 | 
			
		||||
		mr32(P4_SER_CTLSTAT + (port - 4) * 4);
 | 
			
		||||
	return (port < 4) ? mr32(MVS_P0_SER_CTLSTAT + port * 4) :
 | 
			
		||||
		mr32(MVS_P4_SER_CTLSTAT + (port - 4) * 4);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline void mvs_write_phy_ctl(struct mvs_info *mvi, u32 port, u32 val)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	if (port < 4)
 | 
			
		||||
		mw32(P0_SER_CTLSTAT + port * 4, val);
 | 
			
		||||
		mw32(MVS_P0_SER_CTLSTAT + port * 4, val);
 | 
			
		||||
	else
 | 
			
		||||
		mw32(P4_SER_CTLSTAT + (port - 4) * 4, val);
 | 
			
		||||
		mw32(MVS_P4_SER_CTLSTAT + (port - 4) * 4, val);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline u32 mvs_read_port(struct mvs_info *mvi, u32 off, u32 off2, u32 port)
 | 
			
		||||
static inline u32 mvs_read_port(struct mvs_info *mvi, u32 off,
 | 
			
		||||
				u32 off2, u32 port)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs + off;
 | 
			
		||||
	void __iomem *regs2 = mvi->regs + off2;
 | 
			
		||||
	return (port < 4)?readl(regs + port * 8):
 | 
			
		||||
	return (port < 4) ? readl(regs + port * 8) :
 | 
			
		||||
		readl(regs2 + (port - 4) * 8);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -61,16 +96,19 @@ static inline u32 mvs_read_port_cfg_data(struct mvs_info *mvi, u32 port)
 | 
			
		|||
			MVS_P4_CFG_DATA, port);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline void mvs_write_port_cfg_data(struct mvs_info *mvi, u32 port, u32 val)
 | 
			
		||||
static inline void mvs_write_port_cfg_data(struct mvs_info *mvi,
 | 
			
		||||
						u32 port, u32 val)
 | 
			
		||||
{
 | 
			
		||||
	mvs_write_port(mvi, MVS_P0_CFG_DATA,
 | 
			
		||||
			MVS_P4_CFG_DATA, port, val);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline void mvs_write_port_cfg_addr(struct mvs_info *mvi, u32 port, u32 addr)
 | 
			
		||||
static inline void mvs_write_port_cfg_addr(struct mvs_info *mvi,
 | 
			
		||||
						u32 port, u32 addr)
 | 
			
		||||
{
 | 
			
		||||
	mvs_write_port(mvi, MVS_P0_CFG_ADDR,
 | 
			
		||||
			MVS_P4_CFG_ADDR, port, addr);
 | 
			
		||||
	mdelay(10);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline u32 mvs_read_port_vsr_data(struct mvs_info *mvi, u32 port)
 | 
			
		||||
| 
						 | 
				
			
			@ -79,16 +117,19 @@ static inline u32 mvs_read_port_vsr_data(struct mvs_info *mvi, u32 port)
 | 
			
		|||
			MVS_P4_VSR_DATA, port);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline void mvs_write_port_vsr_data(struct mvs_info *mvi, u32 port, u32 val)
 | 
			
		||||
static inline void mvs_write_port_vsr_data(struct mvs_info *mvi,
 | 
			
		||||
						u32 port, u32 val)
 | 
			
		||||
{
 | 
			
		||||
	mvs_write_port(mvi, MVS_P0_VSR_DATA,
 | 
			
		||||
			MVS_P4_VSR_DATA, port, val);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline void mvs_write_port_vsr_addr(struct mvs_info *mvi, u32 port, u32 addr)
 | 
			
		||||
static inline void mvs_write_port_vsr_addr(struct mvs_info *mvi,
 | 
			
		||||
						u32 port, u32 addr)
 | 
			
		||||
{
 | 
			
		||||
	mvs_write_port(mvi, MVS_P0_VSR_ADDR,
 | 
			
		||||
			MVS_P4_VSR_ADDR, port, addr);
 | 
			
		||||
	mdelay(10);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline u32 mvs_read_port_irq_stat(struct mvs_info *mvi, u32 port)
 | 
			
		||||
| 
						 | 
				
			
			@ -97,7 +138,8 @@ static inline u32 mvs_read_port_irq_stat(struct mvs_info *mvi, u32 port)
 | 
			
		|||
			MVS_P4_INT_STAT, port);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline void mvs_write_port_irq_stat(struct mvs_info *mvi, u32 port, u32 val)
 | 
			
		||||
static inline void mvs_write_port_irq_stat(struct mvs_info *mvi,
 | 
			
		||||
						u32 port, u32 val)
 | 
			
		||||
{
 | 
			
		||||
	mvs_write_port(mvi, MVS_P0_INT_STAT,
 | 
			
		||||
			MVS_P4_INT_STAT, port, val);
 | 
			
		||||
| 
						 | 
				
			
			@ -107,12 +149,132 @@ static inline u32 mvs_read_port_irq_mask(struct mvs_info *mvi, u32 port)
 | 
			
		|||
{
 | 
			
		||||
	return mvs_read_port(mvi, MVS_P0_INT_MASK,
 | 
			
		||||
			MVS_P4_INT_MASK, port);
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline void mvs_write_port_irq_mask(struct mvs_info *mvi, u32 port, u32 val)
 | 
			
		||||
static inline void mvs_write_port_irq_mask(struct mvs_info *mvi,
 | 
			
		||||
						u32 port, u32 val)
 | 
			
		||||
{
 | 
			
		||||
	mvs_write_port(mvi, MVS_P0_INT_MASK,
 | 
			
		||||
			MVS_P4_INT_MASK, port, val);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#endif
 | 
			
		||||
static inline void __devinit mvs_phy_hacks(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
 | 
			
		||||
	/* workaround for SATA R-ERR, to ignore phy glitch */
 | 
			
		||||
	tmp = mvs_cr32(mvi, CMD_PHY_TIMER);
 | 
			
		||||
	tmp &= ~(1 << 9);
 | 
			
		||||
	tmp |= (1 << 10);
 | 
			
		||||
	mvs_cw32(mvi, CMD_PHY_TIMER, tmp);
 | 
			
		||||
 | 
			
		||||
	/* enable retry 127 times */
 | 
			
		||||
	mvs_cw32(mvi, CMD_SAS_CTL1, 0x7f7f);
 | 
			
		||||
 | 
			
		||||
	/* extend open frame timeout to max */
 | 
			
		||||
	tmp = mvs_cr32(mvi, CMD_SAS_CTL0);
 | 
			
		||||
	tmp &= ~0xffff;
 | 
			
		||||
	tmp |= 0x3fff;
 | 
			
		||||
	mvs_cw32(mvi, CMD_SAS_CTL0, tmp);
 | 
			
		||||
 | 
			
		||||
	/* workaround for WDTIMEOUT , set to 550 ms */
 | 
			
		||||
	mvs_cw32(mvi, CMD_WD_TIMER, 0x7a0000);
 | 
			
		||||
 | 
			
		||||
	/* not to halt for different port op during wideport link change */
 | 
			
		||||
	mvs_cw32(mvi, CMD_APP_ERR_CONFIG, 0xffefbf7d);
 | 
			
		||||
 | 
			
		||||
	/* workaround for Seagate disk not-found OOB sequence, recv
 | 
			
		||||
	 * COMINIT before sending out COMWAKE */
 | 
			
		||||
	tmp = mvs_cr32(mvi, CMD_PHY_MODE_21);
 | 
			
		||||
	tmp &= 0x0000ffff;
 | 
			
		||||
	tmp |= 0x00fa0000;
 | 
			
		||||
	mvs_cw32(mvi, CMD_PHY_MODE_21, tmp);
 | 
			
		||||
 | 
			
		||||
	tmp = mvs_cr32(mvi, CMD_PHY_TIMER);
 | 
			
		||||
	tmp &= 0x1fffffff;
 | 
			
		||||
	tmp |= (2U << 29);	/* 8 ms retry */
 | 
			
		||||
	mvs_cw32(mvi, CMD_PHY_TIMER, tmp);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline void mvs_int_sata(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	u32 tmp;
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	tmp = mr32(MVS_INT_STAT_SRS_0);
 | 
			
		||||
	if (tmp)
 | 
			
		||||
		mw32(MVS_INT_STAT_SRS_0, tmp);
 | 
			
		||||
	MVS_CHIP_DISP->clear_active_cmds(mvi);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline void mvs_int_full(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	u32 tmp, stat;
 | 
			
		||||
	int i;
 | 
			
		||||
 | 
			
		||||
	stat = mr32(MVS_INT_STAT);
 | 
			
		||||
	mvs_int_rx(mvi, false);
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < mvi->chip->n_phy; i++) {
 | 
			
		||||
		tmp = (stat >> i) & (CINT_PORT | CINT_PORT_STOPPED);
 | 
			
		||||
		if (tmp)
 | 
			
		||||
			mvs_int_port(mvi, i, tmp);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (stat & CINT_SRS)
 | 
			
		||||
		mvs_int_sata(mvi);
 | 
			
		||||
 | 
			
		||||
	mw32(MVS_INT_STAT, stat);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline void mvs_start_delivery(struct mvs_info *mvi, u32 tx)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	mw32(MVS_TX_PROD_IDX, tx);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline u32 mvs_rx_update(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	return mr32(MVS_RX_CONS_IDX);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline u32 mvs_get_prd_size(void)
 | 
			
		||||
{
 | 
			
		||||
	return sizeof(struct mvs_prd);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline u32 mvs_get_prd_count(void)
 | 
			
		||||
{
 | 
			
		||||
	return MAX_SG_ENTRY;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline void mvs_show_pcie_usage(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	u16 link_stat, link_spd;
 | 
			
		||||
	const char *spd[] = {
 | 
			
		||||
		"UnKnown",
 | 
			
		||||
		"2.5",
 | 
			
		||||
		"5.0",
 | 
			
		||||
	};
 | 
			
		||||
	if (mvi->flags & MVF_FLAG_SOC || mvi->id > 0)
 | 
			
		||||
		return;
 | 
			
		||||
 | 
			
		||||
	pci_read_config_word(mvi->pdev, PCR_LINK_STAT, &link_stat);
 | 
			
		||||
	link_spd = (link_stat & PLS_LINK_SPD) >> PLS_LINK_SPD_OFFS;
 | 
			
		||||
	if (link_spd >= 3)
 | 
			
		||||
		link_spd = 0;
 | 
			
		||||
	dev_printk(KERN_INFO, mvi->dev,
 | 
			
		||||
		"mvsas: PCI-E x%u, Bandwidth Usage: %s Gbps\n",
 | 
			
		||||
	       (link_stat & PLS_NEG_LINK_WD) >> PLS_NEG_LINK_WD_OFFS,
 | 
			
		||||
	       spd[link_spd]);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline u32 mvs_hw_max_link_rate(void)
 | 
			
		||||
{
 | 
			
		||||
	return MAX_LINK_RATE;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#endif  /* _MV_CHIPS_H_ */
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,53 +1,66 @@
 | 
			
		|||
/*
 | 
			
		||||
	mv_defs.h - Marvell 88SE6440 SAS/SATA support
 | 
			
		||||
 | 
			
		||||
	Copyright 2007 Red Hat, Inc.
 | 
			
		||||
	Copyright 2008 Marvell. <kewei@marvell.com>
 | 
			
		||||
 | 
			
		||||
	This program is free software; you can redistribute it and/or
 | 
			
		||||
	modify it under the terms of the GNU General Public License as
 | 
			
		||||
	published by the Free Software Foundation; either version 2,
 | 
			
		||||
	or (at your option) any later version.
 | 
			
		||||
 | 
			
		||||
	This program is distributed in the hope that 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.
 | 
			
		||||
 | 
			
		||||
	You should have received a copy of the GNU General Public
 | 
			
		||||
	License along with this program; see the file COPYING.	If not,
 | 
			
		||||
	write to the Free Software Foundation, 675 Mass Ave, Cambridge,
 | 
			
		||||
	MA 02139, USA.
 | 
			
		||||
 | 
			
		||||
 */
 | 
			
		||||
 * Marvell 88SE64xx/88SE94xx const head file
 | 
			
		||||
 *
 | 
			
		||||
 * Copyright 2007 Red Hat, Inc.
 | 
			
		||||
 * Copyright 2008 Marvell. <kewei@marvell.com>
 | 
			
		||||
 *
 | 
			
		||||
 * This file is licensed under GPLv2.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is free software; you can redistribute it and/or
 | 
			
		||||
 * modify it under the terms of the GNU General Public License as
 | 
			
		||||
 * published by the Free Software Foundation; version 2 of the
 | 
			
		||||
 * License.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is distributed in the hope that 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.
 | 
			
		||||
 *
 | 
			
		||||
 * You should have received a copy of the GNU General Public License
 | 
			
		||||
 * along with this program; if not, write to the Free Software
 | 
			
		||||
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
 | 
			
		||||
 * USA
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
#ifndef _MV_DEFS_H_
 | 
			
		||||
#define _MV_DEFS_H_
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
enum chip_flavors {
 | 
			
		||||
	chip_6320,
 | 
			
		||||
	chip_6440,
 | 
			
		||||
	chip_6485,
 | 
			
		||||
	chip_9480,
 | 
			
		||||
	chip_9180,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/* driver compile-time configuration */
 | 
			
		||||
enum driver_configuration {
 | 
			
		||||
	MVS_SLOTS		= 512,	/* command slots */
 | 
			
		||||
	MVS_TX_RING_SZ		= 1024,	/* TX ring size (12-bit) */
 | 
			
		||||
	MVS_RX_RING_SZ		= 1024, /* RX ring size (12-bit) */
 | 
			
		||||
					/* software requires power-of-2
 | 
			
		||||
					   ring size */
 | 
			
		||||
	MVS_SOC_SLOTS		= 64,
 | 
			
		||||
	MVS_SOC_TX_RING_SZ	= MVS_SOC_SLOTS * 2,
 | 
			
		||||
	MVS_SOC_RX_RING_SZ	= MVS_SOC_SLOTS * 2,
 | 
			
		||||
 | 
			
		||||
	MVS_SLOTS		= 512,	/* command slots */
 | 
			
		||||
	MVS_SLOT_BUF_SZ		= 8192, /* cmd tbl + IU + status + PRD */
 | 
			
		||||
	MVS_SSP_CMD_SZ		= 64,	/* SSP command table buffer size */
 | 
			
		||||
	MVS_ATA_CMD_SZ		= 96,	/* SATA command table buffer size */
 | 
			
		||||
	MVS_OAF_SZ		= 64,	/* Open address frame buffer size */
 | 
			
		||||
 | 
			
		||||
	MVS_RX_FIS_COUNT	= 17,	/* Optional rx'd FISs (max 17) */
 | 
			
		||||
 | 
			
		||||
	MVS_QUEUE_SIZE		= 30,	/* Support Queue depth */
 | 
			
		||||
	MVS_CAN_QUEUE		= MVS_SLOTS - 1,	/* SCSI Queue depth */
 | 
			
		||||
	MVS_QUEUE_SIZE	= 32,	/* Support Queue depth */
 | 
			
		||||
	MVS_CAN_QUEUE		= MVS_SLOTS - 2,	/* SCSI Queue depth */
 | 
			
		||||
	MVS_SOC_CAN_QUEUE	= MVS_SOC_SLOTS - 2,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/* unchangeable hardware details */
 | 
			
		||||
enum hardware_details {
 | 
			
		||||
	MVS_MAX_PHYS		= 8,	/* max. possible phys */
 | 
			
		||||
	MVS_MAX_PORTS		= 8,	/* max. possible ports */
 | 
			
		||||
	MVS_RX_FISL_SZ		= 0x400 + (MVS_RX_FIS_COUNT * 0x100),
 | 
			
		||||
	MVS_SOC_PHYS		= 4,	/* soc phys */
 | 
			
		||||
	MVS_SOC_PORTS		= 4,	/* soc phys */
 | 
			
		||||
	MVS_MAX_DEVICES	= 1024,	/* max supported device */
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/* peripheral registers (BAR2) */
 | 
			
		||||
| 
						 | 
				
			
			@ -133,6 +146,8 @@ enum hw_register_bits {
 | 
			
		|||
	CINT_PORT		= (1U << 8),	/* port0 event */
 | 
			
		||||
	CINT_PORT_MASK_OFFSET	= 8,
 | 
			
		||||
	CINT_PORT_MASK		= (0xFF << CINT_PORT_MASK_OFFSET),
 | 
			
		||||
	CINT_PHY_MASK_OFFSET	= 4,
 | 
			
		||||
	CINT_PHY_MASK		= (0x0F << CINT_PHY_MASK_OFFSET),
 | 
			
		||||
 | 
			
		||||
	/* TX (delivery) ring bits */
 | 
			
		||||
	TXQ_CMD_SHIFT		= 29,
 | 
			
		||||
| 
						 | 
				
			
			@ -142,7 +157,11 @@ enum hw_register_bits {
 | 
			
		|||
	TXQ_CMD_SSP_FREE_LIST	= 4,		/* add to SSP targ free list */
 | 
			
		||||
	TXQ_CMD_SLOT_RESET	= 7,		/* reset command slot */
 | 
			
		||||
	TXQ_MODE_I		= (1U << 28),	/* mode: 0=target,1=initiator */
 | 
			
		||||
	TXQ_MODE_TARGET 	= 0,
 | 
			
		||||
	TXQ_MODE_INITIATOR	= 1,
 | 
			
		||||
	TXQ_PRIO_HI		= (1U << 27),	/* priority: 0=normal, 1=high */
 | 
			
		||||
	TXQ_PRI_NORMAL		= 0,
 | 
			
		||||
	TXQ_PRI_HIGH		= 1,
 | 
			
		||||
	TXQ_SRS_SHIFT		= 20,		/* SATA register set */
 | 
			
		||||
	TXQ_SRS_MASK		= 0x7f,
 | 
			
		||||
	TXQ_PHY_SHIFT		= 12,		/* PHY bitmap */
 | 
			
		||||
| 
						 | 
				
			
			@ -175,6 +194,8 @@ enum hw_register_bits {
 | 
			
		|||
	MCH_SSP_FR_READ		= 0x6,		/* Read DATA frame(s) */
 | 
			
		||||
	MCH_SSP_FR_READ_RESP	= 0x7,		/* ditto, plus RESPONSE */
 | 
			
		||||
 | 
			
		||||
	MCH_SSP_MODE_PASSTHRU	= 1,
 | 
			
		||||
	MCH_SSP_MODE_NORMAL	= 0,
 | 
			
		||||
	MCH_PASSTHRU		= (1U << 12),	/* pass-through (SSP) */
 | 
			
		||||
	MCH_FBURST		= (1U << 11),	/* first burst (SSP) */
 | 
			
		||||
	MCH_CHK_LEN		= (1U << 10),	/* chk xfer len (SSP) */
 | 
			
		||||
| 
						 | 
				
			
			@ -199,15 +220,12 @@ enum hw_register_bits {
 | 
			
		|||
	PHY_BCAST_CHG		= (1U << 2),	/* broadcast(change) notif */
 | 
			
		||||
	PHY_RST_HARD		= (1U << 1),	/* hard reset + phy reset */
 | 
			
		||||
	PHY_RST			= (1U << 0),	/* phy reset */
 | 
			
		||||
	PHY_MIN_SPP_PHYS_LINK_RATE_MASK = (0xF << 8),
 | 
			
		||||
	PHY_MAX_SPP_PHYS_LINK_RATE_MASK = (0xF << 12),
 | 
			
		||||
	PHY_NEG_SPP_PHYS_LINK_RATE_MASK_OFFSET = (16),
 | 
			
		||||
	PHY_NEG_SPP_PHYS_LINK_RATE_MASK =
 | 
			
		||||
			(0xF << PHY_NEG_SPP_PHYS_LINK_RATE_MASK_OFFSET),
 | 
			
		||||
	PHY_READY_MASK		= (1U << 20),
 | 
			
		||||
 | 
			
		||||
	/* MVS_Px_INT_STAT, MVS_Px_INT_MASK (per-phy events) */
 | 
			
		||||
	PHYEV_DEC_ERR		= (1U << 24),	/* Phy Decoding Error */
 | 
			
		||||
	PHYEV_DCDR_ERR		= (1U << 23),	/* STP Deocder Error */
 | 
			
		||||
	PHYEV_CRC_ERR		= (1U << 22),	/* STP CRC Error */
 | 
			
		||||
	PHYEV_UNASSOC_FIS	= (1U << 19),	/* unassociated FIS rx'd */
 | 
			
		||||
	PHYEV_AN		= (1U << 18),	/* SATA async notification */
 | 
			
		||||
	PHYEV_BIST_ACT		= (1U << 17),	/* BIST activate FIS */
 | 
			
		||||
| 
						 | 
				
			
			@ -229,9 +247,10 @@ enum hw_register_bits {
 | 
			
		|||
	/* MVS_PCS */
 | 
			
		||||
	PCS_EN_SATA_REG_SHIFT	= (16),		/* Enable SATA Register Set */
 | 
			
		||||
	PCS_EN_PORT_XMT_SHIFT	= (12),		/* Enable Port Transmit */
 | 
			
		||||
	PCS_EN_PORT_XMT_SHIFT2	= (8),		/* For 6480 */
 | 
			
		||||
	PCS_EN_PORT_XMT_SHIFT2	= (8),		/* For 6485 */
 | 
			
		||||
	PCS_SATA_RETRY		= (1U << 8),	/* retry ctl FIS on R_ERR */
 | 
			
		||||
	PCS_RSP_RX_EN		= (1U << 7),	/* raw response rx */
 | 
			
		||||
	PCS_SATA_RETRY_2	= (1U << 6),	/* For 9180 */
 | 
			
		||||
	PCS_SELF_CLEAR		= (1U << 5),	/* self-clearing int mode */
 | 
			
		||||
	PCS_FIS_RX_EN		= (1U << 4),	/* FIS rx enable */
 | 
			
		||||
	PCS_CMD_STOP_ERR	= (1U << 3),	/* cmd stop-on-err enable */
 | 
			
		||||
| 
						 | 
				
			
			@ -246,6 +265,8 @@ enum hw_register_bits {
 | 
			
		|||
	PORT_DEV_SMP_INIT	= (1U << 10),
 | 
			
		||||
	PORT_DEV_STP_INIT	= (1U << 9),
 | 
			
		||||
	PORT_PHY_ID_MASK	= (0xFFU << 24),
 | 
			
		||||
	PORT_SSP_TRGT_MASK	= (0x1U << 19),
 | 
			
		||||
	PORT_SSP_INIT_MASK	= (0x1U << 11),
 | 
			
		||||
	PORT_DEV_TRGT_MASK	= (0x7U << 17),
 | 
			
		||||
	PORT_DEV_INIT_MASK	= (0x7U << 9),
 | 
			
		||||
	PORT_DEV_TYPE_MASK	= (0x7U << 0),
 | 
			
		||||
| 
						 | 
				
			
			@ -283,21 +304,30 @@ enum sas_sata_config_port_regs {
 | 
			
		|||
	PHYR_ATT_ADDR_HI	= 0x14,	/* attached dev SAS addr (high) */
 | 
			
		||||
	PHYR_SATA_CTL		= 0x18,	/* SATA control */
 | 
			
		||||
	PHYR_PHY_STAT		= 0x1C,	/* PHY status */
 | 
			
		||||
	PHYR_SATA_SIG0		= 0x20,	/*port SATA signature FIS(Byte 0-3) */
 | 
			
		||||
	PHYR_SATA_SIG1		= 0x24,	/*port SATA signature FIS(Byte 4-7) */
 | 
			
		||||
	PHYR_SATA_SIG2		= 0x28,	/*port SATA signature FIS(Byte 8-11) */
 | 
			
		||||
	PHYR_SATA_SIG3		= 0x2c,	/*port SATA signature FIS(Byte 12-15) */
 | 
			
		||||
	PHYR_SATA_SIG0	= 0x20,	/*port SATA signature FIS(Byte 0-3) */
 | 
			
		||||
	PHYR_SATA_SIG1	= 0x24,	/*port SATA signature FIS(Byte 4-7) */
 | 
			
		||||
	PHYR_SATA_SIG2	= 0x28,	/*port SATA signature FIS(Byte 8-11) */
 | 
			
		||||
	PHYR_SATA_SIG3	= 0x2c,	/*port SATA signature FIS(Byte 12-15) */
 | 
			
		||||
	PHYR_R_ERR_COUNT	= 0x30, /* port R_ERR count register */
 | 
			
		||||
	PHYR_CRC_ERR_COUNT	= 0x34, /* port CRC error count register */
 | 
			
		||||
	PHYR_WIDE_PORT		= 0x38,	/* wide port participating */
 | 
			
		||||
	PHYR_WIDE_PORT	= 0x38,	/* wide port participating */
 | 
			
		||||
	PHYR_CURRENT0		= 0x80,	/* current connection info 0 */
 | 
			
		||||
	PHYR_CURRENT1		= 0x84,	/* current connection info 1 */
 | 
			
		||||
	PHYR_CURRENT2		= 0x88,	/* current connection info 2 */
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum mvs_info_flags {
 | 
			
		||||
	MVF_MSI			= (1U << 0),	/* MSI is enabled */
 | 
			
		||||
	MVF_PHY_PWR_FIX		= (1U << 1),	/* bug workaround */
 | 
			
		||||
	CONFIG_ID_FRAME0       = 0x100, /* Port device ID frame register 0 */
 | 
			
		||||
	CONFIG_ID_FRAME1       = 0x104, /* Port device ID frame register 1 */
 | 
			
		||||
	CONFIG_ID_FRAME2       = 0x108, /* Port device ID frame register 2 */
 | 
			
		||||
	CONFIG_ID_FRAME3       = 0x10c, /* Port device ID frame register 3 */
 | 
			
		||||
	CONFIG_ID_FRAME4       = 0x110, /* Port device ID frame register 4 */
 | 
			
		||||
	CONFIG_ID_FRAME5       = 0x114, /* Port device ID frame register 5 */
 | 
			
		||||
	CONFIG_ID_FRAME6       = 0x118, /* Port device ID frame register 6 */
 | 
			
		||||
	CONFIG_ATT_ID_FRAME0   = 0x11c, /* attached ID frame register 0 */
 | 
			
		||||
	CONFIG_ATT_ID_FRAME1   = 0x120, /* attached ID frame register 1 */
 | 
			
		||||
	CONFIG_ATT_ID_FRAME2   = 0x124, /* attached ID frame register 2 */
 | 
			
		||||
	CONFIG_ATT_ID_FRAME3   = 0x128, /* attached ID frame register 3 */
 | 
			
		||||
	CONFIG_ATT_ID_FRAME4   = 0x12c, /* attached ID frame register 4 */
 | 
			
		||||
	CONFIG_ATT_ID_FRAME5   = 0x130, /* attached ID frame register 5 */
 | 
			
		||||
	CONFIG_ATT_ID_FRAME6   = 0x134, /* attached ID frame register 6 */
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum sas_cmd_port_registers {
 | 
			
		||||
| 
						 | 
				
			
			@ -305,11 +335,11 @@ enum sas_cmd_port_registers {
 | 
			
		|||
	CMD_CMWK_OOB_DET	= 0x104, /* COMWAKE OOB detect register */
 | 
			
		||||
	CMD_CMSAS_OOB_DET	= 0x108, /* COMSAS OOB detect register */
 | 
			
		||||
	CMD_BRST_OOB_DET	= 0x10c, /* burst OOB detect register */
 | 
			
		||||
	CMD_OOB_SPACE		= 0x110, /* OOB space control register */
 | 
			
		||||
	CMD_OOB_BURST		= 0x114, /* OOB burst control register */
 | 
			
		||||
	CMD_OOB_SPACE	= 0x110, /* OOB space control register */
 | 
			
		||||
	CMD_OOB_BURST	= 0x114, /* OOB burst control register */
 | 
			
		||||
	CMD_PHY_TIMER		= 0x118, /* PHY timer control register */
 | 
			
		||||
	CMD_PHY_CONFIG0		= 0x11c, /* PHY config register 0 */
 | 
			
		||||
	CMD_PHY_CONFIG1		= 0x120, /* PHY config register 1 */
 | 
			
		||||
	CMD_PHY_CONFIG0	= 0x11c, /* PHY config register 0 */
 | 
			
		||||
	CMD_PHY_CONFIG1	= 0x120, /* PHY config register 1 */
 | 
			
		||||
	CMD_SAS_CTL0		= 0x124, /* SAS control register 0 */
 | 
			
		||||
	CMD_SAS_CTL1		= 0x128, /* SAS control register 1 */
 | 
			
		||||
	CMD_SAS_CTL2		= 0x12c, /* SAS control register 2 */
 | 
			
		||||
| 
						 | 
				
			
			@ -318,9 +348,9 @@ enum sas_cmd_port_registers {
 | 
			
		|||
	CMD_PL_TIMER		= 0x138, /* PL timer register */
 | 
			
		||||
	CMD_WD_TIMER		= 0x13c, /* WD timer register */
 | 
			
		||||
	CMD_PORT_SEL_COUNT	= 0x140, /* port selector count register */
 | 
			
		||||
	CMD_APP_MEM_CTL		= 0x144, /* Application Memory Control */
 | 
			
		||||
	CMD_XOR_MEM_CTL		= 0x148, /* XOR Block Memory Control */
 | 
			
		||||
	CMD_DMA_MEM_CTL		= 0x14c, /* DMA Block Memory Control */
 | 
			
		||||
	CMD_APP_MEM_CTL	= 0x144, /* Application Memory Control */
 | 
			
		||||
	CMD_XOR_MEM_CTL	= 0x148, /* XOR Block Memory Control */
 | 
			
		||||
	CMD_DMA_MEM_CTL	= 0x14c, /* DMA Block Memory Control */
 | 
			
		||||
	CMD_PORT_MEM_CTL0	= 0x150, /* Port Memory Control 0 */
 | 
			
		||||
	CMD_PORT_MEM_CTL1	= 0x154, /* Port Memory Control 1 */
 | 
			
		||||
	CMD_SATA_PORT_MEM_CTL0	= 0x158, /* SATA Port Memory Control 0 */
 | 
			
		||||
| 
						 | 
				
			
			@ -353,27 +383,25 @@ enum sas_cmd_port_registers {
 | 
			
		|||
	CMD_PND_FIFO_CTL1	= 0x1C4, /* Pending FIFO Control 1 */
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum pci_cfg_register_bits {
 | 
			
		||||
	PCTL_PWR_ON	= (0xFU << 24),
 | 
			
		||||
	PCTL_OFF	= (0xFU << 12),
 | 
			
		||||
	PRD_REQ_SIZE	= (0x4000),
 | 
			
		||||
	PRD_REQ_MASK	= (0x00007000),
 | 
			
		||||
enum mvs_info_flags {
 | 
			
		||||
	MVF_MSI		= (1U << 0),	/* MSI is enabled */
 | 
			
		||||
	MVF_PHY_PWR_FIX	= (1U << 1),	/* bug workaround */
 | 
			
		||||
	MVF_FLAG_SOC		= (1U << 2),	/* SoC integrated controllers */
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum nvram_layout_offsets {
 | 
			
		||||
	NVR_SIG		= 0x00,		/* 0xAA, 0x55 */
 | 
			
		||||
	NVR_SAS_ADDR	= 0x02,		/* 8-byte SAS address */
 | 
			
		||||
enum mvs_event_flags {
 | 
			
		||||
	PHY_PLUG_EVENT	= (3U),
 | 
			
		||||
	PHY_PLUG_IN		= (1U << 0),	/* phy plug in */
 | 
			
		||||
	PHY_PLUG_OUT		= (1U << 1),	/* phy plug out */
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum chip_flavors {
 | 
			
		||||
	chip_6320,
 | 
			
		||||
	chip_6440,
 | 
			
		||||
	chip_6480,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum port_type {
 | 
			
		||||
	PORT_TYPE_SAS	=  (1L << 1),
 | 
			
		||||
	PORT_TYPE_SATA	=  (1L << 0),
 | 
			
		||||
enum mvs_port_type {
 | 
			
		||||
	PORT_TGT_MASK	=  (1U << 5),
 | 
			
		||||
	PORT_INIT_PORT	=  (1U << 4),
 | 
			
		||||
	PORT_TGT_PORT	=  (1U << 3),
 | 
			
		||||
	PORT_INIT_TGT_PORT = (PORT_INIT_PORT | PORT_TGT_PORT),
 | 
			
		||||
	PORT_TYPE_SAS	=  (1U << 1),
 | 
			
		||||
	PORT_TYPE_SATA	=  (1U << 0),
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/* Command Table Format */
 | 
			
		||||
| 
						 | 
				
			
			@ -438,4 +466,37 @@ enum error_info_rec_2 {
 | 
			
		|||
	USR_BLK_NM	= (1U << 0),	/* User Block Number */
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum pci_cfg_register_bits {
 | 
			
		||||
	PCTL_PWR_OFF	= (0xFU << 24),
 | 
			
		||||
	PCTL_COM_ON	= (0xFU << 20),
 | 
			
		||||
	PCTL_LINK_RST	= (0xFU << 16),
 | 
			
		||||
	PCTL_LINK_OFFS	= (16),
 | 
			
		||||
	PCTL_PHY_DSBL	= (0xFU << 12),
 | 
			
		||||
	PCTL_PHY_DSBL_OFFS	= (12),
 | 
			
		||||
	PRD_REQ_SIZE	= (0x4000),
 | 
			
		||||
	PRD_REQ_MASK	= (0x00007000),
 | 
			
		||||
	PLS_NEG_LINK_WD		= (0x3FU << 4),
 | 
			
		||||
	PLS_NEG_LINK_WD_OFFS	= 4,
 | 
			
		||||
	PLS_LINK_SPD		= (0x0FU << 0),
 | 
			
		||||
	PLS_LINK_SPD_OFFS	= 0,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum open_frame_protocol {
 | 
			
		||||
	PROTOCOL_SMP	= 0x0,
 | 
			
		||||
	PROTOCOL_SSP	= 0x1,
 | 
			
		||||
	PROTOCOL_STP	= 0x2,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/* define for response frame datapres field */
 | 
			
		||||
enum datapres_field {
 | 
			
		||||
	NO_DATA		= 0,
 | 
			
		||||
	RESPONSE_DATA	= 1,
 | 
			
		||||
	SENSE_DATA	= 2,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/* define task management IU */
 | 
			
		||||
struct mvs_tmf_task{
 | 
			
		||||
	u8 tmf;
 | 
			
		||||
	u16 tag_of_task_to_be_managed;
 | 
			
		||||
};
 | 
			
		||||
#endif
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,38 +1,41 @@
 | 
			
		|||
/*
 | 
			
		||||
	mv_init.c - Marvell 88SE6440 SAS/SATA init support
 | 
			
		||||
 * Marvell 88SE64xx/88SE94xx pci init
 | 
			
		||||
 *
 | 
			
		||||
 * Copyright 2007 Red Hat, Inc.
 | 
			
		||||
 * Copyright 2008 Marvell. <kewei@marvell.com>
 | 
			
		||||
 *
 | 
			
		||||
 * This file is licensed under GPLv2.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is free software; you can redistribute it and/or
 | 
			
		||||
 * modify it under the terms of the GNU General Public License as
 | 
			
		||||
 * published by the Free Software Foundation; version 2 of the
 | 
			
		||||
 * License.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is distributed in the hope that 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.
 | 
			
		||||
 *
 | 
			
		||||
 * You should have received a copy of the GNU General Public License
 | 
			
		||||
 * along with this program; if not, write to the Free Software
 | 
			
		||||
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
 | 
			
		||||
 * USA
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
	Copyright 2007 Red Hat, Inc.
 | 
			
		||||
	Copyright 2008 Marvell. <kewei@marvell.com>
 | 
			
		||||
 | 
			
		||||
	This program is free software; you can redistribute it and/or
 | 
			
		||||
	modify it under the terms of the GNU General Public License as
 | 
			
		||||
	published by the Free Software Foundation; either version 2,
 | 
			
		||||
	or (at your option) any later version.
 | 
			
		||||
 | 
			
		||||
	This program is distributed in the hope that 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.
 | 
			
		||||
 | 
			
		||||
	You should have received a copy of the GNU General Public
 | 
			
		||||
	License along with this program; see the file COPYING.	If not,
 | 
			
		||||
	write to the Free Software Foundation, 675 Mass Ave, Cambridge,
 | 
			
		||||
	MA 02139, USA.
 | 
			
		||||
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
#include "mv_sas.h"
 | 
			
		||||
#include "mv_64xx.h"
 | 
			
		||||
#include "mv_chips.h"
 | 
			
		||||
 | 
			
		||||
static struct scsi_transport_template *mvs_stt;
 | 
			
		||||
 | 
			
		||||
static const struct mvs_chip_info mvs_chips[] = {
 | 
			
		||||
	[chip_6320] =		{ 2, 16, 9  },
 | 
			
		||||
	[chip_6440] =		{ 4, 16, 9  },
 | 
			
		||||
	[chip_6480] =		{ 8, 32, 10 },
 | 
			
		||||
	[chip_6320] =	{ 1, 2, 0x400, 17, 16,  9, &mvs_64xx_dispatch, },
 | 
			
		||||
	[chip_6440] =	{ 1, 4, 0x400, 17, 16,  9, &mvs_64xx_dispatch, },
 | 
			
		||||
	[chip_6485] =	{ 1, 8, 0x800, 33, 32, 10, &mvs_64xx_dispatch, },
 | 
			
		||||
	[chip_9180] =	{ 2, 4, 0x800, 17, 64,  9, &mvs_94xx_dispatch, },
 | 
			
		||||
	[chip_9480] =	{ 2, 4, 0x800, 17, 64,  9, &mvs_94xx_dispatch, },
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
#define SOC_SAS_NUM 2
 | 
			
		||||
 | 
			
		||||
static struct scsi_host_template mvs_sht = {
 | 
			
		||||
	.module			= THIS_MODULE,
 | 
			
		||||
	.name			= DRV_NAME,
 | 
			
		||||
| 
						 | 
				
			
			@ -53,17 +56,29 @@ static struct scsi_host_template mvs_sht = {
 | 
			
		|||
	.use_clustering		= ENABLE_CLUSTERING,
 | 
			
		||||
	.eh_device_reset_handler	= sas_eh_device_reset_handler,
 | 
			
		||||
	.eh_bus_reset_handler	= sas_eh_bus_reset_handler,
 | 
			
		||||
	.slave_alloc		= sas_slave_alloc,
 | 
			
		||||
	.slave_alloc		= mvs_slave_alloc,
 | 
			
		||||
	.target_destroy		= sas_target_destroy,
 | 
			
		||||
	.ioctl			= sas_ioctl,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static struct sas_domain_function_template mvs_transport_ops = {
 | 
			
		||||
	.lldd_execute_task	= mvs_task_exec,
 | 
			
		||||
	.lldd_dev_found 	= mvs_dev_found,
 | 
			
		||||
	.lldd_dev_gone	= mvs_dev_gone,
 | 
			
		||||
 | 
			
		||||
	.lldd_execute_task	= mvs_queue_command,
 | 
			
		||||
	.lldd_control_phy	= mvs_phy_control,
 | 
			
		||||
	.lldd_abort_task	= mvs_task_abort,
 | 
			
		||||
	.lldd_port_formed	= mvs_port_formed,
 | 
			
		||||
 | 
			
		||||
	.lldd_abort_task	= mvs_abort_task,
 | 
			
		||||
	.lldd_abort_task_set    = mvs_abort_task_set,
 | 
			
		||||
	.lldd_clear_aca         = mvs_clear_aca,
 | 
			
		||||
       .lldd_clear_task_set    = mvs_clear_task_set,
 | 
			
		||||
	.lldd_I_T_nexus_reset	= mvs_I_T_nexus_reset,
 | 
			
		||||
	.lldd_lu_reset 		= mvs_lu_reset,
 | 
			
		||||
	.lldd_query_task	= mvs_query_task,
 | 
			
		||||
 | 
			
		||||
	.lldd_port_formed	= mvs_port_formed,
 | 
			
		||||
	.lldd_port_deformed     = mvs_port_deformed,
 | 
			
		||||
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static void __devinit mvs_phy_init(struct mvs_info *mvi, int phy_id)
 | 
			
		||||
| 
						 | 
				
			
			@ -71,6 +86,8 @@ static void __devinit mvs_phy_init(struct mvs_info *mvi, int phy_id)
 | 
			
		|||
	struct mvs_phy *phy = &mvi->phy[phy_id];
 | 
			
		||||
	struct asd_sas_phy *sas_phy = &phy->sas_phy;
 | 
			
		||||
 | 
			
		||||
	phy->mvi = mvi;
 | 
			
		||||
	init_timer(&phy->timer);
 | 
			
		||||
	sas_phy->enabled = (phy_id < mvi->chip->n_phy) ? 1 : 0;
 | 
			
		||||
	sas_phy->class = SAS;
 | 
			
		||||
	sas_phy->iproto = SAS_PROTOCOL_ALL;
 | 
			
		||||
| 
						 | 
				
			
			@ -83,248 +100,283 @@ static void __devinit mvs_phy_init(struct mvs_info *mvi, int phy_id)
 | 
			
		|||
	sas_phy->id = phy_id;
 | 
			
		||||
	sas_phy->sas_addr = &mvi->sas_addr[0];
 | 
			
		||||
	sas_phy->frame_rcvd = &phy->frame_rcvd[0];
 | 
			
		||||
	sas_phy->ha = &mvi->sas;
 | 
			
		||||
	sas_phy->ha = (struct sas_ha_struct *)mvi->shost->hostdata;
 | 
			
		||||
	sas_phy->lldd_phy = phy;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_free(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
	struct mvs_wq *mwq;
 | 
			
		||||
	int slot_nr;
 | 
			
		||||
 | 
			
		||||
	if (!mvi)
 | 
			
		||||
		return;
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < MVS_SLOTS; i++) {
 | 
			
		||||
		struct mvs_slot_info *slot = &mvi->slot_info[i];
 | 
			
		||||
	if (mvi->flags & MVF_FLAG_SOC)
 | 
			
		||||
		slot_nr = MVS_SOC_SLOTS;
 | 
			
		||||
	else
 | 
			
		||||
		slot_nr = MVS_SLOTS;
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < mvi->tags_num; i++) {
 | 
			
		||||
		struct mvs_slot_info *slot = &mvi->slot_info[i];
 | 
			
		||||
		if (slot->buf)
 | 
			
		||||
			dma_free_coherent(&mvi->pdev->dev, MVS_SLOT_BUF_SZ,
 | 
			
		||||
			dma_free_coherent(mvi->dev, MVS_SLOT_BUF_SZ,
 | 
			
		||||
					  slot->buf, slot->buf_dma);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (mvi->tx)
 | 
			
		||||
		dma_free_coherent(&mvi->pdev->dev,
 | 
			
		||||
		dma_free_coherent(mvi->dev,
 | 
			
		||||
				  sizeof(*mvi->tx) * MVS_CHIP_SLOT_SZ,
 | 
			
		||||
				  mvi->tx, mvi->tx_dma);
 | 
			
		||||
	if (mvi->rx_fis)
 | 
			
		||||
		dma_free_coherent(&mvi->pdev->dev, MVS_RX_FISL_SZ,
 | 
			
		||||
		dma_free_coherent(mvi->dev, MVS_RX_FISL_SZ,
 | 
			
		||||
				  mvi->rx_fis, mvi->rx_fis_dma);
 | 
			
		||||
	if (mvi->rx)
 | 
			
		||||
		dma_free_coherent(&mvi->pdev->dev,
 | 
			
		||||
		dma_free_coherent(mvi->dev,
 | 
			
		||||
				  sizeof(*mvi->rx) * (MVS_RX_RING_SZ + 1),
 | 
			
		||||
				  mvi->rx, mvi->rx_dma);
 | 
			
		||||
	if (mvi->slot)
 | 
			
		||||
		dma_free_coherent(&mvi->pdev->dev,
 | 
			
		||||
				  sizeof(*mvi->slot) * MVS_SLOTS,
 | 
			
		||||
		dma_free_coherent(mvi->dev,
 | 
			
		||||
				  sizeof(*mvi->slot) * slot_nr,
 | 
			
		||||
				  mvi->slot, mvi->slot_dma);
 | 
			
		||||
#ifdef MVS_ENABLE_PERI
 | 
			
		||||
	if (mvi->peri_regs)
 | 
			
		||||
		iounmap(mvi->peri_regs);
 | 
			
		||||
#ifndef DISABLE_HOTPLUG_DMA_FIX
 | 
			
		||||
	if (mvi->bulk_buffer)
 | 
			
		||||
		dma_free_coherent(mvi->dev, TRASH_BUCKET_SIZE,
 | 
			
		||||
				  mvi->bulk_buffer, mvi->bulk_buffer_dma);
 | 
			
		||||
#endif
 | 
			
		||||
	if (mvi->regs)
 | 
			
		||||
		iounmap(mvi->regs);
 | 
			
		||||
 | 
			
		||||
	MVS_CHIP_DISP->chip_iounmap(mvi);
 | 
			
		||||
	if (mvi->shost)
 | 
			
		||||
		scsi_host_put(mvi->shost);
 | 
			
		||||
	kfree(mvi->sas.sas_port);
 | 
			
		||||
	kfree(mvi->sas.sas_phy);
 | 
			
		||||
	list_for_each_entry(mwq, &mvi->wq_list, entry)
 | 
			
		||||
		cancel_delayed_work(&mwq->work_q);
 | 
			
		||||
	kfree(mvi);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#ifdef MVS_USE_TASKLET
 | 
			
		||||
static void mvs_tasklet(unsigned long data)
 | 
			
		||||
struct tasklet_struct	mv_tasklet;
 | 
			
		||||
static void mvs_tasklet(unsigned long opaque)
 | 
			
		||||
{
 | 
			
		||||
	struct mvs_info *mvi = (struct mvs_info *) data;
 | 
			
		||||
	unsigned long flags;
 | 
			
		||||
	u32 stat;
 | 
			
		||||
	u16 core_nr, i = 0;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irqsave(&mvi->lock, flags);
 | 
			
		||||
	struct mvs_info *mvi;
 | 
			
		||||
	struct sas_ha_struct *sha = (struct sas_ha_struct *)opaque;
 | 
			
		||||
 | 
			
		||||
	core_nr = ((struct mvs_prv_info *)sha->lldd_ha)->n_host;
 | 
			
		||||
	mvi = ((struct mvs_prv_info *)sha->lldd_ha)->mvi[0];
 | 
			
		||||
 | 
			
		||||
	if (unlikely(!mvi))
 | 
			
		||||
		BUG_ON(1);
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < core_nr; i++) {
 | 
			
		||||
		mvi = ((struct mvs_prv_info *)sha->lldd_ha)->mvi[i];
 | 
			
		||||
		stat = MVS_CHIP_DISP->isr_status(mvi, mvi->irq);
 | 
			
		||||
		if (stat)
 | 
			
		||||
			MVS_CHIP_DISP->isr(mvi, mvi->irq, stat);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
#ifdef MVS_DISABLE_MSI
 | 
			
		||||
	mvs_int_full(mvi);
 | 
			
		||||
#else
 | 
			
		||||
	mvs_int_rx(mvi, true);
 | 
			
		||||
#endif
 | 
			
		||||
	spin_unlock_irqrestore(&mvi->lock, flags);
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
static irqreturn_t mvs_interrupt(int irq, void *opaque)
 | 
			
		||||
{
 | 
			
		||||
	struct mvs_info *mvi = opaque;
 | 
			
		||||
	void __iomem *regs = mvi->regs;
 | 
			
		||||
	u32 core_nr, i = 0;
 | 
			
		||||
	u32 stat;
 | 
			
		||||
	struct mvs_info *mvi;
 | 
			
		||||
	struct sas_ha_struct *sha = opaque;
 | 
			
		||||
 | 
			
		||||
	stat = mr32(GBL_INT_STAT);
 | 
			
		||||
	core_nr = ((struct mvs_prv_info *)sha->lldd_ha)->n_host;
 | 
			
		||||
	mvi = ((struct mvs_prv_info *)sha->lldd_ha)->mvi[0];
 | 
			
		||||
 | 
			
		||||
	if (stat == 0 || stat == 0xffffffff)
 | 
			
		||||
	if (unlikely(!mvi))
 | 
			
		||||
		return IRQ_NONE;
 | 
			
		||||
 | 
			
		||||
	/* clear CMD_CMPLT ASAP */
 | 
			
		||||
	mw32_f(INT_STAT, CINT_DONE);
 | 
			
		||||
	stat = MVS_CHIP_DISP->isr_status(mvi, irq);
 | 
			
		||||
	if (!stat)
 | 
			
		||||
		return IRQ_NONE;
 | 
			
		||||
 | 
			
		||||
#ifndef MVS_USE_TASKLET
 | 
			
		||||
	spin_lock(&mvi->lock);
 | 
			
		||||
 | 
			
		||||
	mvs_int_full(mvi);
 | 
			
		||||
 | 
			
		||||
	spin_unlock(&mvi->lock);
 | 
			
		||||
#ifdef MVS_USE_TASKLET
 | 
			
		||||
	tasklet_schedule(&mv_tasklet);
 | 
			
		||||
#else
 | 
			
		||||
	tasklet_schedule(&mvi->tasklet);
 | 
			
		||||
	for (i = 0; i < core_nr; i++) {
 | 
			
		||||
		mvi = ((struct mvs_prv_info *)sha->lldd_ha)->mvi[i];
 | 
			
		||||
		MVS_CHIP_DISP->isr(mvi, irq, stat);
 | 
			
		||||
	}
 | 
			
		||||
#endif
 | 
			
		||||
	return IRQ_HANDLED;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static struct mvs_info *__devinit mvs_alloc(struct pci_dev *pdev,
 | 
			
		||||
					    const struct pci_device_id *ent)
 | 
			
		||||
static int __devinit mvs_alloc(struct mvs_info *mvi, struct Scsi_Host *shost)
 | 
			
		||||
{
 | 
			
		||||
	struct mvs_info *mvi;
 | 
			
		||||
	unsigned long res_start, res_len, res_flag;
 | 
			
		||||
	struct asd_sas_phy **arr_phy;
 | 
			
		||||
	struct asd_sas_port **arr_port;
 | 
			
		||||
	const struct mvs_chip_info *chip = &mvs_chips[ent->driver_data];
 | 
			
		||||
	int i;
 | 
			
		||||
	int i, slot_nr;
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * alloc and init our per-HBA mvs_info struct
 | 
			
		||||
	 */
 | 
			
		||||
 | 
			
		||||
	mvi = kzalloc(sizeof(*mvi), GFP_KERNEL);
 | 
			
		||||
	if (!mvi)
 | 
			
		||||
		return NULL;
 | 
			
		||||
	if (mvi->flags & MVF_FLAG_SOC)
 | 
			
		||||
		slot_nr = MVS_SOC_SLOTS;
 | 
			
		||||
	else
 | 
			
		||||
		slot_nr = MVS_SLOTS;
 | 
			
		||||
 | 
			
		||||
	spin_lock_init(&mvi->lock);
 | 
			
		||||
#ifdef MVS_USE_TASKLET
 | 
			
		||||
	tasklet_init(&mvi->tasklet, mvs_tasklet, (unsigned long)mvi);
 | 
			
		||||
#endif
 | 
			
		||||
	mvi->pdev = pdev;
 | 
			
		||||
	mvi->chip = chip;
 | 
			
		||||
 | 
			
		||||
	if (pdev->device == 0x6440 && pdev->revision == 0)
 | 
			
		||||
		mvi->flags |= MVF_PHY_PWR_FIX;
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * alloc and init SCSI, SAS glue
 | 
			
		||||
	 */
 | 
			
		||||
 | 
			
		||||
	mvi->shost = scsi_host_alloc(&mvs_sht, sizeof(void *));
 | 
			
		||||
	if (!mvi->shost)
 | 
			
		||||
		goto err_out;
 | 
			
		||||
 | 
			
		||||
	arr_phy = kcalloc(MVS_MAX_PHYS, sizeof(void *), GFP_KERNEL);
 | 
			
		||||
	arr_port = kcalloc(MVS_MAX_PHYS, sizeof(void *), GFP_KERNEL);
 | 
			
		||||
	if (!arr_phy || !arr_port)
 | 
			
		||||
		goto err_out;
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < MVS_MAX_PHYS; i++) {
 | 
			
		||||
	for (i = 0; i < mvi->chip->n_phy; i++) {
 | 
			
		||||
		mvs_phy_init(mvi, i);
 | 
			
		||||
		arr_phy[i] = &mvi->phy[i].sas_phy;
 | 
			
		||||
		arr_port[i] = &mvi->port[i].sas_port;
 | 
			
		||||
		mvi->port[i].taskfileset = MVS_ID_NOT_MAPPED;
 | 
			
		||||
		mvi->port[i].wide_port_phymap = 0;
 | 
			
		||||
		mvi->port[i].port_attached = 0;
 | 
			
		||||
		INIT_LIST_HEAD(&mvi->port[i].list);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	SHOST_TO_SAS_HA(mvi->shost) = &mvi->sas;
 | 
			
		||||
	mvi->shost->transportt = mvs_stt;
 | 
			
		||||
	mvi->shost->max_id = 21;
 | 
			
		||||
	mvi->shost->max_lun = ~0;
 | 
			
		||||
	mvi->shost->max_channel = 0;
 | 
			
		||||
	mvi->shost->max_cmd_len = 16;
 | 
			
		||||
 | 
			
		||||
	mvi->sas.sas_ha_name = DRV_NAME;
 | 
			
		||||
	mvi->sas.dev = &pdev->dev;
 | 
			
		||||
	mvi->sas.lldd_module = THIS_MODULE;
 | 
			
		||||
	mvi->sas.sas_addr = &mvi->sas_addr[0];
 | 
			
		||||
	mvi->sas.sas_phy = arr_phy;
 | 
			
		||||
	mvi->sas.sas_port = arr_port;
 | 
			
		||||
	mvi->sas.num_phys = chip->n_phy;
 | 
			
		||||
	mvi->sas.lldd_max_execute_num = 1;
 | 
			
		||||
	mvi->sas.lldd_queue_size = MVS_QUEUE_SIZE;
 | 
			
		||||
	mvi->shost->can_queue = MVS_CAN_QUEUE;
 | 
			
		||||
	mvi->shost->cmd_per_lun = MVS_SLOTS / mvi->sas.num_phys;
 | 
			
		||||
	mvi->sas.lldd_ha = mvi;
 | 
			
		||||
	mvi->sas.core.shost = mvi->shost;
 | 
			
		||||
 | 
			
		||||
	mvs_tag_init(mvi);
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * ioremap main and peripheral registers
 | 
			
		||||
	 */
 | 
			
		||||
 | 
			
		||||
#ifdef MVS_ENABLE_PERI
 | 
			
		||||
	res_start = pci_resource_start(pdev, 2);
 | 
			
		||||
	res_len = pci_resource_len(pdev, 2);
 | 
			
		||||
	if (!res_start || !res_len)
 | 
			
		||||
		goto err_out;
 | 
			
		||||
 | 
			
		||||
	mvi->peri_regs = ioremap_nocache(res_start, res_len);
 | 
			
		||||
	if (!mvi->peri_regs)
 | 
			
		||||
		goto err_out;
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
	res_start = pci_resource_start(pdev, 4);
 | 
			
		||||
	res_len = pci_resource_len(pdev, 4);
 | 
			
		||||
	if (!res_start || !res_len)
 | 
			
		||||
		goto err_out;
 | 
			
		||||
 | 
			
		||||
	res_flag = pci_resource_flags(pdev, 4);
 | 
			
		||||
	if (res_flag & IORESOURCE_CACHEABLE)
 | 
			
		||||
		mvi->regs = ioremap(res_start, res_len);
 | 
			
		||||
	else
 | 
			
		||||
		mvi->regs = ioremap_nocache(res_start, res_len);
 | 
			
		||||
 | 
			
		||||
	if (!mvi->regs)
 | 
			
		||||
		goto err_out;
 | 
			
		||||
	for (i = 0; i < MVS_MAX_DEVICES; i++) {
 | 
			
		||||
		mvi->devices[i].taskfileset = MVS_ID_NOT_MAPPED;
 | 
			
		||||
		mvi->devices[i].dev_type = NO_DEVICE;
 | 
			
		||||
		mvi->devices[i].device_id = i;
 | 
			
		||||
		mvi->devices[i].dev_status = MVS_DEV_NORMAL;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * alloc and init our DMA areas
 | 
			
		||||
	 */
 | 
			
		||||
 | 
			
		||||
	mvi->tx = dma_alloc_coherent(&pdev->dev,
 | 
			
		||||
	mvi->tx = dma_alloc_coherent(mvi->dev,
 | 
			
		||||
				     sizeof(*mvi->tx) * MVS_CHIP_SLOT_SZ,
 | 
			
		||||
				     &mvi->tx_dma, GFP_KERNEL);
 | 
			
		||||
	if (!mvi->tx)
 | 
			
		||||
		goto err_out;
 | 
			
		||||
	memset(mvi->tx, 0, sizeof(*mvi->tx) * MVS_CHIP_SLOT_SZ);
 | 
			
		||||
 | 
			
		||||
	mvi->rx_fis = dma_alloc_coherent(&pdev->dev, MVS_RX_FISL_SZ,
 | 
			
		||||
	mvi->rx_fis = dma_alloc_coherent(mvi->dev, MVS_RX_FISL_SZ,
 | 
			
		||||
					 &mvi->rx_fis_dma, GFP_KERNEL);
 | 
			
		||||
	if (!mvi->rx_fis)
 | 
			
		||||
		goto err_out;
 | 
			
		||||
	memset(mvi->rx_fis, 0, MVS_RX_FISL_SZ);
 | 
			
		||||
 | 
			
		||||
	mvi->rx = dma_alloc_coherent(&pdev->dev,
 | 
			
		||||
	mvi->rx = dma_alloc_coherent(mvi->dev,
 | 
			
		||||
				     sizeof(*mvi->rx) * (MVS_RX_RING_SZ + 1),
 | 
			
		||||
				     &mvi->rx_dma, GFP_KERNEL);
 | 
			
		||||
	if (!mvi->rx)
 | 
			
		||||
		goto err_out;
 | 
			
		||||
	memset(mvi->rx, 0, sizeof(*mvi->rx) * (MVS_RX_RING_SZ + 1));
 | 
			
		||||
 | 
			
		||||
	mvi->rx[0] = cpu_to_le32(0xfff);
 | 
			
		||||
	mvi->rx_cons = 0xfff;
 | 
			
		||||
 | 
			
		||||
	mvi->slot = dma_alloc_coherent(&pdev->dev,
 | 
			
		||||
				       sizeof(*mvi->slot) * MVS_SLOTS,
 | 
			
		||||
	mvi->slot = dma_alloc_coherent(mvi->dev,
 | 
			
		||||
				       sizeof(*mvi->slot) * slot_nr,
 | 
			
		||||
				       &mvi->slot_dma, GFP_KERNEL);
 | 
			
		||||
	if (!mvi->slot)
 | 
			
		||||
		goto err_out;
 | 
			
		||||
	memset(mvi->slot, 0, sizeof(*mvi->slot) * MVS_SLOTS);
 | 
			
		||||
	memset(mvi->slot, 0, sizeof(*mvi->slot) * slot_nr);
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < MVS_SLOTS; i++) {
 | 
			
		||||
#ifndef DISABLE_HOTPLUG_DMA_FIX
 | 
			
		||||
	mvi->bulk_buffer = dma_alloc_coherent(mvi->dev,
 | 
			
		||||
				       TRASH_BUCKET_SIZE,
 | 
			
		||||
				       &mvi->bulk_buffer_dma, GFP_KERNEL);
 | 
			
		||||
	if (!mvi->bulk_buffer)
 | 
			
		||||
		goto err_out;
 | 
			
		||||
#endif
 | 
			
		||||
	for (i = 0; i < slot_nr; i++) {
 | 
			
		||||
		struct mvs_slot_info *slot = &mvi->slot_info[i];
 | 
			
		||||
 | 
			
		||||
		slot->buf = dma_alloc_coherent(&pdev->dev, MVS_SLOT_BUF_SZ,
 | 
			
		||||
		slot->buf = dma_alloc_coherent(mvi->dev, MVS_SLOT_BUF_SZ,
 | 
			
		||||
					       &slot->buf_dma, GFP_KERNEL);
 | 
			
		||||
		if (!slot->buf)
 | 
			
		||||
		if (!slot->buf) {
 | 
			
		||||
			printk(KERN_DEBUG"failed to allocate slot->buf.\n");
 | 
			
		||||
			goto err_out;
 | 
			
		||||
		}
 | 
			
		||||
		memset(slot->buf, 0, MVS_SLOT_BUF_SZ);
 | 
			
		||||
		++mvi->tags_num;
 | 
			
		||||
	}
 | 
			
		||||
	/* Initialize tags */
 | 
			
		||||
	mvs_tag_init(mvi);
 | 
			
		||||
	return 0;
 | 
			
		||||
err_out:
 | 
			
		||||
	return 1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
int mvs_ioremap(struct mvs_info *mvi, int bar, int bar_ex)
 | 
			
		||||
{
 | 
			
		||||
	unsigned long res_start, res_len, res_flag, res_flag_ex = 0;
 | 
			
		||||
	struct pci_dev *pdev = mvi->pdev;
 | 
			
		||||
	if (bar_ex != -1) {
 | 
			
		||||
		/*
 | 
			
		||||
		 * ioremap main and peripheral registers
 | 
			
		||||
		 */
 | 
			
		||||
		res_start = pci_resource_start(pdev, bar_ex);
 | 
			
		||||
		res_len = pci_resource_len(pdev, bar_ex);
 | 
			
		||||
		if (!res_start || !res_len)
 | 
			
		||||
			goto err_out;
 | 
			
		||||
 | 
			
		||||
		res_flag_ex = pci_resource_flags(pdev, bar_ex);
 | 
			
		||||
		if (res_flag_ex & IORESOURCE_MEM) {
 | 
			
		||||
			if (res_flag_ex & IORESOURCE_CACHEABLE)
 | 
			
		||||
				mvi->regs_ex = ioremap(res_start, res_len);
 | 
			
		||||
			else
 | 
			
		||||
				mvi->regs_ex = ioremap_nocache(res_start,
 | 
			
		||||
						res_len);
 | 
			
		||||
		} else
 | 
			
		||||
			mvi->regs_ex = (void *)res_start;
 | 
			
		||||
		if (!mvi->regs_ex)
 | 
			
		||||
			goto err_out;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* finally, read NVRAM to get our SAS address */
 | 
			
		||||
	if (mvs_nvram_read(mvi, NVR_SAS_ADDR, &mvi->sas_addr, 8))
 | 
			
		||||
	res_start = pci_resource_start(pdev, bar);
 | 
			
		||||
	res_len = pci_resource_len(pdev, bar);
 | 
			
		||||
	if (!res_start || !res_len)
 | 
			
		||||
		goto err_out;
 | 
			
		||||
	return mvi;
 | 
			
		||||
 | 
			
		||||
	res_flag = pci_resource_flags(pdev, bar);
 | 
			
		||||
	if (res_flag & IORESOURCE_CACHEABLE)
 | 
			
		||||
		mvi->regs = ioremap(res_start, res_len);
 | 
			
		||||
	else
 | 
			
		||||
		mvi->regs = ioremap_nocache(res_start, res_len);
 | 
			
		||||
 | 
			
		||||
	if (!mvi->regs) {
 | 
			
		||||
		if (mvi->regs_ex && (res_flag_ex & IORESOURCE_MEM))
 | 
			
		||||
			iounmap(mvi->regs_ex);
 | 
			
		||||
		mvi->regs_ex = NULL;
 | 
			
		||||
		goto err_out;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
err_out:
 | 
			
		||||
	return -1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mvs_iounmap(void __iomem *regs)
 | 
			
		||||
{
 | 
			
		||||
	iounmap(regs);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static struct mvs_info *__devinit mvs_pci_alloc(struct pci_dev *pdev,
 | 
			
		||||
				const struct pci_device_id *ent,
 | 
			
		||||
				struct Scsi_Host *shost, unsigned int id)
 | 
			
		||||
{
 | 
			
		||||
	struct mvs_info *mvi;
 | 
			
		||||
	struct sas_ha_struct *sha = SHOST_TO_SAS_HA(shost);
 | 
			
		||||
 | 
			
		||||
	mvi = kzalloc(sizeof(*mvi) + MVS_SLOTS * sizeof(struct mvs_slot_info),
 | 
			
		||||
			GFP_KERNEL);
 | 
			
		||||
	if (!mvi)
 | 
			
		||||
		return NULL;
 | 
			
		||||
 | 
			
		||||
	mvi->pdev = pdev;
 | 
			
		||||
	mvi->dev = &pdev->dev;
 | 
			
		||||
	mvi->chip_id = ent->driver_data;
 | 
			
		||||
	mvi->chip = &mvs_chips[mvi->chip_id];
 | 
			
		||||
	INIT_LIST_HEAD(&mvi->wq_list);
 | 
			
		||||
	mvi->irq = pdev->irq;
 | 
			
		||||
 | 
			
		||||
	((struct mvs_prv_info *)sha->lldd_ha)->mvi[id] = mvi;
 | 
			
		||||
	((struct mvs_prv_info *)sha->lldd_ha)->n_phy = mvi->chip->n_phy;
 | 
			
		||||
 | 
			
		||||
	mvi->id = id;
 | 
			
		||||
	mvi->sas = sha;
 | 
			
		||||
	mvi->shost = shost;
 | 
			
		||||
#ifdef MVS_USE_TASKLET
 | 
			
		||||
	tasklet_init(&mv_tasklet, mvs_tasklet, (unsigned long)sha);
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
	if (MVS_CHIP_DISP->chip_ioremap(mvi))
 | 
			
		||||
		goto err_out;
 | 
			
		||||
	if (!mvs_alloc(mvi, shost))
 | 
			
		||||
		return mvi;
 | 
			
		||||
err_out:
 | 
			
		||||
	mvs_free(mvi);
 | 
			
		||||
	return NULL;
 | 
			
		||||
| 
						 | 
				
			
			@ -363,16 +415,111 @@ static int pci_go_64(struct pci_dev *pdev)
 | 
			
		|||
	return rc;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int __devinit mvs_prep_sas_ha_init(struct Scsi_Host *shost,
 | 
			
		||||
				const struct mvs_chip_info *chip_info)
 | 
			
		||||
{
 | 
			
		||||
	int phy_nr, port_nr; unsigned short core_nr;
 | 
			
		||||
	struct asd_sas_phy **arr_phy;
 | 
			
		||||
	struct asd_sas_port **arr_port;
 | 
			
		||||
	struct sas_ha_struct *sha = SHOST_TO_SAS_HA(shost);
 | 
			
		||||
 | 
			
		||||
	core_nr = chip_info->n_host;
 | 
			
		||||
	phy_nr  = core_nr * chip_info->n_phy;
 | 
			
		||||
	port_nr = phy_nr;
 | 
			
		||||
 | 
			
		||||
	memset(sha, 0x00, sizeof(struct sas_ha_struct));
 | 
			
		||||
	arr_phy  = kcalloc(phy_nr, sizeof(void *), GFP_KERNEL);
 | 
			
		||||
	arr_port = kcalloc(port_nr, sizeof(void *), GFP_KERNEL);
 | 
			
		||||
	if (!arr_phy || !arr_port)
 | 
			
		||||
		goto exit_free;
 | 
			
		||||
 | 
			
		||||
	sha->sas_phy = arr_phy;
 | 
			
		||||
	sha->sas_port = arr_port;
 | 
			
		||||
 | 
			
		||||
	sha->lldd_ha = kzalloc(sizeof(struct mvs_prv_info), GFP_KERNEL);
 | 
			
		||||
	if (!sha->lldd_ha)
 | 
			
		||||
		goto exit_free;
 | 
			
		||||
 | 
			
		||||
	((struct mvs_prv_info *)sha->lldd_ha)->n_host = core_nr;
 | 
			
		||||
 | 
			
		||||
	shost->transportt = mvs_stt;
 | 
			
		||||
	shost->max_id = 128;
 | 
			
		||||
	shost->max_lun = ~0;
 | 
			
		||||
	shost->max_channel = 1;
 | 
			
		||||
	shost->max_cmd_len = 16;
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
exit_free:
 | 
			
		||||
	kfree(arr_phy);
 | 
			
		||||
	kfree(arr_port);
 | 
			
		||||
	return -1;
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void  __devinit mvs_post_sas_ha_init(struct Scsi_Host *shost,
 | 
			
		||||
			const struct mvs_chip_info *chip_info)
 | 
			
		||||
{
 | 
			
		||||
	int can_queue, i = 0, j = 0;
 | 
			
		||||
	struct mvs_info *mvi = NULL;
 | 
			
		||||
	struct sas_ha_struct *sha = SHOST_TO_SAS_HA(shost);
 | 
			
		||||
	unsigned short nr_core = ((struct mvs_prv_info *)sha->lldd_ha)->n_host;
 | 
			
		||||
 | 
			
		||||
	for (j = 0; j < nr_core; j++) {
 | 
			
		||||
		mvi = ((struct mvs_prv_info *)sha->lldd_ha)->mvi[j];
 | 
			
		||||
		for (i = 0; i < chip_info->n_phy; i++) {
 | 
			
		||||
			sha->sas_phy[j * chip_info->n_phy  + i] =
 | 
			
		||||
				&mvi->phy[i].sas_phy;
 | 
			
		||||
			sha->sas_port[j * chip_info->n_phy + i] =
 | 
			
		||||
				&mvi->port[i].sas_port;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	sha->sas_ha_name = DRV_NAME;
 | 
			
		||||
	sha->dev = mvi->dev;
 | 
			
		||||
	sha->lldd_module = THIS_MODULE;
 | 
			
		||||
	sha->sas_addr = &mvi->sas_addr[0];
 | 
			
		||||
 | 
			
		||||
	sha->num_phys = nr_core * chip_info->n_phy;
 | 
			
		||||
 | 
			
		||||
	sha->lldd_max_execute_num = 1;
 | 
			
		||||
 | 
			
		||||
	if (mvi->flags & MVF_FLAG_SOC)
 | 
			
		||||
		can_queue = MVS_SOC_CAN_QUEUE;
 | 
			
		||||
	else
 | 
			
		||||
		can_queue = MVS_CAN_QUEUE;
 | 
			
		||||
 | 
			
		||||
	sha->lldd_queue_size = can_queue;
 | 
			
		||||
	shost->can_queue = can_queue;
 | 
			
		||||
	mvi->shost->cmd_per_lun = MVS_SLOTS/sha->num_phys;
 | 
			
		||||
	sha->core.shost = mvi->shost;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void mvs_init_sas_add(struct mvs_info *mvi)
 | 
			
		||||
{
 | 
			
		||||
	u8 i;
 | 
			
		||||
	for (i = 0; i < mvi->chip->n_phy; i++) {
 | 
			
		||||
		mvi->phy[i].dev_sas_addr = 0x5005043011ab0000ULL;
 | 
			
		||||
		mvi->phy[i].dev_sas_addr =
 | 
			
		||||
			cpu_to_be64((u64)(*(u64 *)&mvi->phy[i].dev_sas_addr));
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	memcpy(mvi->sas_addr, &mvi->phy[0].dev_sas_addr, SAS_ADDR_SIZE);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int __devinit mvs_pci_init(struct pci_dev *pdev,
 | 
			
		||||
				  const struct pci_device_id *ent)
 | 
			
		||||
{
 | 
			
		||||
	int rc;
 | 
			
		||||
	unsigned int rc, nhost = 0;
 | 
			
		||||
	struct mvs_info *mvi;
 | 
			
		||||
	irq_handler_t irq_handler = mvs_interrupt;
 | 
			
		||||
	struct Scsi_Host *shost = NULL;
 | 
			
		||||
	const struct mvs_chip_info *chip;
 | 
			
		||||
 | 
			
		||||
	dev_printk(KERN_INFO, &pdev->dev,
 | 
			
		||||
		"mvsas: driver version %s\n", DRV_VERSION);
 | 
			
		||||
	rc = pci_enable_device(pdev);
 | 
			
		||||
	if (rc)
 | 
			
		||||
		return rc;
 | 
			
		||||
		goto err_out_enable;
 | 
			
		||||
 | 
			
		||||
	pci_set_master(pdev);
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -384,84 +531,110 @@ static int __devinit mvs_pci_init(struct pci_dev *pdev,
 | 
			
		|||
	if (rc)
 | 
			
		||||
		goto err_out_regions;
 | 
			
		||||
 | 
			
		||||
	mvi = mvs_alloc(pdev, ent);
 | 
			
		||||
	if (!mvi) {
 | 
			
		||||
	shost = scsi_host_alloc(&mvs_sht, sizeof(void *));
 | 
			
		||||
	if (!shost) {
 | 
			
		||||
		rc = -ENOMEM;
 | 
			
		||||
		goto err_out_regions;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	rc = mvs_hw_init(mvi);
 | 
			
		||||
	if (rc)
 | 
			
		||||
		goto err_out_mvi;
 | 
			
		||||
 | 
			
		||||
#ifndef MVS_DISABLE_MSI
 | 
			
		||||
	if (!pci_enable_msi(pdev)) {
 | 
			
		||||
		u32 tmp;
 | 
			
		||||
		void __iomem *regs = mvi->regs;
 | 
			
		||||
		mvi->flags |= MVF_MSI;
 | 
			
		||||
		irq_handler = mvs_msi_interrupt;
 | 
			
		||||
		tmp = mr32(PCS);
 | 
			
		||||
		mw32(PCS, tmp | PCS_SELF_CLEAR);
 | 
			
		||||
	chip = &mvs_chips[ent->driver_data];
 | 
			
		||||
	SHOST_TO_SAS_HA(shost) =
 | 
			
		||||
		kcalloc(1, sizeof(struct sas_ha_struct), GFP_KERNEL);
 | 
			
		||||
	if (!SHOST_TO_SAS_HA(shost)) {
 | 
			
		||||
		kfree(shost);
 | 
			
		||||
		rc = -ENOMEM;
 | 
			
		||||
		goto err_out_regions;
 | 
			
		||||
	}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
	rc = request_irq(pdev->irq, irq_handler, IRQF_SHARED, DRV_NAME, mvi);
 | 
			
		||||
	if (rc)
 | 
			
		||||
		goto err_out_msi;
 | 
			
		||||
	rc = mvs_prep_sas_ha_init(shost, chip);
 | 
			
		||||
	if (rc) {
 | 
			
		||||
		kfree(shost);
 | 
			
		||||
		rc = -ENOMEM;
 | 
			
		||||
		goto err_out_regions;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	rc = scsi_add_host(mvi->shost, &pdev->dev);
 | 
			
		||||
	if (rc)
 | 
			
		||||
		goto err_out_irq;
 | 
			
		||||
	pci_set_drvdata(pdev, SHOST_TO_SAS_HA(shost));
 | 
			
		||||
 | 
			
		||||
	rc = sas_register_ha(&mvi->sas);
 | 
			
		||||
	do {
 | 
			
		||||
		mvi = mvs_pci_alloc(pdev, ent, shost, nhost);
 | 
			
		||||
		if (!mvi) {
 | 
			
		||||
			rc = -ENOMEM;
 | 
			
		||||
			goto err_out_regions;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		mvs_init_sas_add(mvi);
 | 
			
		||||
 | 
			
		||||
		mvi->instance = nhost;
 | 
			
		||||
		rc = MVS_CHIP_DISP->chip_init(mvi);
 | 
			
		||||
		if (rc) {
 | 
			
		||||
			mvs_free(mvi);
 | 
			
		||||
			goto err_out_regions;
 | 
			
		||||
		}
 | 
			
		||||
		nhost++;
 | 
			
		||||
	} while (nhost < chip->n_host);
 | 
			
		||||
 | 
			
		||||
	mvs_post_sas_ha_init(shost, chip);
 | 
			
		||||
 | 
			
		||||
	rc = scsi_add_host(shost, &pdev->dev);
 | 
			
		||||
	if (rc)
 | 
			
		||||
		goto err_out_shost;
 | 
			
		||||
 | 
			
		||||
	pci_set_drvdata(pdev, mvi);
 | 
			
		||||
	rc = sas_register_ha(SHOST_TO_SAS_HA(shost));
 | 
			
		||||
	if (rc)
 | 
			
		||||
		goto err_out_shost;
 | 
			
		||||
	rc = request_irq(pdev->irq, irq_handler, IRQF_SHARED,
 | 
			
		||||
		DRV_NAME, SHOST_TO_SAS_HA(shost));
 | 
			
		||||
	if (rc)
 | 
			
		||||
		goto err_not_sas;
 | 
			
		||||
 | 
			
		||||
	mvs_print_info(mvi);
 | 
			
		||||
 | 
			
		||||
	mvs_hba_interrupt_enable(mvi);
 | 
			
		||||
	MVS_CHIP_DISP->interrupt_enable(mvi);
 | 
			
		||||
 | 
			
		||||
	scsi_scan_host(mvi->shost);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
 | 
			
		||||
err_not_sas:
 | 
			
		||||
	sas_unregister_ha(SHOST_TO_SAS_HA(shost));
 | 
			
		||||
err_out_shost:
 | 
			
		||||
	scsi_remove_host(mvi->shost);
 | 
			
		||||
err_out_irq:
 | 
			
		||||
	free_irq(pdev->irq, mvi);
 | 
			
		||||
err_out_msi:
 | 
			
		||||
	if (mvi->flags |= MVF_MSI)
 | 
			
		||||
		pci_disable_msi(pdev);
 | 
			
		||||
err_out_mvi:
 | 
			
		||||
	mvs_free(mvi);
 | 
			
		||||
err_out_regions:
 | 
			
		||||
	pci_release_regions(pdev);
 | 
			
		||||
err_out_disable:
 | 
			
		||||
	pci_disable_device(pdev);
 | 
			
		||||
err_out_enable:
 | 
			
		||||
	return rc;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void __devexit mvs_pci_remove(struct pci_dev *pdev)
 | 
			
		||||
{
 | 
			
		||||
	struct mvs_info *mvi = pci_get_drvdata(pdev);
 | 
			
		||||
	unsigned short core_nr, i = 0;
 | 
			
		||||
	struct sas_ha_struct *sha = pci_get_drvdata(pdev);
 | 
			
		||||
	struct mvs_info *mvi = NULL;
 | 
			
		||||
 | 
			
		||||
	core_nr = ((struct mvs_prv_info *)sha->lldd_ha)->n_host;
 | 
			
		||||
	mvi = ((struct mvs_prv_info *)sha->lldd_ha)->mvi[0];
 | 
			
		||||
 | 
			
		||||
#ifdef MVS_USE_TASKLET
 | 
			
		||||
	tasklet_kill(&mv_tasklet);
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
	pci_set_drvdata(pdev, NULL);
 | 
			
		||||
	sas_unregister_ha(sha);
 | 
			
		||||
	sas_remove_host(mvi->shost);
 | 
			
		||||
	scsi_remove_host(mvi->shost);
 | 
			
		||||
 | 
			
		||||
	if (mvi) {
 | 
			
		||||
		sas_unregister_ha(&mvi->sas);
 | 
			
		||||
		mvs_hba_interrupt_disable(mvi);
 | 
			
		||||
		sas_remove_host(mvi->shost);
 | 
			
		||||
		scsi_remove_host(mvi->shost);
 | 
			
		||||
 | 
			
		||||
		free_irq(pdev->irq, mvi);
 | 
			
		||||
		if (mvi->flags & MVF_MSI)
 | 
			
		||||
			pci_disable_msi(pdev);
 | 
			
		||||
	MVS_CHIP_DISP->interrupt_disable(mvi);
 | 
			
		||||
	free_irq(mvi->irq, sha);
 | 
			
		||||
	for (i = 0; i < core_nr; i++) {
 | 
			
		||||
		mvi = ((struct mvs_prv_info *)sha->lldd_ha)->mvi[i];
 | 
			
		||||
		mvs_free(mvi);
 | 
			
		||||
		pci_release_regions(pdev);
 | 
			
		||||
	}
 | 
			
		||||
	kfree(sha->sas_phy);
 | 
			
		||||
	kfree(sha->sas_port);
 | 
			
		||||
	kfree(sha);
 | 
			
		||||
	pci_release_regions(pdev);
 | 
			
		||||
	pci_disable_device(pdev);
 | 
			
		||||
	return;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static struct pci_device_id __devinitdata mvs_pci_table[] = {
 | 
			
		||||
| 
						 | 
				
			
			@ -474,10 +647,12 @@ static struct pci_device_id __devinitdata mvs_pci_table[] = {
 | 
			
		|||
		.subdevice	= 0x6480,
 | 
			
		||||
		.class		= 0,
 | 
			
		||||
		.class_mask	= 0,
 | 
			
		||||
		.driver_data	= chip_6480,
 | 
			
		||||
		.driver_data	= chip_6485,
 | 
			
		||||
	},
 | 
			
		||||
	{ PCI_VDEVICE(MARVELL, 0x6440), chip_6440 },
 | 
			
		||||
	{ PCI_VDEVICE(MARVELL, 0x6480), chip_6480 },
 | 
			
		||||
	{ PCI_VDEVICE(MARVELL, 0x6485), chip_6485 },
 | 
			
		||||
	{ PCI_VDEVICE(MARVELL, 0x9480), chip_9480 },
 | 
			
		||||
	{ PCI_VDEVICE(MARVELL, 0x9180), chip_9180 },
 | 
			
		||||
 | 
			
		||||
	{ }	/* terminate list */
 | 
			
		||||
};
 | 
			
		||||
| 
						 | 
				
			
			@ -489,15 +664,17 @@ static struct pci_driver mvs_pci_driver = {
 | 
			
		|||
	.remove		= __devexit_p(mvs_pci_remove),
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/* task handler */
 | 
			
		||||
struct task_struct *mvs_th;
 | 
			
		||||
static int __init mvs_init(void)
 | 
			
		||||
{
 | 
			
		||||
	int rc;
 | 
			
		||||
 | 
			
		||||
	mvs_stt = sas_domain_attach_transport(&mvs_transport_ops);
 | 
			
		||||
	if (!mvs_stt)
 | 
			
		||||
		return -ENOMEM;
 | 
			
		||||
 | 
			
		||||
	rc = pci_register_driver(&mvs_pci_driver);
 | 
			
		||||
 | 
			
		||||
	if (rc)
 | 
			
		||||
		goto err_out;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -521,4 +698,6 @@ MODULE_AUTHOR("Jeff Garzik <jgarzik@pobox.com>");
 | 
			
		|||
MODULE_DESCRIPTION("Marvell 88SE6440 SAS/SATA controller driver");
 | 
			
		||||
MODULE_VERSION(DRV_VERSION);
 | 
			
		||||
MODULE_LICENSE("GPL");
 | 
			
		||||
#ifdef CONFIG_PCI
 | 
			
		||||
MODULE_DEVICE_TABLE(pci, mvs_pci_table);
 | 
			
		||||
#endif
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							| 
						 | 
				
			
			@ -1,25 +1,26 @@
 | 
			
		|||
/*
 | 
			
		||||
	mv_sas.h - Marvell 88SE6440 SAS/SATA support
 | 
			
		||||
 | 
			
		||||
	Copyright 2007 Red Hat, Inc.
 | 
			
		||||
	Copyright 2008 Marvell. <kewei@marvell.com>
 | 
			
		||||
 | 
			
		||||
	This program is free software; you can redistribute it and/or
 | 
			
		||||
	modify it under the terms of the GNU General Public License as
 | 
			
		||||
	published by the Free Software Foundation; either version 2,
 | 
			
		||||
	or (at your option) any later version.
 | 
			
		||||
 | 
			
		||||
	This program is distributed in the hope that 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.
 | 
			
		||||
 | 
			
		||||
	You should have received a copy of the GNU General Public
 | 
			
		||||
	License along with this program; see the file COPYING.	If not,
 | 
			
		||||
	write to the Free Software Foundation, 675 Mass Ave, Cambridge,
 | 
			
		||||
	MA 02139, USA.
 | 
			
		||||
 | 
			
		||||
 */
 | 
			
		||||
 * Marvell 88SE64xx/88SE94xx main function head file
 | 
			
		||||
 *
 | 
			
		||||
 * Copyright 2007 Red Hat, Inc.
 | 
			
		||||
 * Copyright 2008 Marvell. <kewei@marvell.com>
 | 
			
		||||
 *
 | 
			
		||||
 * This file is licensed under GPLv2.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is free software; you can redistribute it and/or
 | 
			
		||||
 * modify it under the terms of the GNU General Public License as
 | 
			
		||||
 * published by the Free Software Foundation; version 2 of the
 | 
			
		||||
 * License.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is distributed in the hope that 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.
 | 
			
		||||
 *
 | 
			
		||||
 * You should have received a copy of the GNU General Public License
 | 
			
		||||
 * along with this program; if not, write to the Free Software
 | 
			
		||||
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
 | 
			
		||||
 * USA
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
#ifndef _MV_SAS_H_
 | 
			
		||||
#define _MV_SAS_H_
 | 
			
		||||
| 
						 | 
				
			
			@ -42,26 +43,145 @@
 | 
			
		|||
#include <linux/version.h>
 | 
			
		||||
#include "mv_defs.h"
 | 
			
		||||
 | 
			
		||||
#define DRV_NAME	"mvsas"
 | 
			
		||||
#define DRV_VERSION	"0.5.2"
 | 
			
		||||
#define _MV_DUMP	0
 | 
			
		||||
#define MVS_DISABLE_NVRAM
 | 
			
		||||
#define MVS_DISABLE_MSI
 | 
			
		||||
 | 
			
		||||
#define DRV_NAME		"mvsas"
 | 
			
		||||
#define DRV_VERSION		"0.8.2"
 | 
			
		||||
#define _MV_DUMP		0
 | 
			
		||||
#define MVS_ID_NOT_MAPPED	0x7f
 | 
			
		||||
#define MVS_CHIP_SLOT_SZ	(1U << mvi->chip->slot_width)
 | 
			
		||||
/* #define DISABLE_HOTPLUG_DMA_FIX */
 | 
			
		||||
#define MAX_EXP_RUNNING_REQ	2
 | 
			
		||||
#define WIDE_PORT_MAX_PHY		4
 | 
			
		||||
#define	MV_DISABLE_NCQ	0
 | 
			
		||||
#define mv_printk(fmt, arg ...)	\
 | 
			
		||||
	printk(KERN_DEBUG"%s %d:" fmt, __FILE__, __LINE__, ## arg)
 | 
			
		||||
#ifdef MV_DEBUG
 | 
			
		||||
#define mv_dprintk(format, arg...)	\
 | 
			
		||||
	printk(KERN_DEBUG"%s %d:" format, __FILE__, __LINE__, ## arg)
 | 
			
		||||
#else
 | 
			
		||||
#define mv_dprintk(format, arg...)
 | 
			
		||||
#endif
 | 
			
		||||
#define MV_MAX_U32			0xffffffff
 | 
			
		||||
 | 
			
		||||
#define for_each_phy(__lseq_mask, __mc, __lseq, __rest)			\
 | 
			
		||||
	for ((__mc) = (__lseq_mask), (__lseq) = 0;			\
 | 
			
		||||
					(__mc) != 0 && __rest;		\
 | 
			
		||||
extern struct mvs_tgt_initiator mvs_tgt;
 | 
			
		||||
extern struct mvs_info *tgt_mvi;
 | 
			
		||||
extern const struct mvs_dispatch mvs_64xx_dispatch;
 | 
			
		||||
extern const struct mvs_dispatch mvs_94xx_dispatch;
 | 
			
		||||
 | 
			
		||||
#define DEV_IS_EXPANDER(type)	\
 | 
			
		||||
	((type == EDGE_DEV) || (type == FANOUT_DEV))
 | 
			
		||||
 | 
			
		||||
#define bit(n) ((u32)1 << n)
 | 
			
		||||
 | 
			
		||||
#define for_each_phy(__lseq_mask, __mc, __lseq)			\
 | 
			
		||||
	for ((__mc) = (__lseq_mask), (__lseq) = 0;		\
 | 
			
		||||
					(__mc) != 0 ;		\
 | 
			
		||||
					(++__lseq), (__mc) >>= 1)
 | 
			
		||||
 | 
			
		||||
struct mvs_chip_info {
 | 
			
		||||
	u32		n_phy;
 | 
			
		||||
	u32		srs_sz;
 | 
			
		||||
	u32		slot_width;
 | 
			
		||||
#define MV_INIT_DELAYED_WORK(w, f, d)	INIT_DELAYED_WORK(w, f)
 | 
			
		||||
#define UNASSOC_D2H_FIS(id)		\
 | 
			
		||||
	((void *) mvi->rx_fis + 0x100 * id)
 | 
			
		||||
#define SATA_RECEIVED_FIS_LIST(reg_set)	\
 | 
			
		||||
	((void *) mvi->rx_fis + mvi->chip->fis_offs + 0x100 * reg_set)
 | 
			
		||||
#define SATA_RECEIVED_SDB_FIS(reg_set)	\
 | 
			
		||||
	(SATA_RECEIVED_FIS_LIST(reg_set) + 0x58)
 | 
			
		||||
#define SATA_RECEIVED_D2H_FIS(reg_set)	\
 | 
			
		||||
	(SATA_RECEIVED_FIS_LIST(reg_set) + 0x40)
 | 
			
		||||
#define SATA_RECEIVED_PIO_FIS(reg_set)	\
 | 
			
		||||
	(SATA_RECEIVED_FIS_LIST(reg_set) + 0x20)
 | 
			
		||||
#define SATA_RECEIVED_DMA_FIS(reg_set)	\
 | 
			
		||||
	(SATA_RECEIVED_FIS_LIST(reg_set) + 0x00)
 | 
			
		||||
 | 
			
		||||
enum dev_status {
 | 
			
		||||
	MVS_DEV_NORMAL = 0x0,
 | 
			
		||||
	MVS_DEV_EH	= 0x1,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
struct mvs_info;
 | 
			
		||||
 | 
			
		||||
struct mvs_dispatch {
 | 
			
		||||
	char *name;
 | 
			
		||||
	int (*chip_init)(struct mvs_info *mvi);
 | 
			
		||||
	int (*spi_init)(struct mvs_info *mvi);
 | 
			
		||||
	int (*chip_ioremap)(struct mvs_info *mvi);
 | 
			
		||||
	void (*chip_iounmap)(struct mvs_info *mvi);
 | 
			
		||||
	irqreturn_t (*isr)(struct mvs_info *mvi, int irq, u32 stat);
 | 
			
		||||
	u32 (*isr_status)(struct mvs_info *mvi, int irq);
 | 
			
		||||
	void (*interrupt_enable)(struct mvs_info *mvi);
 | 
			
		||||
	void (*interrupt_disable)(struct mvs_info *mvi);
 | 
			
		||||
 | 
			
		||||
	u32 (*read_phy_ctl)(struct mvs_info *mvi, u32 port);
 | 
			
		||||
	void (*write_phy_ctl)(struct mvs_info *mvi, u32 port, u32 val);
 | 
			
		||||
 | 
			
		||||
	u32 (*read_port_cfg_data)(struct mvs_info *mvi, u32 port);
 | 
			
		||||
	void (*write_port_cfg_data)(struct mvs_info *mvi, u32 port, u32 val);
 | 
			
		||||
	void (*write_port_cfg_addr)(struct mvs_info *mvi, u32 port, u32 addr);
 | 
			
		||||
 | 
			
		||||
	u32 (*read_port_vsr_data)(struct mvs_info *mvi, u32 port);
 | 
			
		||||
	void (*write_port_vsr_data)(struct mvs_info *mvi, u32 port, u32 val);
 | 
			
		||||
	void (*write_port_vsr_addr)(struct mvs_info *mvi, u32 port, u32 addr);
 | 
			
		||||
 | 
			
		||||
	u32 (*read_port_irq_stat)(struct mvs_info *mvi, u32 port);
 | 
			
		||||
	void (*write_port_irq_stat)(struct mvs_info *mvi, u32 port, u32 val);
 | 
			
		||||
 | 
			
		||||
	u32 (*read_port_irq_mask)(struct mvs_info *mvi, u32 port);
 | 
			
		||||
	void (*write_port_irq_mask)(struct mvs_info *mvi, u32 port, u32 val);
 | 
			
		||||
 | 
			
		||||
	void (*get_sas_addr)(void *buf, u32 buflen);
 | 
			
		||||
	void (*command_active)(struct mvs_info *mvi, u32 slot_idx);
 | 
			
		||||
	void (*issue_stop)(struct mvs_info *mvi, enum mvs_port_type type,
 | 
			
		||||
				u32 tfs);
 | 
			
		||||
	void (*start_delivery)(struct mvs_info *mvi, u32 tx);
 | 
			
		||||
	u32 (*rx_update)(struct mvs_info *mvi);
 | 
			
		||||
	void (*int_full)(struct mvs_info *mvi);
 | 
			
		||||
	u8 (*assign_reg_set)(struct mvs_info *mvi, u8 *tfs);
 | 
			
		||||
	void (*free_reg_set)(struct mvs_info *mvi, u8 *tfs);
 | 
			
		||||
	u32 (*prd_size)(void);
 | 
			
		||||
	u32 (*prd_count)(void);
 | 
			
		||||
	void (*make_prd)(struct scatterlist *scatter, int nr, void *prd);
 | 
			
		||||
	void (*detect_porttype)(struct mvs_info *mvi, int i);
 | 
			
		||||
	int (*oob_done)(struct mvs_info *mvi, int i);
 | 
			
		||||
	void (*fix_phy_info)(struct mvs_info *mvi, int i,
 | 
			
		||||
				struct sas_identify_frame *id);
 | 
			
		||||
	void (*phy_work_around)(struct mvs_info *mvi, int i);
 | 
			
		||||
	void (*phy_set_link_rate)(struct mvs_info *mvi, u32 phy_id,
 | 
			
		||||
				struct sas_phy_linkrates *rates);
 | 
			
		||||
	u32 (*phy_max_link_rate)(void);
 | 
			
		||||
	void (*phy_disable)(struct mvs_info *mvi, u32 phy_id);
 | 
			
		||||
	void (*phy_enable)(struct mvs_info *mvi, u32 phy_id);
 | 
			
		||||
	void (*phy_reset)(struct mvs_info *mvi, u32 phy_id, int hard);
 | 
			
		||||
	void (*stp_reset)(struct mvs_info *mvi, u32 phy_id);
 | 
			
		||||
	void (*clear_active_cmds)(struct mvs_info *mvi);
 | 
			
		||||
	u32 (*spi_read_data)(struct mvs_info *mvi);
 | 
			
		||||
	void (*spi_write_data)(struct mvs_info *mvi, u32 data);
 | 
			
		||||
	int (*spi_buildcmd)(struct mvs_info *mvi,
 | 
			
		||||
						u32      *dwCmd,
 | 
			
		||||
						u8       cmd,
 | 
			
		||||
						u8       read,
 | 
			
		||||
						u8       length,
 | 
			
		||||
						u32      addr
 | 
			
		||||
						);
 | 
			
		||||
	int (*spi_issuecmd)(struct mvs_info *mvi, u32 cmd);
 | 
			
		||||
	int (*spi_waitdataready)(struct mvs_info *mvi, u32 timeout);
 | 
			
		||||
#ifndef DISABLE_HOTPLUG_DMA_FIX
 | 
			
		||||
	void (*dma_fix)(dma_addr_t buf_dma, int buf_len, int from, void *prd);
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct mvs_chip_info {
 | 
			
		||||
	u32 		n_host;
 | 
			
		||||
	u32 		n_phy;
 | 
			
		||||
	u32 		fis_offs;
 | 
			
		||||
	u32 		fis_count;
 | 
			
		||||
	u32 		srs_sz;
 | 
			
		||||
	u32 		slot_width;
 | 
			
		||||
	const struct mvs_dispatch *dispatch;
 | 
			
		||||
};
 | 
			
		||||
#define MVS_CHIP_SLOT_SZ	(1U << mvi->chip->slot_width)
 | 
			
		||||
#define MVS_RX_FISL_SZ		\
 | 
			
		||||
	(mvi->chip->fis_offs + (mvi->chip->fis_count * 0x100))
 | 
			
		||||
#define MVS_CHIP_DISP		(mvi->chip->dispatch)
 | 
			
		||||
 | 
			
		||||
struct mvs_err_info {
 | 
			
		||||
	__le32			flags;
 | 
			
		||||
	__le32			flags2;
 | 
			
		||||
| 
						 | 
				
			
			@ -72,7 +192,7 @@ struct mvs_cmd_hdr {
 | 
			
		|||
	__le32			lens;	/* cmd, max resp frame len */
 | 
			
		||||
	__le32			tags;	/* targ port xfer tag; tag */
 | 
			
		||||
	__le32			data_len;	/* data xfer len */
 | 
			
		||||
	__le64			cmd_tbl;	/* command table address */
 | 
			
		||||
	__le64			cmd_tbl;  	/* command table address */
 | 
			
		||||
	__le64			open_frame;	/* open addr frame address */
 | 
			
		||||
	__le64			status_buf;	/* status buffer address */
 | 
			
		||||
	__le64			prd_tbl;		/* PRD tbl address */
 | 
			
		||||
| 
						 | 
				
			
			@ -82,16 +202,17 @@ struct mvs_cmd_hdr {
 | 
			
		|||
struct mvs_port {
 | 
			
		||||
	struct asd_sas_port	sas_port;
 | 
			
		||||
	u8			port_attached;
 | 
			
		||||
	u8			taskfileset;
 | 
			
		||||
	u8			wide_port_phymap;
 | 
			
		||||
	struct list_head	list;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct mvs_phy {
 | 
			
		||||
	struct mvs_info 		*mvi;
 | 
			
		||||
	struct mvs_port		*port;
 | 
			
		||||
	struct asd_sas_phy	sas_phy;
 | 
			
		||||
	struct sas_identify	identify;
 | 
			
		||||
	struct scsi_device	*sdev;
 | 
			
		||||
	struct timer_list timer;
 | 
			
		||||
	u64		dev_sas_addr;
 | 
			
		||||
	u64		att_dev_sas_addr;
 | 
			
		||||
	u32		att_dev_info;
 | 
			
		||||
| 
						 | 
				
			
			@ -102,15 +223,34 @@ struct mvs_phy {
 | 
			
		|||
	u32		frame_rcvd_size;
 | 
			
		||||
	u8		frame_rcvd[32];
 | 
			
		||||
	u8		phy_attached;
 | 
			
		||||
	u8		phy_mode;
 | 
			
		||||
	u8		reserved[2];
 | 
			
		||||
	u32		phy_event;
 | 
			
		||||
	enum sas_linkrate	minimum_linkrate;
 | 
			
		||||
	enum sas_linkrate	maximum_linkrate;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct mvs_device {
 | 
			
		||||
	enum sas_dev_type dev_type;
 | 
			
		||||
	struct domain_device *sas_device;
 | 
			
		||||
	u32 attached_phy;
 | 
			
		||||
	u32 device_id;
 | 
			
		||||
	u32 runing_req;
 | 
			
		||||
	u8 taskfileset;
 | 
			
		||||
	u8 dev_status;
 | 
			
		||||
	u16 reserved;
 | 
			
		||||
	struct list_head		dev_entry;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct mvs_slot_info {
 | 
			
		||||
	struct list_head list;
 | 
			
		||||
	struct sas_task *task;
 | 
			
		||||
	struct list_head entry;
 | 
			
		||||
	union {
 | 
			
		||||
		struct sas_task *task;
 | 
			
		||||
		void *tdata;
 | 
			
		||||
	};
 | 
			
		||||
	u32 n_elem;
 | 
			
		||||
	u32 tx;
 | 
			
		||||
	u32 slot_tag;
 | 
			
		||||
 | 
			
		||||
	/* DMA buffer for storing cmd tbl, open addr frame, status buffer,
 | 
			
		||||
	 * and PRD table
 | 
			
		||||
| 
						 | 
				
			
			@ -120,9 +260,10 @@ struct mvs_slot_info {
 | 
			
		|||
#if _MV_DUMP
 | 
			
		||||
	u32 cmd_size;
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
	void *response;
 | 
			
		||||
	struct mvs_port *port;
 | 
			
		||||
	struct mvs_device	*device;
 | 
			
		||||
	void *open_frame;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct mvs_info {
 | 
			
		||||
| 
						 | 
				
			
			@ -133,17 +274,17 @@ struct mvs_info {
 | 
			
		|||
 | 
			
		||||
	/* our device */
 | 
			
		||||
	struct pci_dev *pdev;
 | 
			
		||||
	struct device *dev;
 | 
			
		||||
 | 
			
		||||
	/* enhanced mode registers */
 | 
			
		||||
	void __iomem *regs;
 | 
			
		||||
 | 
			
		||||
	/* peripheral registers */
 | 
			
		||||
	void __iomem *peri_regs;
 | 
			
		||||
 | 
			
		||||
	/* peripheral or soc registers */
 | 
			
		||||
	void __iomem *regs_ex;
 | 
			
		||||
	u8 sas_addr[SAS_ADDR_SIZE];
 | 
			
		||||
 | 
			
		||||
	/* SCSI/SAS glue */
 | 
			
		||||
	struct sas_ha_struct sas;
 | 
			
		||||
	struct sas_ha_struct *sas;
 | 
			
		||||
	struct Scsi_Host *shost;
 | 
			
		||||
 | 
			
		||||
	/* TX (delivery) DMA ring */
 | 
			
		||||
| 
						 | 
				
			
			@ -154,7 +295,7 @@ struct mvs_info {
 | 
			
		|||
	u32 tx_prod;
 | 
			
		||||
 | 
			
		||||
	/* RX (completion) DMA ring */
 | 
			
		||||
	__le32 *rx;
 | 
			
		||||
	__le32	*rx;
 | 
			
		||||
	dma_addr_t rx_dma;
 | 
			
		||||
 | 
			
		||||
	/* RX consumer idx */
 | 
			
		||||
| 
						 | 
				
			
			@ -168,38 +309,98 @@ struct mvs_info {
 | 
			
		|||
	struct mvs_cmd_hdr *slot;
 | 
			
		||||
	dma_addr_t slot_dma;
 | 
			
		||||
 | 
			
		||||
	u32 chip_id;
 | 
			
		||||
	const struct mvs_chip_info *chip;
 | 
			
		||||
 | 
			
		||||
	u8 tags[MVS_SLOTS];
 | 
			
		||||
	struct mvs_slot_info slot_info[MVS_SLOTS];
 | 
			
		||||
				/* further per-slot information */
 | 
			
		||||
	int tags_num;
 | 
			
		||||
	u8 tags[MVS_SLOTS >> 3];
 | 
			
		||||
 | 
			
		||||
	/* further per-slot information */
 | 
			
		||||
	struct mvs_phy phy[MVS_MAX_PHYS];
 | 
			
		||||
	struct mvs_port port[MVS_MAX_PHYS];
 | 
			
		||||
#ifdef MVS_USE_TASKLET
 | 
			
		||||
	struct tasklet_struct tasklet;
 | 
			
		||||
	u32 irq;
 | 
			
		||||
	u32 exp_req;
 | 
			
		||||
	u32 id;
 | 
			
		||||
	u64 sata_reg_set;
 | 
			
		||||
	struct list_head *hba_list;
 | 
			
		||||
	struct list_head soc_entry;
 | 
			
		||||
	struct list_head wq_list;
 | 
			
		||||
	unsigned long instance;
 | 
			
		||||
	u16 flashid;
 | 
			
		||||
	u32 flashsize;
 | 
			
		||||
	u32 flashsectSize;
 | 
			
		||||
 | 
			
		||||
	void *addon;
 | 
			
		||||
	struct mvs_device	devices[MVS_MAX_DEVICES];
 | 
			
		||||
#ifndef DISABLE_HOTPLUG_DMA_FIX
 | 
			
		||||
	void *bulk_buffer;
 | 
			
		||||
	dma_addr_t bulk_buffer_dma;
 | 
			
		||||
#define TRASH_BUCKET_SIZE    	0x20000
 | 
			
		||||
#endif
 | 
			
		||||
	struct mvs_slot_info slot_info[0];
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct mvs_prv_info{
 | 
			
		||||
	u8 n_host;
 | 
			
		||||
	u8 n_phy;
 | 
			
		||||
	u16 reserve;
 | 
			
		||||
	struct mvs_info *mvi[2];
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct mvs_wq {
 | 
			
		||||
	struct delayed_work work_q;
 | 
			
		||||
	struct mvs_info *mvi;
 | 
			
		||||
	void *data;
 | 
			
		||||
	int handler;
 | 
			
		||||
	struct list_head entry;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct mvs_task_exec_info {
 | 
			
		||||
	struct sas_task *task;
 | 
			
		||||
	struct mvs_cmd_hdr *hdr;
 | 
			
		||||
	struct mvs_port *port;
 | 
			
		||||
	u32 tag;
 | 
			
		||||
	int n_elem;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/******************** function prototype *********************/
 | 
			
		||||
void mvs_get_sas_addr(void *buf, u32 buflen);
 | 
			
		||||
void mvs_tag_clear(struct mvs_info *mvi, u32 tag);
 | 
			
		||||
void mvs_tag_free(struct mvs_info *mvi, u32 tag);
 | 
			
		||||
void mvs_tag_set(struct mvs_info *mvi, unsigned int tag);
 | 
			
		||||
int mvs_tag_alloc(struct mvs_info *mvi, u32 *tag_out);
 | 
			
		||||
void mvs_tag_init(struct mvs_info *mvi);
 | 
			
		||||
void mvs_iounmap(void __iomem *regs);
 | 
			
		||||
int mvs_ioremap(struct mvs_info *mvi, int bar, int bar_ex);
 | 
			
		||||
void mvs_phys_reset(struct mvs_info *mvi, u32 phy_mask, int hard);
 | 
			
		||||
int mvs_phy_control(struct asd_sas_phy *sas_phy, enum phy_func func,
 | 
			
		||||
			void *funcdata);
 | 
			
		||||
void __devinit mvs_set_sas_addr(struct mvs_info *mvi, int port_id,
 | 
			
		||||
				u32 off_lo, u32 off_hi, u64 sas_addr);
 | 
			
		||||
int mvs_slave_alloc(struct scsi_device *scsi_dev);
 | 
			
		||||
int mvs_slave_configure(struct scsi_device *sdev);
 | 
			
		||||
void mvs_scan_start(struct Scsi_Host *shost);
 | 
			
		||||
int mvs_scan_finished(struct Scsi_Host *shost, unsigned long time);
 | 
			
		||||
int mvs_task_exec(struct sas_task *task, const int num, gfp_t gfp_flags);
 | 
			
		||||
int mvs_task_abort(struct sas_task *task);
 | 
			
		||||
int mvs_queue_command(struct sas_task *task, const int num,
 | 
			
		||||
			gfp_t gfp_flags);
 | 
			
		||||
int mvs_abort_task(struct sas_task *task);
 | 
			
		||||
int mvs_abort_task_set(struct domain_device *dev, u8 *lun);
 | 
			
		||||
int mvs_clear_aca(struct domain_device *dev, u8 *lun);
 | 
			
		||||
int mvs_clear_task_set(struct domain_device *dev, u8 * lun);
 | 
			
		||||
void mvs_port_formed(struct asd_sas_phy *sas_phy);
 | 
			
		||||
void mvs_port_deformed(struct asd_sas_phy *sas_phy);
 | 
			
		||||
int mvs_dev_found(struct domain_device *dev);
 | 
			
		||||
void mvs_dev_gone(struct domain_device *dev);
 | 
			
		||||
int mvs_lu_reset(struct domain_device *dev, u8 *lun);
 | 
			
		||||
int mvs_slot_complete(struct mvs_info *mvi, u32 rx_desc, u32 flags);
 | 
			
		||||
int mvs_I_T_nexus_reset(struct domain_device *dev);
 | 
			
		||||
void mvs_int_full(struct mvs_info *mvi);
 | 
			
		||||
void mvs_tag_init(struct mvs_info *mvi);
 | 
			
		||||
int mvs_nvram_read(struct mvs_info *mvi, u32 addr, void *buf, u32 buflen);
 | 
			
		||||
int __devinit mvs_hw_init(struct mvs_info *mvi);
 | 
			
		||||
void __devinit mvs_print_info(struct mvs_info *mvi);
 | 
			
		||||
void mvs_hba_interrupt_enable(struct mvs_info *mvi);
 | 
			
		||||
void mvs_hba_interrupt_disable(struct mvs_info *mvi);
 | 
			
		||||
void mvs_detect_porttype(struct mvs_info *mvi, int i);
 | 
			
		||||
u8 mvs_assign_reg_set(struct mvs_info *mvi, struct mvs_port *port);
 | 
			
		||||
void mvs_enable_xmt(struct mvs_info *mvi, int PhyId);
 | 
			
		||||
void __devinit mvs_phy_hacks(struct mvs_info *mvi);
 | 
			
		||||
void mvs_free_reg_set(struct mvs_info *mvi, struct mvs_port *port);
 | 
			
		||||
 | 
			
		||||
int mvs_query_task(struct sas_task *task);
 | 
			
		||||
void mvs_release_task(struct mvs_info *mvi, int phy_no,
 | 
			
		||||
			struct domain_device *dev);
 | 
			
		||||
void mvs_int_port(struct mvs_info *mvi, int phy_no, u32 events);
 | 
			
		||||
void mvs_update_phyinfo(struct mvs_info *mvi, int i, int get_st);
 | 
			
		||||
int mvs_int_rx(struct mvs_info *mvi, bool self_clear);
 | 
			
		||||
void mvs_hexdump(u32 size, u8 *data, u32 baseaddr);
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
		Reference in a new issue