diff options
| author | Linus Torvalds <torvalds@linux-foundation.org> | 2009-06-16 11:48:13 -0700 | 
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2009-06-16 11:48:13 -0700 | 
| commit | 6a454f71d795368c00d9c329b60cc4d58929e7bc (patch) | |
| tree | f85a7ed30ba1d25606a22cc07e7383e73fc49972 /arch | |
| parent | d613839ef987d20f7c9347732b452efd921b97d9 (diff) | |
| parent | 155af2f95f905c830688dd0ca7c7cac4107334fd (diff) | |
Merge branch 'for-linus' of git://git390.marist.edu/pub/scm/linux-2.6
* 'for-linus' of git://git390.marist.edu/pub/scm/linux-2.6: (33 commits)
  [S390] s390: hibernation support for s390
  [S390] pm: dcssblk power management callbacks.
  [S390] pm: monreader power management callbacks.
  [S390] pm: monwriter power management callbacks.
  [S390] pm: memory hotplug power management callbacks
  [S390] pm: con3270 power management callbacks.
  [S390] pm: smsgiucv power management callbacks.
  [S390] pm: hvc_iucv power management callbacks
  [S390] PM: af_iucv power management callbacks.
  [S390] pm: netiucv power management callbacks.
  [S390] pm: iucv power management callbacks.
  [S390] iucv: establish reboot notifier
  [S390] pm: power management support for SCLP drivers.
  [S390] pm: tape power management callbacks
  [S390] pm: vmlogrdr power management callbacks
  [S390] pm: vmur driver power management callbacks
  [S390] pm: appldata power management callbacks
  [S390] pm: vmwatchdog power management callbacks.
  [S390] pm: zfcp driver power management callbacks
  [S390] pm: claw driver power management callbacks
  ...
