diff options
author | Jiang Liu <jiang.liu@linux.intel.com> | 2014-11-15 09:24:04 -0500 |
---|---|---|
committer | Thomas Gleixner <tglx@linutronix.de> | 2014-11-23 07:01:47 -0500 |
commit | d9109698be6e7439e6082aa00d79d4556114739b (patch) | |
tree | 58b79edd974b2b6d57bd927375946eed9109c333 /kernel/irq/msi.c | |
parent | 926ff9ad76e097011030feaee904395e06eea17a (diff) |
genirq: Introduce msi_domain_alloc/free_irqs()
Introduce msi_domain_{alloc|free}_irqs() to alloc/free interrupts
from generic MSI irqdomain.
Signed-off-by: Jiang Liu <jiang.liu@linux.intel.com>
Cc: Tony Luck <tony.luck@intel.com>
Cc: linux-arm-kernel@lists.infradead.org
Cc: Bjorn Helgaas <bhelgaas@google.com>
Cc: Grant Likely <grant.likely@linaro.org>
Cc: Marc Zyngier <marc.zyngier@arm.com>
Cc: Yijing Wang <wangyijing@huawei.com>
Cc: Yingjoe Chen <yingjoe.chen@mediatek.com>
Cc: Borislav Petkov <bp@alien8.de>
Cc: Matthias Brugger <matthias.bgg@gmail.com>
Cc: Alexander Gordeev <agordeev@redhat.com>
Link: http://lkml.kernel.org/r/1416061447-9472-7-git-send-email-jiang.liu@linux.intel.com
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Diffstat (limited to 'kernel/irq/msi.c')
-rw-r--r-- | kernel/irq/msi.c | 75 |
1 files changed, 75 insertions, 0 deletions
diff --git a/kernel/irq/msi.c b/kernel/irq/msi.c index 5e0cef4741d9..23111aaa06b2 100644 --- a/kernel/irq/msi.c +++ b/kernel/irq/msi.c | |||
@@ -13,6 +13,9 @@ | |||
13 | #include <linux/irqdomain.h> | 13 | #include <linux/irqdomain.h> |
14 | #include <linux/msi.h> | 14 | #include <linux/msi.h> |
15 | 15 | ||
16 | /* Temparory solution for building, will be removed later */ | ||
17 | #include <linux/pci.h> | ||
18 | |||
16 | #ifdef CONFIG_GENERIC_MSI_IRQ_DOMAIN | 19 | #ifdef CONFIG_GENERIC_MSI_IRQ_DOMAIN |
17 | /** | 20 | /** |
18 | * msi_domain_set_affinity - Generic affinity setter function for MSI domains | 21 | * msi_domain_set_affinity - Generic affinity setter function for MSI domains |
@@ -127,6 +130,78 @@ struct irq_domain *msi_create_irq_domain(struct device_node *of_node, | |||
127 | } | 130 | } |
128 | 131 | ||
129 | /** | 132 | /** |
133 | * msi_domain_alloc_irqs - Allocate interrupts from a MSI interrupt domain | ||
134 | * @domain: The domain to allocate from | ||
135 | * @dev: Pointer to device struct of the device for which the interrupts | ||
136 | * are allocated | ||
137 | * @nvec: The number of interrupts to allocate | ||
138 | * | ||
139 | * Returns 0 on success or an error code. | ||
140 | */ | ||
141 | int msi_domain_alloc_irqs(struct irq_domain *domain, struct device *dev, | ||
142 | int nvec) | ||
143 | { | ||
144 | struct msi_domain_info *info = domain->host_data; | ||
145 | struct msi_domain_ops *ops = info->ops; | ||
146 | msi_alloc_info_t arg; | ||
147 | struct msi_desc *desc; | ||
148 | int i, ret, virq = -1; | ||
149 | |||
150 | ret = ops->msi_check(domain, info, dev); | ||
151 | if (ret == 0) | ||
152 | ret = ops->msi_prepare(domain, dev, nvec, &arg); | ||
153 | if (ret) | ||
154 | return ret; | ||
155 | |||
156 | for_each_msi_entry(desc, dev) { | ||
157 | ops->set_desc(&arg, desc); | ||
158 | |||
159 | virq = __irq_domain_alloc_irqs(domain, -1, desc->nvec_used, | ||
160 | dev_to_node(dev), &arg, false); | ||
161 | if (virq < 0) { | ||
162 | ret = -ENOSPC; | ||
163 | if (ops->handle_error) | ||
164 | ret = ops->handle_error(domain, desc, ret); | ||
165 | if (ops->msi_finish) | ||
166 | ops->msi_finish(&arg, ret); | ||
167 | return ret; | ||
168 | } | ||
169 | |||
170 | for (i = 0; i < desc->nvec_used; i++) | ||
171 | irq_set_msi_desc_off(virq, i, desc); | ||
172 | } | ||
173 | |||
174 | if (ops->msi_finish) | ||
175 | ops->msi_finish(&arg, 0); | ||
176 | |||
177 | for_each_msi_entry(desc, dev) { | ||
178 | if (desc->nvec_used == 1) | ||
179 | dev_dbg(dev, "irq %d for MSI\n", virq); | ||
180 | else | ||
181 | dev_dbg(dev, "irq [%d-%d] for MSI\n", | ||
182 | virq, virq + desc->nvec_used - 1); | ||
183 | } | ||
184 | |||
185 | return 0; | ||
186 | } | ||
187 | |||
188 | /** | ||
189 | * msi_domain_free_irqs - Free interrupts from a MSI interrupt @domain associated tp @dev | ||
190 | * @domain: The domain to managing the interrupts | ||
191 | * @dev: Pointer to device struct of the device for which the interrupts | ||
192 | * are free | ||
193 | */ | ||
194 | void msi_domain_free_irqs(struct irq_domain *domain, struct device *dev) | ||
195 | { | ||
196 | struct msi_desc *desc; | ||
197 | |||
198 | for_each_msi_entry(desc, dev) { | ||
199 | irq_domain_free_irqs(desc->irq, desc->nvec_used); | ||
200 | desc->irq = 0; | ||
201 | } | ||
202 | } | ||
203 | |||
204 | /** | ||
130 | * msi_get_domain_info - Get the MSI interrupt domain info for @domain | 205 | * msi_get_domain_info - Get the MSI interrupt domain info for @domain |
131 | * @domain: The interrupt domain to retrieve data from | 206 | * @domain: The interrupt domain to retrieve data from |
132 | * | 207 | * |