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 | |
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>
-rw-r--r-- | include/linux/msi.h | 29 | ||||
-rw-r--r-- | kernel/irq/msi.c | 75 |
2 files changed, 104 insertions, 0 deletions
diff --git a/include/linux/msi.h b/include/linux/msi.h index b5ca2456769c..7a93a988dce8 100644 --- a/include/linux/msi.h +++ b/include/linux/msi.h | |||
@@ -115,6 +115,9 @@ struct msi_controller { | |||
115 | }; | 115 | }; |
116 | 116 | ||
117 | #ifdef CONFIG_GENERIC_MSI_IRQ_DOMAIN | 117 | #ifdef CONFIG_GENERIC_MSI_IRQ_DOMAIN |
118 | |||
119 | #include <asm/msi.h> | ||
120 | |||
118 | struct irq_domain; | 121 | struct irq_domain; |
119 | struct irq_chip; | 122 | struct irq_chip; |
120 | struct device_node; | 123 | struct device_node; |
@@ -125,6 +128,18 @@ struct msi_domain_info; | |||
125 | * @get_hwirq: Retrieve the resulting hw irq number | 128 | * @get_hwirq: Retrieve the resulting hw irq number |
126 | * @msi_init: Domain specific init function for MSI interrupts | 129 | * @msi_init: Domain specific init function for MSI interrupts |
127 | * @msi_free: Domain specific function to free a MSI interrupts | 130 | * @msi_free: Domain specific function to free a MSI interrupts |
131 | * @msi_check: Callback for verification of the domain/info/dev data | ||
132 | * @msi_prepare: Prepare the allocation of the interrupts in the domain | ||
133 | * @msi_finish: Optional callbacl to finalize the allocation | ||
134 | * @set_desc: Set the msi descriptor for an interrupt | ||
135 | * @handle_error: Optional error handler if the allocation fails | ||
136 | * | ||
137 | * @get_hwirq, @msi_init and @msi_free are callbacks used by | ||
138 | * msi_create_irq_domain() and related interfaces | ||
139 | * | ||
140 | * @msi_check, @msi_prepare, @msi_finish, @set_desc and @handle_error | ||
141 | * are callbacks used by msi_irq_domain_alloc_irqs() and related | ||
142 | * interfaces which are based on msi_desc. | ||
128 | */ | 143 | */ |
129 | struct msi_domain_ops { | 144 | struct msi_domain_ops { |
130 | irq_hw_number_t (*get_hwirq)(struct msi_domain_info *info, void *arg); | 145 | irq_hw_number_t (*get_hwirq)(struct msi_domain_info *info, void *arg); |
@@ -135,6 +150,17 @@ struct msi_domain_ops { | |||
135 | void (*msi_free)(struct irq_domain *domain, | 150 | void (*msi_free)(struct irq_domain *domain, |
136 | struct msi_domain_info *info, | 151 | struct msi_domain_info *info, |
137 | unsigned int virq); | 152 | unsigned int virq); |
153 | int (*msi_check)(struct irq_domain *domain, | ||
154 | struct msi_domain_info *info, | ||
155 | struct device *dev); | ||
156 | int (*msi_prepare)(struct irq_domain *domain, | ||
157 | struct device *dev, int nvec, | ||
158 | msi_alloc_info_t *arg); | ||
159 | void (*msi_finish)(msi_alloc_info_t *arg, int retval); | ||
160 | void (*set_desc)(msi_alloc_info_t *arg, | ||
161 | struct msi_desc *desc); | ||
162 | int (*handle_error)(struct irq_domain *domain, | ||
163 | struct msi_desc *desc, int error); | ||
138 | }; | 164 | }; |
139 | 165 | ||
140 | /** | 166 | /** |
@@ -155,6 +181,9 @@ int msi_domain_set_affinity(struct irq_data *data, const struct cpumask *mask, | |||
155 | struct irq_domain *msi_create_irq_domain(struct device_node *of_node, | 181 | struct irq_domain *msi_create_irq_domain(struct device_node *of_node, |
156 | struct msi_domain_info *info, | 182 | struct msi_domain_info *info, |
157 | struct irq_domain *parent); | 183 | struct irq_domain *parent); |
184 | int msi_domain_alloc_irqs(struct irq_domain *domain, struct device *dev, | ||
185 | int nvec); | ||
186 | void msi_domain_free_irqs(struct irq_domain *domain, struct device *dev); | ||
158 | struct msi_domain_info *msi_get_domain_info(struct irq_domain *domain); | 187 | struct msi_domain_info *msi_get_domain_info(struct irq_domain *domain); |
159 | 188 | ||
160 | #endif /* CONFIG_GENERIC_MSI_IRQ_DOMAIN */ | 189 | #endif /* CONFIG_GENERIC_MSI_IRQ_DOMAIN */ |
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 | * |