diff options
Diffstat (limited to 'drivers/pci/pci.c')
-rw-r--r-- | drivers/pci/pci.c | 107 |
1 files changed, 95 insertions, 12 deletions
diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c index 2cfa41e367a7..47663dc0daf7 100644 --- a/drivers/pci/pci.c +++ b/drivers/pci/pci.c | |||
@@ -1395,7 +1395,8 @@ void pci_release_region(struct pci_dev *pdev, int bar) | |||
1395 | * Returns 0 on success, or %EBUSY on error. A warning | 1395 | * Returns 0 on success, or %EBUSY on error. A warning |
1396 | * message is also printed on failure. | 1396 | * message is also printed on failure. |
1397 | */ | 1397 | */ |
1398 | int pci_request_region(struct pci_dev *pdev, int bar, const char *res_name) | 1398 | static int __pci_request_region(struct pci_dev *pdev, int bar, const char *res_name, |
1399 | int exclusive) | ||
1399 | { | 1400 | { |
1400 | struct pci_devres *dr; | 1401 | struct pci_devres *dr; |
1401 | 1402 | ||
@@ -1408,8 +1409,9 @@ int pci_request_region(struct pci_dev *pdev, int bar, const char *res_name) | |||
1408 | goto err_out; | 1409 | goto err_out; |
1409 | } | 1410 | } |
1410 | else if (pci_resource_flags(pdev, bar) & IORESOURCE_MEM) { | 1411 | else if (pci_resource_flags(pdev, bar) & IORESOURCE_MEM) { |
1411 | if (!request_mem_region(pci_resource_start(pdev, bar), | 1412 | if (!__request_mem_region(pci_resource_start(pdev, bar), |
1412 | pci_resource_len(pdev, bar), res_name)) | 1413 | pci_resource_len(pdev, bar), res_name, |
1414 | exclusive)) | ||
1413 | goto err_out; | 1415 | goto err_out; |
1414 | } | 1416 | } |
1415 | 1417 | ||
@@ -1428,6 +1430,47 @@ err_out: | |||
1428 | } | 1430 | } |
1429 | 1431 | ||
1430 | /** | 1432 | /** |
1433 | * pci_request_region - Reserved PCI I/O and memory resource | ||
1434 | * @pdev: PCI device whose resources are to be reserved | ||
1435 | * @bar: BAR to be reserved | ||
1436 | * @res_name: Name to be associated with resource. | ||
1437 | * | ||
1438 | * Mark the PCI region associated with PCI device @pdev BR @bar as | ||
1439 | * being reserved by owner @res_name. Do not access any | ||
1440 | * address inside the PCI regions unless this call returns | ||
1441 | * successfully. | ||
1442 | * | ||
1443 | * Returns 0 on success, or %EBUSY on error. A warning | ||
1444 | * message is also printed on failure. | ||
1445 | */ | ||
1446 | int pci_request_region(struct pci_dev *pdev, int bar, const char *res_name) | ||
1447 | { | ||
1448 | return __pci_request_region(pdev, bar, res_name, 0); | ||
1449 | } | ||
1450 | |||
1451 | /** | ||
1452 | * pci_request_region_exclusive - Reserved PCI I/O and memory resource | ||
1453 | * @pdev: PCI device whose resources are to be reserved | ||
1454 | * @bar: BAR to be reserved | ||
1455 | * @res_name: Name to be associated with resource. | ||
1456 | * | ||
1457 | * Mark the PCI region associated with PCI device @pdev BR @bar as | ||
1458 | * being reserved by owner @res_name. Do not access any | ||
1459 | * address inside the PCI regions unless this call returns | ||
1460 | * successfully. | ||
1461 | * | ||
1462 | * Returns 0 on success, or %EBUSY on error. A warning | ||
1463 | * message is also printed on failure. | ||
1464 | * | ||
1465 | * The key difference that _exclusive makes it that userspace is | ||
1466 | * explicitly not allowed to map the resource via /dev/mem or | ||
1467 | * sysfs. | ||
1468 | */ | ||
1469 | int pci_request_region_exclusive(struct pci_dev *pdev, int bar, const char *res_name) | ||
1470 | { | ||
1471 | return __pci_request_region(pdev, bar, res_name, IORESOURCE_EXCLUSIVE); | ||
1472 | } | ||
1473 | /** | ||
1431 | * pci_release_selected_regions - Release selected PCI I/O and memory resources | 1474 | * pci_release_selected_regions - Release selected PCI I/O and memory resources |
1432 | * @pdev: PCI device whose resources were previously reserved | 1475 | * @pdev: PCI device whose resources were previously reserved |
1433 | * @bars: Bitmask of BARs to be released | 1476 | * @bars: Bitmask of BARs to be released |
@@ -1444,20 +1487,14 @@ void pci_release_selected_regions(struct pci_dev *pdev, int bars) | |||
1444 | pci_release_region(pdev, i); | 1487 | pci_release_region(pdev, i); |
1445 | } | 1488 | } |
1446 | 1489 | ||
1447 | /** | 1490 | int __pci_request_selected_regions(struct pci_dev *pdev, int bars, |
1448 | * pci_request_selected_regions - Reserve selected PCI I/O and memory resources | 1491 | const char *res_name, int excl) |
1449 | * @pdev: PCI device whose resources are to be reserved | ||
1450 | * @bars: Bitmask of BARs to be requested | ||
1451 | * @res_name: Name to be associated with resource | ||
1452 | */ | ||
1453 | int pci_request_selected_regions(struct pci_dev *pdev, int bars, | ||
1454 | const char *res_name) | ||
1455 | { | 1492 | { |
1456 | int i; | 1493 | int i; |
1457 | 1494 | ||
1458 | for (i = 0; i < 6; i++) | 1495 | for (i = 0; i < 6; i++) |
1459 | if (bars & (1 << i)) | 1496 | if (bars & (1 << i)) |
1460 | if(pci_request_region(pdev, i, res_name)) | 1497 | if (__pci_request_region(pdev, i, res_name, excl)) |
1461 | goto err_out; | 1498 | goto err_out; |
1462 | return 0; | 1499 | return 0; |
1463 | 1500 | ||
@@ -1469,6 +1506,26 @@ err_out: | |||
1469 | return -EBUSY; | 1506 | return -EBUSY; |
1470 | } | 1507 | } |
1471 | 1508 | ||
1509 | |||
1510 | /** | ||
1511 | * pci_request_selected_regions - Reserve selected PCI I/O and memory resources | ||
1512 | * @pdev: PCI device whose resources are to be reserved | ||
1513 | * @bars: Bitmask of BARs to be requested | ||
1514 | * @res_name: Name to be associated with resource | ||
1515 | */ | ||
1516 | int pci_request_selected_regions(struct pci_dev *pdev, int bars, | ||
1517 | const char *res_name) | ||
1518 | { | ||
1519 | return __pci_request_selected_regions(pdev, bars, res_name, 0); | ||
1520 | } | ||
1521 | |||
1522 | int pci_request_selected_regions_exclusive(struct pci_dev *pdev, | ||
1523 | int bars, const char *res_name) | ||
1524 | { | ||
1525 | return __pci_request_selected_regions(pdev, bars, res_name, | ||
1526 | IORESOURCE_EXCLUSIVE); | ||
1527 | } | ||
1528 | |||
1472 | /** | 1529 | /** |
1473 | * pci_release_regions - Release reserved PCI I/O and memory resources | 1530 | * pci_release_regions - Release reserved PCI I/O and memory resources |
1474 | * @pdev: PCI device whose resources were previously reserved by pci_request_regions | 1531 | * @pdev: PCI device whose resources were previously reserved by pci_request_regions |
@@ -1502,6 +1559,29 @@ int pci_request_regions(struct pci_dev *pdev, const char *res_name) | |||
1502 | } | 1559 | } |
1503 | 1560 | ||
1504 | /** | 1561 | /** |
1562 | * pci_request_regions_exclusive - Reserved PCI I/O and memory resources | ||
1563 | * @pdev: PCI device whose resources are to be reserved | ||
1564 | * @res_name: Name to be associated with resource. | ||
1565 | * | ||
1566 | * Mark all PCI regions associated with PCI device @pdev as | ||
1567 | * being reserved by owner @res_name. Do not access any | ||
1568 | * address inside the PCI regions unless this call returns | ||
1569 | * successfully. | ||
1570 | * | ||
1571 | * pci_request_regions_exclusive() will mark the region so that | ||
1572 | * /dev/mem and the sysfs MMIO access will not be allowed. | ||
1573 | * | ||
1574 | * Returns 0 on success, or %EBUSY on error. A warning | ||
1575 | * message is also printed on failure. | ||
1576 | */ | ||
1577 | int pci_request_regions_exclusive(struct pci_dev *pdev, const char *res_name) | ||
1578 | { | ||
1579 | return pci_request_selected_regions_exclusive(pdev, | ||
1580 | ((1 << 6) - 1), res_name); | ||
1581 | } | ||
1582 | |||
1583 | |||
1584 | /** | ||
1505 | * pci_set_master - enables bus-mastering for device dev | 1585 | * pci_set_master - enables bus-mastering for device dev |
1506 | * @dev: the PCI device to enable | 1586 | * @dev: the PCI device to enable |
1507 | * | 1587 | * |
@@ -2149,10 +2229,13 @@ EXPORT_SYMBOL(pci_find_capability); | |||
2149 | EXPORT_SYMBOL(pci_bus_find_capability); | 2229 | EXPORT_SYMBOL(pci_bus_find_capability); |
2150 | EXPORT_SYMBOL(pci_release_regions); | 2230 | EXPORT_SYMBOL(pci_release_regions); |
2151 | EXPORT_SYMBOL(pci_request_regions); | 2231 | EXPORT_SYMBOL(pci_request_regions); |
2232 | EXPORT_SYMBOL(pci_request_regions_exclusive); | ||
2152 | EXPORT_SYMBOL(pci_release_region); | 2233 | EXPORT_SYMBOL(pci_release_region); |
2153 | EXPORT_SYMBOL(pci_request_region); | 2234 | EXPORT_SYMBOL(pci_request_region); |
2235 | EXPORT_SYMBOL(pci_request_region_exclusive); | ||
2154 | EXPORT_SYMBOL(pci_release_selected_regions); | 2236 | EXPORT_SYMBOL(pci_release_selected_regions); |
2155 | EXPORT_SYMBOL(pci_request_selected_regions); | 2237 | EXPORT_SYMBOL(pci_request_selected_regions); |
2238 | EXPORT_SYMBOL(pci_request_selected_regions_exclusive); | ||
2156 | EXPORT_SYMBOL(pci_set_master); | 2239 | EXPORT_SYMBOL(pci_set_master); |
2157 | EXPORT_SYMBOL(pci_set_mwi); | 2240 | EXPORT_SYMBOL(pci_set_mwi); |
2158 | EXPORT_SYMBOL(pci_try_set_mwi); | 2241 | EXPORT_SYMBOL(pci_try_set_mwi); |