msi-altix.c 5.1 KB
Newer Older
1 2 3 4 5 6 7 8
/*
 * This file is subject to the terms and conditions of the GNU General Public
 * License.  See the file "COPYING" in the main directory of this archive
 * for more details.
 *
 * Copyright (C) 2006 Silicon Graphics, Inc.  All Rights Reserved.
 */

M
Mark Maule 已提交
9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28
#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
29
sn_msi_teardown(unsigned int irq)
M
Mark Maule 已提交
30 31 32 33 34 35 36 37 38
{
	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;

39
	sn_irq_info = sn_msi_info[irq].sn_irq_info;
M
Mark Maule 已提交
40 41 42 43 44 45 46 47
	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,
48
			       sn_msi_info[irq].pci_addr,
M
Mark Maule 已提交
49
			       PCI_DMA_FROMDEVICE);
50
	sn_msi_info[irq].pci_addr = 0;
M
Mark Maule 已提交
51 52 53 54 55 56 57 58

	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);
59
	sn_msi_info[irq].sn_irq_info = NULL;
M
Mark Maule 已提交
60 61 62

	return;
}
63 64

int
65
sn_msi_setup(struct pci_dev *pdev, unsigned int irq, struct msi_msg *msg)
66
{
M
Mark Maule 已提交
67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94
	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;

95
	status = sn_intr_alloc(nasid, widget, sn_irq_info, irq, -1, -1);
M
Mark Maule 已提交
96 97 98 99 100 101 102 103 104 105 106 107
	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;

108
	/*
M
Mark Maule 已提交
109
	 * Map the xio address into bus space
110
	 */
M
Mark Maule 已提交
111 112 113 114 115 116 117 118 119 120
	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;
	}

121 122
	sn_msi_info[irq].sn_irq_info = sn_irq_info;
	sn_msi_info[irq].pci_addr = bus_addr;
M
Mark Maule 已提交
123

124 125
	msg->address_hi = (u32)(bus_addr >> 32);
	msg->address_lo = (u32)(bus_addr & 0x00000000ffffffff);
M
Mark Maule 已提交
126 127 128 129 130

	/*
	 * In the SN platform, bit 16 is a "send vector" bit which
	 * must be present in order to move the vector through the system.
	 */
131
	msg->data = 0x100 + irq;
M
Mark Maule 已提交
132 133

#ifdef CONFIG_SMP
134
	set_irq_affinity_info(irq, sn_irq_info->irq_cpuid, 0);
M
Mark Maule 已提交
135 136 137 138 139 140
#endif

	return 0;
}

static void
141
sn_msi_target(unsigned int irq, cpumask_t cpu_mask, struct msi_msg *msg)
M
Mark Maule 已提交
142 143 144 145 146 147 148 149 150
{
	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;
151
	unsigned int cpu;
M
Mark Maule 已提交
152

153 154
	cpu = first_cpu(cpu_mask);
	sn_irq_info = sn_msi_info[irq].sn_irq_info;
M
Mark Maule 已提交
155 156 157 158 159 160 161 162 163 164 165
	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);

166
	bus_addr = (u64)(msg->address_hi) << 32 | (u64)(msg->address_lo);
M
Mark Maule 已提交
167
	(*provider->dma_unmap)(pdev, bus_addr, PCI_DMA_FROMDEVICE);
168
	sn_msi_info[irq].pci_addr = 0;
M
Mark Maule 已提交
169 170 171 172 173

	nasid = cpuid_to_nasid(cpu);
	slice = cpuid_to_slice(cpu);

	new_irq_info = sn_retarget_vector(sn_irq_info, nasid, slice);
174
	sn_msi_info[irq].sn_irq_info = new_irq_info;
M
Mark Maule 已提交
175 176 177 178 179 180 181 182 183 184 185 186
	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);

187 188 189
	sn_msi_info[irq].pci_addr = bus_addr;
	msg->address_hi = (u32)(bus_addr >> 32);
	msg->address_lo = (u32)(bus_addr & 0x00000000ffffffff);
M
Mark Maule 已提交
190 191 192
}

struct msi_ops sn_msi_ops = {
193
	.needs_64bit_address = 1,
M
Mark Maule 已提交
194 195 196 197 198 199 200 201 202 203 204
	.setup = sn_msi_setup,
	.teardown = sn_msi_teardown,
#ifdef CONFIG_SMP
	.target = sn_msi_target,
#endif
};

int
sn_msi_init(void)
{
	sn_msi_info =
205
		kzalloc(sizeof(struct sn_msi_info) * NR_IRQS, GFP_KERNEL);
M
Mark Maule 已提交
206 207 208 209 210
	if (! sn_msi_info)
		return -ENOMEM;

	msi_register(&sn_msi_ops);
	return 0;
211
}