2 * resource.c - Contains functions for registering and analyzing resource information
4 * based on isapnp.c resource management (c) Jaroslav Kysela <perex@perex.cz>
5 * Copyright 2003 Adam Belay <ambx1@neo.rr.com>
6 * Copyright (C) 2008 Hewlett-Packard Development Company, L.P.
7 * Bjorn Helgaas <bjorn.helgaas@hp.com>
10 #include <linux/module.h>
11 #include <linux/slab.h>
12 #include <linux/errno.h>
13 #include <linux/interrupt.h>
14 #include <linux/kernel.h>
18 #include <linux/pci.h>
19 #include <linux/ioport.h>
20 #include <linux/init.h>
22 #include <linux/pnp.h>
25 static int pnp_reserve_irq
[16] = {[0 ... 15] = -1 }; /* reserve (don't use) some IRQ */
26 static int pnp_reserve_dma
[8] = {[0 ... 7] = -1 }; /* reserve (don't use) some DMA */
27 static int pnp_reserve_io
[16] = {[0 ... 15] = -1 }; /* reserve (don't use) some I/O region */
28 static int pnp_reserve_mem
[16] = {[0 ... 15] = -1 }; /* reserve (don't use) some memory region */
34 static struct pnp_option
*pnp_build_option(struct pnp_dev
*dev
, unsigned long type
,
35 unsigned int option_flags
)
37 struct pnp_option
*option
;
39 option
= kzalloc(sizeof(struct pnp_option
), GFP_KERNEL
);
43 option
->flags
= option_flags
;
46 list_add_tail(&option
->list
, &dev
->options
);
50 int pnp_register_irq_resource(struct pnp_dev
*dev
, unsigned int option_flags
,
51 pnp_irq_mask_t
*map
, unsigned char flags
)
53 struct pnp_option
*option
;
56 option
= pnp_build_option(dev
, IORESOURCE_IRQ
, option_flags
);
68 for (i
= 0; i
< 16; i
++)
69 if (test_bit(i
, irq
->map
.bits
))
70 pcibios_penalize_isa_irq(i
, 0);
74 dbg_pnp_show_option(dev
, option
);
78 int pnp_register_dma_resource(struct pnp_dev
*dev
, unsigned int option_flags
,
79 unsigned char map
, unsigned char flags
)
81 struct pnp_option
*option
;
84 option
= pnp_build_option(dev
, IORESOURCE_DMA
, option_flags
);
92 dbg_pnp_show_option(dev
, option
);
96 int pnp_register_port_resource(struct pnp_dev
*dev
, unsigned int option_flags
,
97 resource_size_t min
, resource_size_t max
,
98 resource_size_t align
, resource_size_t size
,
101 struct pnp_option
*option
;
102 struct pnp_port
*port
;
104 option
= pnp_build_option(dev
, IORESOURCE_IO
, option_flags
);
108 port
= &option
->u
.port
;
115 dbg_pnp_show_option(dev
, option
);
119 int pnp_register_mem_resource(struct pnp_dev
*dev
, unsigned int option_flags
,
120 resource_size_t min
, resource_size_t max
,
121 resource_size_t align
, resource_size_t size
,
124 struct pnp_option
*option
;
127 option
= pnp_build_option(dev
, IORESOURCE_MEM
, option_flags
);
131 mem
= &option
->u
.mem
;
138 dbg_pnp_show_option(dev
, option
);
142 void pnp_free_options(struct pnp_dev
*dev
)
144 struct pnp_option
*option
, *tmp
;
146 list_for_each_entry_safe(option
, tmp
, &dev
->options
, list
) {
147 list_del(&option
->list
);
153 * resource validity checking
156 #define length(start, end) (*(end) - *(start) + 1)
158 /* Two ranges conflict if one doesn't end before the other starts */
159 #define ranged_conflict(starta, enda, startb, endb) \
160 !((*(enda) < *(startb)) || (*(endb) < *(starta)))
162 #define cannot_compare(flags) \
163 ((flags) & IORESOURCE_DISABLED)
165 int pnp_check_port(struct pnp_dev
*dev
, struct resource
*res
)
168 struct pnp_dev
*tdev
;
169 struct resource
*tres
;
170 resource_size_t
*port
, *end
, *tport
, *tend
;
175 /* if the resource doesn't exist, don't complain about it */
176 if (cannot_compare(res
->flags
))
179 /* check if the resource is already in use, skip if the
180 * device is active because it itself may be in use */
182 if (!request_region(*port
, length(port
, end
), "pnp"))
184 release_region(*port
, length(port
, end
));
187 /* check if the resource is reserved */
188 for (i
= 0; i
< 8; i
++) {
189 int rport
= pnp_reserve_io
[i
<< 1];
190 int rend
= pnp_reserve_io
[(i
<< 1) + 1] + rport
- 1;
191 if (ranged_conflict(port
, end
, &rport
, &rend
))
195 /* check for internal conflicts */
196 for (i
= 0; (tres
= pnp_get_resource(dev
, IORESOURCE_IO
, i
)); i
++) {
197 if (tres
!= res
&& tres
->flags
& IORESOURCE_IO
) {
198 tport
= &tres
->start
;
200 if (ranged_conflict(port
, end
, tport
, tend
))
205 /* check for conflicts with other pnp devices */
206 pnp_for_each_dev(tdev
) {
210 (tres
= pnp_get_resource(tdev
, IORESOURCE_IO
, i
));
212 if (tres
->flags
& IORESOURCE_IO
) {
213 if (cannot_compare(tres
->flags
))
215 if (tres
->flags
& IORESOURCE_WINDOW
)
217 tport
= &tres
->start
;
219 if (ranged_conflict(port
, end
, tport
, tend
))
228 int pnp_check_mem(struct pnp_dev
*dev
, struct resource
*res
)
231 struct pnp_dev
*tdev
;
232 struct resource
*tres
;
233 resource_size_t
*addr
, *end
, *taddr
, *tend
;
238 /* if the resource doesn't exist, don't complain about it */
239 if (cannot_compare(res
->flags
))
242 /* check if the resource is already in use, skip if the
243 * device is active because it itself may be in use */
245 if (!request_mem_region(*addr
, length(addr
, end
), "pnp"))
247 release_mem_region(*addr
, length(addr
, end
));
250 /* check if the resource is reserved */
251 for (i
= 0; i
< 8; i
++) {
252 int raddr
= pnp_reserve_mem
[i
<< 1];
253 int rend
= pnp_reserve_mem
[(i
<< 1) + 1] + raddr
- 1;
254 if (ranged_conflict(addr
, end
, &raddr
, &rend
))
258 /* check for internal conflicts */
259 for (i
= 0; (tres
= pnp_get_resource(dev
, IORESOURCE_MEM
, i
)); i
++) {
260 if (tres
!= res
&& tres
->flags
& IORESOURCE_MEM
) {
261 taddr
= &tres
->start
;
263 if (ranged_conflict(addr
, end
, taddr
, tend
))
268 /* check for conflicts with other pnp devices */
269 pnp_for_each_dev(tdev
) {
273 (tres
= pnp_get_resource(tdev
, IORESOURCE_MEM
, i
));
275 if (tres
->flags
& IORESOURCE_MEM
) {
276 if (cannot_compare(tres
->flags
))
278 if (tres
->flags
& IORESOURCE_WINDOW
)
280 taddr
= &tres
->start
;
282 if (ranged_conflict(addr
, end
, taddr
, tend
))
291 static irqreturn_t
pnp_test_handler(int irq
, void *dev_id
)
297 static int pci_dev_uses_irq(struct pnp_dev
*pnp
, struct pci_dev
*pci
,
303 if (pci
->irq
== irq
) {
304 pnp_dbg(&pnp
->dev
, " device %s using irq %d\n",
310 * See pci_setup_device() and ata_pci_sff_activate_host() for
311 * similar IDE legacy detection.
313 pci_read_config_dword(pci
, PCI_CLASS_REVISION
, &class);
314 class >>= 8; /* discard revision ID */
315 progif
= class & 0xff;
318 if (class == PCI_CLASS_STORAGE_IDE
) {
320 * Unless both channels are native-PCI mode only,
321 * treat the compatibility IRQs as busy.
323 if ((progif
& 0x5) != 0x5)
324 if (pci_get_legacy_ide_irq(pci
, 0) == irq
||
325 pci_get_legacy_ide_irq(pci
, 1) == irq
) {
326 pnp_dbg(&pnp
->dev
, " legacy IDE device %s "
327 "using irq %d\n", pci_name(pci
), irq
);
336 static int pci_uses_irq(struct pnp_dev
*pnp
, unsigned int irq
)
339 struct pci_dev
*pci
= NULL
;
341 for_each_pci_dev(pci
) {
342 if (pci_dev_uses_irq(pnp
, pci
, irq
)) {
351 int pnp_check_irq(struct pnp_dev
*dev
, struct resource
*res
)
354 struct pnp_dev
*tdev
;
355 struct resource
*tres
;
356 resource_size_t
*irq
;
360 /* if the resource doesn't exist, don't complain about it */
361 if (cannot_compare(res
->flags
))
364 /* check if the resource is valid */
368 /* check if the resource is reserved */
369 for (i
= 0; i
< 16; i
++) {
370 if (pnp_reserve_irq
[i
] == *irq
)
374 /* check for internal conflicts */
375 for (i
= 0; (tres
= pnp_get_resource(dev
, IORESOURCE_IRQ
, i
)); i
++) {
376 if (tres
!= res
&& tres
->flags
& IORESOURCE_IRQ
) {
377 if (tres
->start
== *irq
)
382 /* check if the resource is being used by a pci device */
383 if (pci_uses_irq(dev
, *irq
))
386 /* check if the resource is already in use, skip if the
387 * device is active because it itself may be in use */
389 if (request_irq(*irq
, pnp_test_handler
,
390 IRQF_PROBE_SHARED
, "pnp", NULL
))
392 free_irq(*irq
, NULL
);
395 /* check for conflicts with other pnp devices */
396 pnp_for_each_dev(tdev
) {
400 (tres
= pnp_get_resource(tdev
, IORESOURCE_IRQ
, i
));
402 if (tres
->flags
& IORESOURCE_IRQ
) {
403 if (cannot_compare(tres
->flags
))
405 if (tres
->start
== *irq
)
414 #ifdef CONFIG_ISA_DMA_API
415 int pnp_check_dma(struct pnp_dev
*dev
, struct resource
*res
)
418 struct pnp_dev
*tdev
;
419 struct resource
*tres
;
420 resource_size_t
*dma
;
424 /* if the resource doesn't exist, don't complain about it */
425 if (cannot_compare(res
->flags
))
428 /* check if the resource is valid */
429 if (*dma
== 4 || *dma
> 7)
432 /* check if the resource is reserved */
433 for (i
= 0; i
< 8; i
++) {
434 if (pnp_reserve_dma
[i
] == *dma
)
438 /* check for internal conflicts */
439 for (i
= 0; (tres
= pnp_get_resource(dev
, IORESOURCE_DMA
, i
)); i
++) {
440 if (tres
!= res
&& tres
->flags
& IORESOURCE_DMA
) {
441 if (tres
->start
== *dma
)
446 /* check if the resource is already in use, skip if the
447 * device is active because it itself may be in use */
449 if (request_dma(*dma
, "pnp"))
454 /* check for conflicts with other pnp devices */
455 pnp_for_each_dev(tdev
) {
459 (tres
= pnp_get_resource(tdev
, IORESOURCE_DMA
, i
));
461 if (tres
->flags
& IORESOURCE_DMA
) {
462 if (cannot_compare(tres
->flags
))
464 if (tres
->start
== *dma
)
472 #endif /* CONFIG_ISA_DMA_API */
474 unsigned long pnp_resource_type(struct resource
*res
)
476 return res
->flags
& (IORESOURCE_IO
| IORESOURCE_MEM
|
477 IORESOURCE_IRQ
| IORESOURCE_DMA
|
481 struct resource
*pnp_get_resource(struct pnp_dev
*dev
,
482 unsigned long type
, unsigned int num
)
484 struct pnp_resource
*pnp_res
;
485 struct resource
*res
;
487 list_for_each_entry(pnp_res
, &dev
->resources
, list
) {
489 if (pnp_resource_type(res
) == type
&& num
-- == 0)
494 EXPORT_SYMBOL(pnp_get_resource
);
496 static struct pnp_resource
*pnp_new_resource(struct pnp_dev
*dev
)
498 struct pnp_resource
*pnp_res
;
500 pnp_res
= kzalloc(sizeof(struct pnp_resource
), GFP_KERNEL
);
504 list_add_tail(&pnp_res
->list
, &dev
->resources
);
508 struct pnp_resource
*pnp_add_resource(struct pnp_dev
*dev
,
509 struct resource
*res
)
511 struct pnp_resource
*pnp_res
;
513 pnp_res
= pnp_new_resource(dev
);
515 dev_err(&dev
->dev
, "can't add resource %pR\n", res
);
520 pnp_res
->res
.name
= dev
->name
;
521 dev_dbg(&dev
->dev
, "%pR\n", res
);
525 struct pnp_resource
*pnp_add_irq_resource(struct pnp_dev
*dev
, int irq
,
528 struct pnp_resource
*pnp_res
;
529 struct resource
*res
;
531 pnp_res
= pnp_new_resource(dev
);
533 dev_err(&dev
->dev
, "can't add resource for IRQ %d\n", irq
);
538 res
->flags
= IORESOURCE_IRQ
| flags
;
542 dev_printk(KERN_DEBUG
, &dev
->dev
, "%pR\n", res
);
546 struct pnp_resource
*pnp_add_dma_resource(struct pnp_dev
*dev
, int dma
,
549 struct pnp_resource
*pnp_res
;
550 struct resource
*res
;
552 pnp_res
= pnp_new_resource(dev
);
554 dev_err(&dev
->dev
, "can't add resource for DMA %d\n", dma
);
559 res
->flags
= IORESOURCE_DMA
| flags
;
563 dev_printk(KERN_DEBUG
, &dev
->dev
, "%pR\n", res
);
567 struct pnp_resource
*pnp_add_io_resource(struct pnp_dev
*dev
,
568 resource_size_t start
,
569 resource_size_t end
, int flags
)
571 struct pnp_resource
*pnp_res
;
572 struct resource
*res
;
574 pnp_res
= pnp_new_resource(dev
);
576 dev_err(&dev
->dev
, "can't add resource for IO %#llx-%#llx\n",
577 (unsigned long long) start
,
578 (unsigned long long) end
);
583 res
->flags
= IORESOURCE_IO
| flags
;
587 dev_printk(KERN_DEBUG
, &dev
->dev
, "%pR\n", res
);
591 struct pnp_resource
*pnp_add_mem_resource(struct pnp_dev
*dev
,
592 resource_size_t start
,
593 resource_size_t end
, int flags
)
595 struct pnp_resource
*pnp_res
;
596 struct resource
*res
;
598 pnp_res
= pnp_new_resource(dev
);
600 dev_err(&dev
->dev
, "can't add resource for MEM %#llx-%#llx\n",
601 (unsigned long long) start
,
602 (unsigned long long) end
);
607 res
->flags
= IORESOURCE_MEM
| flags
;
611 dev_printk(KERN_DEBUG
, &dev
->dev
, "%pR\n", res
);
615 struct pnp_resource
*pnp_add_bus_resource(struct pnp_dev
*dev
,
616 resource_size_t start
,
619 struct pnp_resource
*pnp_res
;
620 struct resource
*res
;
622 pnp_res
= pnp_new_resource(dev
);
624 dev_err(&dev
->dev
, "can't add resource for BUS %#llx-%#llx\n",
625 (unsigned long long) start
,
626 (unsigned long long) end
);
631 res
->flags
= IORESOURCE_BUS
;
635 dev_printk(KERN_DEBUG
, &dev
->dev
, "%pR\n", res
);
640 * Determine whether the specified resource is a possible configuration
643 int pnp_possible_config(struct pnp_dev
*dev
, int type
, resource_size_t start
,
644 resource_size_t size
)
646 struct pnp_option
*option
;
647 struct pnp_port
*port
;
652 list_for_each_entry(option
, &dev
->options
, list
) {
653 if (option
->type
!= type
)
656 switch (option
->type
) {
658 port
= &option
->u
.port
;
659 if (port
->min
== start
&& port
->size
== size
)
663 mem
= &option
->u
.mem
;
664 if (mem
->min
== start
&& mem
->size
== size
)
668 irq
= &option
->u
.irq
;
669 if (start
< PNP_IRQ_NR
&&
670 test_bit(start
, irq
->map
.bits
))
674 dma
= &option
->u
.dma
;
675 if (dma
->map
& (1 << start
))
683 EXPORT_SYMBOL(pnp_possible_config
);
685 int pnp_range_reserved(resource_size_t start
, resource_size_t end
)
688 struct pnp_resource
*pnp_res
;
689 resource_size_t
*dev_start
, *dev_end
;
691 pnp_for_each_dev(dev
) {
692 list_for_each_entry(pnp_res
, &dev
->resources
, list
) {
693 dev_start
= &pnp_res
->res
.start
;
694 dev_end
= &pnp_res
->res
.end
;
695 if (ranged_conflict(&start
, &end
, dev_start
, dev_end
))
701 EXPORT_SYMBOL(pnp_range_reserved
);
703 /* format is: pnp_reserve_irq=irq1[,irq2] .... */
704 static int __init
pnp_setup_reserve_irq(char *str
)
708 for (i
= 0; i
< 16; i
++)
709 if (get_option(&str
, &pnp_reserve_irq
[i
]) != 2)
714 __setup("pnp_reserve_irq=", pnp_setup_reserve_irq
);
716 /* format is: pnp_reserve_dma=dma1[,dma2] .... */
717 static int __init
pnp_setup_reserve_dma(char *str
)
721 for (i
= 0; i
< 8; i
++)
722 if (get_option(&str
, &pnp_reserve_dma
[i
]) != 2)
727 __setup("pnp_reserve_dma=", pnp_setup_reserve_dma
);
729 /* format is: pnp_reserve_io=io1,size1[,io2,size2] .... */
730 static int __init
pnp_setup_reserve_io(char *str
)
734 for (i
= 0; i
< 16; i
++)
735 if (get_option(&str
, &pnp_reserve_io
[i
]) != 2)
740 __setup("pnp_reserve_io=", pnp_setup_reserve_io
);
742 /* format is: pnp_reserve_mem=mem1,size1[,mem2,size2] .... */
743 static int __init
pnp_setup_reserve_mem(char *str
)
747 for (i
= 0; i
< 16; i
++)
748 if (get_option(&str
, &pnp_reserve_mem
[i
]) != 2)
753 __setup("pnp_reserve_mem=", pnp_setup_reserve_mem
);