[PATCH] PCI: altix: msi support
MSI callouts for altix. Involves a fair amount of code reorg in sn irq.c code as well as adding some extensions to the altix PCI provider abstaction. Signed-off-by: Mark Maule <maule@sgi.com> Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
This commit is contained in:
parent
10083072bf
commit
83821d3f55
|
@ -58,7 +58,7 @@ static int max_pcibus_number = 255; /* Default highest pci bus number */
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static dma_addr_t
|
static dma_addr_t
|
||||||
sn_default_pci_map(struct pci_dev *pdev, unsigned long paddr, size_t size)
|
sn_default_pci_map(struct pci_dev *pdev, unsigned long paddr, size_t size, int type)
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -457,13 +457,6 @@ void sn_pci_fixup_slot(struct pci_dev *dev)
|
||||||
pcidev_info->pdi_sn_irq_info = NULL;
|
pcidev_info->pdi_sn_irq_info = NULL;
|
||||||
kfree(sn_irq_info);
|
kfree(sn_irq_info);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* MSI currently not supported on altix. Remove this when
|
|
||||||
* the MSI abstraction patches are integrated into the kernel
|
|
||||||
* (sometime after 2.6.16 releases)
|
|
||||||
*/
|
|
||||||
dev->no_msi = 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -26,11 +26,11 @@ static void unregister_intr_pda(struct sn_irq_info *sn_irq_info);
|
||||||
|
|
||||||
int sn_force_interrupt_flag = 1;
|
int sn_force_interrupt_flag = 1;
|
||||||
extern int sn_ioif_inited;
|
extern int sn_ioif_inited;
|
||||||
static struct list_head **sn_irq_lh;
|
struct list_head **sn_irq_lh;
|
||||||
static spinlock_t sn_irq_info_lock = SPIN_LOCK_UNLOCKED; /* non-IRQ lock */
|
static spinlock_t sn_irq_info_lock = SPIN_LOCK_UNLOCKED; /* non-IRQ lock */
|
||||||
|
|
||||||
static inline u64 sn_intr_alloc(nasid_t local_nasid, int local_widget,
|
u64 sn_intr_alloc(nasid_t local_nasid, int local_widget,
|
||||||
u64 sn_irq_info,
|
struct sn_irq_info *sn_irq_info,
|
||||||
int req_irq, nasid_t req_nasid,
|
int req_irq, nasid_t req_nasid,
|
||||||
int req_slice)
|
int req_slice)
|
||||||
{
|
{
|
||||||
|
@ -40,12 +40,13 @@ static inline u64 sn_intr_alloc(nasid_t local_nasid, int local_widget,
|
||||||
|
|
||||||
SAL_CALL_NOLOCK(ret_stuff, (u64) SN_SAL_IOIF_INTERRUPT,
|
SAL_CALL_NOLOCK(ret_stuff, (u64) SN_SAL_IOIF_INTERRUPT,
|
||||||
(u64) SAL_INTR_ALLOC, (u64) local_nasid,
|
(u64) SAL_INTR_ALLOC, (u64) local_nasid,
|
||||||
(u64) local_widget, (u64) sn_irq_info, (u64) req_irq,
|
(u64) local_widget, __pa(sn_irq_info), (u64) req_irq,
|
||||||
(u64) req_nasid, (u64) req_slice);
|
(u64) req_nasid, (u64) req_slice);
|
||||||
|
|
||||||
return ret_stuff.status;
|
return ret_stuff.status;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void sn_intr_free(nasid_t local_nasid, int local_widget,
|
void sn_intr_free(nasid_t local_nasid, int local_widget,
|
||||||
struct sn_irq_info *sn_irq_info)
|
struct sn_irq_info *sn_irq_info)
|
||||||
{
|
{
|
||||||
struct ia64_sal_retval ret_stuff;
|
struct ia64_sal_retval ret_stuff;
|
||||||
|
@ -112,73 +113,91 @@ static void sn_end_irq(unsigned int irq)
|
||||||
|
|
||||||
static void sn_irq_info_free(struct rcu_head *head);
|
static void sn_irq_info_free(struct rcu_head *head);
|
||||||
|
|
||||||
|
struct sn_irq_info *sn_retarget_vector(struct sn_irq_info *sn_irq_info,
|
||||||
|
nasid_t nasid, int slice)
|
||||||
|
{
|
||||||
|
int vector;
|
||||||
|
int cpuphys;
|
||||||
|
int64_t bridge;
|
||||||
|
int local_widget, status;
|
||||||
|
nasid_t local_nasid;
|
||||||
|
struct sn_irq_info *new_irq_info;
|
||||||
|
struct sn_pcibus_provider *pci_provider;
|
||||||
|
|
||||||
|
new_irq_info = kmalloc(sizeof(struct sn_irq_info), GFP_ATOMIC);
|
||||||
|
if (new_irq_info == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
memcpy(new_irq_info, sn_irq_info, sizeof(struct sn_irq_info));
|
||||||
|
|
||||||
|
bridge = (u64) new_irq_info->irq_bridge;
|
||||||
|
if (!bridge) {
|
||||||
|
kfree(new_irq_info);
|
||||||
|
return NULL; /* irq is not a device interrupt */
|
||||||
|
}
|
||||||
|
|
||||||
|
local_nasid = NASID_GET(bridge);
|
||||||
|
|
||||||
|
if (local_nasid & 1)
|
||||||
|
local_widget = TIO_SWIN_WIDGETNUM(bridge);
|
||||||
|
else
|
||||||
|
local_widget = SWIN_WIDGETNUM(bridge);
|
||||||
|
|
||||||
|
vector = sn_irq_info->irq_irq;
|
||||||
|
/* Free the old PROM new_irq_info structure */
|
||||||
|
sn_intr_free(local_nasid, local_widget, new_irq_info);
|
||||||
|
/* Update kernels new_irq_info with new target info */
|
||||||
|
unregister_intr_pda(new_irq_info);
|
||||||
|
|
||||||
|
/* allocate a new PROM new_irq_info struct */
|
||||||
|
status = sn_intr_alloc(local_nasid, local_widget,
|
||||||
|
new_irq_info, vector,
|
||||||
|
nasid, slice);
|
||||||
|
|
||||||
|
/* SAL call failed */
|
||||||
|
if (status) {
|
||||||
|
kfree(new_irq_info);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
cpuphys = nasid_slice_to_cpuid(nasid, slice);
|
||||||
|
new_irq_info->irq_cpuid = cpuphys;
|
||||||
|
register_intr_pda(new_irq_info);
|
||||||
|
|
||||||
|
pci_provider = sn_pci_provider[new_irq_info->irq_bridge_type];
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If this represents a line interrupt, target it. If it's
|
||||||
|
* an msi (irq_int_bit < 0), it's already targeted.
|
||||||
|
*/
|
||||||
|
if (new_irq_info->irq_int_bit >= 0 &&
|
||||||
|
pci_provider && pci_provider->target_interrupt)
|
||||||
|
(pci_provider->target_interrupt)(new_irq_info);
|
||||||
|
|
||||||
|
spin_lock(&sn_irq_info_lock);
|
||||||
|
list_replace_rcu(&sn_irq_info->list, &new_irq_info->list);
|
||||||
|
spin_unlock(&sn_irq_info_lock);
|
||||||
|
call_rcu(&sn_irq_info->rcu, sn_irq_info_free);
|
||||||
|
|
||||||
|
#ifdef CONFIG_SMP
|
||||||
|
set_irq_affinity_info((vector & 0xff), cpuphys, 0);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return new_irq_info;
|
||||||
|
}
|
||||||
|
|
||||||
static void sn_set_affinity_irq(unsigned int irq, cpumask_t mask)
|
static void sn_set_affinity_irq(unsigned int irq, cpumask_t mask)
|
||||||
{
|
{
|
||||||
struct sn_irq_info *sn_irq_info, *sn_irq_info_safe;
|
struct sn_irq_info *sn_irq_info, *sn_irq_info_safe;
|
||||||
int cpuid, cpuphys;
|
nasid_t nasid;
|
||||||
|
int slice;
|
||||||
|
|
||||||
cpuid = first_cpu(mask);
|
nasid = cpuid_to_nasid(first_cpu(mask));
|
||||||
cpuphys = cpu_physical_id(cpuid);
|
slice = cpuid_to_slice(first_cpu(mask));
|
||||||
|
|
||||||
list_for_each_entry_safe(sn_irq_info, sn_irq_info_safe,
|
list_for_each_entry_safe(sn_irq_info, sn_irq_info_safe,
|
||||||
sn_irq_lh[irq], list) {
|
sn_irq_lh[irq], list)
|
||||||
u64 bridge;
|
(void)sn_retarget_vector(sn_irq_info, nasid, slice);
|
||||||
int local_widget, status;
|
|
||||||
nasid_t local_nasid;
|
|
||||||
struct sn_irq_info *new_irq_info;
|
|
||||||
struct sn_pcibus_provider *pci_provider;
|
|
||||||
|
|
||||||
new_irq_info = kmalloc(sizeof(struct sn_irq_info), GFP_ATOMIC);
|
|
||||||
if (new_irq_info == NULL)
|
|
||||||
break;
|
|
||||||
memcpy(new_irq_info, sn_irq_info, sizeof(struct sn_irq_info));
|
|
||||||
|
|
||||||
bridge = (u64) new_irq_info->irq_bridge;
|
|
||||||
if (!bridge) {
|
|
||||||
kfree(new_irq_info);
|
|
||||||
break; /* irq is not a device interrupt */
|
|
||||||
}
|
|
||||||
|
|
||||||
local_nasid = NASID_GET(bridge);
|
|
||||||
|
|
||||||
if (local_nasid & 1)
|
|
||||||
local_widget = TIO_SWIN_WIDGETNUM(bridge);
|
|
||||||
else
|
|
||||||
local_widget = SWIN_WIDGETNUM(bridge);
|
|
||||||
|
|
||||||
/* Free the old PROM new_irq_info structure */
|
|
||||||
sn_intr_free(local_nasid, local_widget, new_irq_info);
|
|
||||||
/* Update kernels new_irq_info with new target info */
|
|
||||||
unregister_intr_pda(new_irq_info);
|
|
||||||
|
|
||||||
/* allocate a new PROM new_irq_info struct */
|
|
||||||
status = sn_intr_alloc(local_nasid, local_widget,
|
|
||||||
__pa(new_irq_info), irq,
|
|
||||||
cpuid_to_nasid(cpuid),
|
|
||||||
cpuid_to_slice(cpuid));
|
|
||||||
|
|
||||||
/* SAL call failed */
|
|
||||||
if (status) {
|
|
||||||
kfree(new_irq_info);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
new_irq_info->irq_cpuid = cpuid;
|
|
||||||
register_intr_pda(new_irq_info);
|
|
||||||
|
|
||||||
pci_provider = sn_pci_provider[new_irq_info->irq_bridge_type];
|
|
||||||
if (pci_provider && pci_provider->target_interrupt)
|
|
||||||
(pci_provider->target_interrupt)(new_irq_info);
|
|
||||||
|
|
||||||
spin_lock(&sn_irq_info_lock);
|
|
||||||
list_replace_rcu(&sn_irq_info->list, &new_irq_info->list);
|
|
||||||
spin_unlock(&sn_irq_info_lock);
|
|
||||||
call_rcu(&sn_irq_info->rcu, sn_irq_info_free);
|
|
||||||
|
|
||||||
#ifdef CONFIG_SMP
|
|
||||||
set_irq_affinity_info((irq & 0xff), cpuphys, 0);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
struct hw_interrupt_type irq_type_sn = {
|
struct hw_interrupt_type irq_type_sn = {
|
||||||
|
|
|
@ -11,7 +11,7 @@
|
||||||
|
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
#include <asm/dma.h>
|
#include <asm/dma.h>
|
||||||
#include <asm/sn/pcibr_provider.h>
|
#include <asm/sn/intr.h>
|
||||||
#include <asm/sn/pcibus_provider_defs.h>
|
#include <asm/sn/pcibus_provider_defs.h>
|
||||||
#include <asm/sn/pcidev.h>
|
#include <asm/sn/pcidev.h>
|
||||||
#include <asm/sn/sn_sal.h>
|
#include <asm/sn/sn_sal.h>
|
||||||
|
@ -113,7 +113,8 @@ void *sn_dma_alloc_coherent(struct device *dev, size_t size,
|
||||||
* resources.
|
* resources.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
*dma_handle = provider->dma_map_consistent(pdev, phys_addr, size);
|
*dma_handle = provider->dma_map_consistent(pdev, phys_addr, size,
|
||||||
|
SN_DMA_ADDR_PHYS);
|
||||||
if (!*dma_handle) {
|
if (!*dma_handle) {
|
||||||
printk(KERN_ERR "%s: out of ATEs\n", __FUNCTION__);
|
printk(KERN_ERR "%s: out of ATEs\n", __FUNCTION__);
|
||||||
free_pages((unsigned long)cpuaddr, get_order(size));
|
free_pages((unsigned long)cpuaddr, get_order(size));
|
||||||
|
@ -176,7 +177,7 @@ dma_addr_t sn_dma_map_single(struct device *dev, void *cpu_addr, size_t size,
|
||||||
BUG_ON(dev->bus != &pci_bus_type);
|
BUG_ON(dev->bus != &pci_bus_type);
|
||||||
|
|
||||||
phys_addr = __pa(cpu_addr);
|
phys_addr = __pa(cpu_addr);
|
||||||
dma_addr = provider->dma_map(pdev, phys_addr, size);
|
dma_addr = provider->dma_map(pdev, phys_addr, size, SN_DMA_ADDR_PHYS);
|
||||||
if (!dma_addr) {
|
if (!dma_addr) {
|
||||||
printk(KERN_ERR "%s: out of ATEs\n", __FUNCTION__);
|
printk(KERN_ERR "%s: out of ATEs\n", __FUNCTION__);
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -260,7 +261,8 @@ int sn_dma_map_sg(struct device *dev, struct scatterlist *sg, int nhwentries,
|
||||||
for (i = 0; i < nhwentries; i++, sg++) {
|
for (i = 0; i < nhwentries; i++, sg++) {
|
||||||
phys_addr = SG_ENT_PHYS_ADDRESS(sg);
|
phys_addr = SG_ENT_PHYS_ADDRESS(sg);
|
||||||
sg->dma_address = provider->dma_map(pdev,
|
sg->dma_address = provider->dma_map(pdev,
|
||||||
phys_addr, sg->length);
|
phys_addr, sg->length,
|
||||||
|
SN_DMA_ADDR_PHYS);
|
||||||
|
|
||||||
if (!sg->dma_address) {
|
if (!sg->dma_address) {
|
||||||
printk(KERN_ERR "%s: out of ATEs\n", __FUNCTION__);
|
printk(KERN_ERR "%s: out of ATEs\n", __FUNCTION__);
|
||||||
|
|
|
@ -41,7 +41,7 @@ extern int sn_ioif_inited;
|
||||||
|
|
||||||
static dma_addr_t
|
static dma_addr_t
|
||||||
pcibr_dmamap_ate32(struct pcidev_info *info,
|
pcibr_dmamap_ate32(struct pcidev_info *info,
|
||||||
u64 paddr, size_t req_size, u64 flags)
|
u64 paddr, size_t req_size, u64 flags, int dma_flags)
|
||||||
{
|
{
|
||||||
|
|
||||||
struct pcidev_info *pcidev_info = info->pdi_host_pcidev_info;
|
struct pcidev_info *pcidev_info = info->pdi_host_pcidev_info;
|
||||||
|
@ -81,9 +81,12 @@ pcibr_dmamap_ate32(struct pcidev_info *info,
|
||||||
if (IS_PCIX(pcibus_info))
|
if (IS_PCIX(pcibus_info))
|
||||||
ate_flags &= ~(PCI32_ATE_PREF);
|
ate_flags &= ~(PCI32_ATE_PREF);
|
||||||
|
|
||||||
xio_addr =
|
if (SN_DMA_ADDRTYPE(dma_flags == SN_DMA_ADDR_PHYS))
|
||||||
IS_PIC_SOFT(pcibus_info) ? PHYS_TO_DMA(paddr) :
|
xio_addr = IS_PIC_SOFT(pcibus_info) ? PHYS_TO_DMA(paddr) :
|
||||||
PHYS_TO_TIODMA(paddr);
|
PHYS_TO_TIODMA(paddr);
|
||||||
|
else
|
||||||
|
xio_addr = paddr;
|
||||||
|
|
||||||
offset = IOPGOFF(xio_addr);
|
offset = IOPGOFF(xio_addr);
|
||||||
ate = ate_flags | (xio_addr - offset);
|
ate = ate_flags | (xio_addr - offset);
|
||||||
|
|
||||||
|
@ -91,6 +94,13 @@ pcibr_dmamap_ate32(struct pcidev_info *info,
|
||||||
if (IS_PIC_SOFT(pcibus_info)) {
|
if (IS_PIC_SOFT(pcibus_info)) {
|
||||||
ate |= (pcibus_info->pbi_hub_xid << PIC_ATE_TARGETID_SHFT);
|
ate |= (pcibus_info->pbi_hub_xid << PIC_ATE_TARGETID_SHFT);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If we're mapping for MSI, set the MSI bit in the ATE
|
||||||
|
*/
|
||||||
|
if (dma_flags & SN_DMA_MSI)
|
||||||
|
ate |= PCI32_ATE_MSI;
|
||||||
|
|
||||||
ate_write(pcibus_info, ate_index, ate_count, ate);
|
ate_write(pcibus_info, ate_index, ate_count, ate);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -105,20 +115,27 @@ pcibr_dmamap_ate32(struct pcidev_info *info,
|
||||||
if (pcibus_info->pbi_devreg[internal_device] & PCIBR_DEV_SWAP_DIR)
|
if (pcibus_info->pbi_devreg[internal_device] & PCIBR_DEV_SWAP_DIR)
|
||||||
ATE_SWAP_ON(pci_addr);
|
ATE_SWAP_ON(pci_addr);
|
||||||
|
|
||||||
|
|
||||||
return pci_addr;
|
return pci_addr;
|
||||||
}
|
}
|
||||||
|
|
||||||
static dma_addr_t
|
static dma_addr_t
|
||||||
pcibr_dmatrans_direct64(struct pcidev_info * info, u64 paddr,
|
pcibr_dmatrans_direct64(struct pcidev_info * info, u64 paddr,
|
||||||
u64 dma_attributes)
|
u64 dma_attributes, int dma_flags)
|
||||||
{
|
{
|
||||||
struct pcibus_info *pcibus_info = (struct pcibus_info *)
|
struct pcibus_info *pcibus_info = (struct pcibus_info *)
|
||||||
((info->pdi_host_pcidev_info)->pdi_pcibus_info);
|
((info->pdi_host_pcidev_info)->pdi_pcibus_info);
|
||||||
u64 pci_addr;
|
u64 pci_addr;
|
||||||
|
|
||||||
/* Translate to Crosstalk View of Physical Address */
|
/* Translate to Crosstalk View of Physical Address */
|
||||||
pci_addr = (IS_PIC_SOFT(pcibus_info) ? PHYS_TO_DMA(paddr) :
|
if (SN_DMA_ADDRTYPE(dma_flags) == SN_DMA_ADDR_PHYS)
|
||||||
PHYS_TO_TIODMA(paddr)) | dma_attributes;
|
pci_addr = IS_PIC_SOFT(pcibus_info) ?
|
||||||
|
PHYS_TO_DMA(paddr) :
|
||||||
|
PHYS_TO_TIODMA(paddr) | dma_attributes;
|
||||||
|
else
|
||||||
|
pci_addr = IS_PIC_SOFT(pcibus_info) ?
|
||||||
|
paddr :
|
||||||
|
paddr | dma_attributes;
|
||||||
|
|
||||||
/* Handle Bus mode */
|
/* Handle Bus mode */
|
||||||
if (IS_PCIX(pcibus_info))
|
if (IS_PCIX(pcibus_info))
|
||||||
|
@ -130,7 +147,9 @@ pcibr_dmatrans_direct64(struct pcidev_info * info, u64 paddr,
|
||||||
((u64) pcibus_info->
|
((u64) pcibus_info->
|
||||||
pbi_hub_xid << PIC_PCI64_ATTR_TARG_SHFT);
|
pbi_hub_xid << PIC_PCI64_ATTR_TARG_SHFT);
|
||||||
} else
|
} else
|
||||||
pci_addr |= TIOCP_PCI64_CMDTYPE_MEM;
|
pci_addr |= (dma_flags & SN_DMA_MSI) ?
|
||||||
|
TIOCP_PCI64_CMDTYPE_MSI :
|
||||||
|
TIOCP_PCI64_CMDTYPE_MEM;
|
||||||
|
|
||||||
/* If PCI mode, func zero uses VCHAN0, every other func uses VCHAN1 */
|
/* If PCI mode, func zero uses VCHAN0, every other func uses VCHAN1 */
|
||||||
if (!IS_PCIX(pcibus_info) && PCI_FUNC(info->pdi_linux_pcidev->devfn))
|
if (!IS_PCIX(pcibus_info) && PCI_FUNC(info->pdi_linux_pcidev->devfn))
|
||||||
|
@ -141,7 +160,7 @@ pcibr_dmatrans_direct64(struct pcidev_info * info, u64 paddr,
|
||||||
|
|
||||||
static dma_addr_t
|
static dma_addr_t
|
||||||
pcibr_dmatrans_direct32(struct pcidev_info * info,
|
pcibr_dmatrans_direct32(struct pcidev_info * info,
|
||||||
u64 paddr, size_t req_size, u64 flags)
|
u64 paddr, size_t req_size, u64 flags, int dma_flags)
|
||||||
{
|
{
|
||||||
struct pcidev_info *pcidev_info = info->pdi_host_pcidev_info;
|
struct pcidev_info *pcidev_info = info->pdi_host_pcidev_info;
|
||||||
struct pcibus_info *pcibus_info = (struct pcibus_info *)pcidev_info->
|
struct pcibus_info *pcibus_info = (struct pcibus_info *)pcidev_info->
|
||||||
|
@ -156,8 +175,14 @@ pcibr_dmatrans_direct32(struct pcidev_info * info,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
xio_addr = IS_PIC_SOFT(pcibus_info) ? PHYS_TO_DMA(paddr) :
|
if (dma_flags & SN_DMA_MSI)
|
||||||
PHYS_TO_TIODMA(paddr);
|
return 0;
|
||||||
|
|
||||||
|
if (SN_DMA_ADDRTYPE(dma_flags) == SN_DMA_ADDR_PHYS)
|
||||||
|
xio_addr = IS_PIC_SOFT(pcibus_info) ? PHYS_TO_DMA(paddr) :
|
||||||
|
PHYS_TO_TIODMA(paddr);
|
||||||
|
else
|
||||||
|
xio_addr = paddr;
|
||||||
|
|
||||||
xio_base = pcibus_info->pbi_dir_xbase;
|
xio_base = pcibus_info->pbi_dir_xbase;
|
||||||
offset = xio_addr - xio_base;
|
offset = xio_addr - xio_base;
|
||||||
|
@ -327,7 +352,7 @@ void sn_dma_flush(u64 addr)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
dma_addr_t
|
dma_addr_t
|
||||||
pcibr_dma_map(struct pci_dev * hwdev, unsigned long phys_addr, size_t size)
|
pcibr_dma_map(struct pci_dev * hwdev, unsigned long phys_addr, size_t size, int dma_flags)
|
||||||
{
|
{
|
||||||
dma_addr_t dma_handle;
|
dma_addr_t dma_handle;
|
||||||
struct pcidev_info *pcidev_info = SN_PCIDEV_INFO(hwdev);
|
struct pcidev_info *pcidev_info = SN_PCIDEV_INFO(hwdev);
|
||||||
|
@ -344,11 +369,11 @@ pcibr_dma_map(struct pci_dev * hwdev, unsigned long phys_addr, size_t size)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
dma_handle = pcibr_dmatrans_direct64(pcidev_info, phys_addr,
|
dma_handle = pcibr_dmatrans_direct64(pcidev_info, phys_addr,
|
||||||
PCI64_ATTR_PREF);
|
PCI64_ATTR_PREF, dma_flags);
|
||||||
} else {
|
} else {
|
||||||
/* Handle 32-63 bit cards via direct mapping */
|
/* Handle 32-63 bit cards via direct mapping */
|
||||||
dma_handle = pcibr_dmatrans_direct32(pcidev_info, phys_addr,
|
dma_handle = pcibr_dmatrans_direct32(pcidev_info, phys_addr,
|
||||||
size, 0);
|
size, 0, dma_flags);
|
||||||
if (!dma_handle) {
|
if (!dma_handle) {
|
||||||
/*
|
/*
|
||||||
* It is a 32 bit card and we cannot do direct mapping,
|
* It is a 32 bit card and we cannot do direct mapping,
|
||||||
|
@ -356,7 +381,8 @@ pcibr_dma_map(struct pci_dev * hwdev, unsigned long phys_addr, size_t size)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
dma_handle = pcibr_dmamap_ate32(pcidev_info, phys_addr,
|
dma_handle = pcibr_dmamap_ate32(pcidev_info, phys_addr,
|
||||||
size, PCI32_ATE_PREF);
|
size, PCI32_ATE_PREF,
|
||||||
|
dma_flags);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -365,18 +391,18 @@ pcibr_dma_map(struct pci_dev * hwdev, unsigned long phys_addr, size_t size)
|
||||||
|
|
||||||
dma_addr_t
|
dma_addr_t
|
||||||
pcibr_dma_map_consistent(struct pci_dev * hwdev, unsigned long phys_addr,
|
pcibr_dma_map_consistent(struct pci_dev * hwdev, unsigned long phys_addr,
|
||||||
size_t size)
|
size_t size, int dma_flags)
|
||||||
{
|
{
|
||||||
dma_addr_t dma_handle;
|
dma_addr_t dma_handle;
|
||||||
struct pcidev_info *pcidev_info = SN_PCIDEV_INFO(hwdev);
|
struct pcidev_info *pcidev_info = SN_PCIDEV_INFO(hwdev);
|
||||||
|
|
||||||
if (hwdev->dev.coherent_dma_mask == ~0UL) {
|
if (hwdev->dev.coherent_dma_mask == ~0UL) {
|
||||||
dma_handle = pcibr_dmatrans_direct64(pcidev_info, phys_addr,
|
dma_handle = pcibr_dmatrans_direct64(pcidev_info, phys_addr,
|
||||||
PCI64_ATTR_BAR);
|
PCI64_ATTR_BAR, dma_flags);
|
||||||
} else {
|
} else {
|
||||||
dma_handle = (dma_addr_t) pcibr_dmamap_ate32(pcidev_info,
|
dma_handle = (dma_addr_t) pcibr_dmamap_ate32(pcidev_info,
|
||||||
phys_addr, size,
|
phys_addr, size,
|
||||||
PCI32_ATE_BAR);
|
PCI32_ATE_BAR, dma_flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
return dma_handle;
|
return dma_handle;
|
||||||
|
|
|
@ -515,10 +515,16 @@ tioca_dma_unmap(struct pci_dev *pdev, dma_addr_t bus_addr, int dir)
|
||||||
* use the GART mapped mode.
|
* use the GART mapped mode.
|
||||||
*/
|
*/
|
||||||
static u64
|
static u64
|
||||||
tioca_dma_map(struct pci_dev *pdev, u64 paddr, size_t byte_count)
|
tioca_dma_map(struct pci_dev *pdev, u64 paddr, size_t byte_count, int dma_flags)
|
||||||
{
|
{
|
||||||
u64 mapaddr;
|
u64 mapaddr;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Not supported for now ...
|
||||||
|
*/
|
||||||
|
if (dma_flags & SN_DMA_MSI)
|
||||||
|
return 0;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If card is 64 or 48 bit addresable, use a direct mapping. 32
|
* If card is 64 or 48 bit addresable, use a direct mapping. 32
|
||||||
* bit direct is so restrictive w.r.t. where the memory resides that
|
* bit direct is so restrictive w.r.t. where the memory resides that
|
||||||
|
|
|
@ -170,7 +170,8 @@ tioce_mmr_war_post(struct tioce_kernel *kern, void *mmr_addr)
|
||||||
(ATE_PAGE((start)+(len)-1, pagesize) - ATE_PAGE(start, pagesize) + 1)
|
(ATE_PAGE((start)+(len)-1, pagesize) - ATE_PAGE(start, pagesize) + 1)
|
||||||
|
|
||||||
#define ATE_VALID(ate) ((ate) & (1UL << 63))
|
#define ATE_VALID(ate) ((ate) & (1UL << 63))
|
||||||
#define ATE_MAKE(addr, ps) (((addr) & ~ATE_PAGEMASK(ps)) | (1UL << 63))
|
#define ATE_MAKE(addr, ps, msi) \
|
||||||
|
(((addr) & ~ATE_PAGEMASK(ps)) | (1UL << 63) | ((msi)?(1UL << 62):0))
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Flavors of ate-based mapping supported by tioce_alloc_map()
|
* Flavors of ate-based mapping supported by tioce_alloc_map()
|
||||||
|
@ -196,15 +197,17 @@ tioce_mmr_war_post(struct tioce_kernel *kern, void *mmr_addr)
|
||||||
*
|
*
|
||||||
* 63 - must be 1 to indicate d64 mode to CE hardware
|
* 63 - must be 1 to indicate d64 mode to CE hardware
|
||||||
* 62 - barrier bit ... controlled with tioce_dma_barrier()
|
* 62 - barrier bit ... controlled with tioce_dma_barrier()
|
||||||
* 61 - 0 since this is not an MSI transaction
|
* 61 - msi bit ... specified through dma_flags
|
||||||
* 60:54 - reserved, MBZ
|
* 60:54 - reserved, MBZ
|
||||||
*/
|
*/
|
||||||
static u64
|
static u64
|
||||||
tioce_dma_d64(unsigned long ct_addr)
|
tioce_dma_d64(unsigned long ct_addr, int dma_flags)
|
||||||
{
|
{
|
||||||
u64 bus_addr;
|
u64 bus_addr;
|
||||||
|
|
||||||
bus_addr = ct_addr | (1UL << 63);
|
bus_addr = ct_addr | (1UL << 63);
|
||||||
|
if (dma_flags & SN_DMA_MSI)
|
||||||
|
bus_addr |= (1UL << 61);
|
||||||
|
|
||||||
return bus_addr;
|
return bus_addr;
|
||||||
}
|
}
|
||||||
|
@ -261,7 +264,7 @@ pcidev_to_tioce(struct pci_dev *pdev, struct tioce **base,
|
||||||
*/
|
*/
|
||||||
static u64
|
static u64
|
||||||
tioce_alloc_map(struct tioce_kernel *ce_kern, int type, int port,
|
tioce_alloc_map(struct tioce_kernel *ce_kern, int type, int port,
|
||||||
u64 ct_addr, int len)
|
u64 ct_addr, int len, int dma_flags)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
int j;
|
int j;
|
||||||
|
@ -270,6 +273,7 @@ tioce_alloc_map(struct tioce_kernel *ce_kern, int type, int port,
|
||||||
int entries;
|
int entries;
|
||||||
int nates;
|
int nates;
|
||||||
u64 pagesize;
|
u64 pagesize;
|
||||||
|
int msi_capable, msi_wanted;
|
||||||
u64 *ate_shadow;
|
u64 *ate_shadow;
|
||||||
u64 *ate_reg;
|
u64 *ate_reg;
|
||||||
u64 addr;
|
u64 addr;
|
||||||
|
@ -291,6 +295,7 @@ tioce_alloc_map(struct tioce_kernel *ce_kern, int type, int port,
|
||||||
ate_reg = ce_mmr->ce_ure_ate3240;
|
ate_reg = ce_mmr->ce_ure_ate3240;
|
||||||
pagesize = ce_kern->ce_ate3240_pagesize;
|
pagesize = ce_kern->ce_ate3240_pagesize;
|
||||||
bus_base = TIOCE_M32_MIN;
|
bus_base = TIOCE_M32_MIN;
|
||||||
|
msi_capable = 1;
|
||||||
break;
|
break;
|
||||||
case TIOCE_ATE_M40:
|
case TIOCE_ATE_M40:
|
||||||
first = 0;
|
first = 0;
|
||||||
|
@ -299,6 +304,7 @@ tioce_alloc_map(struct tioce_kernel *ce_kern, int type, int port,
|
||||||
ate_reg = ce_mmr->ce_ure_ate40;
|
ate_reg = ce_mmr->ce_ure_ate40;
|
||||||
pagesize = MB(64);
|
pagesize = MB(64);
|
||||||
bus_base = TIOCE_M40_MIN;
|
bus_base = TIOCE_M40_MIN;
|
||||||
|
msi_capable = 0;
|
||||||
break;
|
break;
|
||||||
case TIOCE_ATE_M40S:
|
case TIOCE_ATE_M40S:
|
||||||
/*
|
/*
|
||||||
|
@ -311,11 +317,16 @@ tioce_alloc_map(struct tioce_kernel *ce_kern, int type, int port,
|
||||||
ate_reg = ce_mmr->ce_ure_ate3240;
|
ate_reg = ce_mmr->ce_ure_ate3240;
|
||||||
pagesize = GB(16);
|
pagesize = GB(16);
|
||||||
bus_base = TIOCE_M40S_MIN;
|
bus_base = TIOCE_M40S_MIN;
|
||||||
|
msi_capable = 0;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
msi_wanted = dma_flags & SN_DMA_MSI;
|
||||||
|
if (msi_wanted && !msi_capable)
|
||||||
|
return 0;
|
||||||
|
|
||||||
nates = ATE_NPAGES(ct_addr, len, pagesize);
|
nates = ATE_NPAGES(ct_addr, len, pagesize);
|
||||||
if (nates > entries)
|
if (nates > entries)
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -344,7 +355,7 @@ tioce_alloc_map(struct tioce_kernel *ce_kern, int type, int port,
|
||||||
for (j = 0; j < nates; j++) {
|
for (j = 0; j < nates; j++) {
|
||||||
u64 ate;
|
u64 ate;
|
||||||
|
|
||||||
ate = ATE_MAKE(addr, pagesize);
|
ate = ATE_MAKE(addr, pagesize, msi_wanted);
|
||||||
ate_shadow[i + j] = ate;
|
ate_shadow[i + j] = ate;
|
||||||
tioce_mmr_storei(ce_kern, &ate_reg[i + j], ate);
|
tioce_mmr_storei(ce_kern, &ate_reg[i + j], ate);
|
||||||
addr += pagesize;
|
addr += pagesize;
|
||||||
|
@ -371,7 +382,7 @@ tioce_alloc_map(struct tioce_kernel *ce_kern, int type, int port,
|
||||||
* Map @paddr into 32-bit bus space of the CE associated with @pcidev_info.
|
* Map @paddr into 32-bit bus space of the CE associated with @pcidev_info.
|
||||||
*/
|
*/
|
||||||
static u64
|
static u64
|
||||||
tioce_dma_d32(struct pci_dev *pdev, u64 ct_addr)
|
tioce_dma_d32(struct pci_dev *pdev, u64 ct_addr, int dma_flags)
|
||||||
{
|
{
|
||||||
int dma_ok;
|
int dma_ok;
|
||||||
int port;
|
int port;
|
||||||
|
@ -381,6 +392,9 @@ tioce_dma_d32(struct pci_dev *pdev, u64 ct_addr)
|
||||||
u64 ct_lower;
|
u64 ct_lower;
|
||||||
dma_addr_t bus_addr;
|
dma_addr_t bus_addr;
|
||||||
|
|
||||||
|
if (dma_flags & SN_DMA_MSI)
|
||||||
|
return 0;
|
||||||
|
|
||||||
ct_upper = ct_addr & ~0x3fffffffUL;
|
ct_upper = ct_addr & ~0x3fffffffUL;
|
||||||
ct_lower = ct_addr & 0x3fffffffUL;
|
ct_lower = ct_addr & 0x3fffffffUL;
|
||||||
|
|
||||||
|
@ -507,7 +521,7 @@ tioce_dma_unmap(struct pci_dev *pdev, dma_addr_t bus_addr, int dir)
|
||||||
*/
|
*/
|
||||||
static u64
|
static u64
|
||||||
tioce_do_dma_map(struct pci_dev *pdev, u64 paddr, size_t byte_count,
|
tioce_do_dma_map(struct pci_dev *pdev, u64 paddr, size_t byte_count,
|
||||||
int barrier)
|
int barrier, int dma_flags)
|
||||||
{
|
{
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
u64 ct_addr;
|
u64 ct_addr;
|
||||||
|
@ -523,15 +537,18 @@ tioce_do_dma_map(struct pci_dev *pdev, u64 paddr, size_t byte_count,
|
||||||
if (dma_mask < 0x7fffffffUL)
|
if (dma_mask < 0x7fffffffUL)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
ct_addr = PHYS_TO_TIODMA(paddr);
|
if (SN_DMA_ADDRTYPE(dma_flags) == SN_DMA_ADDR_PHYS)
|
||||||
|
ct_addr = PHYS_TO_TIODMA(paddr);
|
||||||
|
else
|
||||||
|
ct_addr = paddr;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If the device can generate 64 bit addresses, create a D64 map.
|
* If the device can generate 64 bit addresses, create a D64 map.
|
||||||
* Since this should never fail, bypass the rest of the checks.
|
|
||||||
*/
|
*/
|
||||||
if (dma_mask == ~0UL) {
|
if (dma_mask == ~0UL) {
|
||||||
mapaddr = tioce_dma_d64(ct_addr);
|
mapaddr = tioce_dma_d64(ct_addr, dma_flags);
|
||||||
goto dma_map_done;
|
if (mapaddr)
|
||||||
|
goto dma_map_done;
|
||||||
}
|
}
|
||||||
|
|
||||||
pcidev_to_tioce(pdev, NULL, &ce_kern, &port);
|
pcidev_to_tioce(pdev, NULL, &ce_kern, &port);
|
||||||
|
@ -574,18 +591,22 @@ tioce_do_dma_map(struct pci_dev *pdev, u64 paddr, size_t byte_count,
|
||||||
|
|
||||||
if (byte_count > MB(64)) {
|
if (byte_count > MB(64)) {
|
||||||
mapaddr = tioce_alloc_map(ce_kern, TIOCE_ATE_M40S,
|
mapaddr = tioce_alloc_map(ce_kern, TIOCE_ATE_M40S,
|
||||||
port, ct_addr, byte_count);
|
port, ct_addr, byte_count,
|
||||||
|
dma_flags);
|
||||||
if (!mapaddr)
|
if (!mapaddr)
|
||||||
mapaddr =
|
mapaddr =
|
||||||
tioce_alloc_map(ce_kern, TIOCE_ATE_M40, -1,
|
tioce_alloc_map(ce_kern, TIOCE_ATE_M40, -1,
|
||||||
ct_addr, byte_count);
|
ct_addr, byte_count,
|
||||||
|
dma_flags);
|
||||||
} else {
|
} else {
|
||||||
mapaddr = tioce_alloc_map(ce_kern, TIOCE_ATE_M40, -1,
|
mapaddr = tioce_alloc_map(ce_kern, TIOCE_ATE_M40, -1,
|
||||||
ct_addr, byte_count);
|
ct_addr, byte_count,
|
||||||
|
dma_flags);
|
||||||
if (!mapaddr)
|
if (!mapaddr)
|
||||||
mapaddr =
|
mapaddr =
|
||||||
tioce_alloc_map(ce_kern, TIOCE_ATE_M40S,
|
tioce_alloc_map(ce_kern, TIOCE_ATE_M40S,
|
||||||
port, ct_addr, byte_count);
|
port, ct_addr, byte_count,
|
||||||
|
dma_flags);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -593,7 +614,7 @@ tioce_do_dma_map(struct pci_dev *pdev, u64 paddr, size_t byte_count,
|
||||||
* 32-bit direct is the next mode to try
|
* 32-bit direct is the next mode to try
|
||||||
*/
|
*/
|
||||||
if (!mapaddr && dma_mask >= 0xffffffffUL)
|
if (!mapaddr && dma_mask >= 0xffffffffUL)
|
||||||
mapaddr = tioce_dma_d32(pdev, ct_addr);
|
mapaddr = tioce_dma_d32(pdev, ct_addr, dma_flags);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Last resort, try 32-bit ATE-based map.
|
* Last resort, try 32-bit ATE-based map.
|
||||||
|
@ -601,7 +622,7 @@ tioce_do_dma_map(struct pci_dev *pdev, u64 paddr, size_t byte_count,
|
||||||
if (!mapaddr)
|
if (!mapaddr)
|
||||||
mapaddr =
|
mapaddr =
|
||||||
tioce_alloc_map(ce_kern, TIOCE_ATE_M32, -1, ct_addr,
|
tioce_alloc_map(ce_kern, TIOCE_ATE_M32, -1, ct_addr,
|
||||||
byte_count);
|
byte_count, dma_flags);
|
||||||
|
|
||||||
spin_unlock_irqrestore(&ce_kern->ce_lock, flags);
|
spin_unlock_irqrestore(&ce_kern->ce_lock, flags);
|
||||||
|
|
||||||
|
@ -622,9 +643,9 @@ dma_map_done:
|
||||||
* in the address.
|
* in the address.
|
||||||
*/
|
*/
|
||||||
static u64
|
static u64
|
||||||
tioce_dma(struct pci_dev *pdev, u64 paddr, size_t byte_count)
|
tioce_dma(struct pci_dev *pdev, u64 paddr, size_t byte_count, int dma_flags)
|
||||||
{
|
{
|
||||||
return tioce_do_dma_map(pdev, paddr, byte_count, 0);
|
return tioce_do_dma_map(pdev, paddr, byte_count, 0, dma_flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -636,9 +657,9 @@ tioce_dma(struct pci_dev *pdev, u64 paddr, size_t byte_count)
|
||||||
* Simply call tioce_do_dma_map() to create a map with the barrier bit set
|
* Simply call tioce_do_dma_map() to create a map with the barrier bit set
|
||||||
* in the address.
|
* in the address.
|
||||||
*/ static u64
|
*/ static u64
|
||||||
tioce_dma_consistent(struct pci_dev *pdev, u64 paddr, size_t byte_count)
|
tioce_dma_consistent(struct pci_dev *pdev, u64 paddr, size_t byte_count, int dma_flags)
|
||||||
{
|
{
|
||||||
return tioce_do_dma_map(pdev, paddr, byte_count, 1);
|
return tioce_do_dma_map(pdev, paddr, byte_count, 1, dma_flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -696,7 +717,7 @@ tioce_reserve_m32(struct tioce_kernel *ce_kern, u64 base, u64 limit)
|
||||||
while (ate_index <= last_ate) {
|
while (ate_index <= last_ate) {
|
||||||
u64 ate;
|
u64 ate;
|
||||||
|
|
||||||
ate = ATE_MAKE(0xdeadbeef, ps);
|
ate = ATE_MAKE(0xdeadbeef, ps, 0);
|
||||||
ce_kern->ce_ate3240_shadow[ate_index] = ate;
|
ce_kern->ce_ate3240_shadow[ate_index] = ate;
|
||||||
tioce_mmr_storei(ce_kern, &ce_mmr->ce_ure_ate3240[ate_index],
|
tioce_mmr_storei(ce_kern, &ce_mmr->ce_ure_ate3240[ate_index],
|
||||||
ate);
|
ate);
|
||||||
|
|
|
@ -6,13 +6,205 @@
|
||||||
* Copyright (C) 2006 Silicon Graphics, Inc. All Rights Reserved.
|
* Copyright (C) 2006 Silicon Graphics, Inc. All Rights Reserved.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <asm/errno.h>
|
#include <linux/types.h>
|
||||||
|
#include <linux/pci.h>
|
||||||
|
#include <linux/cpumask.h>
|
||||||
|
|
||||||
|
#include <asm/sn/addrs.h>
|
||||||
|
#include <asm/sn/intr.h>
|
||||||
|
#include <asm/sn/pcibus_provider_defs.h>
|
||||||
|
#include <asm/sn/pcidev.h>
|
||||||
|
#include <asm/sn/nodepda.h>
|
||||||
|
|
||||||
|
#include "msi.h"
|
||||||
|
|
||||||
|
struct sn_msi_info {
|
||||||
|
u64 pci_addr;
|
||||||
|
struct sn_irq_info *sn_irq_info;
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct sn_msi_info *sn_msi_info;
|
||||||
|
|
||||||
|
static void
|
||||||
|
sn_msi_teardown(unsigned int vector)
|
||||||
|
{
|
||||||
|
nasid_t nasid;
|
||||||
|
int widget;
|
||||||
|
struct pci_dev *pdev;
|
||||||
|
struct pcidev_info *sn_pdev;
|
||||||
|
struct sn_irq_info *sn_irq_info;
|
||||||
|
struct pcibus_bussoft *bussoft;
|
||||||
|
struct sn_pcibus_provider *provider;
|
||||||
|
|
||||||
|
sn_irq_info = sn_msi_info[vector].sn_irq_info;
|
||||||
|
if (sn_irq_info == NULL || sn_irq_info->irq_int_bit >= 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
sn_pdev = (struct pcidev_info *)sn_irq_info->irq_pciioinfo;
|
||||||
|
pdev = sn_pdev->pdi_linux_pcidev;
|
||||||
|
provider = SN_PCIDEV_BUSPROVIDER(pdev);
|
||||||
|
|
||||||
|
(*provider->dma_unmap)(pdev,
|
||||||
|
sn_msi_info[vector].pci_addr,
|
||||||
|
PCI_DMA_FROMDEVICE);
|
||||||
|
sn_msi_info[vector].pci_addr = 0;
|
||||||
|
|
||||||
|
bussoft = SN_PCIDEV_BUSSOFT(pdev);
|
||||||
|
nasid = NASID_GET(bussoft->bs_base);
|
||||||
|
widget = (nasid & 1) ?
|
||||||
|
TIO_SWIN_WIDGETNUM(bussoft->bs_base) :
|
||||||
|
SWIN_WIDGETNUM(bussoft->bs_base);
|
||||||
|
|
||||||
|
sn_intr_free(nasid, widget, sn_irq_info);
|
||||||
|
sn_msi_info[vector].sn_irq_info = NULL;
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
sn_msi_setup(struct pci_dev *pdev, unsigned int vector,
|
||||||
|
u32 *addr_hi, u32 *addr_lo, u32 *data)
|
||||||
|
{
|
||||||
|
int widget;
|
||||||
|
int status;
|
||||||
|
nasid_t nasid;
|
||||||
|
u64 bus_addr;
|
||||||
|
struct sn_irq_info *sn_irq_info;
|
||||||
|
struct pcibus_bussoft *bussoft = SN_PCIDEV_BUSSOFT(pdev);
|
||||||
|
struct sn_pcibus_provider *provider = SN_PCIDEV_BUSPROVIDER(pdev);
|
||||||
|
|
||||||
|
if (bussoft == NULL)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (provider == NULL || provider->dma_map_consistent == NULL)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Set up the vector plumbing. Let the prom (via sn_intr_alloc)
|
||||||
|
* decide which cpu to direct this msi at by default.
|
||||||
|
*/
|
||||||
|
|
||||||
|
nasid = NASID_GET(bussoft->bs_base);
|
||||||
|
widget = (nasid & 1) ?
|
||||||
|
TIO_SWIN_WIDGETNUM(bussoft->bs_base) :
|
||||||
|
SWIN_WIDGETNUM(bussoft->bs_base);
|
||||||
|
|
||||||
|
sn_irq_info = kzalloc(sizeof(struct sn_irq_info), GFP_KERNEL);
|
||||||
|
if (! sn_irq_info)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
status = sn_intr_alloc(nasid, widget, sn_irq_info, vector, -1, -1);
|
||||||
|
if (status) {
|
||||||
|
kfree(sn_irq_info);
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
|
||||||
|
sn_irq_info->irq_int_bit = -1; /* mark this as an MSI irq */
|
||||||
|
sn_irq_fixup(pdev, sn_irq_info);
|
||||||
|
|
||||||
|
/* Prom probably should fill these in, but doesn't ... */
|
||||||
|
sn_irq_info->irq_bridge_type = bussoft->bs_asic_type;
|
||||||
|
sn_irq_info->irq_bridge = (void *)bussoft->bs_base;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Map the xio address into bus space
|
||||||
|
*/
|
||||||
|
bus_addr = (*provider->dma_map_consistent)(pdev,
|
||||||
|
sn_irq_info->irq_xtalkaddr,
|
||||||
|
sizeof(sn_irq_info->irq_xtalkaddr),
|
||||||
|
SN_DMA_MSI|SN_DMA_ADDR_XIO);
|
||||||
|
if (! bus_addr) {
|
||||||
|
sn_intr_free(nasid, widget, sn_irq_info);
|
||||||
|
kfree(sn_irq_info);
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
|
||||||
|
sn_msi_info[vector].sn_irq_info = sn_irq_info;
|
||||||
|
sn_msi_info[vector].pci_addr = bus_addr;
|
||||||
|
|
||||||
|
*addr_hi = (u32)(bus_addr >> 32);
|
||||||
|
*addr_lo = (u32)(bus_addr & 0x00000000ffffffff);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* In the SN platform, bit 16 is a "send vector" bit which
|
||||||
|
* must be present in order to move the vector through the system.
|
||||||
|
*/
|
||||||
|
*data = 0x100 + (unsigned int)vector;
|
||||||
|
|
||||||
|
#ifdef CONFIG_SMP
|
||||||
|
set_irq_affinity_info((vector & 0xff), sn_irq_info->irq_cpuid, 0);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
sn_msi_target(unsigned int vector, unsigned int cpu,
|
||||||
|
u32 *addr_hi, u32 *addr_lo)
|
||||||
|
{
|
||||||
|
int slice;
|
||||||
|
nasid_t nasid;
|
||||||
|
u64 bus_addr;
|
||||||
|
struct pci_dev *pdev;
|
||||||
|
struct pcidev_info *sn_pdev;
|
||||||
|
struct sn_irq_info *sn_irq_info;
|
||||||
|
struct sn_irq_info *new_irq_info;
|
||||||
|
struct sn_pcibus_provider *provider;
|
||||||
|
|
||||||
|
sn_irq_info = sn_msi_info[vector].sn_irq_info;
|
||||||
|
if (sn_irq_info == NULL || sn_irq_info->irq_int_bit >= 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Release XIO resources for the old MSI PCI address
|
||||||
|
*/
|
||||||
|
|
||||||
|
sn_pdev = (struct pcidev_info *)sn_irq_info->irq_pciioinfo;
|
||||||
|
pdev = sn_pdev->pdi_linux_pcidev;
|
||||||
|
provider = SN_PCIDEV_BUSPROVIDER(pdev);
|
||||||
|
|
||||||
|
bus_addr = (u64)(*addr_hi) << 32 | (u64)(*addr_lo);
|
||||||
|
(*provider->dma_unmap)(pdev, bus_addr, PCI_DMA_FROMDEVICE);
|
||||||
|
sn_msi_info[vector].pci_addr = 0;
|
||||||
|
|
||||||
|
nasid = cpuid_to_nasid(cpu);
|
||||||
|
slice = cpuid_to_slice(cpu);
|
||||||
|
|
||||||
|
new_irq_info = sn_retarget_vector(sn_irq_info, nasid, slice);
|
||||||
|
sn_msi_info[vector].sn_irq_info = new_irq_info;
|
||||||
|
if (new_irq_info == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Map the xio address into bus space
|
||||||
|
*/
|
||||||
|
|
||||||
|
bus_addr = (*provider->dma_map_consistent)(pdev,
|
||||||
|
new_irq_info->irq_xtalkaddr,
|
||||||
|
sizeof(new_irq_info->irq_xtalkaddr),
|
||||||
|
SN_DMA_MSI|SN_DMA_ADDR_XIO);
|
||||||
|
|
||||||
|
sn_msi_info[vector].pci_addr = bus_addr;
|
||||||
|
*addr_hi = (u32)(bus_addr >> 32);
|
||||||
|
*addr_lo = (u32)(bus_addr & 0x00000000ffffffff);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct msi_ops sn_msi_ops = {
|
||||||
|
.setup = sn_msi_setup,
|
||||||
|
.teardown = sn_msi_teardown,
|
||||||
|
#ifdef CONFIG_SMP
|
||||||
|
.target = sn_msi_target,
|
||||||
|
#endif
|
||||||
|
};
|
||||||
|
|
||||||
int
|
int
|
||||||
sn_msi_init(void)
|
sn_msi_init(void)
|
||||||
{
|
{
|
||||||
/*
|
sn_msi_info =
|
||||||
* return error until MSI is supported on altix platforms
|
kzalloc(sizeof(struct sn_msi_info) * NR_VECTORS, GFP_KERNEL);
|
||||||
*/
|
if (! sn_msi_info)
|
||||||
return -EINVAL;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
msi_register(&sn_msi_ops);
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,6 +10,7 @@
|
||||||
#define _ASM_IA64_SN_INTR_H
|
#define _ASM_IA64_SN_INTR_H
|
||||||
|
|
||||||
#include <linux/rcupdate.h>
|
#include <linux/rcupdate.h>
|
||||||
|
#include <asm/sn/types.h>
|
||||||
|
|
||||||
#define SGI_UART_VECTOR 0xe9
|
#define SGI_UART_VECTOR 0xe9
|
||||||
|
|
||||||
|
@ -40,6 +41,7 @@ struct sn_irq_info {
|
||||||
int irq_cpuid; /* kernel logical cpuid */
|
int irq_cpuid; /* kernel logical cpuid */
|
||||||
int irq_irq; /* the IRQ number */
|
int irq_irq; /* the IRQ number */
|
||||||
int irq_int_bit; /* Bridge interrupt pin */
|
int irq_int_bit; /* Bridge interrupt pin */
|
||||||
|
/* <0 means MSI */
|
||||||
u64 irq_xtalkaddr; /* xtalkaddr IRQ is sent to */
|
u64 irq_xtalkaddr; /* xtalkaddr IRQ is sent to */
|
||||||
int irq_bridge_type;/* pciio asic type (pciio.h) */
|
int irq_bridge_type;/* pciio asic type (pciio.h) */
|
||||||
void *irq_bridge; /* bridge generating irq */
|
void *irq_bridge; /* bridge generating irq */
|
||||||
|
@ -53,6 +55,12 @@ struct sn_irq_info {
|
||||||
};
|
};
|
||||||
|
|
||||||
extern void sn_send_IPI_phys(int, long, int, int);
|
extern void sn_send_IPI_phys(int, long, int, int);
|
||||||
|
extern u64 sn_intr_alloc(nasid_t, int,
|
||||||
|
struct sn_irq_info *,
|
||||||
|
int, nasid_t, int);
|
||||||
|
extern void sn_intr_free(nasid_t, int, struct sn_irq_info *);
|
||||||
|
extern struct sn_irq_info *sn_retarget_vector(struct sn_irq_info *, nasid_t, int);
|
||||||
|
extern struct list_head **sn_irq_lh;
|
||||||
|
|
||||||
#define CPU_VECTOR_TO_IRQ(cpuid,vector) (vector)
|
#define CPU_VECTOR_TO_IRQ(cpuid,vector) (vector)
|
||||||
|
|
||||||
|
|
|
@ -55,6 +55,7 @@
|
||||||
#define PCI32_ATE_V (0x1 << 0)
|
#define PCI32_ATE_V (0x1 << 0)
|
||||||
#define PCI32_ATE_CO (0x1 << 1)
|
#define PCI32_ATE_CO (0x1 << 1)
|
||||||
#define PCI32_ATE_PREC (0x1 << 2)
|
#define PCI32_ATE_PREC (0x1 << 2)
|
||||||
|
#define PCI32_ATE_MSI (0x1 << 2)
|
||||||
#define PCI32_ATE_PREF (0x1 << 3)
|
#define PCI32_ATE_PREF (0x1 << 3)
|
||||||
#define PCI32_ATE_BAR (0x1 << 4)
|
#define PCI32_ATE_BAR (0x1 << 4)
|
||||||
#define PCI32_ATE_ADDR_SHFT 12
|
#define PCI32_ATE_ADDR_SHFT 12
|
||||||
|
@ -117,8 +118,8 @@ struct pcibus_info {
|
||||||
|
|
||||||
extern int pcibr_init_provider(void);
|
extern int pcibr_init_provider(void);
|
||||||
extern void *pcibr_bus_fixup(struct pcibus_bussoft *, struct pci_controller *);
|
extern void *pcibr_bus_fixup(struct pcibus_bussoft *, struct pci_controller *);
|
||||||
extern dma_addr_t pcibr_dma_map(struct pci_dev *, unsigned long, size_t);
|
extern dma_addr_t pcibr_dma_map(struct pci_dev *, unsigned long, size_t, int type);
|
||||||
extern dma_addr_t pcibr_dma_map_consistent(struct pci_dev *, unsigned long, size_t);
|
extern dma_addr_t pcibr_dma_map_consistent(struct pci_dev *, unsigned long, size_t, int type);
|
||||||
extern void pcibr_dma_unmap(struct pci_dev *, dma_addr_t, int);
|
extern void pcibr_dma_unmap(struct pci_dev *, dma_addr_t, int);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
* License. See the file "COPYING" in the main directory of this archive
|
* License. See the file "COPYING" in the main directory of this archive
|
||||||
* for more details.
|
* for more details.
|
||||||
*
|
*
|
||||||
* Copyright (C) 1992 - 1997, 2000-2004 Silicon Graphics, Inc. All rights reserved.
|
* Copyright (C) 1992 - 1997, 2000-2005 Silicon Graphics, Inc. All rights reserved.
|
||||||
*/
|
*/
|
||||||
#ifndef _ASM_IA64_SN_PCI_PCIBUS_PROVIDER_H
|
#ifndef _ASM_IA64_SN_PCI_PCIBUS_PROVIDER_H
|
||||||
#define _ASM_IA64_SN_PCI_PCIBUS_PROVIDER_H
|
#define _ASM_IA64_SN_PCI_PCIBUS_PROVIDER_H
|
||||||
|
@ -45,13 +45,24 @@ struct pci_controller;
|
||||||
*/
|
*/
|
||||||
|
|
||||||
struct sn_pcibus_provider {
|
struct sn_pcibus_provider {
|
||||||
dma_addr_t (*dma_map)(struct pci_dev *, unsigned long, size_t);
|
dma_addr_t (*dma_map)(struct pci_dev *, unsigned long, size_t, int flags);
|
||||||
dma_addr_t (*dma_map_consistent)(struct pci_dev *, unsigned long, size_t);
|
dma_addr_t (*dma_map_consistent)(struct pci_dev *, unsigned long, size_t, int flags);
|
||||||
void (*dma_unmap)(struct pci_dev *, dma_addr_t, int);
|
void (*dma_unmap)(struct pci_dev *, dma_addr_t, int);
|
||||||
void * (*bus_fixup)(struct pcibus_bussoft *, struct pci_controller *);
|
void * (*bus_fixup)(struct pcibus_bussoft *, struct pci_controller *);
|
||||||
void (*force_interrupt)(struct sn_irq_info *);
|
void (*force_interrupt)(struct sn_irq_info *);
|
||||||
void (*target_interrupt)(struct sn_irq_info *);
|
void (*target_interrupt)(struct sn_irq_info *);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Flags used by the map interfaces
|
||||||
|
* bits 3:0 specifies format of passed in address
|
||||||
|
* bit 4 specifies that address is to be used for MSI
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define SN_DMA_ADDRTYPE(x) ((x) & 0xf)
|
||||||
|
#define SN_DMA_ADDR_PHYS 1 /* address is an xio address. */
|
||||||
|
#define SN_DMA_ADDR_XIO 2 /* address is phys memory */
|
||||||
|
#define SN_DMA_MSI 0x10 /* Bus address is to be used for MSI */
|
||||||
|
|
||||||
extern struct sn_pcibus_provider *sn_pci_provider[];
|
extern struct sn_pcibus_provider *sn_pci_provider[];
|
||||||
#endif /* _ASM_IA64_SN_PCI_PCIBUS_PROVIDER_H */
|
#endif /* _ASM_IA64_SN_PCI_PCIBUS_PROVIDER_H */
|
||||||
|
|
|
@ -3,13 +3,14 @@
|
||||||
* License. See the file "COPYING" in the main directory of this archive
|
* License. See the file "COPYING" in the main directory of this archive
|
||||||
* for more details.
|
* for more details.
|
||||||
*
|
*
|
||||||
* Copyright (C) 2003-2004 Silicon Graphics, Inc. All rights reserved.
|
* Copyright (C) 2003-2005 Silicon Graphics, Inc. All rights reserved.
|
||||||
*/
|
*/
|
||||||
#ifndef _ASM_IA64_SN_PCI_TIOCP_H
|
#ifndef _ASM_IA64_SN_PCI_TIOCP_H
|
||||||
#define _ASM_IA64_SN_PCI_TIOCP_H
|
#define _ASM_IA64_SN_PCI_TIOCP_H
|
||||||
|
|
||||||
#define TIOCP_HOST_INTR_ADDR 0x003FFFFFFFFFFFFFUL
|
#define TIOCP_HOST_INTR_ADDR 0x003FFFFFFFFFFFFFUL
|
||||||
#define TIOCP_PCI64_CMDTYPE_MEM (0x1ull << 60)
|
#define TIOCP_PCI64_CMDTYPE_MEM (0x1ull << 60)
|
||||||
|
#define TIOCP_PCI64_CMDTYPE_MSI (0x3ull << 60)
|
||||||
|
|
||||||
|
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
|
|
Loading…
Reference in New Issue