diff options
Diffstat (limited to 'drivers/ieee1394')
-rw-r--r-- | drivers/ieee1394/nodemgr.c | 147 | ||||
-rw-r--r-- | drivers/ieee1394/nodemgr.h | 2 |
2 files changed, 51 insertions, 98 deletions
diff --git a/drivers/ieee1394/nodemgr.c b/drivers/ieee1394/nodemgr.c index b9d3f46c2b06..2376b729e876 100644 --- a/drivers/ieee1394/nodemgr.c +++ b/drivers/ieee1394/nodemgr.c | |||
@@ -154,9 +154,6 @@ struct host_info { | |||
154 | 154 | ||
155 | static int nodemgr_bus_match(struct device * dev, struct device_driver * drv); | 155 | static int nodemgr_bus_match(struct device * dev, struct device_driver * drv); |
156 | static int nodemgr_uevent(struct device *dev, struct kobj_uevent_env *env); | 156 | static int nodemgr_uevent(struct device *dev, struct kobj_uevent_env *env); |
157 | static void nodemgr_reactivate_ne(struct node_entry *ne); | ||
158 | static void nodemgr_remove_ne(struct node_entry *ne); | ||
159 | static struct node_entry *find_entry_by_guid(u64 guid); | ||
160 | 157 | ||
161 | struct bus_type ieee1394_bus_type = { | 158 | struct bus_type ieee1394_bus_type = { |
162 | .name = "ieee1394", | 159 | .name = "ieee1394", |
@@ -385,27 +382,6 @@ static ssize_t fw_get_ignore_driver(struct device *dev, struct device_attribute | |||
385 | static DEVICE_ATTR(ignore_driver, S_IWUSR | S_IRUGO, fw_get_ignore_driver, fw_set_ignore_driver); | 382 | static DEVICE_ATTR(ignore_driver, S_IWUSR | S_IRUGO, fw_get_ignore_driver, fw_set_ignore_driver); |
386 | 383 | ||
387 | 384 | ||
388 | static ssize_t fw_set_destroy_node(struct bus_type *bus, const char *buf, size_t count) | ||
389 | { | ||
390 | struct node_entry *ne; | ||
391 | u64 guid = (u64)simple_strtoull(buf, NULL, 16); | ||
392 | |||
393 | ne = find_entry_by_guid(guid); | ||
394 | |||
395 | if (ne == NULL || !ne->in_limbo) | ||
396 | return -EINVAL; | ||
397 | |||
398 | nodemgr_remove_ne(ne); | ||
399 | |||
400 | return count; | ||
401 | } | ||
402 | static ssize_t fw_get_destroy_node(struct bus_type *bus, char *buf) | ||
403 | { | ||
404 | return sprintf(buf, "You can destroy in_limbo nodes by writing their GUID to this file\n"); | ||
405 | } | ||
406 | static BUS_ATTR(destroy_node, S_IWUSR | S_IRUGO, fw_get_destroy_node, fw_set_destroy_node); | ||
407 | |||
408 | |||
409 | static ssize_t fw_set_rescan(struct bus_type *bus, const char *buf, | 385 | static ssize_t fw_set_rescan(struct bus_type *bus, const char *buf, |
410 | size_t count) | 386 | size_t count) |
411 | { | 387 | { |
@@ -442,7 +418,6 @@ static BUS_ATTR(ignore_drivers, S_IWUSR | S_IRUGO, fw_get_ignore_drivers, fw_set | |||
442 | 418 | ||
443 | 419 | ||
444 | struct bus_attribute *const fw_bus_attrs[] = { | 420 | struct bus_attribute *const fw_bus_attrs[] = { |
445 | &bus_attr_destroy_node, | ||
446 | &bus_attr_rescan, | 421 | &bus_attr_rescan, |
447 | &bus_attr_ignore_drivers, | 422 | &bus_attr_ignore_drivers, |
448 | NULL | 423 | NULL |
@@ -1300,14 +1275,19 @@ static void nodemgr_update_node(struct node_entry *ne, struct csr1212_csr *csr, | |||
1300 | csr1212_destroy_csr(csr); | 1275 | csr1212_destroy_csr(csr); |
1301 | } | 1276 | } |
1302 | 1277 | ||
1303 | if (ne->in_limbo) | ||
1304 | nodemgr_reactivate_ne(ne); | ||
1305 | |||
1306 | /* Mark the node current */ | 1278 | /* Mark the node current */ |
1307 | ne->generation = generation; | 1279 | ne->generation = generation; |
1308 | } | ||
1309 | 1280 | ||
1281 | if (ne->in_limbo) { | ||
1282 | device_remove_file(&ne->device, &dev_attr_ne_in_limbo); | ||
1283 | ne->in_limbo = false; | ||
1310 | 1284 | ||
1285 | HPSB_DEBUG("Node reactivated: " | ||
1286 | "ID:BUS[" NODE_BUS_FMT "] GUID[%016Lx]", | ||
1287 | NODE_BUS_ARGS(ne->host, ne->nodeid), | ||
1288 | (unsigned long long)ne->guid); | ||
1289 | } | ||
1290 | } | ||
1311 | 1291 | ||
1312 | static void nodemgr_node_scan_one(struct hpsb_host *host, | 1292 | static void nodemgr_node_scan_one(struct hpsb_host *host, |
1313 | nodeid_t nodeid, int generation) | 1293 | nodeid_t nodeid, int generation) |
@@ -1392,75 +1372,14 @@ static void nodemgr_node_scan(struct hpsb_host *host, int generation) | |||
1392 | } | 1372 | } |
1393 | } | 1373 | } |
1394 | 1374 | ||
1395 | static int pause_ne(struct device *dev, void *data) | ||
1396 | { | ||
1397 | struct unit_directory *ud; | ||
1398 | struct device_driver *drv; | ||
1399 | struct node_entry *ne = data; | ||
1400 | int error; | ||
1401 | |||
1402 | ud = container_of(dev, struct unit_directory, unit_dev); | ||
1403 | if (ud->ne == ne) { | ||
1404 | drv = get_driver(ud->device.driver); | ||
1405 | if (drv) { | ||
1406 | error = 1; /* release if suspend is not implemented */ | ||
1407 | if (drv->suspend) { | ||
1408 | down(&ud->device.sem); | ||
1409 | error = drv->suspend(&ud->device, PMSG_SUSPEND); | ||
1410 | up(&ud->device.sem); | ||
1411 | } | ||
1412 | if (error) | ||
1413 | device_release_driver(&ud->device); | ||
1414 | put_driver(drv); | ||
1415 | } | ||
1416 | } | ||
1417 | |||
1418 | return 0; | ||
1419 | } | ||
1420 | |||
1421 | static void nodemgr_pause_ne(struct node_entry *ne) | 1375 | static void nodemgr_pause_ne(struct node_entry *ne) |
1422 | { | 1376 | { |
1423 | HPSB_DEBUG("Node suspended: ID:BUS[" NODE_BUS_FMT "] GUID[%016Lx]", | 1377 | HPSB_DEBUG("Node paused: ID:BUS[" NODE_BUS_FMT "] GUID[%016Lx]", |
1424 | NODE_BUS_ARGS(ne->host, ne->nodeid), | 1378 | NODE_BUS_ARGS(ne->host, ne->nodeid), |
1425 | (unsigned long long)ne->guid); | 1379 | (unsigned long long)ne->guid); |
1426 | 1380 | ||
1427 | ne->in_limbo = 1; | 1381 | ne->in_limbo = true; |
1428 | WARN_ON(device_create_file(&ne->device, &dev_attr_ne_in_limbo)); | 1382 | WARN_ON(device_create_file(&ne->device, &dev_attr_ne_in_limbo)); |
1429 | |||
1430 | class_for_each_device(&nodemgr_ud_class, NULL, ne, pause_ne); | ||
1431 | } | ||
1432 | |||
1433 | static int reactivate_ne(struct device *dev, void *data) | ||
1434 | { | ||
1435 | struct unit_directory *ud; | ||
1436 | struct device_driver *drv; | ||
1437 | struct node_entry *ne = data; | ||
1438 | |||
1439 | ud = container_of(dev, struct unit_directory, unit_dev); | ||
1440 | if (ud->ne == ne) { | ||
1441 | drv = get_driver(ud->device.driver); | ||
1442 | if (drv) { | ||
1443 | if (drv->resume) { | ||
1444 | down(&ud->device.sem); | ||
1445 | drv->resume(&ud->device); | ||
1446 | up(&ud->device.sem); | ||
1447 | } | ||
1448 | put_driver(drv); | ||
1449 | } | ||
1450 | } | ||
1451 | |||
1452 | return 0; | ||
1453 | } | ||
1454 | |||
1455 | static void nodemgr_reactivate_ne(struct node_entry *ne) | ||
1456 | { | ||
1457 | ne->in_limbo = 0; | ||
1458 | device_remove_file(&ne->device, &dev_attr_ne_in_limbo); | ||
1459 | |||
1460 | class_for_each_device(&nodemgr_ud_class, NULL, ne, reactivate_ne); | ||
1461 | HPSB_DEBUG("Node resumed: ID:BUS[" NODE_BUS_FMT "] GUID[%016Lx]", | ||
1462 | NODE_BUS_ARGS(ne->host, ne->nodeid), | ||
1463 | (unsigned long long)ne->guid); | ||
1464 | } | 1383 | } |
1465 | 1384 | ||
1466 | static int update_pdrv(struct device *dev, void *data) | 1385 | static int update_pdrv(struct device *dev, void *data) |
@@ -1497,7 +1416,6 @@ static void nodemgr_update_pdrv(struct node_entry *ne) | |||
1497 | class_for_each_device(&nodemgr_ud_class, NULL, ne, update_pdrv); | 1416 | class_for_each_device(&nodemgr_ud_class, NULL, ne, update_pdrv); |
1498 | } | 1417 | } |
1499 | 1418 | ||
1500 | |||
1501 | /* Write the BROADCAST_CHANNEL as per IEEE1394a 8.3.2.3.11 and 8.4.2.3. This | 1419 | /* Write the BROADCAST_CHANNEL as per IEEE1394a 8.3.2.3.11 and 8.4.2.3. This |
1502 | * seems like an optional service but in the end it is practically mandatory | 1420 | * seems like an optional service but in the end it is practically mandatory |
1503 | * as a consequence of these clauses. | 1421 | * as a consequence of these clauses. |
@@ -1574,7 +1492,7 @@ static int node_probe(struct device *dev, void *data) | |||
1574 | return 0; | 1492 | return 0; |
1575 | } | 1493 | } |
1576 | 1494 | ||
1577 | static void nodemgr_node_probe(struct hpsb_host *host, int generation) | 1495 | static int nodemgr_node_probe(struct hpsb_host *host, int generation) |
1578 | { | 1496 | { |
1579 | struct node_probe_parameter p; | 1497 | struct node_probe_parameter p; |
1580 | 1498 | ||
@@ -1595,11 +1513,11 @@ static void nodemgr_node_probe(struct hpsb_host *host, int generation) | |||
1595 | */ | 1513 | */ |
1596 | p.probe_now = false; | 1514 | p.probe_now = false; |
1597 | if (class_for_each_device(&nodemgr_ne_class, NULL, &p, node_probe) != 0) | 1515 | if (class_for_each_device(&nodemgr_ne_class, NULL, &p, node_probe) != 0) |
1598 | return; | 1516 | return 0; |
1599 | 1517 | ||
1600 | p.probe_now = true; | 1518 | p.probe_now = true; |
1601 | if (class_for_each_device(&nodemgr_ne_class, NULL, &p, node_probe) != 0) | 1519 | if (class_for_each_device(&nodemgr_ne_class, NULL, &p, node_probe) != 0) |
1602 | return; | 1520 | return 0; |
1603 | /* | 1521 | /* |
1604 | * Now let's tell the bus to rescan our devices. This may seem | 1522 | * Now let's tell the bus to rescan our devices. This may seem |
1605 | * like overhead, but the driver-model core will only scan a | 1523 | * like overhead, but the driver-model core will only scan a |
@@ -1611,6 +1529,27 @@ static void nodemgr_node_probe(struct hpsb_host *host, int generation) | |||
1611 | */ | 1529 | */ |
1612 | if (bus_rescan_devices(&ieee1394_bus_type) != 0) | 1530 | if (bus_rescan_devices(&ieee1394_bus_type) != 0) |
1613 | HPSB_DEBUG("bus_rescan_devices had an error"); | 1531 | HPSB_DEBUG("bus_rescan_devices had an error"); |
1532 | |||
1533 | return 1; | ||
1534 | } | ||
1535 | |||
1536 | static int remove_nodes_in_limbo(struct device *dev, void *data) | ||
1537 | { | ||
1538 | struct node_entry *ne; | ||
1539 | |||
1540 | if (dev->bus != &ieee1394_bus_type) | ||
1541 | return 0; | ||
1542 | |||
1543 | ne = container_of(dev, struct node_entry, device); | ||
1544 | if (ne->in_limbo) | ||
1545 | nodemgr_remove_ne(ne); | ||
1546 | |||
1547 | return 0; | ||
1548 | } | ||
1549 | |||
1550 | static void nodemgr_remove_nodes_in_limbo(struct hpsb_host *host) | ||
1551 | { | ||
1552 | device_for_each_child(&host->device, NULL, remove_nodes_in_limbo); | ||
1614 | } | 1553 | } |
1615 | 1554 | ||
1616 | static int nodemgr_send_resume_packet(struct hpsb_host *host) | 1555 | static int nodemgr_send_resume_packet(struct hpsb_host *host) |
@@ -1781,10 +1720,24 @@ static int nodemgr_host_thread(void *data) | |||
1781 | 1720 | ||
1782 | /* This actually does the full probe, with sysfs | 1721 | /* This actually does the full probe, with sysfs |
1783 | * registration. */ | 1722 | * registration. */ |
1784 | nodemgr_node_probe(host, generation); | 1723 | if (!nodemgr_node_probe(host, generation)) |
1724 | continue; | ||
1785 | 1725 | ||
1786 | /* Update some of our sysfs symlinks */ | 1726 | /* Update some of our sysfs symlinks */ |
1787 | nodemgr_update_host_dev_links(host); | 1727 | nodemgr_update_host_dev_links(host); |
1728 | |||
1729 | /* Sleep 3 seconds */ | ||
1730 | for (i = 3000/200; i; i--) { | ||
1731 | msleep_interruptible(200); | ||
1732 | if (kthread_should_stop()) | ||
1733 | goto exit; | ||
1734 | |||
1735 | if (generation != get_hpsb_generation(host)) | ||
1736 | break; | ||
1737 | } | ||
1738 | /* Remove nodes which are gone, unless a bus reset happened */ | ||
1739 | if (!i) | ||
1740 | nodemgr_remove_nodes_in_limbo(host); | ||
1788 | } | 1741 | } |
1789 | exit: | 1742 | exit: |
1790 | HPSB_VERBOSE("NodeMgr: Exiting thread"); | 1743 | HPSB_VERBOSE("NodeMgr: Exiting thread"); |
diff --git a/drivers/ieee1394/nodemgr.h b/drivers/ieee1394/nodemgr.h index 6eb26465a84c..4f287a3561ba 100644 --- a/drivers/ieee1394/nodemgr.h +++ b/drivers/ieee1394/nodemgr.h | |||
@@ -110,7 +110,7 @@ struct node_entry { | |||
110 | struct device node_dev; | 110 | struct device node_dev; |
111 | 111 | ||
112 | /* Means this node is not attached anymore */ | 112 | /* Means this node is not attached anymore */ |
113 | int in_limbo; | 113 | bool in_limbo; |
114 | 114 | ||
115 | struct csr1212_csr *csr; | 115 | struct csr1212_csr *csr; |
116 | }; | 116 | }; |