| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | #include <linux/suspend.h>
 | 
					
						
							| 
									
										
										
										
											2007-10-26 01:05:05 +02:00
										 |  |  | #include <linux/suspend_ioctls.h>
 | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | #include <linux/utsname.h>
 | 
					
						
							| 
									
										
										
										
											2008-01-15 23:17:00 -05:00
										 |  |  | #include <linux/freezer.h>
 | 
					
						
							| 
									
										
										
										
											2014-04-07 15:39:20 -07:00
										 |  |  | #include <linux/compiler.h>
 | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct swsusp_info { | 
					
						
							|  |  |  | 	struct new_utsname	uts; | 
					
						
							|  |  |  | 	u32			version_code; | 
					
						
							|  |  |  | 	unsigned long		num_physpages; | 
					
						
							|  |  |  | 	int			cpus; | 
					
						
							|  |  |  | 	unsigned long		image_pages; | 
					
						
							| 
									
										
										
										
											2006-01-06 00:13:05 -08:00
										 |  |  | 	unsigned long		pages; | 
					
						
							| 
									
										
										
										
											2006-03-23 03:00:03 -08:00
										 |  |  | 	unsigned long		size; | 
					
						
							| 
									
										
										
										
											2014-04-07 15:39:20 -07:00
										 |  |  | } __aligned(PAGE_SIZE); | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-18 03:04:52 -07:00
										 |  |  | #ifdef CONFIG_HIBERNATION
 | 
					
						
							| 
									
										
										
										
											2010-09-20 19:44:56 +02:00
										 |  |  | /* kernel/power/snapshot.c */ | 
					
						
							| 
									
										
										
										
											2011-05-15 11:38:48 +02:00
										 |  |  | extern void __init hibernate_reserved_size_init(void); | 
					
						
							| 
									
										
										
										
											2010-09-20 19:44:56 +02:00
										 |  |  | extern void __init hibernate_image_size_init(void); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-18 03:04:52 -07:00
										 |  |  | #ifdef CONFIG_ARCH_HIBERNATION_HEADER
 | 
					
						
							|  |  |  | /* Maximum size of architecture specific data in a hibernation header */ | 
					
						
							|  |  |  | #define MAX_ARCH_HEADER_SIZE	(sizeof(struct new_utsname) + 4)
 | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-18 03:04:52 -07:00
										 |  |  | extern int arch_hibernation_header_save(void *addr, unsigned int max_size); | 
					
						
							|  |  |  | extern int arch_hibernation_header_restore(void *addr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline int init_header_complete(struct swsusp_info *info) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return arch_hibernation_header_save(info, MAX_ARCH_HEADER_SIZE); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline char *check_image_kernel(struct swsusp_info *info) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return arch_hibernation_header_restore(info) ? | 
					
						
							|  |  |  | 			"architecture specific data" : NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif /* CONFIG_ARCH_HIBERNATION_HEADER */
 | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-05-06 14:50:52 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Keep some memory free so that I/O operations can succeed without paging | 
					
						
							|  |  |  |  * [Might this be more than 4 MB?] | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #define PAGES_FOR_IO	((4096 * 1024) >> PAGE_SHIFT)
 | 
					
						
							| 
									
										
										
										
											2007-10-18 03:04:52 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-05-06 14:50:52 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Keep 1 MB of memory free so that device drivers can allocate some pages in | 
					
						
							|  |  |  |  * their .suspend() routines without breaking the suspend to disk. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #define SPARE_PAGES	((1024 * 1024) >> PAGE_SHIFT)
 | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-27 17:13:53 +05:30
										 |  |  | asmlinkage int swsusp_save(void); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-10 01:27:49 +02:00
										 |  |  | /* kernel/power/hibernate.c */ | 
					
						
							| 
									
										
										
										
											2011-12-01 22:33:10 +01:00
										 |  |  | extern bool freezer_test_done; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-07-19 01:47:29 -07:00
										 |  |  | extern int hibernation_snapshot(int platform_mode); | 
					
						
							| 
									
										
											  
											
												swsusp: introduce restore platform operations
At least on some machines it is necessary to prepare the ACPI firmware for the
restoration of the system memory state from the hibernation image if the
"platform" mode of hibernation has been used.  Namely, in that cases we need
to disable the GPEs before replacing the "boot" kernel with the "frozen"
kernel (cf.  http://bugzilla.kernel.org/show_bug.cgi?id=7887).  After the
restore they will be re-enabled by hibernation_ops->finish(), but if the
restore fails, they have to be re-enabled by the restore code explicitly.
For this purpose we can introduce two additional hibernation operations,
called pre_restore() and restore_cleanup() and call them from the restore code
path.  Still, they should be called if the "platform" mode of hibernation has
been used, so we need to pass the information about the hibernation mode from
the "frozen" kernel to the "boot" kernel in the image header.
Apparently, we can't drop the disabling of GPEs before the restore because of
Bug #7887 .   We also can't do it unconditionally, because the GPEs wouldn't
have been enabled after a successful restore if the suspend had been done in
the 'shutdown' or 'reboot' mode.
In principle we could (and probably should) unconditionally disable the GPEs
before each snapshot creation *and* before the restore, but then we'd have to
unconditionally enable them after the snapshot creation as well as after the
restore (or restore failure)   Still, for this purpose we'd need to modify
acpi_enter_sleep_state_prep() and acpi_leave_sleep_state() and we'd have to
introduce some mechanism synchronizing the disablind/enabling of the GPEs with
the device drivers' .suspend()/.resume() routines and with
disable_/enable_nonboot_cpus().   However, this would have affected the
suspend (ie.  s2ram) code as well as the hibernation, which I'd like to avoid
in this patch series.
Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
Cc: Nigel Cunningham <nigel@nigel.suspend2.net>
Cc: Pavel Machek <pavel@ucw.cz>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
											
										 
											2007-07-19 01:47:30 -07:00
										 |  |  | extern int hibernation_restore(int platform_mode); | 
					
						
							| 
									
										
										
										
											2007-07-19 01:47:29 -07:00
										 |  |  | extern int hibernation_platform_enter(void); | 
					
						
							| 
									
										
										
										
											2010-09-20 19:44:56 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #else /* !CONFIG_HIBERNATION */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-15 11:38:48 +02:00
										 |  |  | static inline void hibernate_reserved_size_init(void) {} | 
					
						
							| 
									
										
										
										
											2010-09-20 19:44:56 +02:00
										 |  |  | static inline void hibernate_image_size_init(void) {} | 
					
						
							|  |  |  | #endif /* !CONFIG_HIBERNATION */
 | 
					
						
							| 
									
										
										
										
											2006-12-06 20:34:35 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-05-02 19:27:07 +02:00
										 |  |  | extern int pfn_is_nosave(unsigned long); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | #define power_attr(_name) \
 | 
					
						
							| 
									
										
										
										
											2007-11-02 13:47:53 +01:00
										 |  |  | static struct kobj_attribute _name##_attr = {	\ | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 	.attr	= {				\ | 
					
						
							|  |  |  | 		.name = __stringify(_name),	\ | 
					
						
							|  |  |  | 		.mode = 0644,			\ | 
					
						
							|  |  |  | 	},					\ | 
					
						
							|  |  |  | 	.show	= _name##_show,			\ | 
					
						
							|  |  |  | 	.store	= _name##_store,		\ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-01 03:05:07 -08:00
										 |  |  | /* Preferred image size in bytes (default 500 MB) */ | 
					
						
							|  |  |  | extern unsigned long image_size; | 
					
						
							| 
									
										
										
										
											2011-05-15 11:38:48 +02:00
										 |  |  | /* Size of memory reserved for drivers (default SPARE_PAGES x PAGE_SIZE) */ | 
					
						
							|  |  |  | extern unsigned long reserved_size; | 
					
						
							| 
									
										
										
										
											2006-03-23 02:59:59 -08:00
										 |  |  | extern int in_suspend; | 
					
						
							| 
									
										
										
										
											2006-03-23 03:00:00 -08:00
										 |  |  | extern dev_t swsusp_resume_device; | 
					
						
							| 
									
										
										
										
											2006-12-06 20:34:12 -08:00
										 |  |  | extern sector_t swsusp_resume_block; | 
					
						
							| 
									
										
										
										
											2006-03-23 02:59:59 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-30 14:59:56 -08:00
										 |  |  | extern asmlinkage int swsusp_arch_suspend(void); | 
					
						
							|  |  |  | extern asmlinkage int swsusp_arch_resume(void); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-05-06 14:50:43 -07:00
										 |  |  | extern int create_basic_memory_bitmaps(void); | 
					
						
							|  |  |  | extern void free_basic_memory_bitmaps(void); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:24:05 +02:00
										 |  |  | extern int hibernate_preallocate_memory(void); | 
					
						
							| 
									
										
										
										
											2006-03-23 02:59:59 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-25 23:32:46 -07:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  *	Auxiliary structure used for reading the snapshot image data and | 
					
						
							|  |  |  |  *	metadata from and writing them to the list of page backup entries | 
					
						
							|  |  |  |  *	(PBEs) which is the main data structure of swsusp. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *	Using struct snapshot_handle we can transfer the image, including its | 
					
						
							|  |  |  |  *	metadata, as a continuous sequence of bytes with the help of | 
					
						
							|  |  |  |  *	snapshot_read_next() and snapshot_write_next(). | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *	The code that writes the image to a storage or transfers it to | 
					
						
							|  |  |  |  *	the user land is required to use snapshot_read_next() for this | 
					
						
							|  |  |  |  *	purpose and it should not make any assumptions regarding the internal | 
					
						
							|  |  |  |  *	structure of the image.  Similarly, the code that reads the image from | 
					
						
							|  |  |  |  *	a storage or transfers it from the user land is required to use | 
					
						
							|  |  |  |  *	snapshot_write_next(). | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *	This may allow us to change the internal structure of the image | 
					
						
							|  |  |  |  *	in the future with considerably less effort. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-03-23 02:59:59 -08:00
										 |  |  | struct snapshot_handle { | 
					
						
							| 
									
										
										
										
											2006-09-25 23:32:46 -07:00
										 |  |  | 	unsigned int	cur;	/* number of the block of PAGE_SIZE bytes the
 | 
					
						
							|  |  |  | 				 * next operation will refer to (ie. current) | 
					
						
							|  |  |  | 				 */ | 
					
						
							|  |  |  | 	void		*buffer;	/* address of the block to read from
 | 
					
						
							|  |  |  | 					 * or write to | 
					
						
							|  |  |  | 					 */ | 
					
						
							|  |  |  | 	int		sync_read;	/* Set to one to notify the caller of
 | 
					
						
							|  |  |  | 					 * snapshot_write_next() that it may | 
					
						
							|  |  |  | 					 * need to call wait_on_bio_chain() | 
					
						
							|  |  |  | 					 */ | 
					
						
							| 
									
										
										
										
											2006-03-23 02:59:59 -08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-25 23:32:46 -07:00
										 |  |  | /* This macro returns the address from/to which the caller of
 | 
					
						
							|  |  |  |  * snapshot_read_next()/snapshot_write_next() is allowed to | 
					
						
							|  |  |  |  * read/write data after the function returns | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2010-05-01 23:52:02 +02:00
										 |  |  | #define data_of(handle)	((handle).buffer)
 | 
					
						
							| 
									
										
										
										
											2006-03-23 02:59:59 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-25 23:32:54 -07:00
										 |  |  | extern unsigned int snapshot_additional_pages(struct zone *zone); | 
					
						
							| 
									
										
										
										
											2007-10-26 00:59:31 +02:00
										 |  |  | extern unsigned long snapshot_get_image_size(void); | 
					
						
							| 
									
										
										
										
											2010-05-01 23:52:02 +02:00
										 |  |  | extern int snapshot_read_next(struct snapshot_handle *handle); | 
					
						
							|  |  |  | extern int snapshot_write_next(struct snapshot_handle *handle); | 
					
						
							| 
									
										
											  
											
												[PATCH] swsusp: Improve handling of highmem
Currently swsusp saves the contents of highmem pages by copying them to the
normal zone which is quite inefficient (eg.  it requires two normal pages
to be used for saving one highmem page).  This may be improved by using
highmem for saving the contents of saveable highmem pages.
Namely, during the suspend phase of the suspend-resume cycle we try to
allocate as many free highmem pages as there are saveable highmem pages.
If there are not enough highmem image pages to store the contents of all of
the saveable highmem pages, some of them will be stored in the "normal"
memory.  Next, we allocate as many free "normal" pages as needed to store
the (remaining) image data.  We use a memory bitmap to mark the allocated
free pages (ie.  highmem as well as "normal" image pages).
Now, we use another memory bitmap to mark all of the saveable pages
(highmem as well as "normal") and the contents of the saveable pages are
copied into the image pages.  Then, the second bitmap is used to save the
pfns corresponding to the saveable pages and the first one is used to save
their data.
During the resume phase the pfns of the pages that were saveable during the
suspend are loaded from the image and used to mark the "unsafe" page
frames.  Next, we try to allocate as many free highmem page frames as to
load all of the image data that had been in the highmem before the suspend
and we allocate so many free "normal" page frames that the total number of
allocated free pages (highmem and "normal") is equal to the size of the
image.  While doing this we have to make sure that there will be some extra
free "normal" and "safe" page frames for two lists of PBEs constructed
later.
Now, the image data are loaded, if possible, into their "original" page
frames.  The image data that cannot be written into their "original" page
frames are loaded into "safe" page frames and their "original" kernel
virtual addresses, as well as the addresses of the "safe" pages containing
their copies, are stored in one of two lists of PBEs.
One list of PBEs is for the copies of "normal" suspend pages (ie.  "normal"
pages that were saveable during the suspend) and it is used in the same way
as previously (ie.  by the architecture-dependent parts of swsusp).  The
other list of PBEs is for the copies of highmem suspend pages.  The pages
in this list are restored (in a reversible way) right before the
arch-dependent code is called.
Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
Cc: Pavel Machek <pavel@ucw.cz>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
											
										 
											2006-12-06 20:34:18 -08:00
										 |  |  | extern void snapshot_write_finalize(struct snapshot_handle *handle); | 
					
						
							| 
									
										
										
										
											2006-09-25 23:32:54 -07:00
										 |  |  | extern int snapshot_image_loaded(struct snapshot_handle *handle); | 
					
						
							| 
									
										
										
										
											2006-03-23 03:00:00 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-05-06 14:50:45 -07:00
										 |  |  | /* If unset, the snapshot device cannot be open. */ | 
					
						
							|  |  |  | extern atomic_t snapshot_device_available; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-05-06 14:50:47 -07:00
										 |  |  | extern sector_t alloc_swapdev_block(int swap); | 
					
						
							|  |  |  | extern void free_all_swap_pages(int swap); | 
					
						
							|  |  |  | extern int swsusp_swap_in_use(void); | 
					
						
							| 
									
										
										
										
											2006-03-23 03:00:00 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												swsusp: introduce restore platform operations
At least on some machines it is necessary to prepare the ACPI firmware for the
restoration of the system memory state from the hibernation image if the
"platform" mode of hibernation has been used.  Namely, in that cases we need
to disable the GPEs before replacing the "boot" kernel with the "frozen"
kernel (cf.  http://bugzilla.kernel.org/show_bug.cgi?id=7887).  After the
restore they will be re-enabled by hibernation_ops->finish(), but if the
restore fails, they have to be re-enabled by the restore code explicitly.
For this purpose we can introduce two additional hibernation operations,
called pre_restore() and restore_cleanup() and call them from the restore code
path.  Still, they should be called if the "platform" mode of hibernation has
been used, so we need to pass the information about the hibernation mode from
the "frozen" kernel to the "boot" kernel in the image header.
Apparently, we can't drop the disabling of GPEs before the restore because of
Bug #7887 .   We also can't do it unconditionally, because the GPEs wouldn't
have been enabled after a successful restore if the suspend had been done in
the 'shutdown' or 'reboot' mode.
In principle we could (and probably should) unconditionally disable the GPEs
before each snapshot creation *and* before the restore, but then we'd have to
unconditionally enable them after the snapshot creation as well as after the
restore (or restore failure)   Still, for this purpose we'd need to modify
acpi_enter_sleep_state_prep() and acpi_leave_sleep_state() and we'd have to
introduce some mechanism synchronizing the disablind/enabling of the GPEs with
the device drivers' .suspend()/.resume() routines and with
disable_/enable_nonboot_cpus().   However, this would have affected the
suspend (ie.  s2ram) code as well as the hibernation, which I'd like to avoid
in this patch series.
Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
Cc: Nigel Cunningham <nigel@nigel.suspend2.net>
Cc: Pavel Machek <pavel@ucw.cz>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
											
										 
											2007-07-19 01:47:30 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Flags that can be passed from the hibernatig hernel to the "boot" kernel in | 
					
						
							|  |  |  |  * the image header. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #define SF_PLATFORM_MODE	1
 | 
					
						
							| 
									
										
										
										
											2010-09-09 23:06:23 +02:00
										 |  |  | #define SF_NOCOMPRESS_MODE	2
 | 
					
						
							| 
									
										
										
										
											2011-10-13 23:58:07 +02:00
										 |  |  | #define SF_CRC32_MODE	        4
 | 
					
						
							| 
									
										
											  
											
												swsusp: introduce restore platform operations
At least on some machines it is necessary to prepare the ACPI firmware for the
restoration of the system memory state from the hibernation image if the
"platform" mode of hibernation has been used.  Namely, in that cases we need
to disable the GPEs before replacing the "boot" kernel with the "frozen"
kernel (cf.  http://bugzilla.kernel.org/show_bug.cgi?id=7887).  After the
restore they will be re-enabled by hibernation_ops->finish(), but if the
restore fails, they have to be re-enabled by the restore code explicitly.
For this purpose we can introduce two additional hibernation operations,
called pre_restore() and restore_cleanup() and call them from the restore code
path.  Still, they should be called if the "platform" mode of hibernation has
been used, so we need to pass the information about the hibernation mode from
the "frozen" kernel to the "boot" kernel in the image header.
Apparently, we can't drop the disabling of GPEs before the restore because of
Bug #7887 .   We also can't do it unconditionally, because the GPEs wouldn't
have been enabled after a successful restore if the suspend had been done in
the 'shutdown' or 'reboot' mode.
In principle we could (and probably should) unconditionally disable the GPEs
before each snapshot creation *and* before the restore, but then we'd have to
unconditionally enable them after the snapshot creation as well as after the
restore (or restore failure)   Still, for this purpose we'd need to modify
acpi_enter_sleep_state_prep() and acpi_leave_sleep_state() and we'd have to
introduce some mechanism synchronizing the disablind/enabling of the GPEs with
the device drivers' .suspend()/.resume() routines and with
disable_/enable_nonboot_cpus().   However, this would have affected the
suspend (ie.  s2ram) code as well as the hibernation, which I'd like to avoid
in this patch series.
Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
Cc: Nigel Cunningham <nigel@nigel.suspend2.net>
Cc: Pavel Machek <pavel@ucw.cz>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
											
										 
											2007-07-19 01:47:30 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-10 01:27:49 +02:00
										 |  |  | /* kernel/power/hibernate.c */ | 
					
						
							| 
									
										
										
										
											2006-03-23 03:00:01 -08:00
										 |  |  | extern int swsusp_check(void); | 
					
						
							|  |  |  | extern void swsusp_free(void); | 
					
						
							| 
									
										
											  
											
												swsusp: introduce restore platform operations
At least on some machines it is necessary to prepare the ACPI firmware for the
restoration of the system memory state from the hibernation image if the
"platform" mode of hibernation has been used.  Namely, in that cases we need
to disable the GPEs before replacing the "boot" kernel with the "frozen"
kernel (cf.  http://bugzilla.kernel.org/show_bug.cgi?id=7887).  After the
restore they will be re-enabled by hibernation_ops->finish(), but if the
restore fails, they have to be re-enabled by the restore code explicitly.
For this purpose we can introduce two additional hibernation operations,
called pre_restore() and restore_cleanup() and call them from the restore code
path.  Still, they should be called if the "platform" mode of hibernation has
been used, so we need to pass the information about the hibernation mode from
the "frozen" kernel to the "boot" kernel in the image header.
Apparently, we can't drop the disabling of GPEs before the restore because of
Bug #7887 .   We also can't do it unconditionally, because the GPEs wouldn't
have been enabled after a successful restore if the suspend had been done in
the 'shutdown' or 'reboot' mode.
In principle we could (and probably should) unconditionally disable the GPEs
before each snapshot creation *and* before the restore, but then we'd have to
unconditionally enable them after the snapshot creation as well as after the
restore (or restore failure)   Still, for this purpose we'd need to modify
acpi_enter_sleep_state_prep() and acpi_leave_sleep_state() and we'd have to
introduce some mechanism synchronizing the disablind/enabling of the GPEs with
the device drivers' .suspend()/.resume() routines and with
disable_/enable_nonboot_cpus().   However, this would have affected the
suspend (ie.  s2ram) code as well as the hibernation, which I'd like to avoid
in this patch series.
Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
Cc: Nigel Cunningham <nigel@nigel.suspend2.net>
Cc: Pavel Machek <pavel@ucw.cz>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
											
										 
											2007-07-19 01:47:30 -07:00
										 |  |  | extern int swsusp_read(unsigned int *flags_p); | 
					
						
							|  |  |  | extern int swsusp_write(unsigned int flags); | 
					
						
							| 
									
										
										
										
											2007-10-08 13:21:10 -04:00
										 |  |  | extern void swsusp_close(fmode_t); | 
					
						
							| 
									
										
										
										
											2012-06-16 00:09:58 +02:00
										 |  |  | #ifdef CONFIG_SUSPEND
 | 
					
						
							|  |  |  | extern int swsusp_unmark(void); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2006-12-06 20:34:32 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-01 23:52:34 +02:00
										 |  |  | /* kernel/power/block_io.c */ | 
					
						
							|  |  |  | extern struct block_device *hib_resume_bdev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | extern int hib_bio_read_page(pgoff_t page_off, void *addr, | 
					
						
							|  |  |  | 		struct bio **bio_chain); | 
					
						
							|  |  |  | extern int hib_bio_write_page(pgoff_t page_off, void *addr, | 
					
						
							|  |  |  | 		struct bio **bio_chain); | 
					
						
							|  |  |  | extern int hib_wait_on_bio_chain(struct bio **bio_chain); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-12-06 20:34:32 -08:00
										 |  |  | struct timeval; | 
					
						
							| 
									
										
										
										
											2007-07-19 01:47:36 -07:00
										 |  |  | /* kernel/power/swsusp.c */ | 
					
						
							| 
									
										
										
										
											2006-12-06 20:34:32 -08:00
										 |  |  | extern void swsusp_show_speed(struct timeval *, struct timeval *, | 
					
						
							|  |  |  | 				unsigned int, char *); | 
					
						
							| 
									
										
										
										
											2007-07-19 01:47:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-07-29 23:27:18 +02:00
										 |  |  | #ifdef CONFIG_SUSPEND
 | 
					
						
							| 
									
										
										
										
											2014-05-26 13:40:47 +02:00
										 |  |  | struct pm_sleep_state { | 
					
						
							|  |  |  | 	const char *label; | 
					
						
							|  |  |  | 	suspend_state_t state; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-10 01:27:12 +02:00
										 |  |  | /* kernel/power/suspend.c */ | 
					
						
							| 
									
										
										
										
											2014-05-26 13:40:47 +02:00
										 |  |  | extern struct pm_sleep_state pm_states[]; | 
					
						
							| 
									
										
										
										
											2009-06-10 01:27:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-07-19 01:47:38 -07:00
										 |  |  | extern int suspend_devices_and_enter(suspend_state_t state); | 
					
						
							| 
									
										
										
										
											2007-07-29 23:27:18 +02:00
										 |  |  | #else /* !CONFIG_SUSPEND */
 | 
					
						
							|  |  |  | static inline int suspend_devices_and_enter(suspend_state_t state) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return -ENOSYS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif /* !CONFIG_SUSPEND */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-10 01:27:12 +02:00
										 |  |  | #ifdef CONFIG_PM_TEST_SUSPEND
 | 
					
						
							|  |  |  | /* kernel/power/suspend_test.c */ | 
					
						
							|  |  |  | extern void suspend_test_start(void); | 
					
						
							|  |  |  | extern void suspend_test_finish(const char *label); | 
					
						
							|  |  |  | #else /* !CONFIG_PM_TEST_SUSPEND */
 | 
					
						
							|  |  |  | static inline void suspend_test_start(void) {} | 
					
						
							|  |  |  | static inline void suspend_test_finish(const char *label) {} | 
					
						
							|  |  |  | #endif /* !CONFIG_PM_TEST_SUSPEND */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-19 23:49:18 +01:00
										 |  |  | #ifdef CONFIG_PM_SLEEP
 | 
					
						
							|  |  |  | /* kernel/power/main.c */ | 
					
						
							|  |  |  | extern int pm_notifier_call_chain(unsigned long val); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2007-11-19 23:36:20 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef CONFIG_HIGHMEM
 | 
					
						
							|  |  |  | int restore_highmem(void); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | static inline unsigned int count_highmem_pages(void) { return 0; } | 
					
						
							|  |  |  | static inline int restore_highmem(void) { return 0; } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
											  
											
												Suspend: Testing facility (rev. 2)
Introduce sysfs attribute /sys/power/pm_test allowing one to test the suspend
core code.  Namely, writing one of the strings:
freezer
devices
platform
processors
core
to this file causes the suspend code to work in one of the test modes defined as
follows:
freezer
- test the freezing of processes
devices
- test the freezing of processes and suspending of devices
platform
- test the freezing of processes, suspending of devices and platform global
  control methods(*)
processors
- test the freezing of processes, suspending of devices, platform global
  control methods and the disabling of nonboot CPUs
core
- test the freezing of processes, suspending of devices, platform global
  control methods, the disabling of nonboot CPUs and suspending of
  platform/system devices
(*) These are ACPI global control methods on ACPI systems
Then, if a suspend is started by normal means, the suspend core will perform
its normal operations up to the point indicated by given test level.  Next, it
will wait for 5 seconds and carry out the resume operations needed to transition
the system back to the fully functional state.
Writing "none" to /sys/power/pm_test turns the testing off.
When open for reading, /sys/power/pm_test contains a space-separated list of all
available tests (including "none" that represents the normal functionality) in
which the current test level is indicated by square brackets.
Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
Acked-by: Pavel Machek <pavel@ucw.cz>
Signed-off-by: Len Brown <len.brown@intel.com>
											
										 
											2007-11-19 23:41:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Suspend test levels | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | enum { | 
					
						
							|  |  |  | 	/* keep first */ | 
					
						
							|  |  |  | 	TEST_NONE, | 
					
						
							|  |  |  | 	TEST_CORE, | 
					
						
							|  |  |  | 	TEST_CPUS, | 
					
						
							|  |  |  | 	TEST_PLATFORM, | 
					
						
							|  |  |  | 	TEST_DEVICES, | 
					
						
							|  |  |  | 	TEST_FREEZER, | 
					
						
							|  |  |  | 	/* keep last */ | 
					
						
							|  |  |  | 	__TEST_AFTER_LAST | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define TEST_FIRST	TEST_NONE
 | 
					
						
							|  |  |  | #define TEST_MAX	(__TEST_AFTER_LAST - 1)
 | 
					
						
							| 
									
										
										
										
											2007-11-19 23:42:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | extern int pm_test_level; | 
					
						
							| 
									
										
										
										
											2008-01-15 23:17:00 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef CONFIG_SUSPEND_FREEZER
 | 
					
						
							|  |  |  | static inline int suspend_freeze_processes(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
											  
											
												PM / Freezer: Thaw only kernel threads if freezing of kernel threads fails
If freezing of kernel threads fails, we are expected to automatically
thaw tasks in the error recovery path. However, at times, we encounter
situations in which we would like the automatic error recovery path
to thaw only the kernel threads, because we want to be able to do
some more cleanup before we thaw userspace. Something like:
error = freeze_kernel_threads();
if (error) {
	/* Do some cleanup */
	/* Only then thaw userspace tasks*/
	thaw_processes();
}
An example of such a situation is where we freeze/thaw filesystems
during suspend/hibernation. There, if freezing of kernel threads
fails, we would like to thaw the frozen filesystems before thawing
the userspace tasks.
So, modify freeze_kernel_threads() to thaw only kernel threads in
case of freezing failure. And change suspend_freeze_processes()
accordingly. (At the same time, let us also get rid of the rather
cryptic usage of the conditional operator (:?) in that function.)
[rjw: In fact, this patch fixes a regression introduced during the
 3.3 merge window, because without it thaw_processes() may be called
 before swsusp_free() in some situations and that may lead to massive
 memory allocation failures.]
Signed-off-by: Srivatsa S. Bhat <srivatsa.bhat@linux.vnet.ibm.com>
Acked-by: Tejun Heo <tj@kernel.org>
Acked-by: Nigel Cunningham <nigel@tuxonice.net>
Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
											
										 
											2012-02-03 22:22:41 +01:00
										 |  |  | 	int error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	error = freeze_processes(); | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * freeze_processes() automatically thaws every task if freezing | 
					
						
							|  |  |  | 	 * fails. So we need not do anything extra upon error. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	if (error) | 
					
						
							| 
									
										
										
										
											2012-02-11 22:40:23 +01:00
										 |  |  | 		return error; | 
					
						
							| 
									
										
											  
											
												PM / Freezer: Thaw only kernel threads if freezing of kernel threads fails
If freezing of kernel threads fails, we are expected to automatically
thaw tasks in the error recovery path. However, at times, we encounter
situations in which we would like the automatic error recovery path
to thaw only the kernel threads, because we want to be able to do
some more cleanup before we thaw userspace. Something like:
error = freeze_kernel_threads();
if (error) {
	/* Do some cleanup */
	/* Only then thaw userspace tasks*/
	thaw_processes();
}
An example of such a situation is where we freeze/thaw filesystems
during suspend/hibernation. There, if freezing of kernel threads
fails, we would like to thaw the frozen filesystems before thawing
the userspace tasks.
So, modify freeze_kernel_threads() to thaw only kernel threads in
case of freezing failure. And change suspend_freeze_processes()
accordingly. (At the same time, let us also get rid of the rather
cryptic usage of the conditional operator (:?) in that function.)
[rjw: In fact, this patch fixes a regression introduced during the
 3.3 merge window, because without it thaw_processes() may be called
 before swsusp_free() in some situations and that may lead to massive
 memory allocation failures.]
Signed-off-by: Srivatsa S. Bhat <srivatsa.bhat@linux.vnet.ibm.com>
Acked-by: Tejun Heo <tj@kernel.org>
Acked-by: Nigel Cunningham <nigel@tuxonice.net>
Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
											
										 
											2012-02-03 22:22:41 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	error = freeze_kernel_threads(); | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * freeze_kernel_threads() thaws only kernel threads upon freezing | 
					
						
							|  |  |  | 	 * failure. So we have to thaw the userspace tasks ourselves. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	if (error) | 
					
						
							|  |  |  | 		thaw_processes(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return error; | 
					
						
							| 
									
										
										
										
											2008-01-15 23:17:00 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void suspend_thaw_processes(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	thaw_processes(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | static inline int suspend_freeze_processes(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void suspend_thaw_processes(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2012-04-29 22:53:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef CONFIG_PM_AUTOSLEEP
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* kernel/power/autosleep.c */ | 
					
						
							|  |  |  | extern int pm_autosleep_init(void); | 
					
						
							|  |  |  | extern int pm_autosleep_lock(void); | 
					
						
							|  |  |  | extern void pm_autosleep_unlock(void); | 
					
						
							|  |  |  | extern suspend_state_t pm_autosleep_state(void); | 
					
						
							|  |  |  | extern int pm_autosleep_set_state(suspend_state_t state); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #else /* !CONFIG_PM_AUTOSLEEP */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline int pm_autosleep_init(void) { return 0; } | 
					
						
							|  |  |  | static inline int pm_autosleep_lock(void) { return 0; } | 
					
						
							|  |  |  | static inline void pm_autosleep_unlock(void) {} | 
					
						
							|  |  |  | static inline suspend_state_t pm_autosleep_state(void) { return PM_SUSPEND_ON; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif /* !CONFIG_PM_AUTOSLEEP */
 | 
					
						
							| 
									
										
											  
											
												PM / Sleep: Add user space interface for manipulating wakeup sources, v3
Android allows user space to manipulate wakelocks using two
sysfs file located in /sys/power/, wake_lock and wake_unlock.
Writing a wakelock name and optionally a timeout to the wake_lock
file causes the wakelock whose name was written to be acquired (it
is created before is necessary), optionally with the given timeout.
Writing the name of a wakelock to wake_unlock causes that wakelock
to be released.
Implement an analogous interface for user space using wakeup sources.
Add the /sys/power/wake_lock and /sys/power/wake_unlock files
allowing user space to create, activate and deactivate wakeup
sources, such that writing a name and optionally a timeout to
wake_lock causes the wakeup source of that name to be activated,
optionally with the given timeout.  If that wakeup source doesn't
exist, it will be created and then activated.  Writing a name to
wake_unlock causes the wakeup source of that name, if there is one,
to be deactivated.  Wakeup sources created with the help of
wake_lock that haven't been used for more than 5 minutes are garbage
collected and destroyed.  Moreover, there can be only WL_NUMBER_LIMIT
wakeup sources created with the help of wake_lock present at a time.
The data type used to track wakeup sources created by user space is
called "struct wakelock" to indicate the origins of this feature.
This version of the patch includes an rbtree manipulation fix from John Stultz.
Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
Acked-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Reviewed-by: NeilBrown <neilb@suse.de>
											
										 
											2012-04-29 22:53:42 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef CONFIG_PM_WAKELOCKS
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* kernel/power/wakelock.c */ | 
					
						
							|  |  |  | extern ssize_t pm_show_wakelocks(char *buf, bool show_active); | 
					
						
							|  |  |  | extern int pm_wake_lock(const char *buf); | 
					
						
							|  |  |  | extern int pm_wake_unlock(const char *buf); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif /* !CONFIG_PM_WAKELOCKS */
 |