| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1 | /* | 
|  | 2 | * File:	msi.c | 
|  | 3 | * Purpose:	PCI Message Signaled Interrupt (MSI) | 
|  | 4 | * | 
|  | 5 | * Copyright (C) 2003-2004 Intel | 
|  | 6 | * Copyright (C) Tom Long Nguyen (tom.l.nguyen@intel.com) | 
| Christoph Hellwig | aff1716 | 2016-07-12 18:20:17 +0900 | [diff] [blame] | 7 | * Copyright (C) 2016 Christoph Hellwig. | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 8 | */ | 
|  | 9 |  | 
| Eric W. Biederman | 1ce0337 | 2006-10-04 02:16:41 -0700 | [diff] [blame] | 10 | #include <linux/err.h> | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 11 | #include <linux/mm.h> | 
|  | 12 | #include <linux/irq.h> | 
|  | 13 | #include <linux/interrupt.h> | 
| Paul Gortmaker | 363c75d | 2011-05-27 09:37:25 -0400 | [diff] [blame] | 14 | #include <linux/export.h> | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 15 | #include <linux/ioport.h> | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 16 | #include <linux/pci.h> | 
|  | 17 | #include <linux/proc_fs.h> | 
| Eric W. Biederman | 3b7d192 | 2006-10-04 02:16:59 -0700 | [diff] [blame] | 18 | #include <linux/msi.h> | 
| Dan Williams | 4fdadeb | 2007-04-26 18:21:38 -0700 | [diff] [blame] | 19 | #include <linux/smp.h> | 
| Hidetoshi Seto | 500559a | 2009-08-10 10:14:15 +0900 | [diff] [blame] | 20 | #include <linux/errno.h> | 
|  | 21 | #include <linux/io.h> | 
| Tomasz Nowicki | be2021b | 2016-09-12 20:32:22 +0200 | [diff] [blame] | 22 | #include <linux/acpi_iort.h> | 
| Tejun Heo | 5a0e3ad | 2010-03-24 17:04:11 +0900 | [diff] [blame] | 23 | #include <linux/slab.h> | 
| Jiang Liu | 3878eae | 2014-11-11 21:02:18 +0800 | [diff] [blame] | 24 | #include <linux/irqdomain.h> | 
| David Daney | b6eec9b | 2015-10-08 15:10:49 -0700 | [diff] [blame] | 25 | #include <linux/of_irq.h> | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 26 |  | 
|  | 27 | #include "pci.h" | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 28 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 29 | static int pci_msi_enable = 1; | 
| Yijing Wang | 38737d8 | 2014-10-27 10:44:36 +0800 | [diff] [blame] | 30 | int pci_msi_ignore_mask; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 31 |  | 
| Bjorn Helgaas | 527eee2 | 2013-04-17 17:44:48 -0600 | [diff] [blame] | 32 | #define msix_table_size(flags)	((flags & PCI_MSIX_FLAGS_QSIZE) + 1) | 
|  | 33 |  | 
| Jiang Liu | 8e047ad | 2014-11-15 22:24:07 +0800 | [diff] [blame] | 34 | #ifdef CONFIG_PCI_MSI_IRQ_DOMAIN | 
|  | 35 | static struct irq_domain *pci_msi_default_domain; | 
|  | 36 | static DEFINE_MUTEX(pci_msi_domain_lock); | 
|  | 37 |  | 
|  | 38 | struct irq_domain * __weak arch_get_pci_msi_domain(struct pci_dev *dev) | 
|  | 39 | { | 
|  | 40 | return pci_msi_default_domain; | 
|  | 41 | } | 
|  | 42 |  | 
| Marc Zyngier | 020c312 | 2014-11-15 10:49:12 +0000 | [diff] [blame] | 43 | static struct irq_domain *pci_msi_get_domain(struct pci_dev *dev) | 
|  | 44 | { | 
| Marc Zyngier | d8a1cb7 | 2015-07-28 14:46:14 +0100 | [diff] [blame] | 45 | struct irq_domain *domain; | 
| Marc Zyngier | 020c312 | 2014-11-15 10:49:12 +0000 | [diff] [blame] | 46 |  | 
| Marc Zyngier | d8a1cb7 | 2015-07-28 14:46:14 +0100 | [diff] [blame] | 47 | domain = dev_get_msi_domain(&dev->dev); | 
|  | 48 | if (domain) | 
|  | 49 | return domain; | 
| Marc Zyngier | 020c312 | 2014-11-15 10:49:12 +0000 | [diff] [blame] | 50 |  | 
| Marc Zyngier | d8a1cb7 | 2015-07-28 14:46:14 +0100 | [diff] [blame] | 51 | return arch_get_pci_msi_domain(dev); | 
| Marc Zyngier | 020c312 | 2014-11-15 10:49:12 +0000 | [diff] [blame] | 52 | } | 
|  | 53 |  | 
| Jiang Liu | 8e047ad | 2014-11-15 22:24:07 +0800 | [diff] [blame] | 54 | static int pci_msi_setup_msi_irqs(struct pci_dev *dev, int nvec, int type) | 
|  | 55 | { | 
|  | 56 | struct irq_domain *domain; | 
|  | 57 |  | 
| Marc Zyngier | 020c312 | 2014-11-15 10:49:12 +0000 | [diff] [blame] | 58 | domain = pci_msi_get_domain(dev); | 
| Marc Zyngier | 3845d29 | 2015-12-04 10:28:14 -0600 | [diff] [blame] | 59 | if (domain && irq_domain_is_hierarchy(domain)) | 
| Jiang Liu | 8e047ad | 2014-11-15 22:24:07 +0800 | [diff] [blame] | 60 | return pci_msi_domain_alloc_irqs(domain, dev, nvec, type); | 
|  | 61 |  | 
|  | 62 | return arch_setup_msi_irqs(dev, nvec, type); | 
|  | 63 | } | 
|  | 64 |  | 
|  | 65 | static void pci_msi_teardown_msi_irqs(struct pci_dev *dev) | 
|  | 66 | { | 
|  | 67 | struct irq_domain *domain; | 
|  | 68 |  | 
| Marc Zyngier | 020c312 | 2014-11-15 10:49:12 +0000 | [diff] [blame] | 69 | domain = pci_msi_get_domain(dev); | 
| Marc Zyngier | 3845d29 | 2015-12-04 10:28:14 -0600 | [diff] [blame] | 70 | if (domain && irq_domain_is_hierarchy(domain)) | 
| Jiang Liu | 8e047ad | 2014-11-15 22:24:07 +0800 | [diff] [blame] | 71 | pci_msi_domain_free_irqs(domain, dev); | 
|  | 72 | else | 
|  | 73 | arch_teardown_msi_irqs(dev); | 
|  | 74 | } | 
|  | 75 | #else | 
|  | 76 | #define pci_msi_setup_msi_irqs		arch_setup_msi_irqs | 
|  | 77 | #define pci_msi_teardown_msi_irqs	arch_teardown_msi_irqs | 
|  | 78 | #endif | 
| Bjorn Helgaas | 527eee2 | 2013-04-17 17:44:48 -0600 | [diff] [blame] | 79 |  | 
| Adrian Bunk | 6a9e7f2 | 2007-12-11 23:19:41 +0100 | [diff] [blame] | 80 | /* Arch hooks */ | 
|  | 81 |  | 
| Thomas Petazzoni | 4287d82 | 2013-08-09 22:27:06 +0200 | [diff] [blame] | 82 | int __weak arch_setup_msi_irq(struct pci_dev *dev, struct msi_desc *desc) | 
|  | 83 | { | 
| Lorenzo Pieralisi | 2291ec0 | 2015-08-03 22:04:06 -0500 | [diff] [blame] | 84 | struct msi_controller *chip = dev->bus->msi; | 
| Thierry Reding | 0cbdcfc | 2013-08-09 22:27:08 +0200 | [diff] [blame] | 85 | int err; | 
|  | 86 |  | 
|  | 87 | if (!chip || !chip->setup_irq) | 
|  | 88 | return -EINVAL; | 
|  | 89 |  | 
|  | 90 | err = chip->setup_irq(chip, dev, desc); | 
|  | 91 | if (err < 0) | 
|  | 92 | return err; | 
|  | 93 |  | 
|  | 94 | irq_set_chip_data(desc->irq, chip); | 
|  | 95 |  | 
|  | 96 | return 0; | 
| Thomas Petazzoni | 4287d82 | 2013-08-09 22:27:06 +0200 | [diff] [blame] | 97 | } | 
|  | 98 |  | 
|  | 99 | void __weak arch_teardown_msi_irq(unsigned int irq) | 
|  | 100 | { | 
| Yijing Wang | c2791b8 | 2014-11-11 17:45:45 -0700 | [diff] [blame] | 101 | struct msi_controller *chip = irq_get_chip_data(irq); | 
| Thierry Reding | 0cbdcfc | 2013-08-09 22:27:08 +0200 | [diff] [blame] | 102 |  | 
|  | 103 | if (!chip || !chip->teardown_irq) | 
|  | 104 | return; | 
|  | 105 |  | 
|  | 106 | chip->teardown_irq(chip, irq); | 
| Thomas Petazzoni | 4287d82 | 2013-08-09 22:27:06 +0200 | [diff] [blame] | 107 | } | 
|  | 108 |  | 
| Thomas Petazzoni | 4287d82 | 2013-08-09 22:27:06 +0200 | [diff] [blame] | 109 | int __weak arch_setup_msi_irqs(struct pci_dev *dev, int nvec, int type) | 
| Adrian Bunk | 6a9e7f2 | 2007-12-11 23:19:41 +0100 | [diff] [blame] | 110 | { | 
| Lucas Stach | 339e5b4 | 2015-09-18 13:58:34 -0500 | [diff] [blame] | 111 | struct msi_controller *chip = dev->bus->msi; | 
| Adrian Bunk | 6a9e7f2 | 2007-12-11 23:19:41 +0100 | [diff] [blame] | 112 | struct msi_desc *entry; | 
|  | 113 | int ret; | 
|  | 114 |  | 
| Lucas Stach | 339e5b4 | 2015-09-18 13:58:34 -0500 | [diff] [blame] | 115 | if (chip && chip->setup_irqs) | 
|  | 116 | return chip->setup_irqs(chip, dev, nvec, type); | 
| Matthew Wilcox | 1c8d7b0 | 2009-03-17 08:54:10 -0400 | [diff] [blame] | 117 | /* | 
|  | 118 | * If an architecture wants to support multiple MSI, it needs to | 
|  | 119 | * override arch_setup_msi_irqs() | 
|  | 120 | */ | 
|  | 121 | if (type == PCI_CAP_ID_MSI && nvec > 1) | 
|  | 122 | return 1; | 
|  | 123 |  | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 124 | for_each_pci_msi_entry(entry, dev) { | 
| Adrian Bunk | 6a9e7f2 | 2007-12-11 23:19:41 +0100 | [diff] [blame] | 125 | ret = arch_setup_msi_irq(dev, entry); | 
| Michael Ellerman | b5fbf53 | 2009-02-11 22:27:02 +1100 | [diff] [blame] | 126 | if (ret < 0) | 
| Adrian Bunk | 6a9e7f2 | 2007-12-11 23:19:41 +0100 | [diff] [blame] | 127 | return ret; | 
| Michael Ellerman | b5fbf53 | 2009-02-11 22:27:02 +1100 | [diff] [blame] | 128 | if (ret > 0) | 
|  | 129 | return -ENOSPC; | 
| Adrian Bunk | 6a9e7f2 | 2007-12-11 23:19:41 +0100 | [diff] [blame] | 130 | } | 
|  | 131 |  | 
|  | 132 | return 0; | 
|  | 133 | } | 
|  | 134 |  | 
| Thomas Petazzoni | 4287d82 | 2013-08-09 22:27:06 +0200 | [diff] [blame] | 135 | /* | 
|  | 136 | * We have a default implementation available as a separate non-weak | 
|  | 137 | * function, as it is used by the Xen x86 PCI code | 
|  | 138 | */ | 
| Thomas Gleixner | 1525bf0 | 2010-10-06 16:05:35 -0400 | [diff] [blame] | 139 | void default_teardown_msi_irqs(struct pci_dev *dev) | 
| Adrian Bunk | 6a9e7f2 | 2007-12-11 23:19:41 +0100 | [diff] [blame] | 140 | { | 
| Jiang Liu | 63a7b17 | 2014-11-06 22:20:32 +0800 | [diff] [blame] | 141 | int i; | 
| Adrian Bunk | 6a9e7f2 | 2007-12-11 23:19:41 +0100 | [diff] [blame] | 142 | struct msi_desc *entry; | 
|  | 143 |  | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 144 | for_each_pci_msi_entry(entry, dev) | 
| Jiang Liu | 63a7b17 | 2014-11-06 22:20:32 +0800 | [diff] [blame] | 145 | if (entry->irq) | 
|  | 146 | for (i = 0; i < entry->nvec_used; i++) | 
|  | 147 | arch_teardown_msi_irq(entry->irq + i); | 
| Adrian Bunk | 6a9e7f2 | 2007-12-11 23:19:41 +0100 | [diff] [blame] | 148 | } | 
|  | 149 |  | 
| Thomas Petazzoni | 4287d82 | 2013-08-09 22:27:06 +0200 | [diff] [blame] | 150 | void __weak arch_teardown_msi_irqs(struct pci_dev *dev) | 
|  | 151 | { | 
|  | 152 | return default_teardown_msi_irqs(dev); | 
|  | 153 | } | 
| Konrad Rzeszutek Wilk | 76ccc29 | 2011-12-16 17:38:18 -0500 | [diff] [blame] | 154 |  | 
| DuanZhenzhong | ac8344c | 2013-12-04 13:09:16 +0800 | [diff] [blame] | 155 | static void default_restore_msi_irq(struct pci_dev *dev, int irq) | 
| Konrad Rzeszutek Wilk | 76ccc29 | 2011-12-16 17:38:18 -0500 | [diff] [blame] | 156 | { | 
|  | 157 | struct msi_desc *entry; | 
|  | 158 |  | 
|  | 159 | entry = NULL; | 
|  | 160 | if (dev->msix_enabled) { | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 161 | for_each_pci_msi_entry(entry, dev) { | 
| Konrad Rzeszutek Wilk | 76ccc29 | 2011-12-16 17:38:18 -0500 | [diff] [blame] | 162 | if (irq == entry->irq) | 
|  | 163 | break; | 
|  | 164 | } | 
|  | 165 | } else if (dev->msi_enabled)  { | 
|  | 166 | entry = irq_get_msi_desc(irq); | 
|  | 167 | } | 
|  | 168 |  | 
|  | 169 | if (entry) | 
| Jiang Liu | 83a1891 | 2014-11-09 23:10:34 +0800 | [diff] [blame] | 170 | __pci_write_msi_msg(entry, &entry->msg); | 
| Konrad Rzeszutek Wilk | 76ccc29 | 2011-12-16 17:38:18 -0500 | [diff] [blame] | 171 | } | 
| Thomas Petazzoni | 4287d82 | 2013-08-09 22:27:06 +0200 | [diff] [blame] | 172 |  | 
| DuanZhenzhong | ac8344c | 2013-12-04 13:09:16 +0800 | [diff] [blame] | 173 | void __weak arch_restore_msi_irqs(struct pci_dev *dev) | 
| Thomas Petazzoni | 4287d82 | 2013-08-09 22:27:06 +0200 | [diff] [blame] | 174 | { | 
| DuanZhenzhong | ac8344c | 2013-12-04 13:09:16 +0800 | [diff] [blame] | 175 | return default_restore_msi_irqs(dev); | 
| Thomas Petazzoni | 4287d82 | 2013-08-09 22:27:06 +0200 | [diff] [blame] | 176 | } | 
| Konrad Rzeszutek Wilk | 76ccc29 | 2011-12-16 17:38:18 -0500 | [diff] [blame] | 177 |  | 
| Matthew Wilcox | bffac3c | 2009-01-21 19:19:19 -0500 | [diff] [blame] | 178 | static inline __attribute_const__ u32 msi_mask(unsigned x) | 
|  | 179 | { | 
| Matthew Wilcox | 0b49ec37a2 | 2009-02-08 20:27:47 -0700 | [diff] [blame] | 180 | /* Don't shift by >= width of type */ | 
|  | 181 | if (x >= 5) | 
|  | 182 | return 0xffffffff; | 
|  | 183 | return (1 << (1 << x)) - 1; | 
| Matthew Wilcox | bffac3c | 2009-01-21 19:19:19 -0500 | [diff] [blame] | 184 | } | 
|  | 185 |  | 
| Matthew Wilcox | ce6fce4 | 2008-07-25 15:42:58 -0600 | [diff] [blame] | 186 | /* | 
|  | 187 | * PCI 2.3 does not specify mask bits for each MSI interrupt.  Attempting to | 
|  | 188 | * mask all MSI interrupts by clearing the MSI enable bit does not work | 
|  | 189 | * reliably as devices without an INTx disable bit will then generate a | 
|  | 190 | * level IRQ which will never be cleared. | 
| Matthew Wilcox | ce6fce4 | 2008-07-25 15:42:58 -0600 | [diff] [blame] | 191 | */ | 
| Thomas Gleixner | 23ed8d5 | 2014-11-23 11:55:58 +0100 | [diff] [blame] | 192 | u32 __pci_msi_desc_mask_irq(struct msi_desc *desc, u32 mask, u32 flag) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 193 | { | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 194 | u32 mask_bits = desc->masked; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 195 |  | 
| Yijing Wang | 38737d8 | 2014-10-27 10:44:36 +0800 | [diff] [blame] | 196 | if (pci_msi_ignore_mask || !desc->msi_attrib.maskbit) | 
| Hidetoshi Seto | 12abb8b | 2009-06-24 12:08:09 +0900 | [diff] [blame] | 197 | return 0; | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 198 |  | 
|  | 199 | mask_bits &= ~mask; | 
|  | 200 | mask_bits |= flag; | 
| Jiang Liu | e39758e | 2015-07-09 16:00:43 +0800 | [diff] [blame] | 201 | pci_write_config_dword(msi_desc_to_pci_dev(desc), desc->mask_pos, | 
|  | 202 | mask_bits); | 
| Hidetoshi Seto | 12abb8b | 2009-06-24 12:08:09 +0900 | [diff] [blame] | 203 |  | 
|  | 204 | return mask_bits; | 
|  | 205 | } | 
|  | 206 |  | 
|  | 207 | static void msi_mask_irq(struct msi_desc *desc, u32 mask, u32 flag) | 
|  | 208 | { | 
| Thomas Gleixner | 23ed8d5 | 2014-11-23 11:55:58 +0100 | [diff] [blame] | 209 | desc->masked = __pci_msi_desc_mask_irq(desc, mask, flag); | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 210 | } | 
|  | 211 |  | 
| Christoph Hellwig | 5eb6d66 | 2016-07-12 18:20:14 +0900 | [diff] [blame] | 212 | static void __iomem *pci_msix_desc_addr(struct msi_desc *desc) | 
|  | 213 | { | 
|  | 214 | return desc->mask_base + | 
|  | 215 | desc->msi_attrib.entry_nr * PCI_MSIX_ENTRY_SIZE; | 
|  | 216 | } | 
|  | 217 |  | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 218 | /* | 
|  | 219 | * This internal function does not flush PCI writes to the device. | 
|  | 220 | * All users must ensure that they read from the device before either | 
|  | 221 | * assuming that the device state is up to date, or returning out of this | 
|  | 222 | * file.  This saves a few milliseconds when initialising devices with lots | 
|  | 223 | * of MSI-X interrupts. | 
|  | 224 | */ | 
| Thomas Gleixner | 23ed8d5 | 2014-11-23 11:55:58 +0100 | [diff] [blame] | 225 | u32 __pci_msix_desc_mask_irq(struct msi_desc *desc, u32 flag) | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 226 | { | 
|  | 227 | u32 mask_bits = desc->masked; | 
| Yijing Wang | 38737d8 | 2014-10-27 10:44:36 +0800 | [diff] [blame] | 228 |  | 
|  | 229 | if (pci_msi_ignore_mask) | 
|  | 230 | return 0; | 
|  | 231 |  | 
| Sheng Yang | 8d80528 | 2010-11-11 15:46:55 +0800 | [diff] [blame] | 232 | mask_bits &= ~PCI_MSIX_ENTRY_CTRL_MASKBIT; | 
|  | 233 | if (flag) | 
|  | 234 | mask_bits |= PCI_MSIX_ENTRY_CTRL_MASKBIT; | 
| Christoph Hellwig | 5eb6d66 | 2016-07-12 18:20:14 +0900 | [diff] [blame] | 235 | writel(mask_bits, pci_msix_desc_addr(desc) + PCI_MSIX_ENTRY_VECTOR_CTRL); | 
| Hidetoshi Seto | 12abb8b | 2009-06-24 12:08:09 +0900 | [diff] [blame] | 236 |  | 
|  | 237 | return mask_bits; | 
|  | 238 | } | 
|  | 239 |  | 
|  | 240 | static void msix_mask_irq(struct msi_desc *desc, u32 flag) | 
|  | 241 | { | 
| Thomas Gleixner | 23ed8d5 | 2014-11-23 11:55:58 +0100 | [diff] [blame] | 242 | desc->masked = __pci_msix_desc_mask_irq(desc, flag); | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 243 | } | 
|  | 244 |  | 
| Thomas Gleixner | 1c9db52 | 2010-09-28 16:46:51 +0200 | [diff] [blame] | 245 | static void msi_set_mask_bit(struct irq_data *data, u32 flag) | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 246 | { | 
| Jiang Liu | c391f26 | 2015-06-01 16:05:41 +0800 | [diff] [blame] | 247 | struct msi_desc *desc = irq_data_get_msi_desc(data); | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 248 |  | 
|  | 249 | if (desc->msi_attrib.is_msix) { | 
|  | 250 | msix_mask_irq(desc, flag); | 
|  | 251 | readl(desc->mask_base);		/* Flush write to device */ | 
| Matthew Wilcox | 24d2755 | 2009-03-17 08:54:06 -0400 | [diff] [blame] | 252 | } else { | 
| Yijing Wang | a281b78 | 2014-07-08 10:08:55 +0800 | [diff] [blame] | 253 | unsigned offset = data->irq - desc->irq; | 
| Matthew Wilcox | 1c8d7b0 | 2009-03-17 08:54:10 -0400 | [diff] [blame] | 254 | msi_mask_irq(desc, 1 << offset, flag << offset); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 255 | } | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 256 | } | 
|  | 257 |  | 
| Thomas Gleixner | 23ed8d5 | 2014-11-23 11:55:58 +0100 | [diff] [blame] | 258 | /** | 
|  | 259 | * pci_msi_mask_irq - Generic irq chip callback to mask PCI/MSI interrupts | 
|  | 260 | * @data:	pointer to irqdata associated to that interrupt | 
|  | 261 | */ | 
|  | 262 | void pci_msi_mask_irq(struct irq_data *data) | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 263 | { | 
| Thomas Gleixner | 1c9db52 | 2010-09-28 16:46:51 +0200 | [diff] [blame] | 264 | msi_set_mask_bit(data, 1); | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 265 | } | 
| Jake Oshins | a4289dc | 2015-12-10 17:52:59 +0000 | [diff] [blame] | 266 | EXPORT_SYMBOL_GPL(pci_msi_mask_irq); | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 267 |  | 
| Thomas Gleixner | 23ed8d5 | 2014-11-23 11:55:58 +0100 | [diff] [blame] | 268 | /** | 
|  | 269 | * pci_msi_unmask_irq - Generic irq chip callback to unmask PCI/MSI interrupts | 
|  | 270 | * @data:	pointer to irqdata associated to that interrupt | 
|  | 271 | */ | 
|  | 272 | void pci_msi_unmask_irq(struct irq_data *data) | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 273 | { | 
| Thomas Gleixner | 1c9db52 | 2010-09-28 16:46:51 +0200 | [diff] [blame] | 274 | msi_set_mask_bit(data, 0); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 275 | } | 
| Jake Oshins | a4289dc | 2015-12-10 17:52:59 +0000 | [diff] [blame] | 276 | EXPORT_SYMBOL_GPL(pci_msi_unmask_irq); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 277 |  | 
| DuanZhenzhong | ac8344c | 2013-12-04 13:09:16 +0800 | [diff] [blame] | 278 | void default_restore_msi_irqs(struct pci_dev *dev) | 
|  | 279 | { | 
|  | 280 | struct msi_desc *entry; | 
|  | 281 |  | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 282 | for_each_pci_msi_entry(entry, dev) | 
| DuanZhenzhong | ac8344c | 2013-12-04 13:09:16 +0800 | [diff] [blame] | 283 | default_restore_msi_irq(dev, entry->irq); | 
| DuanZhenzhong | ac8344c | 2013-12-04 13:09:16 +0800 | [diff] [blame] | 284 | } | 
|  | 285 |  | 
| Jiang Liu | 891d4a4 | 2014-11-09 23:10:33 +0800 | [diff] [blame] | 286 | void __pci_read_msi_msg(struct msi_desc *entry, struct msi_msg *msg) | 
| Eric W. Biederman | 0366f8f | 2006-10-04 02:16:33 -0700 | [diff] [blame] | 287 | { | 
| Jiang Liu | e39758e | 2015-07-09 16:00:43 +0800 | [diff] [blame] | 288 | struct pci_dev *dev = msi_desc_to_pci_dev(entry); | 
|  | 289 |  | 
|  | 290 | BUG_ON(dev->current_state != PCI_D0); | 
| Eric W. Biederman | 0366f8f | 2006-10-04 02:16:33 -0700 | [diff] [blame] | 291 |  | 
| Ben Hutchings | 30da552 | 2010-07-23 14:56:28 +0100 | [diff] [blame] | 292 | if (entry->msi_attrib.is_msix) { | 
| Christoph Hellwig | 5eb6d66 | 2016-07-12 18:20:14 +0900 | [diff] [blame] | 293 | void __iomem *base = pci_msix_desc_addr(entry); | 
| Ben Hutchings | 30da552 | 2010-07-23 14:56:28 +0100 | [diff] [blame] | 294 |  | 
|  | 295 | msg->address_lo = readl(base + PCI_MSIX_ENTRY_LOWER_ADDR); | 
|  | 296 | msg->address_hi = readl(base + PCI_MSIX_ENTRY_UPPER_ADDR); | 
|  | 297 | msg->data = readl(base + PCI_MSIX_ENTRY_DATA); | 
|  | 298 | } else { | 
| Bjorn Helgaas | f532216 | 2013-04-17 17:34:36 -0600 | [diff] [blame] | 299 | int pos = dev->msi_cap; | 
| Ben Hutchings | 30da552 | 2010-07-23 14:56:28 +0100 | [diff] [blame] | 300 | u16 data; | 
|  | 301 |  | 
| Bjorn Helgaas | 9925ad0 | 2013-04-17 17:39:57 -0600 | [diff] [blame] | 302 | pci_read_config_dword(dev, pos + PCI_MSI_ADDRESS_LO, | 
|  | 303 | &msg->address_lo); | 
| Ben Hutchings | 30da552 | 2010-07-23 14:56:28 +0100 | [diff] [blame] | 304 | if (entry->msi_attrib.is_64) { | 
| Bjorn Helgaas | 9925ad0 | 2013-04-17 17:39:57 -0600 | [diff] [blame] | 305 | pci_read_config_dword(dev, pos + PCI_MSI_ADDRESS_HI, | 
|  | 306 | &msg->address_hi); | 
| Bjorn Helgaas | 2f22134 | 2013-04-17 17:41:13 -0600 | [diff] [blame] | 307 | pci_read_config_word(dev, pos + PCI_MSI_DATA_64, &data); | 
| Ben Hutchings | 30da552 | 2010-07-23 14:56:28 +0100 | [diff] [blame] | 308 | } else { | 
|  | 309 | msg->address_hi = 0; | 
| Bjorn Helgaas | 2f22134 | 2013-04-17 17:41:13 -0600 | [diff] [blame] | 310 | pci_read_config_word(dev, pos + PCI_MSI_DATA_32, &data); | 
| Ben Hutchings | 30da552 | 2010-07-23 14:56:28 +0100 | [diff] [blame] | 311 | } | 
|  | 312 | msg->data = data; | 
|  | 313 | } | 
| Eric W. Biederman | 0366f8f | 2006-10-04 02:16:33 -0700 | [diff] [blame] | 314 | } | 
|  | 315 |  | 
| Jiang Liu | 83a1891 | 2014-11-09 23:10:34 +0800 | [diff] [blame] | 316 | void __pci_write_msi_msg(struct msi_desc *entry, struct msi_msg *msg) | 
| Yinghai Lu | 3145e94 | 2008-12-05 18:58:34 -0800 | [diff] [blame] | 317 | { | 
| Jiang Liu | e39758e | 2015-07-09 16:00:43 +0800 | [diff] [blame] | 318 | struct pci_dev *dev = msi_desc_to_pci_dev(entry); | 
|  | 319 |  | 
|  | 320 | if (dev->current_state != PCI_D0) { | 
| Ben Hutchings | fcd097f | 2010-06-17 20:16:36 +0100 | [diff] [blame] | 321 | /* Don't touch the hardware now */ | 
|  | 322 | } else if (entry->msi_attrib.is_msix) { | 
| Christoph Hellwig | 5eb6d66 | 2016-07-12 18:20:14 +0900 | [diff] [blame] | 323 | void __iomem *base = pci_msix_desc_addr(entry); | 
| Matthew Wilcox | 24d2755 | 2009-03-17 08:54:06 -0400 | [diff] [blame] | 324 |  | 
| Hidetoshi Seto | 2c21fd4 | 2009-06-23 17:40:04 +0900 | [diff] [blame] | 325 | writel(msg->address_lo, base + PCI_MSIX_ENTRY_LOWER_ADDR); | 
|  | 326 | writel(msg->address_hi, base + PCI_MSIX_ENTRY_UPPER_ADDR); | 
|  | 327 | writel(msg->data, base + PCI_MSIX_ENTRY_DATA); | 
| Matthew Wilcox | 24d2755 | 2009-03-17 08:54:06 -0400 | [diff] [blame] | 328 | } else { | 
| Bjorn Helgaas | f532216 | 2013-04-17 17:34:36 -0600 | [diff] [blame] | 329 | int pos = dev->msi_cap; | 
| Matthew Wilcox | 1c8d7b0 | 2009-03-17 08:54:10 -0400 | [diff] [blame] | 330 | u16 msgctl; | 
|  | 331 |  | 
| Bjorn Helgaas | f84ecd2 | 2013-04-17 17:38:32 -0600 | [diff] [blame] | 332 | pci_read_config_word(dev, pos + PCI_MSI_FLAGS, &msgctl); | 
| Matthew Wilcox | 1c8d7b0 | 2009-03-17 08:54:10 -0400 | [diff] [blame] | 333 | msgctl &= ~PCI_MSI_FLAGS_QSIZE; | 
|  | 334 | msgctl |= entry->msi_attrib.multiple << 4; | 
| Bjorn Helgaas | f84ecd2 | 2013-04-17 17:38:32 -0600 | [diff] [blame] | 335 | pci_write_config_word(dev, pos + PCI_MSI_FLAGS, msgctl); | 
| Eric W. Biederman | 0366f8f | 2006-10-04 02:16:33 -0700 | [diff] [blame] | 336 |  | 
| Bjorn Helgaas | 9925ad0 | 2013-04-17 17:39:57 -0600 | [diff] [blame] | 337 | pci_write_config_dword(dev, pos + PCI_MSI_ADDRESS_LO, | 
|  | 338 | msg->address_lo); | 
| Eric W. Biederman | 0366f8f | 2006-10-04 02:16:33 -0700 | [diff] [blame] | 339 | if (entry->msi_attrib.is_64) { | 
| Bjorn Helgaas | 9925ad0 | 2013-04-17 17:39:57 -0600 | [diff] [blame] | 340 | pci_write_config_dword(dev, pos + PCI_MSI_ADDRESS_HI, | 
|  | 341 | msg->address_hi); | 
| Bjorn Helgaas | 2f22134 | 2013-04-17 17:41:13 -0600 | [diff] [blame] | 342 | pci_write_config_word(dev, pos + PCI_MSI_DATA_64, | 
|  | 343 | msg->data); | 
| Eric W. Biederman | 0366f8f | 2006-10-04 02:16:33 -0700 | [diff] [blame] | 344 | } else { | 
| Bjorn Helgaas | 2f22134 | 2013-04-17 17:41:13 -0600 | [diff] [blame] | 345 | pci_write_config_word(dev, pos + PCI_MSI_DATA_32, | 
|  | 346 | msg->data); | 
| Eric W. Biederman | 0366f8f | 2006-10-04 02:16:33 -0700 | [diff] [blame] | 347 | } | 
| Eric W. Biederman | 0366f8f | 2006-10-04 02:16:33 -0700 | [diff] [blame] | 348 | } | 
| Eric W. Biederman | 392ee1e | 2007-03-08 13:04:57 -0700 | [diff] [blame] | 349 | entry->msg = *msg; | 
| Eric W. Biederman | 0366f8f | 2006-10-04 02:16:33 -0700 | [diff] [blame] | 350 | } | 
|  | 351 |  | 
| Jiang Liu | 83a1891 | 2014-11-09 23:10:34 +0800 | [diff] [blame] | 352 | void pci_write_msi_msg(unsigned int irq, struct msi_msg *msg) | 
| Yinghai Lu | 3145e94 | 2008-12-05 18:58:34 -0800 | [diff] [blame] | 353 | { | 
| Thomas Gleixner | dced35a | 2011-03-28 17:49:12 +0200 | [diff] [blame] | 354 | struct msi_desc *entry = irq_get_msi_desc(irq); | 
| Yinghai Lu | 3145e94 | 2008-12-05 18:58:34 -0800 | [diff] [blame] | 355 |  | 
| Jiang Liu | 83a1891 | 2014-11-09 23:10:34 +0800 | [diff] [blame] | 356 | __pci_write_msi_msg(entry, msg); | 
| Yinghai Lu | 3145e94 | 2008-12-05 18:58:34 -0800 | [diff] [blame] | 357 | } | 
| Jiang Liu | 83a1891 | 2014-11-09 23:10:34 +0800 | [diff] [blame] | 358 | EXPORT_SYMBOL_GPL(pci_write_msi_msg); | 
| Yinghai Lu | 3145e94 | 2008-12-05 18:58:34 -0800 | [diff] [blame] | 359 |  | 
| Hidetoshi Seto | f56e448 | 2009-08-06 11:32:51 +0900 | [diff] [blame] | 360 | static void free_msi_irqs(struct pci_dev *dev) | 
|  | 361 | { | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 362 | struct list_head *msi_list = dev_to_msi_list(&dev->dev); | 
| Hidetoshi Seto | f56e448 | 2009-08-06 11:32:51 +0900 | [diff] [blame] | 363 | struct msi_desc *entry, *tmp; | 
| Greg Kroah-Hartman | 1c51b50 | 2013-12-19 12:30:17 -0800 | [diff] [blame] | 364 | struct attribute **msi_attrs; | 
|  | 365 | struct device_attribute *dev_attr; | 
| Jiang Liu | 63a7b17 | 2014-11-06 22:20:32 +0800 | [diff] [blame] | 366 | int i, count = 0; | 
| Hidetoshi Seto | f56e448 | 2009-08-06 11:32:51 +0900 | [diff] [blame] | 367 |  | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 368 | for_each_pci_msi_entry(entry, dev) | 
| Jiang Liu | 63a7b17 | 2014-11-06 22:20:32 +0800 | [diff] [blame] | 369 | if (entry->irq) | 
|  | 370 | for (i = 0; i < entry->nvec_used; i++) | 
|  | 371 | BUG_ON(irq_has_action(entry->irq + i)); | 
| Hidetoshi Seto | f56e448 | 2009-08-06 11:32:51 +0900 | [diff] [blame] | 372 |  | 
| Jiang Liu | 8e047ad | 2014-11-15 22:24:07 +0800 | [diff] [blame] | 373 | pci_msi_teardown_msi_irqs(dev); | 
| Hidetoshi Seto | f56e448 | 2009-08-06 11:32:51 +0900 | [diff] [blame] | 374 |  | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 375 | list_for_each_entry_safe(entry, tmp, msi_list, list) { | 
| Hidetoshi Seto | f56e448 | 2009-08-06 11:32:51 +0900 | [diff] [blame] | 376 | if (entry->msi_attrib.is_msix) { | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 377 | if (list_is_last(&entry->list, msi_list)) | 
| Hidetoshi Seto | f56e448 | 2009-08-06 11:32:51 +0900 | [diff] [blame] | 378 | iounmap(entry->mask_base); | 
|  | 379 | } | 
| Neil Horman | 424eb39 | 2012-01-03 10:29:54 -0500 | [diff] [blame] | 380 |  | 
| Hidetoshi Seto | f56e448 | 2009-08-06 11:32:51 +0900 | [diff] [blame] | 381 | list_del(&entry->list); | 
|  | 382 | kfree(entry); | 
|  | 383 | } | 
| Greg Kroah-Hartman | 1c51b50 | 2013-12-19 12:30:17 -0800 | [diff] [blame] | 384 |  | 
|  | 385 | if (dev->msi_irq_groups) { | 
|  | 386 | sysfs_remove_groups(&dev->dev.kobj, dev->msi_irq_groups); | 
|  | 387 | msi_attrs = dev->msi_irq_groups[0]->attrs; | 
| Alexei Starovoitov | b701c0b | 2014-06-04 15:49:50 -0700 | [diff] [blame] | 388 | while (msi_attrs[count]) { | 
| Greg Kroah-Hartman | 1c51b50 | 2013-12-19 12:30:17 -0800 | [diff] [blame] | 389 | dev_attr = container_of(msi_attrs[count], | 
|  | 390 | struct device_attribute, attr); | 
|  | 391 | kfree(dev_attr->attr.name); | 
|  | 392 | kfree(dev_attr); | 
|  | 393 | ++count; | 
|  | 394 | } | 
|  | 395 | kfree(msi_attrs); | 
|  | 396 | kfree(dev->msi_irq_groups[0]); | 
|  | 397 | kfree(dev->msi_irq_groups); | 
|  | 398 | dev->msi_irq_groups = NULL; | 
|  | 399 | } | 
| Hidetoshi Seto | f56e448 | 2009-08-06 11:32:51 +0900 | [diff] [blame] | 400 | } | 
| Satoru Takeuchi | c54c187 | 2007-01-18 13:50:05 +0900 | [diff] [blame] | 401 |  | 
| David Miller | ba698ad | 2007-10-25 01:16:30 -0700 | [diff] [blame] | 402 | static void pci_intx_for_msi(struct pci_dev *dev, int enable) | 
|  | 403 | { | 
|  | 404 | if (!(dev->dev_flags & PCI_DEV_FLAGS_MSI_INTX_DISABLE_BUG)) | 
|  | 405 | pci_intx(dev, enable); | 
|  | 406 | } | 
|  | 407 |  | 
| Michael Ellerman | 8fed4b6 | 2007-01-25 19:34:08 +1100 | [diff] [blame] | 408 | static void __pci_restore_msi_state(struct pci_dev *dev) | 
| Shaohua Li | 41017f0 | 2006-02-08 17:11:38 +0800 | [diff] [blame] | 409 | { | 
| Shaohua Li | 41017f0 | 2006-02-08 17:11:38 +0800 | [diff] [blame] | 410 | u16 control; | 
| Eric W. Biederman | 392ee1e | 2007-03-08 13:04:57 -0700 | [diff] [blame] | 411 | struct msi_desc *entry; | 
| Shaohua Li | 41017f0 | 2006-02-08 17:11:38 +0800 | [diff] [blame] | 412 |  | 
| Eric W. Biederman | b1cbf4e | 2007-03-05 00:30:10 -0800 | [diff] [blame] | 413 | if (!dev->msi_enabled) | 
|  | 414 | return; | 
|  | 415 |  | 
| Thomas Gleixner | dced35a | 2011-03-28 17:49:12 +0200 | [diff] [blame] | 416 | entry = irq_get_msi_desc(dev->irq); | 
| Shaohua Li | 41017f0 | 2006-02-08 17:11:38 +0800 | [diff] [blame] | 417 |  | 
| David Miller | ba698ad | 2007-10-25 01:16:30 -0700 | [diff] [blame] | 418 | pci_intx_for_msi(dev, 0); | 
| Michael S. Tsirkin | 61b64ab | 2015-05-07 09:52:21 -0500 | [diff] [blame] | 419 | pci_msi_set_enable(dev, 0); | 
| DuanZhenzhong | ac8344c | 2013-12-04 13:09:16 +0800 | [diff] [blame] | 420 | arch_restore_msi_irqs(dev); | 
| Eric W. Biederman | 392ee1e | 2007-03-08 13:04:57 -0700 | [diff] [blame] | 421 |  | 
| Bjorn Helgaas | f532216 | 2013-04-17 17:34:36 -0600 | [diff] [blame] | 422 | pci_read_config_word(dev, dev->msi_cap + PCI_MSI_FLAGS, &control); | 
| Yijing Wang | 31ea5d4 | 2014-06-19 16:30:30 +0800 | [diff] [blame] | 423 | msi_mask_irq(entry, msi_mask(entry->msi_attrib.multi_cap), | 
|  | 424 | entry->masked); | 
| Jesse Barnes | abad2ec | 2008-08-07 08:52:37 -0700 | [diff] [blame] | 425 | control &= ~PCI_MSI_FLAGS_QSIZE; | 
| Matthew Wilcox | 1c8d7b0 | 2009-03-17 08:54:10 -0400 | [diff] [blame] | 426 | control |= (entry->msi_attrib.multiple << 4) | PCI_MSI_FLAGS_ENABLE; | 
| Bjorn Helgaas | f532216 | 2013-04-17 17:34:36 -0600 | [diff] [blame] | 427 | pci_write_config_word(dev, dev->msi_cap + PCI_MSI_FLAGS, control); | 
| Michael Ellerman | 8fed4b6 | 2007-01-25 19:34:08 +1100 | [diff] [blame] | 428 | } | 
|  | 429 |  | 
|  | 430 | static void __pci_restore_msix_state(struct pci_dev *dev) | 
| Shaohua Li | 41017f0 | 2006-02-08 17:11:38 +0800 | [diff] [blame] | 431 | { | 
| Shaohua Li | 41017f0 | 2006-02-08 17:11:38 +0800 | [diff] [blame] | 432 | struct msi_desc *entry; | 
| Shaohua Li | 41017f0 | 2006-02-08 17:11:38 +0800 | [diff] [blame] | 433 |  | 
| Eric W. Biederman | ded86d8 | 2007-01-28 12:42:52 -0700 | [diff] [blame] | 434 | if (!dev->msix_enabled) | 
|  | 435 | return; | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 436 | BUG_ON(list_empty(dev_to_msi_list(&dev->dev))); | 
| Eric W. Biederman | ded86d8 | 2007-01-28 12:42:52 -0700 | [diff] [blame] | 437 |  | 
| Shaohua Li | 41017f0 | 2006-02-08 17:11:38 +0800 | [diff] [blame] | 438 | /* route the table */ | 
| David Miller | ba698ad | 2007-10-25 01:16:30 -0700 | [diff] [blame] | 439 | pci_intx_for_msi(dev, 0); | 
| Michael S. Tsirkin | 61b64ab | 2015-05-07 09:52:21 -0500 | [diff] [blame] | 440 | pci_msix_clear_and_set_ctrl(dev, 0, | 
| Yijing Wang | 66f0d0c | 2014-06-19 16:29:53 +0800 | [diff] [blame] | 441 | PCI_MSIX_FLAGS_ENABLE | PCI_MSIX_FLAGS_MASKALL); | 
| Shaohua Li | 41017f0 | 2006-02-08 17:11:38 +0800 | [diff] [blame] | 442 |  | 
| DuanZhenzhong | ac8344c | 2013-12-04 13:09:16 +0800 | [diff] [blame] | 443 | arch_restore_msi_irqs(dev); | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 444 | for_each_pci_msi_entry(entry, dev) | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 445 | msix_mask_irq(entry, entry->masked); | 
| Shaohua Li | 41017f0 | 2006-02-08 17:11:38 +0800 | [diff] [blame] | 446 |  | 
| Michael S. Tsirkin | 61b64ab | 2015-05-07 09:52:21 -0500 | [diff] [blame] | 447 | pci_msix_clear_and_set_ctrl(dev, PCI_MSIX_FLAGS_MASKALL, 0); | 
| Shaohua Li | 41017f0 | 2006-02-08 17:11:38 +0800 | [diff] [blame] | 448 | } | 
| Michael Ellerman | 8fed4b6 | 2007-01-25 19:34:08 +1100 | [diff] [blame] | 449 |  | 
|  | 450 | void pci_restore_msi_state(struct pci_dev *dev) | 
|  | 451 | { | 
|  | 452 | __pci_restore_msi_state(dev); | 
|  | 453 | __pci_restore_msix_state(dev); | 
|  | 454 | } | 
| Linas Vepstas | 94688cf | 2007-11-07 15:43:59 -0600 | [diff] [blame] | 455 | EXPORT_SYMBOL_GPL(pci_restore_msi_state); | 
| Shaohua Li | 41017f0 | 2006-02-08 17:11:38 +0800 | [diff] [blame] | 456 |  | 
| Greg Kroah-Hartman | 1c51b50 | 2013-12-19 12:30:17 -0800 | [diff] [blame] | 457 | static ssize_t msi_mode_show(struct device *dev, struct device_attribute *attr, | 
| Neil Horman | da8d1c8 | 2011-10-06 14:08:18 -0400 | [diff] [blame] | 458 | char *buf) | 
|  | 459 | { | 
| Greg Kroah-Hartman | 1c51b50 | 2013-12-19 12:30:17 -0800 | [diff] [blame] | 460 | struct msi_desc *entry; | 
|  | 461 | unsigned long irq; | 
|  | 462 | int retval; | 
|  | 463 |  | 
|  | 464 | retval = kstrtoul(attr->attr.name, 10, &irq); | 
|  | 465 | if (retval) | 
|  | 466 | return retval; | 
|  | 467 |  | 
| Yijing Wang | e11ece5 | 2014-07-08 10:09:19 +0800 | [diff] [blame] | 468 | entry = irq_get_msi_desc(irq); | 
|  | 469 | if (entry) | 
|  | 470 | return sprintf(buf, "%s\n", | 
|  | 471 | entry->msi_attrib.is_msix ? "msix" : "msi"); | 
|  | 472 |  | 
| Greg Kroah-Hartman | 1c51b50 | 2013-12-19 12:30:17 -0800 | [diff] [blame] | 473 | return -ENODEV; | 
| Neil Horman | da8d1c8 | 2011-10-06 14:08:18 -0400 | [diff] [blame] | 474 | } | 
|  | 475 |  | 
| Neil Horman | da8d1c8 | 2011-10-06 14:08:18 -0400 | [diff] [blame] | 476 | static int populate_msi_sysfs(struct pci_dev *pdev) | 
|  | 477 | { | 
| Greg Kroah-Hartman | 1c51b50 | 2013-12-19 12:30:17 -0800 | [diff] [blame] | 478 | struct attribute **msi_attrs; | 
|  | 479 | struct attribute *msi_attr; | 
|  | 480 | struct device_attribute *msi_dev_attr; | 
|  | 481 | struct attribute_group *msi_irq_group; | 
|  | 482 | const struct attribute_group **msi_irq_groups; | 
| Neil Horman | da8d1c8 | 2011-10-06 14:08:18 -0400 | [diff] [blame] | 483 | struct msi_desc *entry; | 
| Greg Kroah-Hartman | 1c51b50 | 2013-12-19 12:30:17 -0800 | [diff] [blame] | 484 | int ret = -ENOMEM; | 
|  | 485 | int num_msi = 0; | 
| Neil Horman | da8d1c8 | 2011-10-06 14:08:18 -0400 | [diff] [blame] | 486 | int count = 0; | 
| Romain Bezut | a867606 | 2015-09-24 01:31:16 +0200 | [diff] [blame] | 487 | int i; | 
| Neil Horman | da8d1c8 | 2011-10-06 14:08:18 -0400 | [diff] [blame] | 488 |  | 
| Greg Kroah-Hartman | 1c51b50 | 2013-12-19 12:30:17 -0800 | [diff] [blame] | 489 | /* Determine how many msi entries we have */ | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 490 | for_each_pci_msi_entry(entry, pdev) | 
| Romain Bezut | a867606 | 2015-09-24 01:31:16 +0200 | [diff] [blame] | 491 | num_msi += entry->nvec_used; | 
| Greg Kroah-Hartman | 1c51b50 | 2013-12-19 12:30:17 -0800 | [diff] [blame] | 492 | if (!num_msi) | 
|  | 493 | return 0; | 
|  | 494 |  | 
|  | 495 | /* Dynamically create the MSI attributes for the PCI device */ | 
|  | 496 | msi_attrs = kzalloc(sizeof(void *) * (num_msi + 1), GFP_KERNEL); | 
|  | 497 | if (!msi_attrs) | 
|  | 498 | return -ENOMEM; | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 499 | for_each_pci_msi_entry(entry, pdev) { | 
| Romain Bezut | a867606 | 2015-09-24 01:31:16 +0200 | [diff] [blame] | 500 | for (i = 0; i < entry->nvec_used; i++) { | 
|  | 501 | msi_dev_attr = kzalloc(sizeof(*msi_dev_attr), GFP_KERNEL); | 
|  | 502 | if (!msi_dev_attr) | 
|  | 503 | goto error_attrs; | 
|  | 504 | msi_attrs[count] = &msi_dev_attr->attr; | 
| Greg Kroah-Hartman | 86bb4f6 | 2014-02-13 10:47:20 -0700 | [diff] [blame] | 505 |  | 
| Romain Bezut | a867606 | 2015-09-24 01:31:16 +0200 | [diff] [blame] | 506 | sysfs_attr_init(&msi_dev_attr->attr); | 
|  | 507 | msi_dev_attr->attr.name = kasprintf(GFP_KERNEL, "%d", | 
|  | 508 | entry->irq + i); | 
|  | 509 | if (!msi_dev_attr->attr.name) | 
|  | 510 | goto error_attrs; | 
|  | 511 | msi_dev_attr->attr.mode = S_IRUGO; | 
|  | 512 | msi_dev_attr->show = msi_mode_show; | 
|  | 513 | ++count; | 
|  | 514 | } | 
| Greg Kroah-Hartman | 1c51b50 | 2013-12-19 12:30:17 -0800 | [diff] [blame] | 515 | } | 
|  | 516 |  | 
|  | 517 | msi_irq_group = kzalloc(sizeof(*msi_irq_group), GFP_KERNEL); | 
|  | 518 | if (!msi_irq_group) | 
|  | 519 | goto error_attrs; | 
|  | 520 | msi_irq_group->name = "msi_irqs"; | 
|  | 521 | msi_irq_group->attrs = msi_attrs; | 
|  | 522 |  | 
|  | 523 | msi_irq_groups = kzalloc(sizeof(void *) * 2, GFP_KERNEL); | 
|  | 524 | if (!msi_irq_groups) | 
|  | 525 | goto error_irq_group; | 
|  | 526 | msi_irq_groups[0] = msi_irq_group; | 
|  | 527 |  | 
|  | 528 | ret = sysfs_create_groups(&pdev->dev.kobj, msi_irq_groups); | 
|  | 529 | if (ret) | 
|  | 530 | goto error_irq_groups; | 
|  | 531 | pdev->msi_irq_groups = msi_irq_groups; | 
| Neil Horman | da8d1c8 | 2011-10-06 14:08:18 -0400 | [diff] [blame] | 532 |  | 
|  | 533 | return 0; | 
|  | 534 |  | 
| Greg Kroah-Hartman | 1c51b50 | 2013-12-19 12:30:17 -0800 | [diff] [blame] | 535 | error_irq_groups: | 
|  | 536 | kfree(msi_irq_groups); | 
|  | 537 | error_irq_group: | 
|  | 538 | kfree(msi_irq_group); | 
|  | 539 | error_attrs: | 
|  | 540 | count = 0; | 
|  | 541 | msi_attr = msi_attrs[count]; | 
|  | 542 | while (msi_attr) { | 
|  | 543 | msi_dev_attr = container_of(msi_attr, struct device_attribute, attr); | 
|  | 544 | kfree(msi_attr->name); | 
|  | 545 | kfree(msi_dev_attr); | 
|  | 546 | ++count; | 
|  | 547 | msi_attr = msi_attrs[count]; | 
| Neil Horman | da8d1c8 | 2011-10-06 14:08:18 -0400 | [diff] [blame] | 548 | } | 
| Greg Kroah-Hartman | 2923775 | 2014-02-13 10:47:35 -0700 | [diff] [blame] | 549 | kfree(msi_attrs); | 
| Neil Horman | da8d1c8 | 2011-10-06 14:08:18 -0400 | [diff] [blame] | 550 | return ret; | 
|  | 551 | } | 
|  | 552 |  | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 553 | static struct msi_desc * | 
|  | 554 | msi_setup_entry(struct pci_dev *dev, int nvec, bool affinity) | 
| Yijing Wang | d873b4d | 2014-07-08 10:07:23 +0800 | [diff] [blame] | 555 | { | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 556 | struct cpumask *masks = NULL; | 
| Yijing Wang | d873b4d | 2014-07-08 10:07:23 +0800 | [diff] [blame] | 557 | struct msi_desc *entry; | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 558 | u16 control; | 
|  | 559 |  | 
|  | 560 | if (affinity) { | 
|  | 561 | masks = irq_create_affinity_masks(dev->irq_affinity, nvec); | 
|  | 562 | if (!masks) | 
|  | 563 | pr_err("Unable to allocate affinity masks, ignoring\n"); | 
|  | 564 | } | 
| Yijing Wang | d873b4d | 2014-07-08 10:07:23 +0800 | [diff] [blame] | 565 |  | 
|  | 566 | /* MSI Entry Initialization */ | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 567 | entry = alloc_msi_entry(&dev->dev, nvec, masks); | 
| Yijing Wang | d873b4d | 2014-07-08 10:07:23 +0800 | [diff] [blame] | 568 | if (!entry) | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 569 | goto out; | 
| Yijing Wang | d873b4d | 2014-07-08 10:07:23 +0800 | [diff] [blame] | 570 |  | 
|  | 571 | pci_read_config_word(dev, dev->msi_cap + PCI_MSI_FLAGS, &control); | 
|  | 572 |  | 
|  | 573 | entry->msi_attrib.is_msix	= 0; | 
|  | 574 | entry->msi_attrib.is_64		= !!(control & PCI_MSI_FLAGS_64BIT); | 
|  | 575 | entry->msi_attrib.entry_nr	= 0; | 
|  | 576 | entry->msi_attrib.maskbit	= !!(control & PCI_MSI_FLAGS_MASKBIT); | 
|  | 577 | entry->msi_attrib.default_irq	= dev->irq;	/* Save IOAPIC IRQ */ | 
| Yijing Wang | d873b4d | 2014-07-08 10:07:23 +0800 | [diff] [blame] | 578 | entry->msi_attrib.multi_cap	= (control & PCI_MSI_FLAGS_QMASK) >> 1; | 
| Jiang Liu | 63a7b17 | 2014-11-06 22:20:32 +0800 | [diff] [blame] | 579 | entry->msi_attrib.multiple	= ilog2(__roundup_pow_of_two(nvec)); | 
| Yijing Wang | d873b4d | 2014-07-08 10:07:23 +0800 | [diff] [blame] | 580 |  | 
|  | 581 | if (control & PCI_MSI_FLAGS_64BIT) | 
|  | 582 | entry->mask_pos = dev->msi_cap + PCI_MSI_MASK_64; | 
|  | 583 | else | 
|  | 584 | entry->mask_pos = dev->msi_cap + PCI_MSI_MASK_32; | 
|  | 585 |  | 
|  | 586 | /* Save the initial mask status */ | 
|  | 587 | if (entry->msi_attrib.maskbit) | 
|  | 588 | pci_read_config_dword(dev, entry->mask_pos, &entry->masked); | 
|  | 589 |  | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 590 | out: | 
|  | 591 | kfree(masks); | 
| Yijing Wang | d873b4d | 2014-07-08 10:07:23 +0800 | [diff] [blame] | 592 | return entry; | 
|  | 593 | } | 
|  | 594 |  | 
| Benjamin Herrenschmidt | f144d14 | 2014-10-03 15:13:24 +1000 | [diff] [blame] | 595 | static int msi_verify_entries(struct pci_dev *dev) | 
|  | 596 | { | 
|  | 597 | struct msi_desc *entry; | 
|  | 598 |  | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 599 | for_each_pci_msi_entry(entry, dev) { | 
| Benjamin Herrenschmidt | f144d14 | 2014-10-03 15:13:24 +1000 | [diff] [blame] | 600 | if (!dev->no_64bit_msi || !entry->msg.address_hi) | 
|  | 601 | continue; | 
|  | 602 | dev_err(&dev->dev, "Device has broken 64-bit MSI but arch" | 
|  | 603 | " tried to assign one above 4G\n"); | 
|  | 604 | return -EIO; | 
|  | 605 | } | 
|  | 606 | return 0; | 
|  | 607 | } | 
|  | 608 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 609 | /** | 
|  | 610 | * msi_capability_init - configure device's MSI capability structure | 
|  | 611 | * @dev: pointer to the pci_dev data structure of MSI device function | 
| Matthew Wilcox | 1c8d7b0 | 2009-03-17 08:54:10 -0400 | [diff] [blame] | 612 | * @nvec: number of interrupts to allocate | 
| Stephen Hemminger | 62c6151 | 2016-10-23 09:32:34 -0700 | [diff] [blame] | 613 | * @affinity: flag to indicate cpu irq affinity mask should be set | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 614 | * | 
| Matthew Wilcox | 1c8d7b0 | 2009-03-17 08:54:10 -0400 | [diff] [blame] | 615 | * Setup the MSI capability structure of the device with the requested | 
|  | 616 | * number of interrupts.  A return value of zero indicates the successful | 
|  | 617 | * setup of an entry with the new MSI irq.  A negative return value indicates | 
|  | 618 | * an error, and a positive return value indicates the number of interrupts | 
|  | 619 | * which could have been allocated. | 
|  | 620 | */ | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 621 | static int msi_capability_init(struct pci_dev *dev, int nvec, bool affinity) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 622 | { | 
|  | 623 | struct msi_desc *entry; | 
| Gavin Shan | f465136 | 2013-04-04 16:54:32 +0000 | [diff] [blame] | 624 | int ret; | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 625 | unsigned mask; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 626 |  | 
| Michael S. Tsirkin | 61b64ab | 2015-05-07 09:52:21 -0500 | [diff] [blame] | 627 | pci_msi_set_enable(dev, 0);	/* Disable MSI during set up */ | 
| Matthew Wilcox | 110828c | 2009-06-16 06:31:45 -0600 | [diff] [blame] | 628 |  | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 629 | entry = msi_setup_entry(dev, nvec, affinity); | 
| Eric W. Biederman | f7feaca | 2007-01-28 12:56:37 -0700 | [diff] [blame] | 630 | if (!entry) | 
|  | 631 | return -ENOMEM; | 
| Eric W. Biederman | 1ce0337 | 2006-10-04 02:16:41 -0700 | [diff] [blame] | 632 |  | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 633 | /* All MSIs are unmasked by default, Mask them all */ | 
| Yijing Wang | 31ea5d4 | 2014-06-19 16:30:30 +0800 | [diff] [blame] | 634 | mask = msi_mask(entry->msi_attrib.multi_cap); | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 635 | msi_mask_irq(entry, mask, mask); | 
|  | 636 |  | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 637 | list_add_tail(&entry->list, dev_to_msi_list(&dev->dev)); | 
| Michael Ellerman | 9c83133 | 2007-04-18 19:39:21 +1000 | [diff] [blame] | 638 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 639 | /* Configure MSI capability structure */ | 
| Jiang Liu | 8e047ad | 2014-11-15 22:24:07 +0800 | [diff] [blame] | 640 | ret = pci_msi_setup_msi_irqs(dev, nvec, PCI_CAP_ID_MSI); | 
| Michael Ellerman | 7fe3730 | 2007-04-18 19:39:21 +1000 | [diff] [blame] | 641 | if (ret) { | 
| Hidetoshi Seto | 7ba1930 | 2009-06-23 17:39:27 +0900 | [diff] [blame] | 642 | msi_mask_irq(entry, mask, ~mask); | 
| Hidetoshi Seto | f56e448 | 2009-08-06 11:32:51 +0900 | [diff] [blame] | 643 | free_msi_irqs(dev); | 
| Michael Ellerman | 7fe3730 | 2007-04-18 19:39:21 +1000 | [diff] [blame] | 644 | return ret; | 
| Mark Maule | fd58e55 | 2006-04-10 21:17:48 -0500 | [diff] [blame] | 645 | } | 
| Eric W. Biederman | f7feaca | 2007-01-28 12:56:37 -0700 | [diff] [blame] | 646 |  | 
| Benjamin Herrenschmidt | f144d14 | 2014-10-03 15:13:24 +1000 | [diff] [blame] | 647 | ret = msi_verify_entries(dev); | 
|  | 648 | if (ret) { | 
|  | 649 | msi_mask_irq(entry, mask, ~mask); | 
|  | 650 | free_msi_irqs(dev); | 
|  | 651 | return ret; | 
|  | 652 | } | 
|  | 653 |  | 
| Neil Horman | da8d1c8 | 2011-10-06 14:08:18 -0400 | [diff] [blame] | 654 | ret = populate_msi_sysfs(dev); | 
|  | 655 | if (ret) { | 
|  | 656 | msi_mask_irq(entry, mask, ~mask); | 
|  | 657 | free_msi_irqs(dev); | 
|  | 658 | return ret; | 
|  | 659 | } | 
|  | 660 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 661 | /* Set MSI enabled bits	 */ | 
| David Miller | ba698ad | 2007-10-25 01:16:30 -0700 | [diff] [blame] | 662 | pci_intx_for_msi(dev, 0); | 
| Michael S. Tsirkin | 61b64ab | 2015-05-07 09:52:21 -0500 | [diff] [blame] | 663 | pci_msi_set_enable(dev, 1); | 
| Eric W. Biederman | b1cbf4e | 2007-03-05 00:30:10 -0800 | [diff] [blame] | 664 | dev->msi_enabled = 1; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 665 |  | 
| Jiang Liu | 5f22699 | 2015-07-30 14:00:08 -0500 | [diff] [blame] | 666 | pcibios_free_irq(dev); | 
| Michael Ellerman | 7fe3730 | 2007-04-18 19:39:21 +1000 | [diff] [blame] | 667 | dev->irq = entry->irq; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 668 | return 0; | 
|  | 669 | } | 
|  | 670 |  | 
| Gavin Shan | 520fe9d | 2013-04-04 16:54:33 +0000 | [diff] [blame] | 671 | static void __iomem *msix_map_region(struct pci_dev *dev, unsigned nr_entries) | 
| Hidetoshi Seto | 5a05a9d | 2009-08-06 11:34:34 +0900 | [diff] [blame] | 672 | { | 
| Kenji Kaneshige | 4302e0f | 2010-06-17 10:42:44 +0900 | [diff] [blame] | 673 | resource_size_t phys_addr; | 
| Hidetoshi Seto | 5a05a9d | 2009-08-06 11:34:34 +0900 | [diff] [blame] | 674 | u32 table_offset; | 
| Yijing Wang | 6a878e5 | 2015-01-28 09:52:17 +0800 | [diff] [blame] | 675 | unsigned long flags; | 
| Hidetoshi Seto | 5a05a9d | 2009-08-06 11:34:34 +0900 | [diff] [blame] | 676 | u8 bir; | 
|  | 677 |  | 
| Bjorn Helgaas | 909094c | 2013-04-17 17:43:40 -0600 | [diff] [blame] | 678 | pci_read_config_dword(dev, dev->msix_cap + PCI_MSIX_TABLE, | 
|  | 679 | &table_offset); | 
| Bjorn Helgaas | 4d18760 | 2013-04-17 18:10:07 -0600 | [diff] [blame] | 680 | bir = (u8)(table_offset & PCI_MSIX_TABLE_BIR); | 
| Yijing Wang | 6a878e5 | 2015-01-28 09:52:17 +0800 | [diff] [blame] | 681 | flags = pci_resource_flags(dev, bir); | 
|  | 682 | if (!flags || (flags & IORESOURCE_UNSET)) | 
|  | 683 | return NULL; | 
|  | 684 |  | 
| Bjorn Helgaas | 4d18760 | 2013-04-17 18:10:07 -0600 | [diff] [blame] | 685 | table_offset &= PCI_MSIX_TABLE_OFFSET; | 
| Hidetoshi Seto | 5a05a9d | 2009-08-06 11:34:34 +0900 | [diff] [blame] | 686 | phys_addr = pci_resource_start(dev, bir) + table_offset; | 
|  | 687 |  | 
|  | 688 | return ioremap_nocache(phys_addr, nr_entries * PCI_MSIX_ENTRY_SIZE); | 
|  | 689 | } | 
|  | 690 |  | 
| Gavin Shan | 520fe9d | 2013-04-04 16:54:33 +0000 | [diff] [blame] | 691 | static int msix_setup_entries(struct pci_dev *dev, void __iomem *base, | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 692 | struct msix_entry *entries, int nvec, | 
|  | 693 | bool affinity) | 
| Hidetoshi Seto | d9d7070 | 2009-08-06 11:35:48 +0900 | [diff] [blame] | 694 | { | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 695 | struct cpumask *curmsk, *masks = NULL; | 
| Hidetoshi Seto | d9d7070 | 2009-08-06 11:35:48 +0900 | [diff] [blame] | 696 | struct msi_desc *entry; | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 697 | int ret, i; | 
| Hidetoshi Seto | d9d7070 | 2009-08-06 11:35:48 +0900 | [diff] [blame] | 698 |  | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 699 | if (affinity) { | 
|  | 700 | masks = irq_create_affinity_masks(dev->irq_affinity, nvec); | 
|  | 701 | if (!masks) | 
|  | 702 | pr_err("Unable to allocate affinity masks, ignoring\n"); | 
|  | 703 | } | 
| Christoph Hellwig | 4ef3368 | 2016-07-12 18:20:18 +0900 | [diff] [blame] | 704 |  | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 705 | for (i = 0, curmsk = masks; i < nvec; i++) { | 
|  | 706 | entry = alloc_msi_entry(&dev->dev, 1, curmsk); | 
| Hidetoshi Seto | d9d7070 | 2009-08-06 11:35:48 +0900 | [diff] [blame] | 707 | if (!entry) { | 
|  | 708 | if (!i) | 
|  | 709 | iounmap(base); | 
|  | 710 | else | 
|  | 711 | free_msi_irqs(dev); | 
|  | 712 | /* No enough memory. Don't try again */ | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 713 | ret = -ENOMEM; | 
|  | 714 | goto out; | 
| Hidetoshi Seto | d9d7070 | 2009-08-06 11:35:48 +0900 | [diff] [blame] | 715 | } | 
|  | 716 |  | 
|  | 717 | entry->msi_attrib.is_msix	= 1; | 
|  | 718 | entry->msi_attrib.is_64		= 1; | 
| Christoph Hellwig | 3ac020e | 2016-07-12 18:20:16 +0900 | [diff] [blame] | 719 | if (entries) | 
|  | 720 | entry->msi_attrib.entry_nr = entries[i].entry; | 
|  | 721 | else | 
|  | 722 | entry->msi_attrib.entry_nr = i; | 
| Hidetoshi Seto | d9d7070 | 2009-08-06 11:35:48 +0900 | [diff] [blame] | 723 | entry->msi_attrib.default_irq	= dev->irq; | 
| Hidetoshi Seto | d9d7070 | 2009-08-06 11:35:48 +0900 | [diff] [blame] | 724 | entry->mask_base		= base; | 
|  | 725 |  | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 726 | list_add_tail(&entry->list, dev_to_msi_list(&dev->dev)); | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 727 | if (masks) | 
|  | 728 | curmsk++; | 
| Hidetoshi Seto | d9d7070 | 2009-08-06 11:35:48 +0900 | [diff] [blame] | 729 | } | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 730 | ret = 0; | 
|  | 731 | out: | 
|  | 732 | kfree(masks); | 
| Hidetoshi Seto | d9d7070 | 2009-08-06 11:35:48 +0900 | [diff] [blame] | 733 | return 0; | 
|  | 734 | } | 
|  | 735 |  | 
| Hidetoshi Seto | 75cb342 | 2009-08-06 11:35:10 +0900 | [diff] [blame] | 736 | static void msix_program_entries(struct pci_dev *dev, | 
| Gavin Shan | 520fe9d | 2013-04-04 16:54:33 +0000 | [diff] [blame] | 737 | struct msix_entry *entries) | 
| Hidetoshi Seto | 75cb342 | 2009-08-06 11:35:10 +0900 | [diff] [blame] | 738 | { | 
|  | 739 | struct msi_desc *entry; | 
|  | 740 | int i = 0; | 
|  | 741 |  | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 742 | for_each_pci_msi_entry(entry, dev) { | 
| Christoph Hellwig | 3ac020e | 2016-07-12 18:20:16 +0900 | [diff] [blame] | 743 | if (entries) | 
|  | 744 | entries[i++].vector = entry->irq; | 
| Christoph Hellwig | 12eb21d | 2016-07-12 18:20:15 +0900 | [diff] [blame] | 745 | entry->masked = readl(pci_msix_desc_addr(entry) + | 
|  | 746 | PCI_MSIX_ENTRY_VECTOR_CTRL); | 
| Hidetoshi Seto | 75cb342 | 2009-08-06 11:35:10 +0900 | [diff] [blame] | 747 | msix_mask_irq(entry, 1); | 
| Hidetoshi Seto | 75cb342 | 2009-08-06 11:35:10 +0900 | [diff] [blame] | 748 | } | 
|  | 749 | } | 
|  | 750 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 751 | /** | 
|  | 752 | * msix_capability_init - configure device's MSI-X capability | 
|  | 753 | * @dev: pointer to the pci_dev data structure of MSI-X device function | 
| Randy Dunlap | 8f7020d | 2005-10-23 11:57:38 -0700 | [diff] [blame] | 754 | * @entries: pointer to an array of struct msix_entry entries | 
|  | 755 | * @nvec: number of @entries | 
| Stephen Hemminger | 62c6151 | 2016-10-23 09:32:34 -0700 | [diff] [blame] | 756 | * @affinity: flag to indicate cpu irq affinity mask should be set | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 757 | * | 
| Steven Cole | eaae4b3 | 2005-05-03 18:38:30 -0600 | [diff] [blame] | 758 | * Setup the MSI-X capability structure of device function with a | 
| Eric W. Biederman | 1ce0337 | 2006-10-04 02:16:41 -0700 | [diff] [blame] | 759 | * single MSI-X irq. A return of zero indicates the successful setup of | 
|  | 760 | * requested MSI-X entries with allocated irqs or non-zero for otherwise. | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 761 | **/ | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 762 | static int msix_capability_init(struct pci_dev *dev, struct msix_entry *entries, | 
|  | 763 | int nvec, bool affinity) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 764 | { | 
| Gavin Shan | 520fe9d | 2013-04-04 16:54:33 +0000 | [diff] [blame] | 765 | int ret; | 
| Hidetoshi Seto | 5a05a9d | 2009-08-06 11:34:34 +0900 | [diff] [blame] | 766 | u16 control; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 767 | void __iomem *base; | 
|  | 768 |  | 
| Matthew Wilcox | f598282 | 2009-06-18 19:15:59 -0700 | [diff] [blame] | 769 | /* Ensure MSI-X is disabled while it is set up */ | 
| Michael S. Tsirkin | 61b64ab | 2015-05-07 09:52:21 -0500 | [diff] [blame] | 770 | pci_msix_clear_and_set_ctrl(dev, PCI_MSIX_FLAGS_ENABLE, 0); | 
| Matthew Wilcox | f598282 | 2009-06-18 19:15:59 -0700 | [diff] [blame] | 771 |  | 
| Yijing Wang | 66f0d0c | 2014-06-19 16:29:53 +0800 | [diff] [blame] | 772 | pci_read_config_word(dev, dev->msix_cap + PCI_MSIX_FLAGS, &control); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 773 | /* Request & Map MSI-X table region */ | 
| Bjorn Helgaas | 527eee2 | 2013-04-17 17:44:48 -0600 | [diff] [blame] | 774 | base = msix_map_region(dev, msix_table_size(control)); | 
| Hidetoshi Seto | 5a05a9d | 2009-08-06 11:34:34 +0900 | [diff] [blame] | 775 | if (!base) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 776 | return -ENOMEM; | 
|  | 777 |  | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 778 | ret = msix_setup_entries(dev, base, entries, nvec, affinity); | 
| Hidetoshi Seto | d9d7070 | 2009-08-06 11:35:48 +0900 | [diff] [blame] | 779 | if (ret) | 
|  | 780 | return ret; | 
| Michael Ellerman | 9c83133 | 2007-04-18 19:39:21 +1000 | [diff] [blame] | 781 |  | 
| Jiang Liu | 8e047ad | 2014-11-15 22:24:07 +0800 | [diff] [blame] | 782 | ret = pci_msi_setup_msi_irqs(dev, nvec, PCI_CAP_ID_MSIX); | 
| Hidetoshi Seto | 583871d | 2009-08-06 11:33:39 +0900 | [diff] [blame] | 783 | if (ret) | 
| Alexander Gordeev | 2adc790 | 2013-12-16 09:34:56 +0100 | [diff] [blame] | 784 | goto out_avail; | 
| Michael Ellerman | 9c83133 | 2007-04-18 19:39:21 +1000 | [diff] [blame] | 785 |  | 
| Benjamin Herrenschmidt | f144d14 | 2014-10-03 15:13:24 +1000 | [diff] [blame] | 786 | /* Check if all MSI entries honor device restrictions */ | 
|  | 787 | ret = msi_verify_entries(dev); | 
|  | 788 | if (ret) | 
|  | 789 | goto out_free; | 
|  | 790 |  | 
| Matthew Wilcox | f598282 | 2009-06-18 19:15:59 -0700 | [diff] [blame] | 791 | /* | 
|  | 792 | * Some devices require MSI-X to be enabled before we can touch the | 
|  | 793 | * MSI-X registers.  We need to mask all the vectors to prevent | 
|  | 794 | * interrupts coming in before they're fully set up. | 
|  | 795 | */ | 
| Michael S. Tsirkin | 61b64ab | 2015-05-07 09:52:21 -0500 | [diff] [blame] | 796 | pci_msix_clear_and_set_ctrl(dev, 0, | 
| Yijing Wang | 66f0d0c | 2014-06-19 16:29:53 +0800 | [diff] [blame] | 797 | PCI_MSIX_FLAGS_MASKALL | PCI_MSIX_FLAGS_ENABLE); | 
| Matthew Wilcox | f598282 | 2009-06-18 19:15:59 -0700 | [diff] [blame] | 798 |  | 
| Hidetoshi Seto | 75cb342 | 2009-08-06 11:35:10 +0900 | [diff] [blame] | 799 | msix_program_entries(dev, entries); | 
| Matthew Wilcox | f598282 | 2009-06-18 19:15:59 -0700 | [diff] [blame] | 800 |  | 
| Neil Horman | da8d1c8 | 2011-10-06 14:08:18 -0400 | [diff] [blame] | 801 | ret = populate_msi_sysfs(dev); | 
| Alexander Gordeev | 2adc790 | 2013-12-16 09:34:56 +0100 | [diff] [blame] | 802 | if (ret) | 
|  | 803 | goto out_free; | 
| Neil Horman | da8d1c8 | 2011-10-06 14:08:18 -0400 | [diff] [blame] | 804 |  | 
| Matthew Wilcox | f598282 | 2009-06-18 19:15:59 -0700 | [diff] [blame] | 805 | /* Set MSI-X enabled bits and unmask the function */ | 
| David Miller | ba698ad | 2007-10-25 01:16:30 -0700 | [diff] [blame] | 806 | pci_intx_for_msi(dev, 0); | 
| Eric W. Biederman | b1cbf4e | 2007-03-05 00:30:10 -0800 | [diff] [blame] | 807 | dev->msix_enabled = 1; | 
| Michael S. Tsirkin | 61b64ab | 2015-05-07 09:52:21 -0500 | [diff] [blame] | 808 | pci_msix_clear_and_set_ctrl(dev, PCI_MSIX_FLAGS_MASKALL, 0); | 
| Matthew Wilcox | 8d18101 | 2009-05-08 07:13:33 -0600 | [diff] [blame] | 809 |  | 
| Jiang Liu | 5f22699 | 2015-07-30 14:00:08 -0500 | [diff] [blame] | 810 | pcibios_free_irq(dev); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 811 | return 0; | 
| Hidetoshi Seto | 583871d | 2009-08-06 11:33:39 +0900 | [diff] [blame] | 812 |  | 
| Alexander Gordeev | 2adc790 | 2013-12-16 09:34:56 +0100 | [diff] [blame] | 813 | out_avail: | 
| Hidetoshi Seto | 583871d | 2009-08-06 11:33:39 +0900 | [diff] [blame] | 814 | if (ret < 0) { | 
|  | 815 | /* | 
|  | 816 | * If we had some success, report the number of irqs | 
|  | 817 | * we succeeded in setting up. | 
|  | 818 | */ | 
| Hidetoshi Seto | d9d7070 | 2009-08-06 11:35:48 +0900 | [diff] [blame] | 819 | struct msi_desc *entry; | 
| Hidetoshi Seto | 583871d | 2009-08-06 11:33:39 +0900 | [diff] [blame] | 820 | int avail = 0; | 
|  | 821 |  | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 822 | for_each_pci_msi_entry(entry, dev) { | 
| Hidetoshi Seto | 583871d | 2009-08-06 11:33:39 +0900 | [diff] [blame] | 823 | if (entry->irq != 0) | 
|  | 824 | avail++; | 
|  | 825 | } | 
|  | 826 | if (avail != 0) | 
|  | 827 | ret = avail; | 
|  | 828 | } | 
|  | 829 |  | 
| Alexander Gordeev | 2adc790 | 2013-12-16 09:34:56 +0100 | [diff] [blame] | 830 | out_free: | 
| Hidetoshi Seto | 583871d | 2009-08-06 11:33:39 +0900 | [diff] [blame] | 831 | free_msi_irqs(dev); | 
|  | 832 |  | 
|  | 833 | return ret; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 834 | } | 
|  | 835 |  | 
|  | 836 | /** | 
| Alexander Gordeev | a06cd74 | 2014-09-23 12:45:58 -0600 | [diff] [blame] | 837 | * pci_msi_supported - check whether MSI may be enabled on a device | 
| Brice Goglin | 24334a1 | 2006-08-31 01:55:07 -0400 | [diff] [blame] | 838 | * @dev: pointer to the pci_dev data structure of MSI device function | 
| Michael Ellerman | c9953a7 | 2007-04-05 17:19:08 +1000 | [diff] [blame] | 839 | * @nvec: how many MSIs have been requested ? | 
| Brice Goglin | 24334a1 | 2006-08-31 01:55:07 -0400 | [diff] [blame] | 840 | * | 
| Bjorn Helgaas | f762598 | 2013-11-14 11:28:18 -0700 | [diff] [blame] | 841 | * Look at global flags, the device itself, and its parent buses | 
| Michael Ellerman | 17bbc12 | 2007-04-05 17:19:07 +1000 | [diff] [blame] | 842 | * to determine if MSI/-X are supported for the device. If MSI/-X is | 
| Alexander Gordeev | a06cd74 | 2014-09-23 12:45:58 -0600 | [diff] [blame] | 843 | * supported return 1, else return 0. | 
| Brice Goglin | 24334a1 | 2006-08-31 01:55:07 -0400 | [diff] [blame] | 844 | **/ | 
| Alexander Gordeev | a06cd74 | 2014-09-23 12:45:58 -0600 | [diff] [blame] | 845 | static int pci_msi_supported(struct pci_dev *dev, int nvec) | 
| Brice Goglin | 24334a1 | 2006-08-31 01:55:07 -0400 | [diff] [blame] | 846 | { | 
|  | 847 | struct pci_bus *bus; | 
|  | 848 |  | 
| Brice Goglin | 0306ebf | 2006-10-05 10:24:31 +0200 | [diff] [blame] | 849 | /* MSI must be globally enabled and supported by the device */ | 
| Alexander Gordeev | 27e2060 | 2014-09-23 14:25:11 -0600 | [diff] [blame] | 850 | if (!pci_msi_enable) | 
| Alexander Gordeev | a06cd74 | 2014-09-23 12:45:58 -0600 | [diff] [blame] | 851 | return 0; | 
| Alexander Gordeev | 27e2060 | 2014-09-23 14:25:11 -0600 | [diff] [blame] | 852 |  | 
|  | 853 | if (!dev || dev->no_msi || dev->current_state != PCI_D0) | 
| Alexander Gordeev | a06cd74 | 2014-09-23 12:45:58 -0600 | [diff] [blame] | 854 | return 0; | 
| Brice Goglin | 24334a1 | 2006-08-31 01:55:07 -0400 | [diff] [blame] | 855 |  | 
| Michael Ellerman | 314e77b | 2007-04-05 17:19:12 +1000 | [diff] [blame] | 856 | /* | 
|  | 857 | * You can't ask to have 0 or less MSIs configured. | 
|  | 858 | *  a) it's stupid .. | 
|  | 859 | *  b) the list manipulation code assumes nvec >= 1. | 
|  | 860 | */ | 
|  | 861 | if (nvec < 1) | 
| Alexander Gordeev | a06cd74 | 2014-09-23 12:45:58 -0600 | [diff] [blame] | 862 | return 0; | 
| Michael Ellerman | 314e77b | 2007-04-05 17:19:12 +1000 | [diff] [blame] | 863 |  | 
| Hidetoshi Seto | 500559a | 2009-08-10 10:14:15 +0900 | [diff] [blame] | 864 | /* | 
|  | 865 | * Any bridge which does NOT route MSI transactions from its | 
|  | 866 | * secondary bus to its primary bus must set NO_MSI flag on | 
| Brice Goglin | 0306ebf | 2006-10-05 10:24:31 +0200 | [diff] [blame] | 867 | * the secondary pci_bus. | 
|  | 868 | * We expect only arch-specific PCI host bus controller driver | 
|  | 869 | * or quirks for specific PCI bridges to be setting NO_MSI. | 
|  | 870 | */ | 
| Brice Goglin | 24334a1 | 2006-08-31 01:55:07 -0400 | [diff] [blame] | 871 | for (bus = dev->bus; bus; bus = bus->parent) | 
|  | 872 | if (bus->bus_flags & PCI_BUS_FLAGS_NO_MSI) | 
| Alexander Gordeev | a06cd74 | 2014-09-23 12:45:58 -0600 | [diff] [blame] | 873 | return 0; | 
| Brice Goglin | 24334a1 | 2006-08-31 01:55:07 -0400 | [diff] [blame] | 874 |  | 
| Alexander Gordeev | a06cd74 | 2014-09-23 12:45:58 -0600 | [diff] [blame] | 875 | return 1; | 
| Brice Goglin | 24334a1 | 2006-08-31 01:55:07 -0400 | [diff] [blame] | 876 | } | 
|  | 877 |  | 
|  | 878 | /** | 
| Alexander Gordeev | d1ac1d2 | 2013-12-30 08:28:13 +0100 | [diff] [blame] | 879 | * pci_msi_vec_count - Return the number of MSI vectors a device can send | 
|  | 880 | * @dev: device to report about | 
|  | 881 | * | 
|  | 882 | * This function returns the number of MSI vectors a device requested via | 
|  | 883 | * Multiple Message Capable register. It returns a negative errno if the | 
|  | 884 | * device is not capable sending MSI interrupts. Otherwise, the call succeeds | 
|  | 885 | * and returns a power of two, up to a maximum of 2^5 (32), according to the | 
|  | 886 | * MSI specification. | 
|  | 887 | **/ | 
|  | 888 | int pci_msi_vec_count(struct pci_dev *dev) | 
|  | 889 | { | 
|  | 890 | int ret; | 
|  | 891 | u16 msgctl; | 
|  | 892 |  | 
|  | 893 | if (!dev->msi_cap) | 
|  | 894 | return -EINVAL; | 
|  | 895 |  | 
|  | 896 | pci_read_config_word(dev, dev->msi_cap + PCI_MSI_FLAGS, &msgctl); | 
|  | 897 | ret = 1 << ((msgctl & PCI_MSI_FLAGS_QMASK) >> 1); | 
|  | 898 |  | 
|  | 899 | return ret; | 
|  | 900 | } | 
|  | 901 | EXPORT_SYMBOL(pci_msi_vec_count); | 
|  | 902 |  | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 903 | void pci_msi_shutdown(struct pci_dev *dev) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 904 | { | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 905 | struct msi_desc *desc; | 
|  | 906 | u32 mask; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 907 |  | 
| Michael Ellerman | 128bc5f | 2007-03-22 21:51:39 +1100 | [diff] [blame] | 908 | if (!pci_msi_enable || !dev || !dev->msi_enabled) | 
| Eric W. Biederman | ded86d8 | 2007-01-28 12:42:52 -0700 | [diff] [blame] | 909 | return; | 
|  | 910 |  | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 911 | BUG_ON(list_empty(dev_to_msi_list(&dev->dev))); | 
| Jiang Liu | 4a7cc83 | 2015-07-09 16:00:44 +0800 | [diff] [blame] | 912 | desc = first_pci_msi_entry(dev); | 
| Matthew Wilcox | 110828c | 2009-06-16 06:31:45 -0600 | [diff] [blame] | 913 |  | 
| Michael S. Tsirkin | 61b64ab | 2015-05-07 09:52:21 -0500 | [diff] [blame] | 914 | pci_msi_set_enable(dev, 0); | 
| David Miller | ba698ad | 2007-10-25 01:16:30 -0700 | [diff] [blame] | 915 | pci_intx_for_msi(dev, 1); | 
| Eric W. Biederman | b1cbf4e | 2007-03-05 00:30:10 -0800 | [diff] [blame] | 916 | dev->msi_enabled = 0; | 
| Eric W. Biederman | 7bd007e | 2006-10-04 02:16:31 -0700 | [diff] [blame] | 917 |  | 
| Hidetoshi Seto | 12abb8b | 2009-06-24 12:08:09 +0900 | [diff] [blame] | 918 | /* Return the device with MSI unmasked as initial states */ | 
| Yijing Wang | 31ea5d4 | 2014-06-19 16:30:30 +0800 | [diff] [blame] | 919 | mask = msi_mask(desc->msi_attrib.multi_cap); | 
| Hidetoshi Seto | 12abb8b | 2009-06-24 12:08:09 +0900 | [diff] [blame] | 920 | /* Keep cached state to be restored */ | 
| Thomas Gleixner | 23ed8d5 | 2014-11-23 11:55:58 +0100 | [diff] [blame] | 921 | __pci_msi_desc_mask_irq(desc, mask, ~mask); | 
| Michael Ellerman | e387b9e | 2007-03-22 21:51:27 +1100 | [diff] [blame] | 922 |  | 
|  | 923 | /* Restore dev->irq to its default pin-assertion irq */ | 
| Matthew Wilcox | f2440d9 | 2009-03-17 08:54:09 -0400 | [diff] [blame] | 924 | dev->irq = desc->msi_attrib.default_irq; | 
| Jiang Liu | 5f22699 | 2015-07-30 14:00:08 -0500 | [diff] [blame] | 925 | pcibios_alloc_irq(dev); | 
| Yinghai Lu | d52877c | 2008-04-23 14:58:09 -0700 | [diff] [blame] | 926 | } | 
| Matthew Wilcox | 24d2755 | 2009-03-17 08:54:06 -0400 | [diff] [blame] | 927 |  | 
| Hidetoshi Seto | 500559a | 2009-08-10 10:14:15 +0900 | [diff] [blame] | 928 | void pci_disable_msi(struct pci_dev *dev) | 
| Yinghai Lu | d52877c | 2008-04-23 14:58:09 -0700 | [diff] [blame] | 929 | { | 
| Yinghai Lu | d52877c | 2008-04-23 14:58:09 -0700 | [diff] [blame] | 930 | if (!pci_msi_enable || !dev || !dev->msi_enabled) | 
|  | 931 | return; | 
|  | 932 |  | 
|  | 933 | pci_msi_shutdown(dev); | 
| Hidetoshi Seto | f56e448 | 2009-08-06 11:32:51 +0900 | [diff] [blame] | 934 | free_msi_irqs(dev); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 935 | } | 
| Michael Ellerman | 4cc086f | 2007-03-22 21:51:34 +1100 | [diff] [blame] | 936 | EXPORT_SYMBOL(pci_disable_msi); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 937 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 938 | /** | 
| Alexander Gordeev | ff1aa43 | 2013-12-30 08:28:15 +0100 | [diff] [blame] | 939 | * pci_msix_vec_count - return the number of device's MSI-X table entries | 
| Rafael J. Wysocki | a52e2e3 | 2009-01-24 00:21:14 +0100 | [diff] [blame] | 940 | * @dev: pointer to the pci_dev data structure of MSI-X device function | 
| Alexander Gordeev | ff1aa43 | 2013-12-30 08:28:15 +0100 | [diff] [blame] | 941 | * This function returns the number of device's MSI-X table entries and | 
|  | 942 | * therefore the number of MSI-X vectors device is capable of sending. | 
|  | 943 | * It returns a negative errno if the device is not capable of sending MSI-X | 
|  | 944 | * interrupts. | 
|  | 945 | **/ | 
|  | 946 | int pci_msix_vec_count(struct pci_dev *dev) | 
| Rafael J. Wysocki | a52e2e3 | 2009-01-24 00:21:14 +0100 | [diff] [blame] | 947 | { | 
| Rafael J. Wysocki | a52e2e3 | 2009-01-24 00:21:14 +0100 | [diff] [blame] | 948 | u16 control; | 
|  | 949 |  | 
| Gavin Shan | 520fe9d | 2013-04-04 16:54:33 +0000 | [diff] [blame] | 950 | if (!dev->msix_cap) | 
| Alexander Gordeev | ff1aa43 | 2013-12-30 08:28:15 +0100 | [diff] [blame] | 951 | return -EINVAL; | 
| Rafael J. Wysocki | a52e2e3 | 2009-01-24 00:21:14 +0100 | [diff] [blame] | 952 |  | 
| Bjorn Helgaas | f84ecd2 | 2013-04-17 17:38:32 -0600 | [diff] [blame] | 953 | pci_read_config_word(dev, dev->msix_cap + PCI_MSIX_FLAGS, &control); | 
| Bjorn Helgaas | 527eee2 | 2013-04-17 17:44:48 -0600 | [diff] [blame] | 954 | return msix_table_size(control); | 
| Rafael J. Wysocki | a52e2e3 | 2009-01-24 00:21:14 +0100 | [diff] [blame] | 955 | } | 
| Alexander Gordeev | ff1aa43 | 2013-12-30 08:28:15 +0100 | [diff] [blame] | 956 | EXPORT_SYMBOL(pci_msix_vec_count); | 
| Rafael J. Wysocki | a52e2e3 | 2009-01-24 00:21:14 +0100 | [diff] [blame] | 957 |  | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 958 | static int __pci_enable_msix(struct pci_dev *dev, struct msix_entry *entries, | 
|  | 959 | int nvec, bool affinity) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 960 | { | 
| Bjorn Helgaas | 5ec0940 | 2014-09-23 14:38:28 -0600 | [diff] [blame] | 961 | int nr_entries; | 
| Eric W. Biederman | ded86d8 | 2007-01-28 12:42:52 -0700 | [diff] [blame] | 962 | int i, j; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 963 |  | 
| Alexander Gordeev | a06cd74 | 2014-09-23 12:45:58 -0600 | [diff] [blame] | 964 | if (!pci_msi_supported(dev, nvec)) | 
|  | 965 | return -EINVAL; | 
| Michael Ellerman | c9953a7 | 2007-04-05 17:19:08 +1000 | [diff] [blame] | 966 |  | 
| Alexander Gordeev | ff1aa43 | 2013-12-30 08:28:15 +0100 | [diff] [blame] | 967 | nr_entries = pci_msix_vec_count(dev); | 
|  | 968 | if (nr_entries < 0) | 
|  | 969 | return nr_entries; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 970 | if (nvec > nr_entries) | 
| Michael S. Tsirkin | 57fbf52 | 2009-05-07 11:28:41 +0300 | [diff] [blame] | 971 | return nr_entries; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 972 |  | 
| Christoph Hellwig | 3ac020e | 2016-07-12 18:20:16 +0900 | [diff] [blame] | 973 | if (entries) { | 
|  | 974 | /* Check for any invalid entries */ | 
|  | 975 | for (i = 0; i < nvec; i++) { | 
|  | 976 | if (entries[i].entry >= nr_entries) | 
|  | 977 | return -EINVAL;		/* invalid entry */ | 
|  | 978 | for (j = i + 1; j < nvec; j++) { | 
|  | 979 | if (entries[i].entry == entries[j].entry) | 
|  | 980 | return -EINVAL;	/* duplicate entry */ | 
|  | 981 | } | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 982 | } | 
|  | 983 | } | 
| Eric W. Biederman | ded86d8 | 2007-01-28 12:42:52 -0700 | [diff] [blame] | 984 | WARN_ON(!!dev->msix_enabled); | 
| Eric W. Biederman | 7bd007e | 2006-10-04 02:16:31 -0700 | [diff] [blame] | 985 |  | 
| Eric W. Biederman | 1ce0337 | 2006-10-04 02:16:41 -0700 | [diff] [blame] | 986 | /* Check whether driver already requested for MSI irq */ | 
| Hidetoshi Seto | 500559a | 2009-08-10 10:14:15 +0900 | [diff] [blame] | 987 | if (dev->msi_enabled) { | 
| Ryan Desfosses | 227f064 | 2014-04-18 20:13:50 -0400 | [diff] [blame] | 988 | dev_info(&dev->dev, "can't enable MSI-X (MSI IRQ already assigned)\n"); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 989 | return -EINVAL; | 
|  | 990 | } | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 991 | return msix_capability_init(dev, entries, nvec, affinity); | 
|  | 992 | } | 
|  | 993 |  | 
|  | 994 | /** | 
|  | 995 | * pci_enable_msix - configure device's MSI-X capability structure | 
|  | 996 | * @dev: pointer to the pci_dev data structure of MSI-X device function | 
|  | 997 | * @entries: pointer to an array of MSI-X entries (optional) | 
|  | 998 | * @nvec: number of MSI-X irqs requested for allocation by device driver | 
|  | 999 | * | 
|  | 1000 | * Setup the MSI-X capability structure of device function with the number | 
|  | 1001 | * of requested irqs upon its software driver call to request for | 
|  | 1002 | * MSI-X mode enabled on its hardware device function. A return of zero | 
|  | 1003 | * indicates the successful configuration of MSI-X capability structure | 
|  | 1004 | * with new allocated MSI-X irqs. A return of < 0 indicates a failure. | 
|  | 1005 | * Or a return of > 0 indicates that driver request is exceeding the number | 
|  | 1006 | * of irqs or MSI-X vectors available. Driver should use the returned value to | 
|  | 1007 | * re-send its request. | 
|  | 1008 | **/ | 
|  | 1009 | int pci_enable_msix(struct pci_dev *dev, struct msix_entry *entries, int nvec) | 
|  | 1010 | { | 
|  | 1011 | return __pci_enable_msix(dev, entries, nvec, false); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1012 | } | 
| Michael Ellerman | 4cc086f | 2007-03-22 21:51:34 +1100 | [diff] [blame] | 1013 | EXPORT_SYMBOL(pci_enable_msix); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1014 |  | 
| Hidetoshi Seto | 500559a | 2009-08-10 10:14:15 +0900 | [diff] [blame] | 1015 | void pci_msix_shutdown(struct pci_dev *dev) | 
| Michael Ellerman | fc4afc7 | 2007-03-22 21:51:33 +1100 | [diff] [blame] | 1016 | { | 
| Hidetoshi Seto | 12abb8b | 2009-06-24 12:08:09 +0900 | [diff] [blame] | 1017 | struct msi_desc *entry; | 
|  | 1018 |  | 
| Michael Ellerman | 128bc5f | 2007-03-22 21:51:39 +1100 | [diff] [blame] | 1019 | if (!pci_msi_enable || !dev || !dev->msix_enabled) | 
| Eric W. Biederman | ded86d8 | 2007-01-28 12:42:52 -0700 | [diff] [blame] | 1020 | return; | 
|  | 1021 |  | 
| Hidetoshi Seto | 12abb8b | 2009-06-24 12:08:09 +0900 | [diff] [blame] | 1022 | /* Return the device with MSI-X masked as initial states */ | 
| Jiang Liu | 5004e98 | 2015-07-09 16:00:41 +0800 | [diff] [blame] | 1023 | for_each_pci_msi_entry(entry, dev) { | 
| Hidetoshi Seto | 12abb8b | 2009-06-24 12:08:09 +0900 | [diff] [blame] | 1024 | /* Keep cached states to be restored */ | 
| Thomas Gleixner | 23ed8d5 | 2014-11-23 11:55:58 +0100 | [diff] [blame] | 1025 | __pci_msix_desc_mask_irq(entry, 1); | 
| Hidetoshi Seto | 12abb8b | 2009-06-24 12:08:09 +0900 | [diff] [blame] | 1026 | } | 
|  | 1027 |  | 
| Michael S. Tsirkin | 61b64ab | 2015-05-07 09:52:21 -0500 | [diff] [blame] | 1028 | pci_msix_clear_and_set_ctrl(dev, PCI_MSIX_FLAGS_ENABLE, 0); | 
| David Miller | ba698ad | 2007-10-25 01:16:30 -0700 | [diff] [blame] | 1029 | pci_intx_for_msi(dev, 1); | 
| Eric W. Biederman | b1cbf4e | 2007-03-05 00:30:10 -0800 | [diff] [blame] | 1030 | dev->msix_enabled = 0; | 
| Jiang Liu | 5f22699 | 2015-07-30 14:00:08 -0500 | [diff] [blame] | 1031 | pcibios_alloc_irq(dev); | 
| Yinghai Lu | d52877c | 2008-04-23 14:58:09 -0700 | [diff] [blame] | 1032 | } | 
| Hidetoshi Seto | c901851 | 2009-08-06 11:31:27 +0900 | [diff] [blame] | 1033 |  | 
| Hidetoshi Seto | 500559a | 2009-08-10 10:14:15 +0900 | [diff] [blame] | 1034 | void pci_disable_msix(struct pci_dev *dev) | 
| Yinghai Lu | d52877c | 2008-04-23 14:58:09 -0700 | [diff] [blame] | 1035 | { | 
|  | 1036 | if (!pci_msi_enable || !dev || !dev->msix_enabled) | 
|  | 1037 | return; | 
|  | 1038 |  | 
|  | 1039 | pci_msix_shutdown(dev); | 
| Hidetoshi Seto | f56e448 | 2009-08-06 11:32:51 +0900 | [diff] [blame] | 1040 | free_msi_irqs(dev); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1041 | } | 
| Michael Ellerman | 4cc086f | 2007-03-22 21:51:34 +1100 | [diff] [blame] | 1042 | EXPORT_SYMBOL(pci_disable_msix); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1043 |  | 
| Matthew Wilcox | 309e57d | 2006-03-05 22:33:34 -0700 | [diff] [blame] | 1044 | void pci_no_msi(void) | 
|  | 1045 | { | 
|  | 1046 | pci_msi_enable = 0; | 
|  | 1047 | } | 
| Michael Ellerman | c9953a7 | 2007-04-05 17:19:08 +1000 | [diff] [blame] | 1048 |  | 
| Andrew Patterson | 07ae95f | 2008-11-10 15:31:05 -0700 | [diff] [blame] | 1049 | /** | 
|  | 1050 | * pci_msi_enabled - is MSI enabled? | 
|  | 1051 | * | 
|  | 1052 | * Returns true if MSI has not been disabled by the command-line option | 
|  | 1053 | * pci=nomsi. | 
|  | 1054 | **/ | 
|  | 1055 | int pci_msi_enabled(void) | 
|  | 1056 | { | 
|  | 1057 | return pci_msi_enable; | 
|  | 1058 | } | 
|  | 1059 | EXPORT_SYMBOL(pci_msi_enabled); | 
|  | 1060 |  | 
| Christoph Hellwig | 4ef3368 | 2016-07-12 18:20:18 +0900 | [diff] [blame] | 1061 | static int __pci_enable_msi_range(struct pci_dev *dev, int minvec, int maxvec, | 
|  | 1062 | unsigned int flags) | 
| Alexander Gordeev | 302a252 | 2013-12-30 08:28:16 +0100 | [diff] [blame] | 1063 | { | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 1064 | bool affinity = flags & PCI_IRQ_AFFINITY; | 
| Alexander Gordeev | 034cd97 | 2014-04-14 15:28:35 +0200 | [diff] [blame] | 1065 | int nvec; | 
| Alexander Gordeev | 302a252 | 2013-12-30 08:28:16 +0100 | [diff] [blame] | 1066 | int rc; | 
|  | 1067 |  | 
| Alexander Gordeev | a06cd74 | 2014-09-23 12:45:58 -0600 | [diff] [blame] | 1068 | if (!pci_msi_supported(dev, minvec)) | 
|  | 1069 | return -EINVAL; | 
| Alexander Gordeev | 034cd97 | 2014-04-14 15:28:35 +0200 | [diff] [blame] | 1070 |  | 
|  | 1071 | WARN_ON(!!dev->msi_enabled); | 
|  | 1072 |  | 
|  | 1073 | /* Check whether driver already requested MSI-X irqs */ | 
|  | 1074 | if (dev->msix_enabled) { | 
|  | 1075 | dev_info(&dev->dev, | 
|  | 1076 | "can't enable MSI (MSI-X already enabled)\n"); | 
|  | 1077 | return -EINVAL; | 
|  | 1078 | } | 
|  | 1079 |  | 
| Alexander Gordeev | 302a252 | 2013-12-30 08:28:16 +0100 | [diff] [blame] | 1080 | if (maxvec < minvec) | 
|  | 1081 | return -ERANGE; | 
|  | 1082 |  | 
| Alexander Gordeev | 034cd97 | 2014-04-14 15:28:35 +0200 | [diff] [blame] | 1083 | nvec = pci_msi_vec_count(dev); | 
|  | 1084 | if (nvec < 0) | 
|  | 1085 | return nvec; | 
| Christoph Hellwig | 4ef3368 | 2016-07-12 18:20:18 +0900 | [diff] [blame] | 1086 | if (nvec < minvec) | 
| Alexander Gordeev | 034cd97 | 2014-04-14 15:28:35 +0200 | [diff] [blame] | 1087 | return -EINVAL; | 
| Christoph Hellwig | 4ef3368 | 2016-07-12 18:20:18 +0900 | [diff] [blame] | 1088 |  | 
|  | 1089 | if (nvec > maxvec) | 
| Alexander Gordeev | 034cd97 | 2014-04-14 15:28:35 +0200 | [diff] [blame] | 1090 | nvec = maxvec; | 
|  | 1091 |  | 
| Christoph Hellwig | 4ef3368 | 2016-07-12 18:20:18 +0900 | [diff] [blame] | 1092 | for (;;) { | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 1093 | if (affinity) { | 
|  | 1094 | nvec = irq_calc_affinity_vectors(dev->irq_affinity, | 
|  | 1095 | nvec); | 
| Christoph Hellwig | 4ef3368 | 2016-07-12 18:20:18 +0900 | [diff] [blame] | 1096 | if (nvec < minvec) | 
| Alexander Gordeev | 302a252 | 2013-12-30 08:28:16 +0100 | [diff] [blame] | 1097 | return -ENOSPC; | 
| Alexander Gordeev | 302a252 | 2013-12-30 08:28:16 +0100 | [diff] [blame] | 1098 | } | 
| Alexander Gordeev | 302a252 | 2013-12-30 08:28:16 +0100 | [diff] [blame] | 1099 |  | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 1100 | rc = msi_capability_init(dev, nvec, affinity); | 
| Christoph Hellwig | 4ef3368 | 2016-07-12 18:20:18 +0900 | [diff] [blame] | 1101 | if (rc == 0) | 
|  | 1102 | return nvec; | 
|  | 1103 |  | 
| Christoph Hellwig | 4ef3368 | 2016-07-12 18:20:18 +0900 | [diff] [blame] | 1104 | if (rc < 0) | 
|  | 1105 | return rc; | 
|  | 1106 | if (rc < minvec) | 
|  | 1107 | return -ENOSPC; | 
|  | 1108 |  | 
|  | 1109 | nvec = rc; | 
|  | 1110 | } | 
|  | 1111 | } | 
|  | 1112 |  | 
|  | 1113 | /** | 
|  | 1114 | * pci_enable_msi_range - configure device's MSI capability structure | 
|  | 1115 | * @dev: device to configure | 
|  | 1116 | * @minvec: minimal number of interrupts to configure | 
|  | 1117 | * @maxvec: maximum number of interrupts to configure | 
|  | 1118 | * | 
|  | 1119 | * This function tries to allocate a maximum possible number of interrupts in a | 
|  | 1120 | * range between @minvec and @maxvec. It returns a negative errno if an error | 
|  | 1121 | * occurs. If it succeeds, it returns the actual number of interrupts allocated | 
|  | 1122 | * and updates the @dev's irq member to the lowest new interrupt number; | 
|  | 1123 | * the other interrupt numbers allocated to this device are consecutive. | 
|  | 1124 | **/ | 
|  | 1125 | int pci_enable_msi_range(struct pci_dev *dev, int minvec, int maxvec) | 
|  | 1126 | { | 
| Christoph Hellwig | 4fe0d15 | 2016-08-11 07:11:04 -0700 | [diff] [blame] | 1127 | return __pci_enable_msi_range(dev, minvec, maxvec, 0); | 
| Alexander Gordeev | 302a252 | 2013-12-30 08:28:16 +0100 | [diff] [blame] | 1128 | } | 
|  | 1129 | EXPORT_SYMBOL(pci_enable_msi_range); | 
|  | 1130 |  | 
| Christoph Hellwig | 4ef3368 | 2016-07-12 18:20:18 +0900 | [diff] [blame] | 1131 | static int __pci_enable_msix_range(struct pci_dev *dev, | 
|  | 1132 | struct msix_entry *entries, int minvec, int maxvec, | 
|  | 1133 | unsigned int flags) | 
|  | 1134 | { | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 1135 | bool affinity = flags & PCI_IRQ_AFFINITY; | 
|  | 1136 | int rc, nvec = maxvec; | 
| Christoph Hellwig | 4ef3368 | 2016-07-12 18:20:18 +0900 | [diff] [blame] | 1137 |  | 
|  | 1138 | if (maxvec < minvec) | 
|  | 1139 | return -ERANGE; | 
|  | 1140 |  | 
|  | 1141 | for (;;) { | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 1142 | if (affinity) { | 
|  | 1143 | nvec = irq_calc_affinity_vectors(dev->irq_affinity, | 
|  | 1144 | nvec); | 
| Christoph Hellwig | 4ef3368 | 2016-07-12 18:20:18 +0900 | [diff] [blame] | 1145 | if (nvec < minvec) | 
|  | 1146 | return -ENOSPC; | 
|  | 1147 | } | 
|  | 1148 |  | 
| Thomas Gleixner | e75eafb | 2016-09-14 16:18:49 +0200 | [diff] [blame] | 1149 | rc = __pci_enable_msix(dev, entries, nvec, affinity); | 
| Christoph Hellwig | 4ef3368 | 2016-07-12 18:20:18 +0900 | [diff] [blame] | 1150 | if (rc == 0) | 
|  | 1151 | return nvec; | 
|  | 1152 |  | 
| Christoph Hellwig | 4ef3368 | 2016-07-12 18:20:18 +0900 | [diff] [blame] | 1153 | if (rc < 0) | 
|  | 1154 | return rc; | 
|  | 1155 | if (rc < minvec) | 
|  | 1156 | return -ENOSPC; | 
|  | 1157 |  | 
|  | 1158 | nvec = rc; | 
|  | 1159 | } | 
|  | 1160 | } | 
|  | 1161 |  | 
| Alexander Gordeev | 302a252 | 2013-12-30 08:28:16 +0100 | [diff] [blame] | 1162 | /** | 
|  | 1163 | * pci_enable_msix_range - configure device's MSI-X capability structure | 
|  | 1164 | * @dev: pointer to the pci_dev data structure of MSI-X device function | 
|  | 1165 | * @entries: pointer to an array of MSI-X entries | 
|  | 1166 | * @minvec: minimum number of MSI-X irqs requested | 
|  | 1167 | * @maxvec: maximum number of MSI-X irqs requested | 
|  | 1168 | * | 
|  | 1169 | * Setup the MSI-X capability structure of device function with a maximum | 
|  | 1170 | * possible number of interrupts in the range between @minvec and @maxvec | 
|  | 1171 | * upon its software driver call to request for MSI-X mode enabled on its | 
|  | 1172 | * hardware device function. It returns a negative errno if an error occurs. | 
|  | 1173 | * If it succeeds, it returns the actual number of interrupts allocated and | 
|  | 1174 | * indicates the successful configuration of MSI-X capability structure | 
|  | 1175 | * with new allocated MSI-X interrupts. | 
|  | 1176 | **/ | 
|  | 1177 | int pci_enable_msix_range(struct pci_dev *dev, struct msix_entry *entries, | 
| Christoph Hellwig | 4ef3368 | 2016-07-12 18:20:18 +0900 | [diff] [blame] | 1178 | int minvec, int maxvec) | 
| Alexander Gordeev | 302a252 | 2013-12-30 08:28:16 +0100 | [diff] [blame] | 1179 | { | 
| Christoph Hellwig | 4fe0d15 | 2016-08-11 07:11:04 -0700 | [diff] [blame] | 1180 | return __pci_enable_msix_range(dev, entries, minvec, maxvec, 0); | 
| Alexander Gordeev | 302a252 | 2013-12-30 08:28:16 +0100 | [diff] [blame] | 1181 | } | 
|  | 1182 | EXPORT_SYMBOL(pci_enable_msix_range); | 
| Jiang Liu | 3878eae | 2014-11-11 21:02:18 +0800 | [diff] [blame] | 1183 |  | 
| Christoph Hellwig | aff1716 | 2016-07-12 18:20:17 +0900 | [diff] [blame] | 1184 | /** | 
|  | 1185 | * pci_alloc_irq_vectors - allocate multiple IRQs for a device | 
|  | 1186 | * @dev:		PCI device to operate on | 
|  | 1187 | * @min_vecs:		minimum number of vectors required (must be >= 1) | 
|  | 1188 | * @max_vecs:		maximum (desired) number of vectors | 
|  | 1189 | * @flags:		flags or quirks for the allocation | 
|  | 1190 | * | 
|  | 1191 | * Allocate up to @max_vecs interrupt vectors for @dev, using MSI-X or MSI | 
|  | 1192 | * vectors if available, and fall back to a single legacy vector | 
|  | 1193 | * if neither is available.  Return the number of vectors allocated, | 
|  | 1194 | * (which might be smaller than @max_vecs) if successful, or a negative | 
|  | 1195 | * error code on error. If less than @min_vecs interrupt vectors are | 
|  | 1196 | * available for @dev the function will fail with -ENOSPC. | 
|  | 1197 | * | 
|  | 1198 | * To get the Linux IRQ number used for a vector that can be passed to | 
|  | 1199 | * request_irq() use the pci_irq_vector() helper. | 
|  | 1200 | */ | 
|  | 1201 | int pci_alloc_irq_vectors(struct pci_dev *dev, unsigned int min_vecs, | 
|  | 1202 | unsigned int max_vecs, unsigned int flags) | 
|  | 1203 | { | 
|  | 1204 | int vecs = -ENOSPC; | 
|  | 1205 |  | 
| Christoph Hellwig | 4fe0d15 | 2016-08-11 07:11:04 -0700 | [diff] [blame] | 1206 | if (flags & PCI_IRQ_MSIX) { | 
| Christoph Hellwig | 4ef3368 | 2016-07-12 18:20:18 +0900 | [diff] [blame] | 1207 | vecs = __pci_enable_msix_range(dev, NULL, min_vecs, max_vecs, | 
|  | 1208 | flags); | 
| Christoph Hellwig | aff1716 | 2016-07-12 18:20:17 +0900 | [diff] [blame] | 1209 | if (vecs > 0) | 
|  | 1210 | return vecs; | 
|  | 1211 | } | 
|  | 1212 |  | 
| Christoph Hellwig | 4fe0d15 | 2016-08-11 07:11:04 -0700 | [diff] [blame] | 1213 | if (flags & PCI_IRQ_MSI) { | 
| Christoph Hellwig | 4ef3368 | 2016-07-12 18:20:18 +0900 | [diff] [blame] | 1214 | vecs = __pci_enable_msi_range(dev, min_vecs, max_vecs, flags); | 
| Christoph Hellwig | aff1716 | 2016-07-12 18:20:17 +0900 | [diff] [blame] | 1215 | if (vecs > 0) | 
|  | 1216 | return vecs; | 
|  | 1217 | } | 
|  | 1218 |  | 
|  | 1219 | /* use legacy irq if allowed */ | 
| Christoph Hellwig | 5d0bdf2 | 2016-08-11 07:11:05 -0700 | [diff] [blame] | 1220 | if ((flags & PCI_IRQ_LEGACY) && min_vecs == 1) { | 
|  | 1221 | pci_intx(dev, 1); | 
| Christoph Hellwig | aff1716 | 2016-07-12 18:20:17 +0900 | [diff] [blame] | 1222 | return 1; | 
| Christoph Hellwig | 5d0bdf2 | 2016-08-11 07:11:05 -0700 | [diff] [blame] | 1223 | } | 
|  | 1224 |  | 
| Christoph Hellwig | aff1716 | 2016-07-12 18:20:17 +0900 | [diff] [blame] | 1225 | return vecs; | 
|  | 1226 | } | 
|  | 1227 | EXPORT_SYMBOL(pci_alloc_irq_vectors); | 
|  | 1228 |  | 
|  | 1229 | /** | 
|  | 1230 | * pci_free_irq_vectors - free previously allocated IRQs for a device | 
|  | 1231 | * @dev:		PCI device to operate on | 
|  | 1232 | * | 
|  | 1233 | * Undoes the allocations and enabling in pci_alloc_irq_vectors(). | 
|  | 1234 | */ | 
|  | 1235 | void pci_free_irq_vectors(struct pci_dev *dev) | 
|  | 1236 | { | 
|  | 1237 | pci_disable_msix(dev); | 
|  | 1238 | pci_disable_msi(dev); | 
|  | 1239 | } | 
|  | 1240 | EXPORT_SYMBOL(pci_free_irq_vectors); | 
|  | 1241 |  | 
|  | 1242 | /** | 
|  | 1243 | * pci_irq_vector - return Linux IRQ number of a device vector | 
|  | 1244 | * @dev: PCI device to operate on | 
|  | 1245 | * @nr: device-relative interrupt vector index (0-based). | 
|  | 1246 | */ | 
|  | 1247 | int pci_irq_vector(struct pci_dev *dev, unsigned int nr) | 
|  | 1248 | { | 
|  | 1249 | if (dev->msix_enabled) { | 
|  | 1250 | struct msi_desc *entry; | 
|  | 1251 | int i = 0; | 
|  | 1252 |  | 
|  | 1253 | for_each_pci_msi_entry(entry, dev) { | 
|  | 1254 | if (i == nr) | 
|  | 1255 | return entry->irq; | 
|  | 1256 | i++; | 
|  | 1257 | } | 
|  | 1258 | WARN_ON_ONCE(1); | 
|  | 1259 | return -EINVAL; | 
|  | 1260 | } | 
|  | 1261 |  | 
|  | 1262 | if (dev->msi_enabled) { | 
|  | 1263 | struct msi_desc *entry = first_pci_msi_entry(dev); | 
|  | 1264 |  | 
|  | 1265 | if (WARN_ON_ONCE(nr >= entry->nvec_used)) | 
|  | 1266 | return -EINVAL; | 
|  | 1267 | } else { | 
|  | 1268 | if (WARN_ON_ONCE(nr > 0)) | 
|  | 1269 | return -EINVAL; | 
|  | 1270 | } | 
|  | 1271 |  | 
|  | 1272 | return dev->irq + nr; | 
|  | 1273 | } | 
|  | 1274 | EXPORT_SYMBOL(pci_irq_vector); | 
|  | 1275 |  | 
| Thomas Gleixner | ee8d41e | 2016-09-14 16:18:51 +0200 | [diff] [blame] | 1276 | /** | 
|  | 1277 | * pci_irq_get_affinity - return the affinity of a particular msi vector | 
|  | 1278 | * @dev:	PCI device to operate on | 
|  | 1279 | * @nr:		device-relative interrupt vector index (0-based). | 
|  | 1280 | */ | 
|  | 1281 | const struct cpumask *pci_irq_get_affinity(struct pci_dev *dev, int nr) | 
|  | 1282 | { | 
|  | 1283 | if (dev->msix_enabled) { | 
|  | 1284 | struct msi_desc *entry; | 
|  | 1285 | int i = 0; | 
|  | 1286 |  | 
|  | 1287 | for_each_pci_msi_entry(entry, dev) { | 
|  | 1288 | if (i == nr) | 
|  | 1289 | return entry->affinity; | 
|  | 1290 | i++; | 
|  | 1291 | } | 
|  | 1292 | WARN_ON_ONCE(1); | 
|  | 1293 | return NULL; | 
|  | 1294 | } else if (dev->msi_enabled) { | 
|  | 1295 | struct msi_desc *entry = first_pci_msi_entry(dev); | 
|  | 1296 |  | 
|  | 1297 | if (WARN_ON_ONCE(!entry || nr >= entry->nvec_used)) | 
|  | 1298 | return NULL; | 
|  | 1299 |  | 
|  | 1300 | return &entry->affinity[nr]; | 
|  | 1301 | } else { | 
|  | 1302 | return cpu_possible_mask; | 
|  | 1303 | } | 
|  | 1304 | } | 
|  | 1305 | EXPORT_SYMBOL(pci_irq_get_affinity); | 
|  | 1306 |  | 
| Jiang Liu | 25a98bd | 2015-07-09 16:00:45 +0800 | [diff] [blame] | 1307 | struct pci_dev *msi_desc_to_pci_dev(struct msi_desc *desc) | 
|  | 1308 | { | 
|  | 1309 | return to_pci_dev(desc->dev); | 
|  | 1310 | } | 
| Jake Oshins | a4289dc | 2015-12-10 17:52:59 +0000 | [diff] [blame] | 1311 | EXPORT_SYMBOL(msi_desc_to_pci_dev); | 
| Jiang Liu | 25a98bd | 2015-07-09 16:00:45 +0800 | [diff] [blame] | 1312 |  | 
| Jiang Liu | c179c9b | 2015-07-09 16:00:36 +0800 | [diff] [blame] | 1313 | void *msi_desc_to_pci_sysdata(struct msi_desc *desc) | 
|  | 1314 | { | 
|  | 1315 | struct pci_dev *dev = msi_desc_to_pci_dev(desc); | 
|  | 1316 |  | 
|  | 1317 | return dev->bus->sysdata; | 
|  | 1318 | } | 
|  | 1319 | EXPORT_SYMBOL_GPL(msi_desc_to_pci_sysdata); | 
|  | 1320 |  | 
| Jiang Liu | 3878eae | 2014-11-11 21:02:18 +0800 | [diff] [blame] | 1321 | #ifdef CONFIG_PCI_MSI_IRQ_DOMAIN | 
|  | 1322 | /** | 
|  | 1323 | * pci_msi_domain_write_msg - Helper to write MSI message to PCI config space | 
|  | 1324 | * @irq_data:	Pointer to interrupt data of the MSI interrupt | 
|  | 1325 | * @msg:	Pointer to the message | 
|  | 1326 | */ | 
|  | 1327 | void pci_msi_domain_write_msg(struct irq_data *irq_data, struct msi_msg *msg) | 
|  | 1328 | { | 
| Jiang Liu | 507a883 | 2015-06-01 16:05:42 +0800 | [diff] [blame] | 1329 | struct msi_desc *desc = irq_data_get_msi_desc(irq_data); | 
| Jiang Liu | 3878eae | 2014-11-11 21:02:18 +0800 | [diff] [blame] | 1330 |  | 
|  | 1331 | /* | 
|  | 1332 | * For MSI-X desc->irq is always equal to irq_data->irq. For | 
|  | 1333 | * MSI only the first interrupt of MULTI MSI passes the test. | 
|  | 1334 | */ | 
|  | 1335 | if (desc->irq == irq_data->irq) | 
|  | 1336 | __pci_write_msi_msg(desc, msg); | 
|  | 1337 | } | 
|  | 1338 |  | 
|  | 1339 | /** | 
|  | 1340 | * pci_msi_domain_calc_hwirq - Generate a unique ID for an MSI source | 
|  | 1341 | * @dev:	Pointer to the PCI device | 
|  | 1342 | * @desc:	Pointer to the msi descriptor | 
|  | 1343 | * | 
|  | 1344 | * The ID number is only used within the irqdomain. | 
|  | 1345 | */ | 
|  | 1346 | irq_hw_number_t pci_msi_domain_calc_hwirq(struct pci_dev *dev, | 
|  | 1347 | struct msi_desc *desc) | 
|  | 1348 | { | 
|  | 1349 | return (irq_hw_number_t)desc->msi_attrib.entry_nr | | 
|  | 1350 | PCI_DEVID(dev->bus->number, dev->devfn) << 11 | | 
|  | 1351 | (pci_domain_nr(dev->bus) & 0xFFFFFFFF) << 27; | 
|  | 1352 | } | 
|  | 1353 |  | 
|  | 1354 | static inline bool pci_msi_desc_is_multi_msi(struct msi_desc *desc) | 
|  | 1355 | { | 
|  | 1356 | return !desc->msi_attrib.is_msix && desc->nvec_used > 1; | 
|  | 1357 | } | 
|  | 1358 |  | 
|  | 1359 | /** | 
|  | 1360 | * pci_msi_domain_check_cap - Verify that @domain supports the capabilities for @dev | 
|  | 1361 | * @domain:	The interrupt domain to check | 
|  | 1362 | * @info:	The domain info for verification | 
|  | 1363 | * @dev:	The device to check | 
|  | 1364 | * | 
|  | 1365 | * Returns: | 
|  | 1366 | *  0 if the functionality is supported | 
|  | 1367 | *  1 if Multi MSI is requested, but the domain does not support it | 
|  | 1368 | *  -ENOTSUPP otherwise | 
|  | 1369 | */ | 
|  | 1370 | int pci_msi_domain_check_cap(struct irq_domain *domain, | 
|  | 1371 | struct msi_domain_info *info, struct device *dev) | 
|  | 1372 | { | 
|  | 1373 | struct msi_desc *desc = first_pci_msi_entry(to_pci_dev(dev)); | 
|  | 1374 |  | 
|  | 1375 | /* Special handling to support pci_enable_msi_range() */ | 
|  | 1376 | if (pci_msi_desc_is_multi_msi(desc) && | 
|  | 1377 | !(info->flags & MSI_FLAG_MULTI_PCI_MSI)) | 
|  | 1378 | return 1; | 
|  | 1379 | else if (desc->msi_attrib.is_msix && !(info->flags & MSI_FLAG_PCI_MSIX)) | 
|  | 1380 | return -ENOTSUPP; | 
|  | 1381 |  | 
|  | 1382 | return 0; | 
|  | 1383 | } | 
|  | 1384 |  | 
|  | 1385 | static int pci_msi_domain_handle_error(struct irq_domain *domain, | 
|  | 1386 | struct msi_desc *desc, int error) | 
|  | 1387 | { | 
|  | 1388 | /* Special handling to support pci_enable_msi_range() */ | 
|  | 1389 | if (pci_msi_desc_is_multi_msi(desc) && error == -ENOSPC) | 
|  | 1390 | return 1; | 
|  | 1391 |  | 
|  | 1392 | return error; | 
|  | 1393 | } | 
|  | 1394 |  | 
|  | 1395 | #ifdef GENERIC_MSI_DOMAIN_OPS | 
|  | 1396 | static void pci_msi_domain_set_desc(msi_alloc_info_t *arg, | 
|  | 1397 | struct msi_desc *desc) | 
|  | 1398 | { | 
|  | 1399 | arg->desc = desc; | 
|  | 1400 | arg->hwirq = pci_msi_domain_calc_hwirq(msi_desc_to_pci_dev(desc), | 
|  | 1401 | desc); | 
|  | 1402 | } | 
|  | 1403 | #else | 
|  | 1404 | #define pci_msi_domain_set_desc		NULL | 
|  | 1405 | #endif | 
|  | 1406 |  | 
|  | 1407 | static struct msi_domain_ops pci_msi_domain_ops_default = { | 
|  | 1408 | .set_desc	= pci_msi_domain_set_desc, | 
|  | 1409 | .msi_check	= pci_msi_domain_check_cap, | 
|  | 1410 | .handle_error	= pci_msi_domain_handle_error, | 
|  | 1411 | }; | 
|  | 1412 |  | 
|  | 1413 | static void pci_msi_domain_update_dom_ops(struct msi_domain_info *info) | 
|  | 1414 | { | 
|  | 1415 | struct msi_domain_ops *ops = info->ops; | 
|  | 1416 |  | 
|  | 1417 | if (ops == NULL) { | 
|  | 1418 | info->ops = &pci_msi_domain_ops_default; | 
|  | 1419 | } else { | 
|  | 1420 | if (ops->set_desc == NULL) | 
|  | 1421 | ops->set_desc = pci_msi_domain_set_desc; | 
|  | 1422 | if (ops->msi_check == NULL) | 
|  | 1423 | ops->msi_check = pci_msi_domain_check_cap; | 
|  | 1424 | if (ops->handle_error == NULL) | 
|  | 1425 | ops->handle_error = pci_msi_domain_handle_error; | 
|  | 1426 | } | 
|  | 1427 | } | 
|  | 1428 |  | 
|  | 1429 | static void pci_msi_domain_update_chip_ops(struct msi_domain_info *info) | 
|  | 1430 | { | 
|  | 1431 | struct irq_chip *chip = info->chip; | 
|  | 1432 |  | 
|  | 1433 | BUG_ON(!chip); | 
|  | 1434 | if (!chip->irq_write_msi_msg) | 
|  | 1435 | chip->irq_write_msi_msg = pci_msi_domain_write_msg; | 
| Marc Zyngier | 0701c53 | 2015-10-13 19:14:45 +0100 | [diff] [blame] | 1436 | if (!chip->irq_mask) | 
|  | 1437 | chip->irq_mask = pci_msi_mask_irq; | 
|  | 1438 | if (!chip->irq_unmask) | 
|  | 1439 | chip->irq_unmask = pci_msi_unmask_irq; | 
| Jiang Liu | 3878eae | 2014-11-11 21:02:18 +0800 | [diff] [blame] | 1440 | } | 
|  | 1441 |  | 
|  | 1442 | /** | 
| Marc Zyngier | be5436c | 2015-10-13 12:51:44 +0100 | [diff] [blame] | 1443 | * pci_msi_create_irq_domain - Create a MSI interrupt domain | 
|  | 1444 | * @fwnode:	Optional fwnode of the interrupt controller | 
| Jiang Liu | 3878eae | 2014-11-11 21:02:18 +0800 | [diff] [blame] | 1445 | * @info:	MSI domain info | 
|  | 1446 | * @parent:	Parent irq domain | 
|  | 1447 | * | 
|  | 1448 | * Updates the domain and chip ops and creates a MSI interrupt domain. | 
|  | 1449 | * | 
|  | 1450 | * Returns: | 
|  | 1451 | * A domain pointer or NULL in case of failure. | 
|  | 1452 | */ | 
| Marc Zyngier | be5436c | 2015-10-13 12:51:44 +0100 | [diff] [blame] | 1453 | struct irq_domain *pci_msi_create_irq_domain(struct fwnode_handle *fwnode, | 
| Jiang Liu | 3878eae | 2014-11-11 21:02:18 +0800 | [diff] [blame] | 1454 | struct msi_domain_info *info, | 
|  | 1455 | struct irq_domain *parent) | 
|  | 1456 | { | 
| Marc Zyngier | 0380839 | 2015-07-28 14:46:09 +0100 | [diff] [blame] | 1457 | struct irq_domain *domain; | 
|  | 1458 |  | 
| Jiang Liu | 3878eae | 2014-11-11 21:02:18 +0800 | [diff] [blame] | 1459 | if (info->flags & MSI_FLAG_USE_DEF_DOM_OPS) | 
|  | 1460 | pci_msi_domain_update_dom_ops(info); | 
|  | 1461 | if (info->flags & MSI_FLAG_USE_DEF_CHIP_OPS) | 
|  | 1462 | pci_msi_domain_update_chip_ops(info); | 
|  | 1463 |  | 
| Marc Zyngier | f3b0946 | 2016-07-13 17:18:33 +0100 | [diff] [blame] | 1464 | info->flags |= MSI_FLAG_ACTIVATE_EARLY; | 
|  | 1465 |  | 
| Marc Zyngier | be5436c | 2015-10-13 12:51:44 +0100 | [diff] [blame] | 1466 | domain = msi_create_irq_domain(fwnode, info, parent); | 
| Marc Zyngier | 0380839 | 2015-07-28 14:46:09 +0100 | [diff] [blame] | 1467 | if (!domain) | 
|  | 1468 | return NULL; | 
|  | 1469 |  | 
|  | 1470 | domain->bus_token = DOMAIN_BUS_PCI_MSI; | 
|  | 1471 | return domain; | 
| Jiang Liu | 3878eae | 2014-11-11 21:02:18 +0800 | [diff] [blame] | 1472 | } | 
| Jake Oshins | a4289dc | 2015-12-10 17:52:59 +0000 | [diff] [blame] | 1473 | EXPORT_SYMBOL_GPL(pci_msi_create_irq_domain); | 
| Jiang Liu | 3878eae | 2014-11-11 21:02:18 +0800 | [diff] [blame] | 1474 |  | 
|  | 1475 | /** | 
|  | 1476 | * pci_msi_domain_alloc_irqs - Allocate interrupts for @dev in @domain | 
|  | 1477 | * @domain:	The interrupt domain to allocate from | 
|  | 1478 | * @dev:	The device for which to allocate | 
|  | 1479 | * @nvec:	The number of interrupts to allocate | 
|  | 1480 | * @type:	Unused to allow simpler migration from the arch_XXX interfaces | 
|  | 1481 | * | 
|  | 1482 | * Returns: | 
|  | 1483 | * A virtual interrupt number or an error code in case of failure | 
|  | 1484 | */ | 
|  | 1485 | int pci_msi_domain_alloc_irqs(struct irq_domain *domain, struct pci_dev *dev, | 
|  | 1486 | int nvec, int type) | 
|  | 1487 | { | 
|  | 1488 | return msi_domain_alloc_irqs(domain, &dev->dev, nvec); | 
|  | 1489 | } | 
|  | 1490 |  | 
|  | 1491 | /** | 
|  | 1492 | * pci_msi_domain_free_irqs - Free interrupts for @dev in @domain | 
|  | 1493 | * @domain:	The interrupt domain | 
|  | 1494 | * @dev:	The device for which to free interrupts | 
|  | 1495 | */ | 
|  | 1496 | void pci_msi_domain_free_irqs(struct irq_domain *domain, struct pci_dev *dev) | 
|  | 1497 | { | 
|  | 1498 | msi_domain_free_irqs(domain, &dev->dev); | 
|  | 1499 | } | 
| Jiang Liu | 8e047ad | 2014-11-15 22:24:07 +0800 | [diff] [blame] | 1500 |  | 
|  | 1501 | /** | 
|  | 1502 | * pci_msi_create_default_irq_domain - Create a default MSI interrupt domain | 
| Marc Zyngier | be5436c | 2015-10-13 12:51:44 +0100 | [diff] [blame] | 1503 | * @fwnode:	Optional fwnode of the interrupt controller | 
| Jiang Liu | 8e047ad | 2014-11-15 22:24:07 +0800 | [diff] [blame] | 1504 | * @info:	MSI domain info | 
|  | 1505 | * @parent:	Parent irq domain | 
|  | 1506 | * | 
|  | 1507 | * Returns: A domain pointer or NULL in case of failure. If successful | 
|  | 1508 | * the default PCI/MSI irqdomain pointer is updated. | 
|  | 1509 | */ | 
| Marc Zyngier | be5436c | 2015-10-13 12:51:44 +0100 | [diff] [blame] | 1510 | struct irq_domain *pci_msi_create_default_irq_domain(struct fwnode_handle *fwnode, | 
| Jiang Liu | 8e047ad | 2014-11-15 22:24:07 +0800 | [diff] [blame] | 1511 | struct msi_domain_info *info, struct irq_domain *parent) | 
|  | 1512 | { | 
|  | 1513 | struct irq_domain *domain; | 
|  | 1514 |  | 
|  | 1515 | mutex_lock(&pci_msi_domain_lock); | 
|  | 1516 | if (pci_msi_default_domain) { | 
|  | 1517 | pr_err("PCI: default irq domain for PCI MSI has already been created.\n"); | 
|  | 1518 | domain = NULL; | 
|  | 1519 | } else { | 
| Marc Zyngier | be5436c | 2015-10-13 12:51:44 +0100 | [diff] [blame] | 1520 | domain = pci_msi_create_irq_domain(fwnode, info, parent); | 
| Jiang Liu | 8e047ad | 2014-11-15 22:24:07 +0800 | [diff] [blame] | 1521 | pci_msi_default_domain = domain; | 
|  | 1522 | } | 
|  | 1523 | mutex_unlock(&pci_msi_domain_lock); | 
|  | 1524 |  | 
|  | 1525 | return domain; | 
|  | 1526 | } | 
| David Daney | b6eec9b | 2015-10-08 15:10:49 -0700 | [diff] [blame] | 1527 |  | 
|  | 1528 | static int get_msi_id_cb(struct pci_dev *pdev, u16 alias, void *data) | 
|  | 1529 | { | 
|  | 1530 | u32 *pa = data; | 
|  | 1531 |  | 
|  | 1532 | *pa = alias; | 
|  | 1533 | return 0; | 
|  | 1534 | } | 
|  | 1535 | /** | 
|  | 1536 | * pci_msi_domain_get_msi_rid - Get the MSI requester id (RID) | 
|  | 1537 | * @domain:	The interrupt domain | 
|  | 1538 | * @pdev:	The PCI device. | 
|  | 1539 | * | 
|  | 1540 | * The RID for a device is formed from the alias, with a firmware | 
|  | 1541 | * supplied mapping applied | 
|  | 1542 | * | 
|  | 1543 | * Returns: The RID. | 
|  | 1544 | */ | 
|  | 1545 | u32 pci_msi_domain_get_msi_rid(struct irq_domain *domain, struct pci_dev *pdev) | 
|  | 1546 | { | 
|  | 1547 | struct device_node *of_node; | 
|  | 1548 | u32 rid = 0; | 
|  | 1549 |  | 
|  | 1550 | pci_for_each_dma_alias(pdev, get_msi_id_cb, &rid); | 
|  | 1551 |  | 
|  | 1552 | of_node = irq_domain_get_of_node(domain); | 
| Tomasz Nowicki | be2021b | 2016-09-12 20:32:22 +0200 | [diff] [blame] | 1553 | rid = of_node ? of_msi_map_rid(&pdev->dev, of_node, rid) : | 
|  | 1554 | iort_msi_map_rid(&pdev->dev, rid); | 
| David Daney | b6eec9b | 2015-10-08 15:10:49 -0700 | [diff] [blame] | 1555 |  | 
|  | 1556 | return rid; | 
|  | 1557 | } | 
| Marc Zyngier | 54fa97e | 2015-10-02 14:43:06 +0100 | [diff] [blame] | 1558 |  | 
|  | 1559 | /** | 
|  | 1560 | * pci_msi_get_device_domain - Get the MSI domain for a given PCI device | 
|  | 1561 | * @pdev:	The PCI device | 
|  | 1562 | * | 
|  | 1563 | * Use the firmware data to find a device-specific MSI domain | 
|  | 1564 | * (i.e. not one that is ste as a default). | 
|  | 1565 | * | 
|  | 1566 | * Returns: The coresponding MSI domain or NULL if none has been found. | 
|  | 1567 | */ | 
|  | 1568 | struct irq_domain *pci_msi_get_device_domain(struct pci_dev *pdev) | 
|  | 1569 | { | 
| Tomasz Nowicki | be2021b | 2016-09-12 20:32:22 +0200 | [diff] [blame] | 1570 | struct irq_domain *dom; | 
| Marc Zyngier | 54fa97e | 2015-10-02 14:43:06 +0100 | [diff] [blame] | 1571 | u32 rid = 0; | 
|  | 1572 |  | 
|  | 1573 | pci_for_each_dma_alias(pdev, get_msi_id_cb, &rid); | 
| Tomasz Nowicki | be2021b | 2016-09-12 20:32:22 +0200 | [diff] [blame] | 1574 | dom = of_msi_map_get_device_domain(&pdev->dev, rid); | 
|  | 1575 | if (!dom) | 
|  | 1576 | dom = iort_get_device_domain(&pdev->dev, rid); | 
|  | 1577 | return dom; | 
| Marc Zyngier | 54fa97e | 2015-10-02 14:43:06 +0100 | [diff] [blame] | 1578 | } | 
| Jiang Liu | 3878eae | 2014-11-11 21:02:18 +0800 | [diff] [blame] | 1579 | #endif /* CONFIG_PCI_MSI_IRQ_DOMAIN */ |