Diffstat (limited to 'arch')
| -rw-r--r-- | arch/s390/Kconfig | 9 | ||||
| -rw-r--r-- | arch/s390/Makefile | 4 | ||||
| -rw-r--r-- | arch/s390/appldata/appldata_base.c | 119 | ||||
| -rw-r--r-- | arch/s390/include/asm/ccwdev.h | 19 | ||||
| -rw-r--r-- | arch/s390/include/asm/ccwgroup.h | 10 | ||||
| -rw-r--r-- | arch/s390/include/asm/suspend.h | 10 | ||||
| -rw-r--r-- | arch/s390/include/asm/system.h | 22 | ||||
| -rw-r--r-- | arch/s390/kernel/early.c | 6 | ||||
| -rw-r--r-- | arch/s390/kernel/mem_detect.c | 19 | ||||
| -rw-r--r-- | arch/s390/kernel/smp.c | 38 | ||||
| -rw-r--r-- | arch/s390/mm/pgtable.c | 19 | ||||
| -rw-r--r-- | arch/s390/power/Makefile | 8 | ||||
| -rw-r--r-- | arch/s390/power/suspend.c | 40 | ||||
| -rw-r--r-- | arch/s390/power/swsusp.c | 30 | ||||
| -rw-r--r-- | arch/s390/power/swsusp_64.c | 17 | ||||
| -rw-r--r-- | arch/s390/power/swsusp_asm64.S | 199 | 
16 files changed, 533 insertions, 36 deletions
| diff --git a/arch/s390/Kconfig b/arch/s390/Kconfig index 99dc3ded6b4..a14dba0e4d6 100644 --- a/arch/s390/Kconfig +++ b/arch/s390/Kconfig @@ -348,6 +348,9 @@ config ARCH_ENABLE_MEMORY_HOTPLUG  config ARCH_ENABLE_MEMORY_HOTREMOVE  	def_bool y +config ARCH_HIBERNATION_POSSIBLE +       def_bool y if 64BIT +  source "mm/Kconfig"  comment "I/O subsystem configuration" @@ -592,6 +595,12 @@ config SECCOMP  endmenu +menu "Power Management" + +source "kernel/power/Kconfig" + +endmenu +  source "net/Kconfig"  config PCMCIA diff --git a/arch/s390/Makefile b/arch/s390/Makefile index 578c61f15a4..0ff387cebf8 100644 --- a/arch/s390/Makefile +++ b/arch/s390/Makefile @@ -88,7 +88,9 @@ LDFLAGS_vmlinux := -e start  head-y		:= arch/s390/kernel/head.o arch/s390/kernel/init_task.o  core-y		+= arch/s390/mm/ arch/s390/kernel/ arch/s390/crypto/ \ -		   arch/s390/appldata/ arch/s390/hypfs/ arch/s390/kvm/ +		   arch/s390/appldata/ arch/s390/hypfs/ arch/s390/kvm/ \ +		   arch/s390/power/ +  libs-y		+= arch/s390/lib/  drivers-y	+= drivers/s390/  drivers-$(CONFIG_MATHEMU) += arch/s390/math-emu/ diff --git a/arch/s390/appldata/appldata_base.c b/arch/s390/appldata/appldata_base.c index 1dfc7100c7e..264528e4f58 100644 --- a/arch/s390/appldata/appldata_base.c +++ b/arch/s390/appldata/appldata_base.c @@ -5,7 +5,7 @@   * Exports appldata_register_ops() and appldata_unregister_ops() for the   * data gathering modules.   * - * Copyright IBM Corp. 2003, 2008 + * Copyright IBM Corp. 2003, 2009   *   * Author: Gerald Schaefer <gerald.schaefer@de.ibm.com>   */ @@ -26,6 +26,8 @@  #include <linux/notifier.h>  #include <linux/cpu.h>  #include <linux/workqueue.h> +#include <linux/suspend.h> +#include <linux/platform_device.h>  #include <asm/appldata.h>  #include <asm/timer.h>  #include <asm/uaccess.h> @@ -41,6 +43,9 @@  #define TOD_MICRO	0x01000			/* nr. of TOD clock units  						   for 1 microsecond */ + +static struct platform_device *appldata_pdev; +  /*   * /proc entries (sysctl)   */ @@ -86,6 +91,7 @@ static atomic_t appldata_expire_count = ATOMIC_INIT(0);  static DEFINE_SPINLOCK(appldata_timer_lock);  static int appldata_interval = APPLDATA_CPU_INTERVAL;  static int appldata_timer_active; +static int appldata_timer_suspended = 0;  /*   * Work queue @@ -475,6 +481,93 @@ void appldata_unregister_ops(struct appldata_ops *ops)  /********************** module-ops management <END> **************************/ +/**************************** suspend / resume *******************************/ +static int appldata_freeze(struct device *dev) +{ +	struct appldata_ops *ops; +	int rc; +	struct list_head *lh; + +	get_online_cpus(); +	spin_lock(&appldata_timer_lock); +	if (appldata_timer_active) { +		__appldata_vtimer_setup(APPLDATA_DEL_TIMER); +		appldata_timer_suspended = 1; +	} +	spin_unlock(&appldata_timer_lock); +	put_online_cpus(); + +	mutex_lock(&appldata_ops_mutex); +	list_for_each(lh, &appldata_ops_list) { +		ops = list_entry(lh, struct appldata_ops, list); +		if (ops->active == 1) { +			rc = appldata_diag(ops->record_nr, APPLDATA_STOP_REC, +					(unsigned long) ops->data, ops->size, +					ops->mod_lvl); +			if (rc != 0) +				pr_err("Stopping the data collection for %s " +				       "failed with rc=%d\n", ops->name, rc); +		} +	} +	mutex_unlock(&appldata_ops_mutex); +	return 0; +} + +static int appldata_restore(struct device *dev) +{ +	struct appldata_ops *ops; +	int rc; +	struct list_head *lh; + +	get_online_cpus(); +	spin_lock(&appldata_timer_lock); +	if (appldata_timer_suspended) { +		__appldata_vtimer_setup(APPLDATA_ADD_TIMER); +		appldata_timer_suspended = 0; +	} +	spin_unlock(&appldata_timer_lock); +	put_online_cpus(); + +	mutex_lock(&appldata_ops_mutex); +	list_for_each(lh, &appldata_ops_list) { +		ops = list_entry(lh, struct appldata_ops, list); +		if (ops->active == 1) { +			ops->callback(ops->data);	// init record +			rc = appldata_diag(ops->record_nr, +					APPLDATA_START_INTERVAL_REC, +					(unsigned long) ops->data, ops->size, +					ops->mod_lvl); +			if (rc != 0) { +				pr_err("Starting the data collection for %s " +				       "failed with rc=%d\n", ops->name, rc); +			} +		} +	} +	mutex_unlock(&appldata_ops_mutex); +	return 0; +} + +static int appldata_thaw(struct device *dev) +{ +	return appldata_restore(dev); +} + +static struct dev_pm_ops appldata_pm_ops = { +	.freeze		= appldata_freeze, +	.thaw		= appldata_thaw, +	.restore	= appldata_restore, +}; + +static struct platform_driver appldata_pdrv = { +	.driver = { +		.name	= "appldata", +		.owner	= THIS_MODULE, +		.pm	= &appldata_pm_ops, +	}, +}; +/************************* suspend / resume <END> ****************************/ + +  /******************************* init / exit *********************************/  static void __cpuinit appldata_online_cpu(int cpu) @@ -531,11 +624,23 @@ static struct notifier_block __cpuinitdata appldata_nb = {   */  static int __init appldata_init(void)  { -	int i; +	int i, rc; + +	rc = platform_driver_register(&appldata_pdrv); +	if (rc) +		return rc; +	appldata_pdev = platform_device_register_simple("appldata", -1, NULL, +							0); +	if (IS_ERR(appldata_pdev)) { +		rc = PTR_ERR(appldata_pdev); +		goto out_driver; +	}  	appldata_wq = create_singlethread_workqueue("appldata"); -	if (!appldata_wq) -		return -ENOMEM; +	if (!appldata_wq) { +		rc = -ENOMEM; +		goto out_device; +	}  	get_online_cpus();  	for_each_online_cpu(i) @@ -547,6 +652,12 @@ static int __init appldata_init(void)  	appldata_sysctl_header = register_sysctl_table(appldata_dir_table);  	return 0; + +out_device: +	platform_device_unregister(appldata_pdev); +out_driver: +	platform_driver_unregister(&appldata_pdrv); +	return rc;  }  __initcall(appldata_init); diff --git a/arch/s390/include/asm/ccwdev.h b/arch/s390/include/asm/ccwdev.h index ba007d8df94..2a541955117 100644 --- a/arch/s390/include/asm/ccwdev.h +++ b/arch/s390/include/asm/ccwdev.h @@ -1,11 +1,9 @@  /* - *  include/asm-s390/ccwdev.h - *  include/asm-s390x/ccwdev.h + * Copyright  IBM Corp. 2002, 2009   * - *    Copyright (C) 2002 IBM Deutschland Entwicklung GmbH, IBM Corporation - *    Author(s): Arnd Bergmann <arndb@de.ibm.com> + * Author(s): Arnd Bergmann <arndb@de.ibm.com>   * - *  Interface for CCW device drivers + * Interface for CCW device drivers   */  #ifndef _S390_CCWDEV_H_  #define _S390_CCWDEV_H_ @@ -104,6 +102,11 @@ struct ccw_device {   * @set_offline: called when setting device offline   * @notify: notify driver of device state changes   * @shutdown: called at device shutdown + * @prepare: prepare for pm state transition + * @complete: undo work done in @prepare + * @freeze: callback for freezing during hibernation snapshotting + * @thaw: undo work done in @freeze + * @restore: callback for restoring after hibernation   * @driver: embedded device driver structure   * @name: device driver name   */ @@ -116,6 +119,11 @@ struct ccw_driver {  	int (*set_offline) (struct ccw_device *);  	int (*notify) (struct ccw_device *, int);  	void (*shutdown) (struct ccw_device *); +	int (*prepare) (struct ccw_device *); +	void (*complete) (struct ccw_device *); +	int (*freeze)(struct ccw_device *); +	int (*thaw) (struct ccw_device *); +	int (*restore)(struct ccw_device *);  	struct device_driver driver;  	char *name;  }; @@ -184,6 +192,7 @@ extern void ccw_device_get_id(struct ccw_device *, struct ccw_dev_id *);  #define to_ccwdrv(n) container_of(n, struct ccw_driver, driver)  extern struct ccw_device *ccw_device_probe_console(void); +extern int ccw_device_force_console(void);  // FIXME: these have to go  extern int _ccw_device_get_subchannel_number(struct ccw_device *); diff --git a/arch/s390/include/asm/ccwgroup.h b/arch/s390/include/asm/ccwgroup.h index a27f68985a7..c79c1e787b8 100644 --- a/arch/s390/include/asm/ccwgroup.h +++ b/arch/s390/include/asm/ccwgroup.h @@ -38,6 +38,11 @@ struct ccwgroup_device {   * @set_online: function called when device is set online   * @set_offline: function called when device is set offline   * @shutdown: function called when device is shut down + * @prepare: prepare for pm state transition + * @complete: undo work done in @prepare + * @freeze: callback for freezing during hibernation snapshotting + * @thaw: undo work done in @freeze + * @restore: callback for restoring after hibernation   * @driver: embedded driver structure   */  struct ccwgroup_driver { @@ -51,6 +56,11 @@ struct ccwgroup_driver {  	int (*set_online) (struct ccwgroup_device *);  	int (*set_offline) (struct ccwgroup_device *);  	void (*shutdown)(struct ccwgroup_device *); +	int (*prepare) (struct ccwgroup_device *); +	void (*complete) (struct ccwgroup_device *); +	int (*freeze)(struct ccwgroup_device *); +	int (*thaw) (struct ccwgroup_device *); +	int (*restore)(struct ccwgroup_device *);  	struct device_driver driver;  }; diff --git a/arch/s390/include/asm/suspend.h b/arch/s390/include/asm/suspend.h new file mode 100644 index 00000000000..dc75c616eaf --- /dev/null +++ b/arch/s390/include/asm/suspend.h @@ -0,0 +1,10 @@ +#ifndef __ASM_S390_SUSPEND_H +#define __ASM_S390_SUSPEND_H + +static inline int arch_prepare_suspend(void) +{ +	return 0; +} + +#endif + diff --git a/arch/s390/include/asm/system.h b/arch/s390/include/asm/system.h index 3a8b26eb1f2..4fb83c1cdb7 100644 --- a/arch/s390/include/asm/system.h +++ b/arch/s390/include/asm/system.h @@ -1,11 +1,7 @@  /* - *  include/asm-s390/system.h + * Copyright IBM Corp. 1999, 2009   * - *  S390 version - *    Copyright (C) 1999 IBM Deutschland Entwicklung GmbH, IBM Corporation - *    Author(s): Martin Schwidefsky (schwidefsky@de.ibm.com), - * - *  Derived from "include/asm-i386/system.h" + * Author(s): Martin Schwidefsky <schwidefsky@de.ibm.com>   */  #ifndef __ASM_SYSTEM_H @@ -469,6 +465,20 @@ extern psw_t sysc_restore_trace_psw;  extern psw_t io_restore_trace_psw;  #endif +static inline int tprot(unsigned long addr) +{ +	int rc = -EFAULT; + +	asm volatile( +		"	tprot	0(%1),0\n" +		"0:	ipm	%0\n" +		"	srl	%0,28\n" +		"1:\n" +		EX_TABLE(0b,1b) +		: "+d" (rc) : "a" (addr) : "cc"); +	return rc; +} +  #endif /* __KERNEL__ */  #endif diff --git a/arch/s390/kernel/early.c b/arch/s390/kernel/early.c index fb263736826..f9b144049dc 100644 --- a/arch/s390/kernel/early.c +++ b/arch/s390/kernel/early.c @@ -1,7 +1,7 @@  /*   *  arch/s390/kernel/early.c   * - *    Copyright IBM Corp. 2007 + *    Copyright IBM Corp. 2007, 2009   *    Author(s): Hongjie Yang <hongjie@us.ibm.com>,   *		 Heiko Carstens <heiko.carstens@de.ibm.com>   */ @@ -210,7 +210,7 @@ static noinline __init void detect_machine_type(void)  		machine_flags |= MACHINE_FLAG_VM;  } -static __init void early_pgm_check_handler(void) +static void early_pgm_check_handler(void)  {  	unsigned long addr;  	const struct exception_table_entry *fixup; @@ -222,7 +222,7 @@ static __init void early_pgm_check_handler(void)  	S390_lowcore.program_old_psw.addr = fixup->fixup | PSW_ADDR_AMODE;  } -static noinline __init void setup_lowcore_early(void) +void setup_lowcore_early(void)  {  	psw_t psw; diff --git a/arch/s390/kernel/mem_detect.c b/arch/s390/kernel/mem_detect.c index 9872999c66d..559af0d0787 100644 --- a/arch/s390/kernel/mem_detect.c +++ b/arch/s390/kernel/mem_detect.c @@ -1,6 +1,7 @@  /* - *    Copyright IBM Corp. 2008 - *    Author(s): Heiko Carstens <heiko.carstens@de.ibm.com> + * Copyright IBM Corp. 2008, 2009 + * + * Author(s): Heiko Carstens <heiko.carstens@de.ibm.com>   */  #include <linux/kernel.h> @@ -9,20 +10,6 @@  #include <asm/sclp.h>  #include <asm/setup.h> -static inline int tprot(unsigned long addr) -{ -	int rc = -EFAULT; - -	asm volatile( -		"	tprot	0(%1),0\n" -		"0:	ipm	%0\n" -		"	srl	%0,28\n" -		"1:\n" -		EX_TABLE(0b,1b) -		: "+d" (rc) : "a" (addr) : "cc"); -	return rc; -} -  #define ADDR2G (1ULL << 31)  static void find_memory_chunks(struct mem_chunk chunk[]) diff --git a/arch/s390/kernel/smp.c b/arch/s390/kernel/smp.c index cc8c484984e..fd8e3111a4e 100644 --- a/arch/s390/kernel/smp.c +++ b/arch/s390/kernel/smp.c @@ -1,7 +1,7 @@  /*   *  arch/s390/kernel/smp.c   * - *    Copyright IBM Corp. 1999,2007 + *    Copyright IBM Corp. 1999, 2009   *    Author(s): Denis Joseph Barrow (djbarrow@de.ibm.com,barrow_dj@yahoo.com),   *		 Martin Schwidefsky (schwidefsky@de.ibm.com)   *		 Heiko Carstens (heiko.carstens@de.ibm.com) @@ -1031,6 +1031,42 @@ out:  static SYSDEV_CLASS_ATTR(dispatching, 0644, dispatching_show,  			 dispatching_store); +/* + * If the resume kernel runs on another cpu than the suspended kernel, + * we have to switch the cpu IDs in the logical map. + */ +void smp_switch_boot_cpu_in_resume(u32 resume_phys_cpu_id, +				   struct _lowcore *suspend_lowcore) +{ +	int cpu, suspend_cpu_id, resume_cpu_id; +	u32 suspend_phys_cpu_id; + +	suspend_phys_cpu_id = __cpu_logical_map[suspend_lowcore->cpu_nr]; +	suspend_cpu_id = suspend_lowcore->cpu_nr; + +	for_each_present_cpu(cpu) { +		if (__cpu_logical_map[cpu] == resume_phys_cpu_id) { +			resume_cpu_id = cpu; +			goto found; +		} +	} +	panic("Could not find resume cpu in logical map.\n"); + +found: +	printk("Resume  cpu ID: %i/%i\n", resume_phys_cpu_id, resume_cpu_id); +	printk("Suspend cpu ID: %i/%i\n", suspend_phys_cpu_id, suspend_cpu_id); + +	__cpu_logical_map[resume_cpu_id] = suspend_phys_cpu_id; +	__cpu_logical_map[suspend_cpu_id] = resume_phys_cpu_id; + +	lowcore_ptr[suspend_cpu_id]->cpu_addr = resume_phys_cpu_id; +} + +u32 smp_get_phys_cpu_id(void) +{ +	return __cpu_logical_map[smp_processor_id()]; +} +  static int __init topology_init(void)  {  	int cpu; diff --git a/arch/s390/mm/pgtable.c b/arch/s390/mm/pgtable.c index 4ca8e826bf3..56566720798 100644 --- a/arch/s390/mm/pgtable.c +++ b/arch/s390/mm/pgtable.c @@ -313,3 +313,22 @@ int s390_enable_sie(void)  	return 0;  }  EXPORT_SYMBOL_GPL(s390_enable_sie); + +#ifdef CONFIG_DEBUG_PAGEALLOC +#ifdef CONFIG_HIBERNATION +bool kernel_page_present(struct page *page) +{ +	unsigned long addr; +	int cc; + +	addr = page_to_phys(page); +	asm("lra %1,0(%1)\n" +	    "ipm %0\n" +	    "srl %0,28" +	    :"=d"(cc),"+a"(addr)::"cc"); +	return cc == 0; +} + +#endif /* CONFIG_HIBERNATION */ +#endif /* CONFIG_DEBUG_PAGEALLOC */ + diff --git a/arch/s390/power/Makefile b/arch/s390/power/Makefile new file mode 100644 index 00000000000..973bb45a8fe --- /dev/null +++ b/arch/s390/power/Makefile @@ -0,0 +1,8 @@ +# +# Makefile for s390 PM support +# + +obj-$(CONFIG_HIBERNATION) += suspend.o +obj-$(CONFIG_HIBERNATION) += swsusp.o +obj-$(CONFIG_HIBERNATION) += swsusp_64.o +obj-$(CONFIG_HIBERNATION) += swsusp_asm64.o diff --git a/arch/s390/power/suspend.c b/arch/s390/power/suspend.c new file mode 100644 index 00000000000..b3351eceebb --- /dev/null +++ b/arch/s390/power/suspend.c @@ -0,0 +1,40 @@ +/* + * Suspend support specific for s390. + * + * Copyright IBM Corp. 2009 + * + * Author(s): Hans-Joachim Picht <hans@linux.vnet.ibm.com> + */ + +#include <linux/mm.h> +#include <linux/suspend.h> +#include <linux/reboot.h> +#include <linux/pfn.h> +#include <asm/sections.h> +#include <asm/ipl.h> + +/* + * References to section boundaries + */ +extern const void __nosave_begin, __nosave_end; + +/* + *  check if given pfn is in the 'nosave' or in the read only NSS section + */ +int pfn_is_nosave(unsigned long pfn) +{ +	unsigned long nosave_begin_pfn = __pa(&__nosave_begin) >> PAGE_SHIFT; +	unsigned long nosave_end_pfn = PAGE_ALIGN(__pa(&__nosave_end)) +					>> PAGE_SHIFT; +	unsigned long eshared_pfn = PFN_DOWN(__pa(&_eshared)) - 1; +	unsigned long stext_pfn = PFN_DOWN(__pa(&_stext)); + +	if (pfn >= nosave_begin_pfn && pfn < nosave_end_pfn) +		return 1; +	if (pfn >= stext_pfn && pfn <= eshared_pfn) { +		if (ipl_info.type == IPL_TYPE_NSS) +			return 1; +	} else if ((tprot(pfn * PAGE_SIZE) && pfn > 0)) +		return 1; +	return 0; +} diff --git a/arch/s390/power/swsusp.c b/arch/s390/power/swsusp.c new file mode 100644 index 00000000000..e6a4fe9f5f2 --- /dev/null +++ b/arch/s390/power/swsusp.c @@ -0,0 +1,30 @@ +/* + * Support for suspend and resume on s390 + * + * Copyright IBM Corp. 2009 + * + * Author(s): Hans-Joachim Picht <hans@linux.vnet.ibm.com> + * + */ + + +/* + * save CPU registers before creating a hibernation image and before + * restoring the memory state from it + */ +void save_processor_state(void) +{ +	/* implentation contained in the +	 * swsusp_arch_suspend function +	 */ +} + +/* + * restore the contents of CPU registers + */ +void restore_processor_state(void) +{ +	/* implentation contained in the +	 * swsusp_arch_resume function +	 */ +} diff --git a/arch/s390/power/swsusp_64.c b/arch/s390/power/swsusp_64.c new file mode 100644 index 00000000000..9516a517d72 --- /dev/null +++ b/arch/s390/power/swsusp_64.c @@ -0,0 +1,17 @@ +/* + * Support for suspend and resume on s390 + * + * Copyright IBM Corp. 2009 + * + * Author(s): Hans-Joachim Picht <hans@linux.vnet.ibm.com> + * + */ + +#include <asm/system.h> +#include <linux/interrupt.h> + +void do_after_copyback(void) +{ +	mb(); +} + diff --git a/arch/s390/power/swsusp_asm64.S b/arch/s390/power/swsusp_asm64.S new file mode 100644 index 00000000000..3c74e7d827c --- /dev/null +++ b/arch/s390/power/swsusp_asm64.S @@ -0,0 +1,199 @@ +/* + * S390 64-bit swsusp implementation + * + * Copyright IBM Corp. 2009 + * + * Author(s): Hans-Joachim Picht <hans@linux.vnet.ibm.com> + *	      Michael Holzheu <holzheu@linux.vnet.ibm.com> + */ + +#include <asm/page.h> +#include <asm/ptrace.h> +#include <asm/asm-offsets.h> + +/* + * Save register context in absolute 0 lowcore and call swsusp_save() to + * create in-memory kernel image. The context is saved in the designated + * "store status" memory locations (see POP). + * We return from this function twice. The first time during the suspend to + * disk process. The second time via the swsusp_arch_resume() function + * (see below) in the resume process. + * This function runs with disabled interrupts. + */ +	.section .text +	.align	2 +	.globl swsusp_arch_suspend +swsusp_arch_suspend: +	stmg	%r6,%r15,__SF_GPRS(%r15) +	lgr	%r1,%r15 +	aghi	%r15,-STACK_FRAME_OVERHEAD +	stg	%r1,__SF_BACKCHAIN(%r15) + +	/* Deactivate DAT */ +	stnsm	__SF_EMPTY(%r15),0xfb + +	/* Switch off lowcore protection */ +	stctg	%c0,%c0,__SF_EMPTY(%r15) +	ni	__SF_EMPTY+4(%r15),0xef +	lctlg	%c0,%c0,__SF_EMPTY(%r15) + +	/* Store prefix register on stack */ +	stpx	__SF_EMPTY(%r15) + +	/* Setup base register for lowcore (absolute 0) */ +	llgf	%r1,__SF_EMPTY(%r15) + +	/* Get pointer to save area */ +	aghi	%r1,0x1000 + +	/* Store registers */ +	mvc	0x318(4,%r1),__SF_EMPTY(%r15)	/* move prefix to lowcore */ +	stfpc	0x31c(%r1)			/* store fpu control */ +	std	0,0x200(%r1)			/* store f0 */ +	std	1,0x208(%r1)			/* store f1 */ +	std	2,0x210(%r1)			/* store f2 */ +	std	3,0x218(%r1)			/* store f3 */ +	std	4,0x220(%r1)			/* store f4 */ +	std	5,0x228(%r1)			/* store f5 */ +	std	6,0x230(%r1)			/* store f6 */ +	std	7,0x238(%r1)			/* store f7 */ +	std	8,0x240(%r1)			/* store f8 */ +	std	9,0x248(%r1)			/* store f9 */ +	std	10,0x250(%r1)			/* store f10 */ +	std	11,0x258(%r1)			/* store f11 */ +	std	12,0x260(%r1)			/* store f12 */ +	std	13,0x268(%r1)			/* store f13 */ +	std	14,0x270(%r1)			/* store f14 */ +	std	15,0x278(%r1)			/* store f15 */ +	stam	%a0,%a15,0x340(%r1)		/* store access registers */ +	stctg	%c0,%c15,0x380(%r1)		/* store control registers */ +	stmg	%r0,%r15,0x280(%r1)		/* store general registers */ + +	stpt	0x328(%r1)			/* store timer */ +	stckc	0x330(%r1)			/* store clock comparator */ + +	/* Activate DAT */ +	stosm	__SF_EMPTY(%r15),0x04 + +	/* Set prefix page to zero */ +	xc	__SF_EMPTY(4,%r15),__SF_EMPTY(%r15) +	spx	__SF_EMPTY(%r15) + +	/* Setup lowcore */ +	brasl	%r14,setup_lowcore_early + +	/* Save image */ +	brasl	%r14,swsusp_save + +	/* Switch on lowcore protection */ +	stctg	%c0,%c0,__SF_EMPTY(%r15) +	oi	__SF_EMPTY+4(%r15),0x10 +	lctlg	%c0,%c0,__SF_EMPTY(%r15) + +	/* Restore prefix register and return */ +	lghi	%r1,0x1000 +	spx	0x318(%r1) +	lmg	%r6,%r15,STACK_FRAME_OVERHEAD + __SF_GPRS(%r15) +	lghi	%r2,0 +	br	%r14 + +/* + * Restore saved memory image to correct place and restore register context. + * Then we return to the function that called swsusp_arch_suspend(). + * swsusp_arch_resume() runs with disabled interrupts. + */ +	.globl swsusp_arch_resume +swsusp_arch_resume: +	stmg	%r6,%r15,__SF_GPRS(%r15) +	lgr	%r1,%r15 +	aghi	%r15,-STACK_FRAME_OVERHEAD +	stg	%r1,__SF_BACKCHAIN(%r15) + +	/* Save boot cpu number */ +	brasl	%r14,smp_get_phys_cpu_id +	lgr	%r10,%r2 + +	/* Deactivate DAT */ +	stnsm	__SF_EMPTY(%r15),0xfb + +	/* Switch off lowcore protection */ +	stctg	%c0,%c0,__SF_EMPTY(%r15) +	ni	__SF_EMPTY+4(%r15),0xef +	lctlg	%c0,%c0,__SF_EMPTY(%r15) + +	/* Set prefix page to zero */ +	xc	__SF_EMPTY(4,%r15),__SF_EMPTY(%r15) +	spx	__SF_EMPTY(%r15) + +	/* Restore saved image */ +	larl	%r1,restore_pblist +	lg	%r1,0(%r1) +	ltgr	%r1,%r1 +	jz	2f +0: +	lg	%r2,8(%r1) +	lg	%r4,0(%r1) +	lghi	%r3,PAGE_SIZE +	lghi	%r5,PAGE_SIZE +1: +	mvcle	%r2,%r4,0 +	jo	1b +	lg	%r1,16(%r1) +	ltgr	%r1,%r1 +	jnz	0b +2: +	ptlb				/* flush tlb */ + +	/* Restore registers */ +	lghi	%r13,0x1000		/* %r1 = pointer to save arae */ + +	spt	0x328(%r13)		/* reprogram timer */ +	//sckc	0x330(%r13)		/* set clock comparator */ + +	lctlg	%c0,%c15,0x380(%r13)	/* load control registers */ +	lam	%a0,%a15,0x340(%r13)	/* load access registers */ + +	lfpc	0x31c(%r13)		/* load fpu control */ +	ld	0,0x200(%r13)		/* load f0 */ +	ld	1,0x208(%r13)		/* load f1 */ +	ld	2,0x210(%r13)		/* load f2 */ +	ld	3,0x218(%r13)		/* load f3 */ +	ld	4,0x220(%r13)		/* load f4 */ +	ld	5,0x228(%r13)		/* load f5 */ +	ld	6,0x230(%r13)		/* load f6 */ +	ld	7,0x238(%r13)		/* load f7 */ +	ld	8,0x240(%r13)		/* load f8 */ +	ld	9,0x248(%r13)		/* load f9 */ +	ld	10,0x250(%r13)		/* load f10 */ +	ld	11,0x258(%r13)		/* load f11 */ +	ld	12,0x260(%r13)		/* load f12 */ +	ld	13,0x268(%r13)		/* load f13 */ +	ld	14,0x270(%r13)		/* load f14 */ +	ld	15,0x278(%r13)		/* load f15 */ + +	/* Load old stack */ +	lg	%r15,0x2f8(%r13) + +	/* Pointer to save arae */ +	lghi	%r13,0x1000 + +	/* Switch CPUs */ +	lgr	%r2,%r10		/* get cpu id */ +	llgf	%r3,0x318(%r13) +	brasl	%r14,smp_switch_boot_cpu_in_resume + +	/* Restore prefix register */ +	spx	0x318(%r13) + +	/* Switch on lowcore protection */ +	stctg	%c0,%c0,__SF_EMPTY(%r15) +	oi	__SF_EMPTY+4(%r15),0x10 +	lctlg	%c0,%c0,__SF_EMPTY(%r15) + +	/* Activate DAT */ +	stosm	__SF_EMPTY(%r15),0x04 + +	/* Return 0 */ +	lmg	%r6,%r15,STACK_FRAME_OVERHEAD + __SF_GPRS(%r15) +	lghi	%r2,0 +	br	%r14 | 
