diff options
author | Lin Ming <ming.m.lin@intel.com> | 2012-06-25 04:13:06 -0400 |
---|---|---|
committer | Jeff Garzik <jgarzik@redhat.com> | 2012-06-29 11:38:13 -0400 |
commit | 3bd46600a7a7e938c54df8cdbac9910668c7dfb0 (patch) | |
tree | 72272f24842f7487b918a47c6e7244e92cad9ae4 /drivers/ata | |
parent | febe53ba6b781862c12686c7ea1972bdedee457a (diff) |
libata-acpi: add ata port runtime D3Cold support
ATA port may support runtime D3Cold state, for example, Zero-power ODD case.
This patch adds wakeup notifier and enable/disable run_wake during
supend/resume.
Signed-off-by: Lin Ming <ming.m.lin@intel.com>
Signed-off-by: Jeff Garzik <jgarzik@redhat.com>
Diffstat (limited to 'drivers/ata')
-rw-r--r-- | drivers/ata/libata-acpi.c | 78 | ||||
-rw-r--r-- | drivers/ata/libata-scsi.c | 3 | ||||
-rw-r--r-- | drivers/ata/libata.h | 4 |
3 files changed, 81 insertions, 4 deletions
diff --git a/drivers/ata/libata-acpi.c b/drivers/ata/libata-acpi.c index bb20fd597eb1..e01807a1ef31 100644 --- a/drivers/ata/libata-acpi.c +++ b/drivers/ata/libata-acpi.c | |||
@@ -16,6 +16,7 @@ | |||
16 | #include <linux/libata.h> | 16 | #include <linux/libata.h> |
17 | #include <linux/pci.h> | 17 | #include <linux/pci.h> |
18 | #include <linux/slab.h> | 18 | #include <linux/slab.h> |
19 | #include <linux/pm_runtime.h> | ||
19 | #include <scsi/scsi_device.h> | 20 | #include <scsi/scsi_device.h> |
20 | #include "libata.h" | 21 | #include "libata.h" |
21 | 22 | ||
@@ -853,6 +854,7 @@ void ata_acpi_set_state(struct ata_port *ap, pm_message_t state) | |||
853 | { | 854 | { |
854 | struct ata_device *dev; | 855 | struct ata_device *dev; |
855 | acpi_handle handle; | 856 | acpi_handle handle; |
857 | int acpi_state; | ||
856 | 858 | ||
857 | /* channel first and then drives for power on and vica versa | 859 | /* channel first and then drives for power on and vica versa |
858 | for power off */ | 860 | for power off */ |
@@ -862,10 +864,23 @@ void ata_acpi_set_state(struct ata_port *ap, pm_message_t state) | |||
862 | 864 | ||
863 | ata_for_each_dev(dev, &ap->link, ENABLED) { | 865 | ata_for_each_dev(dev, &ap->link, ENABLED) { |
864 | handle = ata_dev_acpi_handle(dev); | 866 | handle = ata_dev_acpi_handle(dev); |
865 | if (handle) | 867 | if (!handle) |
866 | acpi_bus_set_power(handle, | 868 | continue; |
867 | state.event == PM_EVENT_ON ? | 869 | |
868 | ACPI_STATE_D0 : ACPI_STATE_D3); | 870 | if (state.event != PM_EVENT_ON) { |
871 | acpi_state = acpi_pm_device_sleep_state( | ||
872 | &dev->sdev->sdev_gendev, NULL); | ||
873 | if (acpi_state > 0) | ||
874 | acpi_bus_set_power(handle, acpi_state); | ||
875 | /* TBD: need to check if it's runtime pm request */ | ||
876 | acpi_pm_device_run_wake( | ||
877 | &dev->sdev->sdev_gendev, true); | ||
878 | } else { | ||
879 | /* Ditto */ | ||
880 | acpi_pm_device_run_wake( | ||
881 | &dev->sdev->sdev_gendev, false); | ||
882 | acpi_bus_set_power(handle, ACPI_STATE_D0); | ||
883 | } | ||
869 | } | 884 | } |
870 | 885 | ||
871 | handle = ata_ap_acpi_handle(ap); | 886 | handle = ata_ap_acpi_handle(ap); |
@@ -965,6 +980,61 @@ void ata_acpi_on_disable(struct ata_device *dev) | |||
965 | ata_acpi_clear_gtf(dev); | 980 | ata_acpi_clear_gtf(dev); |
966 | } | 981 | } |
967 | 982 | ||
983 | static void ata_acpi_wake_dev(acpi_handle handle, u32 event, void *context) | ||
984 | { | ||
985 | struct ata_device *ata_dev = context; | ||
986 | |||
987 | if (event == ACPI_NOTIFY_DEVICE_WAKE && ata_dev && | ||
988 | pm_runtime_suspended(&ata_dev->sdev->sdev_gendev)) | ||
989 | scsi_autopm_get_device(ata_dev->sdev); | ||
990 | } | ||
991 | |||
992 | static void ata_acpi_add_pm_notifier(struct ata_device *dev) | ||
993 | { | ||
994 | struct acpi_device *acpi_dev; | ||
995 | acpi_handle handle; | ||
996 | acpi_status status; | ||
997 | |||
998 | handle = ata_dev_acpi_handle(dev); | ||
999 | if (!handle) | ||
1000 | return; | ||
1001 | |||
1002 | status = acpi_bus_get_device(handle, &acpi_dev); | ||
1003 | if (ACPI_SUCCESS(status)) { | ||
1004 | acpi_install_notify_handler(handle, ACPI_SYSTEM_NOTIFY, | ||
1005 | ata_acpi_wake_dev, dev); | ||
1006 | device_set_run_wake(&dev->sdev->sdev_gendev, true); | ||
1007 | } | ||
1008 | } | ||
1009 | |||
1010 | static void ata_acpi_remove_pm_notifier(struct ata_device *dev) | ||
1011 | { | ||
1012 | struct acpi_device *acpi_dev; | ||
1013 | acpi_handle handle; | ||
1014 | acpi_status status; | ||
1015 | |||
1016 | handle = ata_dev_acpi_handle(dev); | ||
1017 | if (!handle) | ||
1018 | return; | ||
1019 | |||
1020 | status = acpi_bus_get_device(handle, &acpi_dev); | ||
1021 | if (ACPI_SUCCESS(status)) { | ||
1022 | device_set_run_wake(&dev->sdev->sdev_gendev, false); | ||
1023 | acpi_remove_notify_handler(handle, ACPI_SYSTEM_NOTIFY, | ||
1024 | ata_acpi_wake_dev); | ||
1025 | } | ||
1026 | } | ||
1027 | |||
1028 | void ata_acpi_bind(struct ata_device *dev) | ||
1029 | { | ||
1030 | ata_acpi_add_pm_notifier(dev); | ||
1031 | } | ||
1032 | |||
1033 | void ata_acpi_unbind(struct ata_device *dev) | ||
1034 | { | ||
1035 | ata_acpi_remove_pm_notifier(dev); | ||
1036 | } | ||
1037 | |||
968 | static int compat_pci_ata(struct ata_port *ap) | 1038 | static int compat_pci_ata(struct ata_port *ap) |
969 | { | 1039 | { |
970 | struct device *dev = ap->tdev.parent; | 1040 | struct device *dev = ap->tdev.parent; |
diff --git a/drivers/ata/libata-scsi.c b/drivers/ata/libata-scsi.c index 22226350cd0c..8ec81ca8f659 100644 --- a/drivers/ata/libata-scsi.c +++ b/drivers/ata/libata-scsi.c | |||
@@ -3445,6 +3445,7 @@ void ata_scsi_scan_host(struct ata_port *ap, int sync) | |||
3445 | if (!IS_ERR(sdev)) { | 3445 | if (!IS_ERR(sdev)) { |
3446 | dev->sdev = sdev; | 3446 | dev->sdev = sdev; |
3447 | scsi_device_put(sdev); | 3447 | scsi_device_put(sdev); |
3448 | ata_acpi_bind(dev); | ||
3448 | } else { | 3449 | } else { |
3449 | dev->sdev = NULL; | 3450 | dev->sdev = NULL; |
3450 | } | 3451 | } |
@@ -3541,6 +3542,8 @@ static void ata_scsi_remove_dev(struct ata_device *dev) | |||
3541 | mutex_lock(&ap->scsi_host->scan_mutex); | 3542 | mutex_lock(&ap->scsi_host->scan_mutex); |
3542 | spin_lock_irqsave(ap->lock, flags); | 3543 | spin_lock_irqsave(ap->lock, flags); |
3543 | 3544 | ||
3545 | ata_acpi_unbind(dev); | ||
3546 | |||
3544 | /* clearing dev->sdev is protected by host lock */ | 3547 | /* clearing dev->sdev is protected by host lock */ |
3545 | sdev = dev->sdev; | 3548 | sdev = dev->sdev; |
3546 | dev->sdev = NULL; | 3549 | dev->sdev = NULL; |
diff --git a/drivers/ata/libata.h b/drivers/ata/libata.h index b0d5294982eb..44a7939b5bb6 100644 --- a/drivers/ata/libata.h +++ b/drivers/ata/libata.h | |||
@@ -120,6 +120,8 @@ extern void ata_acpi_on_disable(struct ata_device *dev); | |||
120 | extern void ata_acpi_set_state(struct ata_port *ap, pm_message_t state); | 120 | extern void ata_acpi_set_state(struct ata_port *ap, pm_message_t state); |
121 | extern int ata_acpi_register(void); | 121 | extern int ata_acpi_register(void); |
122 | extern void ata_acpi_unregister(void); | 122 | extern void ata_acpi_unregister(void); |
123 | extern void ata_acpi_bind(struct ata_device *dev); | ||
124 | extern void ata_acpi_unbind(struct ata_device *dev); | ||
123 | #else | 125 | #else |
124 | static inline void ata_acpi_dissociate(struct ata_host *host) { } | 126 | static inline void ata_acpi_dissociate(struct ata_host *host) { } |
125 | static inline int ata_acpi_on_suspend(struct ata_port *ap) { return 0; } | 127 | static inline int ata_acpi_on_suspend(struct ata_port *ap) { return 0; } |
@@ -130,6 +132,8 @@ static inline void ata_acpi_set_state(struct ata_port *ap, | |||
130 | pm_message_t state) { } | 132 | pm_message_t state) { } |
131 | static inline int ata_acpi_register(void) { return 0; } | 133 | static inline int ata_acpi_register(void) { return 0; } |
132 | static void ata_acpi_unregister(void) { } | 134 | static void ata_acpi_unregister(void) { } |
135 | static void ata_acpi_bind(struct ata_device *dev) { } | ||
136 | static void ata_acpi_unbind(struct ata_device *dev) { } | ||
133 | #endif | 137 | #endif |
134 | 138 | ||
135 | /* libata-scsi.c */ | 139 | /* libata-scsi.c */ |