I/OAT: Add support for MSI and MSI-X
Add support for MSI and MSI-X interrupt handling, including the ability to choose the desired interrupt method. Signed-off-by: Shannon Nelson <shannon.nelson@intel.com> Acked-by: David S. Miller <davem@davemloft.net> [bunk@kernel.org: drivers/dma/ioat_dma.c: make 3 functions static] Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
		
					parent
					
						
							
								8ab89567da
							
						
					
				
			
			
				commit
				
					
						3e037454bc
					
				
			
		
					 4 changed files with 304 additions and 66 deletions
				
			
		|  | @ -47,6 +47,65 @@ | ||||||
| static void ioat_dma_start_null_desc(struct ioat_dma_chan *ioat_chan); | static void ioat_dma_start_null_desc(struct ioat_dma_chan *ioat_chan); | ||||||
| static void ioat_dma_memcpy_cleanup(struct ioat_dma_chan *ioat_chan); | static void ioat_dma_memcpy_cleanup(struct ioat_dma_chan *ioat_chan); | ||||||
| 
 | 
 | ||||||
|  | static struct ioat_dma_chan *ioat_lookup_chan_by_index(struct ioatdma_device *device, | ||||||
|  | 						       int index) | ||||||
|  | { | ||||||
|  | 	return device->idx[index]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * ioat_dma_do_interrupt - handler used for single vector interrupt mode | ||||||
|  |  * @irq: interrupt id | ||||||
|  |  * @data: interrupt data | ||||||
|  |  */ | ||||||
|  | static irqreturn_t ioat_dma_do_interrupt(int irq, void *data) | ||||||
|  | { | ||||||
|  | 	struct ioatdma_device *instance = data; | ||||||
|  | 	struct ioat_dma_chan *ioat_chan; | ||||||
|  | 	unsigned long attnstatus; | ||||||
|  | 	int bit; | ||||||
|  | 	u8 intrctrl; | ||||||
|  | 
 | ||||||
|  | 	intrctrl = readb(instance->reg_base + IOAT_INTRCTRL_OFFSET); | ||||||
|  | 
 | ||||||
|  | 	if (!(intrctrl & IOAT_INTRCTRL_MASTER_INT_EN)) | ||||||
|  | 		return IRQ_NONE; | ||||||
|  | 
 | ||||||
|  | 	if (!(intrctrl & IOAT_INTRCTRL_INT_STATUS)) { | ||||||
|  | 		writeb(intrctrl, instance->reg_base + IOAT_INTRCTRL_OFFSET); | ||||||
|  | 		return IRQ_NONE; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	attnstatus = readl(instance->reg_base + IOAT_ATTNSTATUS_OFFSET); | ||||||
|  | 	for_each_bit(bit, &attnstatus, BITS_PER_LONG) { | ||||||
|  | 		ioat_chan = ioat_lookup_chan_by_index(instance, bit); | ||||||
|  | 		tasklet_schedule(&ioat_chan->cleanup_task); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	writeb(intrctrl, instance->reg_base + IOAT_INTRCTRL_OFFSET); | ||||||
|  | 	return IRQ_HANDLED; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * ioat_dma_do_interrupt_msix - handler used for vector-per-channel interrupt mode | ||||||
|  |  * @irq: interrupt id | ||||||
|  |  * @data: interrupt data | ||||||
|  |  */ | ||||||
|  | static irqreturn_t ioat_dma_do_interrupt_msix(int irq, void *data) | ||||||
|  | { | ||||||
|  | 	struct ioat_dma_chan *ioat_chan = data; | ||||||
|  | 
 | ||||||
|  | 	tasklet_schedule(&ioat_chan->cleanup_task); | ||||||
|  | 
 | ||||||
|  | 	return IRQ_HANDLED; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void ioat_dma_cleanup_tasklet(unsigned long data); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * ioat_dma_enumerate_channels - find and initialize the device's channels | ||||||
|  |  * @device: the device to be enumerated | ||||||
|  |  */ | ||||||
| static int ioat_dma_enumerate_channels(struct ioatdma_device *device) | static int ioat_dma_enumerate_channels(struct ioatdma_device *device) | ||||||
| { | { | ||||||
| 	u8 xfercap_scale; | 	u8 xfercap_scale; | ||||||
|  | @ -76,6 +135,11 @@ static int ioat_dma_enumerate_channels(struct ioatdma_device *device) | ||||||
| 		ioat_chan->common.device = &device->common; | 		ioat_chan->common.device = &device->common; | ||||||
| 		list_add_tail(&ioat_chan->common.device_node, | 		list_add_tail(&ioat_chan->common.device_node, | ||||||
| 			      &device->common.channels); | 			      &device->common.channels); | ||||||
|  | 		device->idx[i] = ioat_chan; | ||||||
|  | 		tasklet_init(&ioat_chan->cleanup_task, | ||||||
|  | 			     ioat_dma_cleanup_tasklet, | ||||||
|  | 			     (unsigned long) ioat_chan); | ||||||
|  | 		tasklet_disable(&ioat_chan->cleanup_task); | ||||||
| 	} | 	} | ||||||
| 	return device->common.chancnt; | 	return device->common.chancnt; | ||||||
| } | } | ||||||
|  | @ -234,6 +298,7 @@ static int ioat_dma_alloc_chan_resources(struct dma_chan *chan) | ||||||
| 	writel(((u64) ioat_chan->completion_addr) >> 32, | 	writel(((u64) ioat_chan->completion_addr) >> 32, | ||||||
| 	       ioat_chan->reg_base + IOAT_CHANCMP_OFFSET_HIGH); | 	       ioat_chan->reg_base + IOAT_CHANCMP_OFFSET_HIGH); | ||||||
| 
 | 
 | ||||||
|  | 	tasklet_enable(&ioat_chan->cleanup_task); | ||||||
| 	ioat_dma_start_null_desc(ioat_chan); | 	ioat_dma_start_null_desc(ioat_chan); | ||||||
| 	return i; | 	return i; | ||||||
| } | } | ||||||
|  | @ -245,9 +310,14 @@ static void ioat_dma_free_chan_resources(struct dma_chan *chan) | ||||||
| 	struct ioat_desc_sw *desc, *_desc; | 	struct ioat_desc_sw *desc, *_desc; | ||||||
| 	int in_use_descs = 0; | 	int in_use_descs = 0; | ||||||
| 
 | 
 | ||||||
|  | 	tasklet_disable(&ioat_chan->cleanup_task); | ||||||
| 	ioat_dma_memcpy_cleanup(ioat_chan); | 	ioat_dma_memcpy_cleanup(ioat_chan); | ||||||
| 
 | 
 | ||||||
|  | 	/* Delay 100ms after reset to allow internal DMA logic to quiesce
 | ||||||
|  | 	 * before removing DMA descriptor resources. | ||||||
|  | 	 */ | ||||||
| 	writeb(IOAT_CHANCMD_RESET, ioat_chan->reg_base + IOAT_CHANCMD_OFFSET); | 	writeb(IOAT_CHANCMD_RESET, ioat_chan->reg_base + IOAT_CHANCMD_OFFSET); | ||||||
|  | 	mdelay(100); | ||||||
| 
 | 
 | ||||||
| 	spin_lock_bh(&ioat_chan->desc_lock); | 	spin_lock_bh(&ioat_chan->desc_lock); | ||||||
| 	list_for_each_entry_safe(desc, _desc, &ioat_chan->used_desc, node) { | 	list_for_each_entry_safe(desc, _desc, &ioat_chan->used_desc, node) { | ||||||
|  | @ -276,6 +346,34 @@ static void ioat_dma_free_chan_resources(struct dma_chan *chan) | ||||||
| 			in_use_descs - 1); | 			in_use_descs - 1); | ||||||
| 
 | 
 | ||||||
| 	ioat_chan->last_completion = ioat_chan->completion_addr = 0; | 	ioat_chan->last_completion = ioat_chan->completion_addr = 0; | ||||||
|  | 	ioat_chan->pending = 0; | ||||||
|  | } | ||||||
|  | /**
 | ||||||
|  |  * ioat_dma_get_next_descriptor - return the next available descriptor | ||||||
|  |  * @ioat_chan: IOAT DMA channel handle | ||||||
|  |  * | ||||||
|  |  * Gets the next descriptor from the chain, and must be called with the | ||||||
|  |  * channel's desc_lock held.  Allocates more descriptors if the channel | ||||||
|  |  * has run out. | ||||||
|  |  */ | ||||||
|  | static struct ioat_desc_sw *ioat_dma_get_next_descriptor( | ||||||
|  | 						struct ioat_dma_chan *ioat_chan) | ||||||
|  | { | ||||||
|  | 	struct ioat_desc_sw *new = NULL; | ||||||
|  | 
 | ||||||
|  | 	if (!list_empty(&ioat_chan->free_desc)) { | ||||||
|  | 		new = to_ioat_desc(ioat_chan->free_desc.next); | ||||||
|  | 		list_del(&new->node); | ||||||
|  | 	} else { | ||||||
|  | 		/* try to get another desc */ | ||||||
|  | 		new = ioat_dma_alloc_descriptor(ioat_chan, GFP_ATOMIC); | ||||||
|  | 		/* will this ever happen? */ | ||||||
|  | 		/* TODO add upper limit on these */ | ||||||
|  | 		BUG_ON(!new); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	prefetch(new->hw); | ||||||
|  | 	return new; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static struct dma_async_tx_descriptor *ioat_dma_prep_memcpy( | static struct dma_async_tx_descriptor *ioat_dma_prep_memcpy( | ||||||
|  | @ -300,17 +398,7 @@ static struct dma_async_tx_descriptor *ioat_dma_prep_memcpy( | ||||||
| 
 | 
 | ||||||
| 	spin_lock_bh(&ioat_chan->desc_lock); | 	spin_lock_bh(&ioat_chan->desc_lock); | ||||||
| 	while (len) { | 	while (len) { | ||||||
| 		if (!list_empty(&ioat_chan->free_desc)) { | 		new = ioat_dma_get_next_descriptor(ioat_chan); | ||||||
| 			new = to_ioat_desc(ioat_chan->free_desc.next); |  | ||||||
| 			list_del(&new->node); |  | ||||||
| 		} else { |  | ||||||
| 			/* try to get another desc */ |  | ||||||
| 			new = ioat_dma_alloc_descriptor(ioat_chan, GFP_ATOMIC); |  | ||||||
| 			/* will this ever happen? */ |  | ||||||
| 			/* TODO add upper limit on these */ |  | ||||||
| 			BUG_ON(!new); |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		copy = min((u32) len, ioat_chan->xfercap); | 		copy = min((u32) len, ioat_chan->xfercap); | ||||||
| 
 | 
 | ||||||
| 		new->hw->size = copy; | 		new->hw->size = copy; | ||||||
|  | @ -360,6 +448,14 @@ static void ioat_dma_memcpy_issue_pending(struct dma_chan *chan) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static void ioat_dma_cleanup_tasklet(unsigned long data) | ||||||
|  | { | ||||||
|  | 	struct ioat_dma_chan *chan = (void *)data; | ||||||
|  | 	ioat_dma_memcpy_cleanup(chan); | ||||||
|  | 	writew(IOAT_CHANCTRL_INT_DISABLE, | ||||||
|  | 	       chan->reg_base + IOAT_CHANCTRL_OFFSET); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static void ioat_dma_memcpy_cleanup(struct ioat_dma_chan *ioat_chan) | static void ioat_dma_memcpy_cleanup(struct ioat_dma_chan *ioat_chan) | ||||||
| { | { | ||||||
| 	unsigned long phys_complete; | 	unsigned long phys_complete; | ||||||
|  | @ -397,6 +493,7 @@ static void ioat_dma_memcpy_cleanup(struct ioat_dma_chan *ioat_chan) | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	cookie = 0; | ||||||
| 	spin_lock_bh(&ioat_chan->desc_lock); | 	spin_lock_bh(&ioat_chan->desc_lock); | ||||||
| 	list_for_each_entry_safe(desc, _desc, &ioat_chan->used_desc, node) { | 	list_for_each_entry_safe(desc, _desc, &ioat_chan->used_desc, node) { | ||||||
| 
 | 
 | ||||||
|  | @ -509,48 +606,13 @@ static enum dma_status ioat_dma_is_complete(struct dma_chan *chan, | ||||||
| 
 | 
 | ||||||
| /* PCI API */ | /* PCI API */ | ||||||
| 
 | 
 | ||||||
| static irqreturn_t ioat_do_interrupt(int irq, void *data) |  | ||||||
| { |  | ||||||
| 	struct ioatdma_device *instance = data; |  | ||||||
| 	unsigned long attnstatus; |  | ||||||
| 	u8 intrctrl; |  | ||||||
| 
 |  | ||||||
| 	intrctrl = readb(instance->reg_base + IOAT_INTRCTRL_OFFSET); |  | ||||||
| 
 |  | ||||||
| 	if (!(intrctrl & IOAT_INTRCTRL_MASTER_INT_EN)) |  | ||||||
| 		return IRQ_NONE; |  | ||||||
| 
 |  | ||||||
| 	if (!(intrctrl & IOAT_INTRCTRL_INT_STATUS)) { |  | ||||||
| 		writeb(intrctrl, instance->reg_base + IOAT_INTRCTRL_OFFSET); |  | ||||||
| 		return IRQ_NONE; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	attnstatus = readl(instance->reg_base + IOAT_ATTNSTATUS_OFFSET); |  | ||||||
| 
 |  | ||||||
| 	printk(KERN_ERR "ioatdma: interrupt! status %lx\n", attnstatus); |  | ||||||
| 
 |  | ||||||
| 	writeb(intrctrl, instance->reg_base + IOAT_INTRCTRL_OFFSET); |  | ||||||
| 	return IRQ_HANDLED; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static void ioat_dma_start_null_desc(struct ioat_dma_chan *ioat_chan) | static void ioat_dma_start_null_desc(struct ioat_dma_chan *ioat_chan) | ||||||
| { | { | ||||||
| 	struct ioat_desc_sw *desc; | 	struct ioat_desc_sw *desc; | ||||||
| 
 | 
 | ||||||
| 	spin_lock_bh(&ioat_chan->desc_lock); | 	spin_lock_bh(&ioat_chan->desc_lock); | ||||||
| 
 | 
 | ||||||
| 	if (!list_empty(&ioat_chan->free_desc)) { | 	desc = ioat_dma_get_next_descriptor(ioat_chan); | ||||||
| 		desc = to_ioat_desc(ioat_chan->free_desc.next); |  | ||||||
| 		list_del(&desc->node); |  | ||||||
| 	} else { |  | ||||||
| 		/* try to get another desc */ |  | ||||||
| 		spin_unlock_bh(&ioat_chan->desc_lock); |  | ||||||
| 		desc = ioat_dma_alloc_descriptor(ioat_chan, GFP_KERNEL); |  | ||||||
| 		spin_lock_bh(&ioat_chan->desc_lock); |  | ||||||
| 		/* will this ever happen? */ |  | ||||||
| 		BUG_ON(!desc); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	desc->hw->ctl = IOAT_DMA_DESCRIPTOR_NUL; | 	desc->hw->ctl = IOAT_DMA_DESCRIPTOR_NUL; | ||||||
| 	desc->hw->next = 0; | 	desc->hw->next = 0; | ||||||
| 	desc->async_tx.ack = 1; | 	desc->async_tx.ack = 1; | ||||||
|  | @ -571,7 +633,11 @@ static void ioat_dma_start_null_desc(struct ioat_dma_chan *ioat_chan) | ||||||
|  */ |  */ | ||||||
| #define IOAT_TEST_SIZE 2000 | #define IOAT_TEST_SIZE 2000 | ||||||
| 
 | 
 | ||||||
| static int ioat_self_test(struct ioatdma_device *device) | /**
 | ||||||
|  |  * ioat_dma_self_test - Perform a IOAT transaction to verify the HW works. | ||||||
|  |  * @device: device to be tested | ||||||
|  |  */ | ||||||
|  | static int ioat_dma_self_test(struct ioatdma_device *device) | ||||||
| { | { | ||||||
| 	int i; | 	int i; | ||||||
| 	u8 *src; | 	u8 *src; | ||||||
|  | @ -639,6 +705,161 @@ out: | ||||||
| 	return err; | 	return err; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static char ioat_interrupt_style[32] = "msix"; | ||||||
|  | module_param_string(ioat_interrupt_style, ioat_interrupt_style, | ||||||
|  | 		    sizeof(ioat_interrupt_style), 0644); | ||||||
|  | MODULE_PARM_DESC(ioat_interrupt_style, | ||||||
|  | 		 "set ioat interrupt style: msix (default), " | ||||||
|  | 		 "msix-single-vector, msi, intx)"); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * ioat_dma_setup_interrupts - setup interrupt handler | ||||||
|  |  * @device: ioat device | ||||||
|  |  */ | ||||||
|  | static int ioat_dma_setup_interrupts(struct ioatdma_device *device) | ||||||
|  | { | ||||||
|  | 	struct ioat_dma_chan *ioat_chan; | ||||||
|  | 	int err, i, j, msixcnt; | ||||||
|  | 	u8 intrctrl = 0; | ||||||
|  | 
 | ||||||
|  | 	if (!strcmp(ioat_interrupt_style, "msix")) | ||||||
|  | 		goto msix; | ||||||
|  | 	if (!strcmp(ioat_interrupt_style, "msix-single-vector")) | ||||||
|  | 		goto msix_single_vector; | ||||||
|  | 	if (!strcmp(ioat_interrupt_style, "msi")) | ||||||
|  | 		goto msi; | ||||||
|  | 	if (!strcmp(ioat_interrupt_style, "intx")) | ||||||
|  | 		goto intx; | ||||||
|  | 
 | ||||||
|  | msix: | ||||||
|  | 	/* The number of MSI-X vectors should equal the number of channels */ | ||||||
|  | 	msixcnt = device->common.chancnt; | ||||||
|  | 	for (i = 0; i < msixcnt; i++) | ||||||
|  | 		device->msix_entries[i].entry = i; | ||||||
|  | 
 | ||||||
|  | 	err = pci_enable_msix(device->pdev, device->msix_entries, msixcnt); | ||||||
|  | 	if (err < 0) | ||||||
|  | 		goto msi; | ||||||
|  | 	if (err > 0) | ||||||
|  | 		goto msix_single_vector; | ||||||
|  | 
 | ||||||
|  | 	for (i = 0; i < msixcnt; i++) { | ||||||
|  | 		ioat_chan = ioat_lookup_chan_by_index(device, i); | ||||||
|  | 		err = request_irq(device->msix_entries[i].vector, | ||||||
|  | 				  ioat_dma_do_interrupt_msix, | ||||||
|  | 				  0, "ioat-msix", ioat_chan); | ||||||
|  | 		if (err) { | ||||||
|  | 			for (j = 0; j < i; j++) { | ||||||
|  | 				ioat_chan = | ||||||
|  | 					ioat_lookup_chan_by_index(device, j); | ||||||
|  | 				free_irq(device->msix_entries[j].vector, | ||||||
|  | 					 ioat_chan); | ||||||
|  | 			} | ||||||
|  | 			goto msix_single_vector; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	intrctrl |= IOAT_INTRCTRL_MSIX_VECTOR_CONTROL; | ||||||
|  | 	device->irq_mode = msix_multi_vector; | ||||||
|  | 	goto done; | ||||||
|  | 
 | ||||||
|  | msix_single_vector: | ||||||
|  | 	device->msix_entries[0].entry = 0; | ||||||
|  | 	err = pci_enable_msix(device->pdev, device->msix_entries, 1); | ||||||
|  | 	if (err) | ||||||
|  | 		goto msi; | ||||||
|  | 
 | ||||||
|  | 	err = request_irq(device->msix_entries[0].vector, ioat_dma_do_interrupt, | ||||||
|  | 			  0, "ioat-msix", device); | ||||||
|  | 	if (err) { | ||||||
|  | 		pci_disable_msix(device->pdev); | ||||||
|  | 		goto msi; | ||||||
|  | 	} | ||||||
|  | 	device->irq_mode = msix_single_vector; | ||||||
|  | 	goto done; | ||||||
|  | 
 | ||||||
|  | msi: | ||||||
|  | 	err = pci_enable_msi(device->pdev); | ||||||
|  | 	if (err) | ||||||
|  | 		goto intx; | ||||||
|  | 
 | ||||||
|  | 	err = request_irq(device->pdev->irq, ioat_dma_do_interrupt, | ||||||
|  | 			  0, "ioat-msi", device); | ||||||
|  | 	if (err) { | ||||||
|  | 		pci_disable_msi(device->pdev); | ||||||
|  | 		goto intx; | ||||||
|  | 	} | ||||||
|  | 	/*
 | ||||||
|  | 	 * CB 1.2 devices need a bit set in configuration space to enable MSI | ||||||
|  | 	 */ | ||||||
|  | 	if (device->version == IOAT_VER_1_2) { | ||||||
|  | 		u32 dmactrl; | ||||||
|  | 		pci_read_config_dword(device->pdev, | ||||||
|  | 				      IOAT_PCI_DMACTRL_OFFSET, &dmactrl); | ||||||
|  | 		dmactrl |= IOAT_PCI_DMACTRL_MSI_EN; | ||||||
|  | 		pci_write_config_dword(device->pdev, | ||||||
|  | 				       IOAT_PCI_DMACTRL_OFFSET, dmactrl); | ||||||
|  | 	} | ||||||
|  | 	device->irq_mode = msi; | ||||||
|  | 	goto done; | ||||||
|  | 
 | ||||||
|  | intx: | ||||||
|  | 	err = request_irq(device->pdev->irq, ioat_dma_do_interrupt, | ||||||
|  | 			  IRQF_SHARED, "ioat-intx", device); | ||||||
|  | 	if (err) | ||||||
|  | 		goto err_no_irq; | ||||||
|  | 	device->irq_mode = intx; | ||||||
|  | 
 | ||||||
|  | done: | ||||||
|  | 	intrctrl |= IOAT_INTRCTRL_MASTER_INT_EN; | ||||||
|  | 	writeb(intrctrl, device->reg_base + IOAT_INTRCTRL_OFFSET); | ||||||
|  | 	return 0; | ||||||
|  | 
 | ||||||
|  | err_no_irq: | ||||||
|  | 	/* Disable all interrupt generation */ | ||||||
|  | 	writeb(0, device->reg_base + IOAT_INTRCTRL_OFFSET); | ||||||
|  | 	dev_err(&device->pdev->dev, "no usable interrupts\n"); | ||||||
|  | 	device->irq_mode = none; | ||||||
|  | 	return -1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * ioat_dma_remove_interrupts - remove whatever interrupts were set | ||||||
|  |  * @device: ioat device | ||||||
|  |  */ | ||||||
|  | static void ioat_dma_remove_interrupts(struct ioatdma_device *device) | ||||||
|  | { | ||||||
|  | 	struct ioat_dma_chan *ioat_chan; | ||||||
|  | 	int i; | ||||||
|  | 
 | ||||||
|  | 	/* Disable all interrupt generation */ | ||||||
|  | 	writeb(0, device->reg_base + IOAT_INTRCTRL_OFFSET); | ||||||
|  | 
 | ||||||
|  | 	switch (device->irq_mode) { | ||||||
|  | 	case msix_multi_vector: | ||||||
|  | 		for (i = 0; i < device->common.chancnt; i++) { | ||||||
|  | 			ioat_chan = ioat_lookup_chan_by_index(device, i); | ||||||
|  | 			free_irq(device->msix_entries[i].vector, ioat_chan); | ||||||
|  | 		} | ||||||
|  | 		pci_disable_msix(device->pdev); | ||||||
|  | 		break; | ||||||
|  | 	case msix_single_vector: | ||||||
|  | 		free_irq(device->msix_entries[0].vector, device); | ||||||
|  | 		pci_disable_msix(device->pdev); | ||||||
|  | 		break; | ||||||
|  | 	case msi: | ||||||
|  | 		free_irq(device->pdev->irq, device); | ||||||
|  | 		pci_disable_msi(device->pdev); | ||||||
|  | 		break; | ||||||
|  | 	case intx: | ||||||
|  | 		free_irq(device->pdev->irq, device); | ||||||
|  | 		break; | ||||||
|  | 	case none: | ||||||
|  | 		dev_warn(&device->pdev->dev, | ||||||
|  | 			 "call to %s without interrupts setup\n", __func__); | ||||||
|  | 	} | ||||||
|  | 	device->irq_mode = none; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| struct ioatdma_device *ioat_dma_probe(struct pci_dev *pdev, | struct ioatdma_device *ioat_dma_probe(struct pci_dev *pdev, | ||||||
| 				      void __iomem *iobase) | 				      void __iomem *iobase) | ||||||
| { | { | ||||||
|  | @ -684,21 +905,16 @@ struct ioatdma_device *ioat_dma_probe(struct pci_dev *pdev, | ||||||
| 	device->common.device_issue_pending = ioat_dma_memcpy_issue_pending; | 	device->common.device_issue_pending = ioat_dma_memcpy_issue_pending; | ||||||
| 	device->common.device_dependency_added = ioat_dma_dependency_added; | 	device->common.device_dependency_added = ioat_dma_dependency_added; | ||||||
| 	device->common.dev = &pdev->dev; | 	device->common.dev = &pdev->dev; | ||||||
| 	printk(KERN_INFO "ioatdma: Intel(R) I/OAT DMA Engine found," | 	dev_err(&device->pdev->dev, | ||||||
|  | 		"ioatdma: Intel(R) I/OAT DMA Engine found," | ||||||
| 		" %d channels, device version 0x%02x\n", | 		" %d channels, device version 0x%02x\n", | ||||||
| 		device->common.chancnt, device->version); | 		device->common.chancnt, device->version); | ||||||
| 
 | 
 | ||||||
| 	pci_set_drvdata(pdev, device); | 	err = ioat_dma_setup_interrupts(device); | ||||||
| 	err = request_irq(pdev->irq, &ioat_do_interrupt, IRQF_SHARED, "ioat", |  | ||||||
| 		device); |  | ||||||
| 	if (err) | 	if (err) | ||||||
| 		goto err_irq; | 		goto err_setup_interrupts; | ||||||
| 
 | 
 | ||||||
| 	writeb(IOAT_INTRCTRL_MASTER_INT_EN, | 	err = ioat_dma_self_test(device); | ||||||
| 	       device->reg_base + IOAT_INTRCTRL_OFFSET); |  | ||||||
| 	pci_set_master(pdev); |  | ||||||
| 
 |  | ||||||
| 	err = ioat_self_test(device); |  | ||||||
| 	if (err) | 	if (err) | ||||||
| 		goto err_self_test; | 		goto err_self_test; | ||||||
| 
 | 
 | ||||||
|  | @ -707,8 +923,8 @@ struct ioatdma_device *ioat_dma_probe(struct pci_dev *pdev, | ||||||
| 	return device; | 	return device; | ||||||
| 
 | 
 | ||||||
| err_self_test: | err_self_test: | ||||||
| 	free_irq(device->pdev->irq, device); | 	ioat_dma_remove_interrupts(device); | ||||||
| err_irq: | err_setup_interrupts: | ||||||
| 	pci_pool_destroy(device->completion_pool); | 	pci_pool_destroy(device->completion_pool); | ||||||
| err_completion_pool: | err_completion_pool: | ||||||
| 	pci_pool_destroy(device->dma_pool); | 	pci_pool_destroy(device->dma_pool); | ||||||
|  | @ -716,7 +932,7 @@ err_dma_pool: | ||||||
| 	kfree(device); | 	kfree(device); | ||||||
| err_kzalloc: | err_kzalloc: | ||||||
| 	iounmap(iobase); | 	iounmap(iobase); | ||||||
| 	printk(KERN_ERR | 	dev_err(&device->pdev->dev, | ||||||
| 		"ioatdma: Intel(R) I/OAT DMA Engine initialization failed\n"); | 		"ioatdma: Intel(R) I/OAT DMA Engine initialization failed\n"); | ||||||
| 	return NULL; | 	return NULL; | ||||||
| } | } | ||||||
|  | @ -728,7 +944,7 @@ void ioat_dma_remove(struct ioatdma_device *device) | ||||||
| 
 | 
 | ||||||
| 	dma_async_device_unregister(&device->common); | 	dma_async_device_unregister(&device->common); | ||||||
| 
 | 
 | ||||||
| 	free_irq(device->pdev->irq, device); | 	ioat_dma_remove_interrupts(device); | ||||||
| 
 | 
 | ||||||
| 	pci_pool_destroy(device->dma_pool); | 	pci_pool_destroy(device->dma_pool); | ||||||
| 	pci_pool_destroy(device->completion_pool); | 	pci_pool_destroy(device->completion_pool); | ||||||
|  |  | ||||||
|  | @ -28,6 +28,14 @@ | ||||||
| #include <linux/cache.h> | #include <linux/cache.h> | ||||||
| #include <linux/pci_ids.h> | #include <linux/pci_ids.h> | ||||||
| 
 | 
 | ||||||
|  | enum ioat_interrupt { | ||||||
|  | 	none = 0, | ||||||
|  | 	msix_multi_vector = 1, | ||||||
|  | 	msix_single_vector = 2, | ||||||
|  | 	msi = 3, | ||||||
|  | 	intx = 4, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| #define IOAT_LOW_COMPLETION_MASK	0xffffffc0 | #define IOAT_LOW_COMPLETION_MASK	0xffffffc0 | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  | @ -46,6 +54,9 @@ struct ioatdma_device { | ||||||
| 	struct pci_pool *completion_pool; | 	struct pci_pool *completion_pool; | ||||||
| 	struct dma_device common; | 	struct dma_device common; | ||||||
| 	u8 version; | 	u8 version; | ||||||
|  | 	enum ioat_interrupt irq_mode; | ||||||
|  | 	struct msix_entry msix_entries[4]; | ||||||
|  | 	struct ioat_dma_chan *idx[4]; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  | @ -94,6 +105,7 @@ struct ioat_dma_chan { | ||||||
| 			u32 high; | 			u32 high; | ||||||
| 		}; | 		}; | ||||||
| 	} *completion_virt; | 	} *completion_virt; | ||||||
|  | 	struct tasklet_struct cleanup_task; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| /* wrapper around hardware descriptor format + additional software fields */ | /* wrapper around hardware descriptor format + additional software fields */ | ||||||
|  |  | ||||||
|  | @ -1,5 +1,5 @@ | ||||||
| /*
 | /*
 | ||||||
|  * Copyright(c) 2004 - 2006 Intel Corporation. All rights reserved. |  * Copyright(c) 2004 - 2007 Intel Corporation. All rights reserved. | ||||||
|  * |  * | ||||||
|  * This program is free software; you can redistribute it and/or modify it |  * 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 |  * under the terms of the GNU General Public License as published by the Free | ||||||
|  | @ -21,6 +21,9 @@ | ||||||
| #ifndef _IOAT_REGISTERS_H_ | #ifndef _IOAT_REGISTERS_H_ | ||||||
| #define _IOAT_REGISTERS_H_ | #define _IOAT_REGISTERS_H_ | ||||||
| 
 | 
 | ||||||
|  | #define IOAT_PCI_DMACTRL_OFFSET			0x48 | ||||||
|  | #define IOAT_PCI_DMACTRL_DMA_EN			0x00000001 | ||||||
|  | #define IOAT_PCI_DMACTRL_MSI_EN			0x00000002 | ||||||
| 
 | 
 | ||||||
| /* MMIO Device Registers */ | /* MMIO Device Registers */ | ||||||
| #define IOAT_CHANCNT_OFFSET			0x00	/*  8-bit */ | #define IOAT_CHANCNT_OFFSET			0x00	/*  8-bit */ | ||||||
|  | @ -39,6 +42,7 @@ | ||||||
| #define IOAT_INTRCTRL_MASTER_INT_EN		0x01	/* Master Interrupt Enable */ | #define IOAT_INTRCTRL_MASTER_INT_EN		0x01	/* Master Interrupt Enable */ | ||||||
| #define IOAT_INTRCTRL_INT_STATUS		0x02	/* ATTNSTATUS -or- Channel Int */ | #define IOAT_INTRCTRL_INT_STATUS		0x02	/* ATTNSTATUS -or- Channel Int */ | ||||||
| #define IOAT_INTRCTRL_INT			0x04	/* INT_STATUS -and- MASTER_INT_EN */ | #define IOAT_INTRCTRL_INT			0x04	/* INT_STATUS -and- MASTER_INT_EN */ | ||||||
|  | #define IOAT_INTRCTRL_MSIX_VECTOR_CONTROL	0x08    /* Enable all MSI-X vectors */ | ||||||
| 
 | 
 | ||||||
| #define IOAT_ATTNSTATUS_OFFSET			0x04	/* Each bit is a channel */ | #define IOAT_ATTNSTATUS_OFFSET			0x04	/* Each bit is a channel */ | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -8,6 +8,12 @@ | ||||||
|  */ |  */ | ||||||
| #include <asm/bitops.h> | #include <asm/bitops.h> | ||||||
| 
 | 
 | ||||||
|  | #define for_each_bit(bit, addr, size) \ | ||||||
|  | 	for ((bit) = find_first_bit((addr), (size)); \ | ||||||
|  | 	     (bit) < (size); \ | ||||||
|  | 	     (bit) = find_next_bit((addr), (size), (bit) + 1)) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| static __inline__ int get_bitmask_order(unsigned int count) | static __inline__ int get_bitmask_order(unsigned int count) | ||||||
| { | { | ||||||
| 	int order; | 	int order; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Shannon Nelson
				Shannon Nelson