diff options
| -rw-r--r-- | Documentation/kernel-parameters.txt | 3 | ||||
| -rw-r--r-- | include/linux/suspend.h | 2 | ||||
| -rw-r--r-- | kernel/power/hibernate.c | 31 | ||||
| -rw-r--r-- | kernel/power/main.c | 6 | ||||
| -rw-r--r-- | kernel/power/user.c | 3 |
5 files changed, 40 insertions, 5 deletions
diff --git a/Documentation/kernel-parameters.txt b/Documentation/kernel-parameters.txt index 6eaa9cdb7094..f8f0466b8b1d 100644 --- a/Documentation/kernel-parameters.txt +++ b/Documentation/kernel-parameters.txt | |||
| @@ -2184,6 +2184,8 @@ bytes respectively. Such letter suffixes can also be entirely omitted. | |||
| 2184 | in certain environments such as networked servers or | 2184 | in certain environments such as networked servers or |
| 2185 | real-time systems. | 2185 | real-time systems. |
| 2186 | 2186 | ||
| 2187 | nohibernate [HIBERNATION] Disable hibernation and resume. | ||
| 2188 | |||
| 2187 | nohz= [KNL] Boottime enable/disable dynamic ticks | 2189 | nohz= [KNL] Boottime enable/disable dynamic ticks |
| 2188 | Valid arguments: on, off | 2190 | Valid arguments: on, off |
| 2189 | Default: on | 2191 | Default: on |
| @@ -2980,6 +2982,7 @@ bytes respectively. Such letter suffixes can also be entirely omitted. | |||
| 2980 | noresume Don't check if there's a hibernation image | 2982 | noresume Don't check if there's a hibernation image |
| 2981 | present during boot. | 2983 | present during boot. |
| 2982 | nocompress Don't compress/decompress hibernation images. | 2984 | nocompress Don't compress/decompress hibernation images. |
| 2985 | no Disable hibernation and resume. | ||
| 2983 | 2986 | ||
| 2984 | retain_initrd [RAM] Keep initrd memory after extraction | 2987 | retain_initrd [RAM] Keep initrd memory after extraction |
| 2985 | 2988 | ||
diff --git a/include/linux/suspend.h b/include/linux/suspend.h index f76994b9396c..519064e0c943 100644 --- a/include/linux/suspend.h +++ b/include/linux/suspend.h | |||
| @@ -327,6 +327,7 @@ extern unsigned long get_safe_page(gfp_t gfp_mask); | |||
| 327 | extern void hibernation_set_ops(const struct platform_hibernation_ops *ops); | 327 | extern void hibernation_set_ops(const struct platform_hibernation_ops *ops); |
| 328 | extern int hibernate(void); | 328 | extern int hibernate(void); |
| 329 | extern bool system_entering_hibernation(void); | 329 | extern bool system_entering_hibernation(void); |
| 330 | extern bool hibernation_available(void); | ||
| 330 | asmlinkage int swsusp_save(void); | 331 | asmlinkage int swsusp_save(void); |
| 331 | extern struct pbe *restore_pblist; | 332 | extern struct pbe *restore_pblist; |
| 332 | #else /* CONFIG_HIBERNATION */ | 333 | #else /* CONFIG_HIBERNATION */ |
| @@ -339,6 +340,7 @@ static inline void swsusp_unset_page_free(struct page *p) {} | |||
| 339 | static inline void hibernation_set_ops(const struct platform_hibernation_ops *ops) {} | 340 | static inline void hibernation_set_ops(const struct platform_hibernation_ops *ops) {} |
| 340 | static inline int hibernate(void) { return -ENOSYS; } | 341 | static inline int hibernate(void) { return -ENOSYS; } |
| 341 | static inline bool system_entering_hibernation(void) { return false; } | 342 | static inline bool system_entering_hibernation(void) { return false; } |
| 343 | static inline bool hibernation_available(void) { return false; } | ||
| 342 | #endif /* CONFIG_HIBERNATION */ | 344 | #endif /* CONFIG_HIBERNATION */ |
| 343 | 345 | ||
| 344 | /* Hibernation and suspend events */ | 346 | /* Hibernation and suspend events */ |
diff --git a/kernel/power/hibernate.c b/kernel/power/hibernate.c index 49e0a20fd010..258f492f0347 100644 --- a/kernel/power/hibernate.c +++ b/kernel/power/hibernate.c | |||
| @@ -35,6 +35,7 @@ | |||
| 35 | 35 | ||
| 36 | static int nocompress; | 36 | static int nocompress; |
| 37 | static int noresume; | 37 | static int noresume; |
| 38 | static int nohibernate; | ||
| 38 | static int resume_wait; | 39 | static int resume_wait; |
| 39 | static unsigned int resume_delay; | 40 | static unsigned int resume_delay; |
| 40 | static char resume_file[256] = CONFIG_PM_STD_PARTITION; | 41 | static char resume_file[256] = CONFIG_PM_STD_PARTITION; |
| @@ -62,6 +63,11 @@ bool freezer_test_done; | |||
| 62 | 63 | ||
| 63 | static const struct platform_hibernation_ops *hibernation_ops; | 64 | static const struct platform_hibernation_ops *hibernation_ops; |
| 64 | 65 | ||
| 66 | bool hibernation_available(void) | ||
| 67 | { | ||
| 68 | return (nohibernate == 0); | ||
| 69 | } | ||
| 70 | |||
| 65 | /** | 71 | /** |
| 66 | * hibernation_set_ops - Set the global hibernate operations. | 72 | * hibernation_set_ops - Set the global hibernate operations. |
| 67 | * @ops: Hibernation operations to use in subsequent hibernation transitions. | 73 | * @ops: Hibernation operations to use in subsequent hibernation transitions. |
| @@ -642,6 +648,11 @@ int hibernate(void) | |||
| 642 | { | 648 | { |
| 643 | int error; | 649 | int error; |
| 644 | 650 | ||
| 651 | if (!hibernation_available()) { | ||
| 652 | pr_debug("PM: Hibernation not available.\n"); | ||
| 653 | return -EPERM; | ||
| 654 | } | ||
| 655 | |||
| 645 | lock_system_sleep(); | 656 | lock_system_sleep(); |
| 646 | /* The snapshot device should not be opened while we're running */ | 657 | /* The snapshot device should not be opened while we're running */ |
| 647 | if (!atomic_add_unless(&snapshot_device_available, -1, 0)) { | 658 | if (!atomic_add_unless(&snapshot_device_available, -1, 0)) { |
| @@ -734,7 +745,7 @@ static int software_resume(void) | |||
| 734 | /* | 745 | /* |
| 735 | * If the user said "noresume".. bail out early. | 746 | * If the user said "noresume".. bail out early. |
| 736 | */ | 747 | */ |
| 737 | if (noresume) | 748 | if (noresume || !hibernation_available()) |
| 738 | return 0; | 749 | return 0; |
| 739 | 750 | ||
| 740 | /* | 751 | /* |
| @@ -900,6 +911,9 @@ static ssize_t disk_show(struct kobject *kobj, struct kobj_attribute *attr, | |||
| 900 | int i; | 911 | int i; |
| 901 | char *start = buf; | 912 | char *start = buf; |
| 902 | 913 | ||
| 914 | if (!hibernation_available()) | ||
| 915 | return sprintf(buf, "[disabled]\n"); | ||
| 916 | |||
| 903 | for (i = HIBERNATION_FIRST; i <= HIBERNATION_MAX; i++) { | 917 | for (i = HIBERNATION_FIRST; i <= HIBERNATION_MAX; i++) { |
| 904 | if (!hibernation_modes[i]) | 918 | if (!hibernation_modes[i]) |
| 905 | continue; | 919 | continue; |
| @@ -934,6 +948,9 @@ static ssize_t disk_store(struct kobject *kobj, struct kobj_attribute *attr, | |||
| 934 | char *p; | 948 | char *p; |
| 935 | int mode = HIBERNATION_INVALID; | 949 | int mode = HIBERNATION_INVALID; |
| 936 | 950 | ||
| 951 | if (!hibernation_available()) | ||
| 952 | return -EPERM; | ||
| 953 | |||
| 937 | p = memchr(buf, '\n', n); | 954 | p = memchr(buf, '\n', n); |
| 938 | len = p ? p - buf : n; | 955 | len = p ? p - buf : n; |
| 939 | 956 | ||
| @@ -1101,6 +1118,10 @@ static int __init hibernate_setup(char *str) | |||
| 1101 | noresume = 1; | 1118 | noresume = 1; |
| 1102 | else if (!strncmp(str, "nocompress", 10)) | 1119 | else if (!strncmp(str, "nocompress", 10)) |
| 1103 | nocompress = 1; | 1120 | nocompress = 1; |
| 1121 | else if (!strncmp(str, "no", 2)) { | ||
| 1122 | noresume = 1; | ||
| 1123 | nohibernate = 1; | ||
| 1124 | } | ||
| 1104 | return 1; | 1125 | return 1; |
| 1105 | } | 1126 | } |
| 1106 | 1127 | ||
| @@ -1125,9 +1146,17 @@ static int __init resumedelay_setup(char *str) | |||
| 1125 | return 1; | 1146 | return 1; |
| 1126 | } | 1147 | } |
| 1127 | 1148 | ||
| 1149 | static int __init nohibernate_setup(char *str) | ||
| 1150 | { | ||
| 1151 | noresume = 1; | ||
| 1152 | nohibernate = 1; | ||
| 1153 | return 1; | ||
| 1154 | } | ||
| 1155 | |||
| 1128 | __setup("noresume", noresume_setup); | 1156 | __setup("noresume", noresume_setup); |
| 1129 | __setup("resume_offset=", resume_offset_setup); | 1157 | __setup("resume_offset=", resume_offset_setup); |
| 1130 | __setup("resume=", resume_setup); | 1158 | __setup("resume=", resume_setup); |
| 1131 | __setup("hibernate=", hibernate_setup); | 1159 | __setup("hibernate=", hibernate_setup); |
| 1132 | __setup("resumewait", resumewait_setup); | 1160 | __setup("resumewait", resumewait_setup); |
| 1133 | __setup("resumedelay=", resumedelay_setup); | 1161 | __setup("resumedelay=", resumedelay_setup); |
| 1162 | __setup("nohibernate", nohibernate_setup); | ||
diff --git a/kernel/power/main.c b/kernel/power/main.c index 573410d6647e..8e90f330f139 100644 --- a/kernel/power/main.c +++ b/kernel/power/main.c | |||
| @@ -300,13 +300,11 @@ static ssize_t state_show(struct kobject *kobj, struct kobj_attribute *attr, | |||
| 300 | s += sprintf(s,"%s ", pm_states[i].label); | 300 | s += sprintf(s,"%s ", pm_states[i].label); |
| 301 | 301 | ||
| 302 | #endif | 302 | #endif |
| 303 | #ifdef CONFIG_HIBERNATION | 303 | if (hibernation_available()) |
| 304 | s += sprintf(s, "%s\n", "disk"); | 304 | s += sprintf(s, "disk "); |
| 305 | #else | ||
| 306 | if (s != buf) | 305 | if (s != buf) |
| 307 | /* convert the last space to a newline */ | 306 | /* convert the last space to a newline */ |
| 308 | *(s-1) = '\n'; | 307 | *(s-1) = '\n'; |
| 309 | #endif | ||
| 310 | return (s - buf); | 308 | return (s - buf); |
| 311 | } | 309 | } |
| 312 | 310 | ||
diff --git a/kernel/power/user.c b/kernel/power/user.c index 98d357584cd6..526e8911460a 100644 --- a/kernel/power/user.c +++ b/kernel/power/user.c | |||
| @@ -49,6 +49,9 @@ static int snapshot_open(struct inode *inode, struct file *filp) | |||
| 49 | struct snapshot_data *data; | 49 | struct snapshot_data *data; |
| 50 | int error; | 50 | int error; |
| 51 | 51 | ||
| 52 | if (!hibernation_available()) | ||
| 53 | return -EPERM; | ||
| 54 | |||
| 52 | lock_system_sleep(); | 55 | lock_system_sleep(); |
| 53 | 56 | ||
| 54 | if (!atomic_add_unless(&snapshot_device_available, -1, 0)) { | 57 | if (!atomic_add_unless(&snapshot_device_available, -1, 0)) { |
