diff options
Diffstat (limited to 'arch/x86/kernel/io_apic.c')
-rw-r--r-- | arch/x86/kernel/io_apic.c | 130 |
1 files changed, 4 insertions, 126 deletions
diff --git a/arch/x86/kernel/io_apic.c b/arch/x86/kernel/io_apic.c index f959acbc0db2..683610517d2a 100644 --- a/arch/x86/kernel/io_apic.c +++ b/arch/x86/kernel/io_apic.c | |||
@@ -111,9 +111,6 @@ struct irq_cfg; | |||
111 | struct irq_pin_list; | 111 | struct irq_pin_list; |
112 | struct irq_cfg { | 112 | struct irq_cfg { |
113 | unsigned int irq; | 113 | unsigned int irq; |
114 | #ifdef CONFIG_HAVE_SPARSE_IRQ | ||
115 | struct irq_cfg *next; | ||
116 | #endif | ||
117 | struct irq_pin_list *irq_2_pin; | 114 | struct irq_pin_list *irq_2_pin; |
118 | cpumask_t domain; | 115 | cpumask_t domain; |
119 | cpumask_t old_domain; | 116 | cpumask_t old_domain; |
@@ -151,15 +148,6 @@ static void init_one_irq_cfg(struct irq_cfg *cfg) | |||
151 | 148 | ||
152 | static struct irq_cfg *irq_cfgx; | 149 | static struct irq_cfg *irq_cfgx; |
153 | 150 | ||
154 | #ifdef CONFIG_HAVE_SPARSE_IRQ | ||
155 | /* | ||
156 | * Protect the irq_cfgx_free freelist: | ||
157 | */ | ||
158 | static DEFINE_SPINLOCK(irq_cfg_lock); | ||
159 | |||
160 | static struct irq_cfg *irq_cfgx_free; | ||
161 | #endif | ||
162 | |||
163 | static void __init init_work(void *data) | 151 | static void __init init_work(void *data) |
164 | { | 152 | { |
165 | struct dyn_array *da = data; | 153 | struct dyn_array *da = data; |
@@ -174,114 +162,7 @@ static void __init init_work(void *data) | |||
174 | legacy_count = ARRAY_SIZE(irq_cfg_legacy); | 162 | legacy_count = ARRAY_SIZE(irq_cfg_legacy); |
175 | for (i = legacy_count; i < *da->nr; i++) | 163 | for (i = legacy_count; i < *da->nr; i++) |
176 | init_one_irq_cfg(&cfg[i]); | 164 | init_one_irq_cfg(&cfg[i]); |
177 | |||
178 | #ifdef CONFIG_HAVE_SPARSE_IRQ | ||
179 | for (i = 1; i < *da->nr; i++) | ||
180 | cfg[i-1].next = &cfg[i]; | ||
181 | |||
182 | irq_cfgx_free = &irq_cfgx[legacy_count]; | ||
183 | irq_cfgx[legacy_count - 1].next = NULL; | ||
184 | #endif | ||
185 | } | ||
186 | |||
187 | #ifdef CONFIG_HAVE_SPARSE_IRQ | ||
188 | /* need to be biger than size of irq_cfg_legacy */ | ||
189 | static int nr_irq_cfg = 32; | ||
190 | |||
191 | static int __init parse_nr_irq_cfg(char *arg) | ||
192 | { | ||
193 | if (arg) { | ||
194 | nr_irq_cfg = simple_strtoul(arg, NULL, 0); | ||
195 | if (nr_irq_cfg < 32) | ||
196 | nr_irq_cfg = 32; | ||
197 | } | ||
198 | return 0; | ||
199 | } | ||
200 | |||
201 | early_param("nr_irq_cfg", parse_nr_irq_cfg); | ||
202 | |||
203 | #define for_each_irq_cfg(irqX, cfg) \ | ||
204 | for (cfg = irq_cfgx, irqX = cfg->irq; cfg; cfg = cfg->next, irqX = cfg ? cfg->irq : -1U) | ||
205 | |||
206 | |||
207 | DEFINE_DYN_ARRAY(irq_cfgx, sizeof(struct irq_cfg), nr_irq_cfg, PAGE_SIZE, init_work); | ||
208 | |||
209 | static struct irq_cfg *irq_cfg(unsigned int irq) | ||
210 | { | ||
211 | struct irq_cfg *cfg; | ||
212 | |||
213 | cfg = irq_cfgx; | ||
214 | while (cfg) { | ||
215 | if (cfg->irq == irq) | ||
216 | return cfg; | ||
217 | |||
218 | cfg = cfg->next; | ||
219 | } | ||
220 | |||
221 | return NULL; | ||
222 | } | ||
223 | |||
224 | static struct irq_cfg *irq_cfg_alloc(unsigned int irq) | ||
225 | { | ||
226 | struct irq_cfg *cfg, *cfg_pri; | ||
227 | unsigned long flags; | ||
228 | int count = 0; | ||
229 | int i; | ||
230 | |||
231 | cfg_pri = cfg = irq_cfgx; | ||
232 | while (cfg) { | ||
233 | if (cfg->irq == irq) | ||
234 | return cfg; | ||
235 | |||
236 | cfg_pri = cfg; | ||
237 | cfg = cfg->next; | ||
238 | count++; | ||
239 | } | ||
240 | |||
241 | spin_lock_irqsave(&irq_cfg_lock, flags); | ||
242 | if (!irq_cfgx_free) { | ||
243 | unsigned long phys; | ||
244 | unsigned long total_bytes; | ||
245 | /* | ||
246 | * we run out of pre-allocate ones, allocate more | ||
247 | */ | ||
248 | printk(KERN_DEBUG "try to get more irq_cfg %d\n", nr_irq_cfg); | ||
249 | |||
250 | total_bytes = sizeof(struct irq_cfg) * nr_irq_cfg; | ||
251 | if (after_bootmem) | ||
252 | cfg = kzalloc(total_bytes, GFP_ATOMIC); | ||
253 | else | ||
254 | cfg = __alloc_bootmem_nopanic(total_bytes, PAGE_SIZE, 0); | ||
255 | |||
256 | if (!cfg) | ||
257 | panic("please boot with nr_irq_cfg= %d\n", count * 2); | ||
258 | |||
259 | phys = __pa(cfg); | ||
260 | printk(KERN_DEBUG "irq_cfg ==> [%#lx - %#lx]\n", phys, phys + total_bytes); | ||
261 | |||
262 | for (i = 0; i < nr_irq_cfg; i++) | ||
263 | init_one_irq_cfg(&cfg[i]); | ||
264 | |||
265 | for (i = 1; i < nr_irq_cfg; i++) | ||
266 | cfg[i-1].next = &cfg[i]; | ||
267 | |||
268 | irq_cfgx_free = cfg; | ||
269 | } | ||
270 | |||
271 | cfg = irq_cfgx_free; | ||
272 | irq_cfgx_free = irq_cfgx_free->next; | ||
273 | cfg->next = NULL; | ||
274 | if (cfg_pri) | ||
275 | cfg_pri->next = cfg; | ||
276 | else | ||
277 | irq_cfgx = cfg; | ||
278 | cfg->irq = irq; | ||
279 | |||
280 | spin_unlock_irqrestore(&irq_cfg_lock, flags); | ||
281 | |||
282 | return cfg; | ||
283 | } | 165 | } |
284 | #else | ||
285 | 166 | ||
286 | #define for_each_irq_cfg(irq, cfg) \ | 167 | #define for_each_irq_cfg(irq, cfg) \ |
287 | for (irq = 0, cfg = &irq_cfgx[irq]; irq < nr_irqs; irq++, cfg = &irq_cfgx[irq]) | 168 | for (irq = 0, cfg = &irq_cfgx[irq]; irq < nr_irqs; irq++, cfg = &irq_cfgx[irq]) |
@@ -290,17 +171,16 @@ DEFINE_DYN_ARRAY(irq_cfgx, sizeof(struct irq_cfg), nr_irqs, PAGE_SIZE, init_work | |||
290 | 171 | ||
291 | struct irq_cfg *irq_cfg(unsigned int irq) | 172 | struct irq_cfg *irq_cfg(unsigned int irq) |
292 | { | 173 | { |
293 | if (irq < nr_irqs) | 174 | if (irq < nr_irqs) |
294 | return &irq_cfgx[irq]; | 175 | return &irq_cfgx[irq]; |
295 | 176 | ||
296 | return NULL; | 177 | return NULL; |
297 | } | 178 | } |
298 | struct irq_cfg *irq_cfg_alloc(unsigned int irq) | 179 | struct irq_cfg *irq_cfg_alloc(unsigned int irq) |
299 | { | 180 | { |
300 | return irq_cfg(irq); | 181 | return irq_cfg(irq); |
301 | } | 182 | } |
302 | 183 | ||
303 | #endif | ||
304 | /* | 184 | /* |
305 | * This is performance-critical, we want to do it O(1) | 185 | * This is performance-critical, we want to do it O(1) |
306 | * | 186 | * |
@@ -3068,9 +2948,7 @@ unsigned int create_irq_nr(unsigned int irq_want) | |||
3068 | unsigned long flags; | 2948 | unsigned long flags; |
3069 | struct irq_cfg *cfg_new; | 2949 | struct irq_cfg *cfg_new; |
3070 | 2950 | ||
3071 | #ifndef CONFIG_HAVE_SPARSE_IRQ | ||
3072 | irq_want = nr_irqs - 1; | 2951 | irq_want = nr_irqs - 1; |
3073 | #endif | ||
3074 | 2952 | ||
3075 | irq = 0; | 2953 | irq = 0; |
3076 | spin_lock_irqsave(&vector_lock, flags); | 2954 | spin_lock_irqsave(&vector_lock, flags); |