diff options
author | Zhao Yakui <yakui.zhao@intel.com> | 2008-08-11 01:40:22 -0400 |
---|---|---|
committer | Len Brown <len.brown@intel.com> | 2008-10-22 18:00:50 -0400 |
commit | 39a0ad871000d2a016a4fa113a6e53d22aabf25d (patch) | |
tree | e4ff7eabccc74b2d16e80d78c2e255d2d004cd4d /drivers/acpi/scan.c | |
parent | 6415e12ba0f92a54f02d9c4ecaa3c82f35f3d335 (diff) |
ACPI : Load device driver according to the status of acpi device
According to ACPI spec when the status of some device is not present
but functional, the device is valid and the children of this device
should be enumerated. It means that the device should be added to
linux acpi device tree. But the device driver for this device should not
be loaded.
The detailed info can be found in the section 6.3.7 of ACPI 3.0b spec.
_STA may return bit 0 clear (not present) with bit 3 set (device is
functional). This case is used to indicate a valid device for which no
device driver should be loaded (for example, a bridge device.).
Children of this device may be present and valid. OS should continue
enumeration below a device whose _STA returns this bit combination
http://bugzilla.kernel.org/show_bug.cgi?id=3358
Signed-off-by: Zhao Yakui <yakui.zhao@intel.com>
Signed-off-by: Li Shaohua <shaohua.li@intel.com>
Signed-off-by: Zhang Rui <rui.zhang@intel.com>
Signed-off-by: Andi Kleen <ak@linux.intel.com>
Signed-off-by: Len Brown <len.brown@intel.com>
Diffstat (limited to 'drivers/acpi/scan.c')
-rw-r--r-- | drivers/acpi/scan.c | 35 |
1 files changed, 25 insertions, 10 deletions
diff --git a/drivers/acpi/scan.c b/drivers/acpi/scan.c index ad0679843bd5..89c112ef9e14 100644 --- a/drivers/acpi/scan.c +++ b/drivers/acpi/scan.c | |||
@@ -276,6 +276,13 @@ int acpi_match_device_ids(struct acpi_device *device, | |||
276 | { | 276 | { |
277 | const struct acpi_device_id *id; | 277 | const struct acpi_device_id *id; |
278 | 278 | ||
279 | /* | ||
280 | * If the device is not present, it is unnecessary to load device | ||
281 | * driver for it. | ||
282 | */ | ||
283 | if (!device->status.present) | ||
284 | return -ENODEV; | ||
285 | |||
279 | if (device->flags.hardware_id) { | 286 | if (device->flags.hardware_id) { |
280 | for (id = ids; id->id[0]; id++) { | 287 | for (id = ids; id->id[0]; id++) { |
281 | if (!strcmp((char*)id->id, device->pnp.hardware_id)) | 288 | if (!strcmp((char*)id->id, device->pnp.hardware_id)) |
@@ -1222,15 +1229,18 @@ acpi_add_single_object(struct acpi_device **child, | |||
1222 | result = -ENODEV; | 1229 | result = -ENODEV; |
1223 | goto end; | 1230 | goto end; |
1224 | } | 1231 | } |
1225 | if (!device->status.present) { | 1232 | /* |
1226 | /* Bay and dock should be handled even if absent */ | 1233 | * When the device is neither present nor functional, the |
1227 | if (!ACPI_SUCCESS( | 1234 | * device should not be added to Linux ACPI device tree. |
1228 | acpi_is_child_device(device, acpi_bay_match)) && | 1235 | * When the status of the device is not present but functinal, |
1229 | !ACPI_SUCCESS( | 1236 | * it should be added to Linux ACPI tree. For example : bay |
1230 | acpi_is_child_device(device, acpi_dock_match))) { | 1237 | * device , dock device. |
1231 | result = -ENODEV; | 1238 | * In such conditions it is unncessary to check whether it is |
1232 | goto end; | 1239 | * bay device or dock device. |
1233 | } | 1240 | */ |
1241 | if (!device->status.present && !device->status.functional) { | ||
1242 | result = -ENODEV; | ||
1243 | goto end; | ||
1234 | } | 1244 | } |
1235 | break; | 1245 | break; |
1236 | default: | 1246 | default: |
@@ -1411,7 +1421,12 @@ static int acpi_bus_scan(struct acpi_device *start, struct acpi_bus_ops *ops) | |||
1411 | * TBD: Need notifications and other detection mechanisms | 1421 | * TBD: Need notifications and other detection mechanisms |
1412 | * in place before we can fully implement this. | 1422 | * in place before we can fully implement this. |
1413 | */ | 1423 | */ |
1414 | if (child->status.present) { | 1424 | /* |
1425 | * When the device is not present but functional, it is also | ||
1426 | * necessary to scan the children of this device. | ||
1427 | */ | ||
1428 | if (child->status.present || (!child->status.present && | ||
1429 | child->status.functional)) { | ||
1415 | status = acpi_get_next_object(ACPI_TYPE_ANY, chandle, | 1430 | status = acpi_get_next_object(ACPI_TYPE_ANY, chandle, |
1416 | NULL, NULL); | 1431 | NULL, NULL); |
1417 | if (ACPI_SUCCESS(status)) { | 1432 | if (ACPI_SUCCESS(status)) { |