[PATCH 2/7] powerpc: add ibm,suspend-me rtas interface
Dave C Boutcher
sleddog at us.ibm.com
Thu Jan 12 12:27:01 EST 2006
Create a proc file that lets us call the ibm,suspend-me
RTAS call. This call has to be wrapped in some hypervisor
calls that synchronize all the CPUs, so it can't be done
from userland.
This depends on a prior patch to add additional hypervisor
call constants. Updated to reflect comments from Olof
Johansson
Signed-off-by: Dave Boutcher <sleddog at us.ibm.com>
arch/powerpc/platforms/pseries/Makefile | 2
arch/powerpc/platforms/pseries/sysstate.c | 157 ++++++++++++++++++++++++++++++
2 files changed, 158 insertions(+), 1 deletion(-)
diff -uNr linux-2.6.15-patched/arch/powerpc/platforms/pseries/Makefile linux-2.6.15-patched2/arch/powerpc/platforms/pseries/Makefile
--- linux-2.6.15-patched/arch/powerpc/platforms/pseries/Makefile 2006-01-11 17:46:57.000000000 -0600
+++ linux-2.6.15-patched2/arch/powerpc/platforms/pseries/Makefile 2006-01-11 17:50:21.000000000 -0600
@@ -1,5 +1,5 @@
obj-y := pci.o lpar.o hvCall.o nvram.o reconfig.o \
- setup.o iommu.o ras.o rtasd.o
+ setup.o iommu.o ras.o rtasd.o sysstate.o
obj-$(CONFIG_SMP) += smp.o
obj-$(CONFIG_IBMVIO) += vio.o
obj-$(CONFIG_XICS) += xics.o
diff -uNr linux-2.6.15-patched/arch/powerpc/platforms/pseries/sysstate.c linux-2.6.15-patched2/arch/powerpc/platforms/pseries/sysstate.c
--- linux-2.6.15-patched/arch/powerpc/platforms/pseries/sysstate.c 1969-12-31 18:00:00.000000000 -0600
+++ linux-2.6.15-patched2/arch/powerpc/platforms/pseries/sysstate.c 2006-01-11 17:50:21.000000000 -0600
@@ -0,0 +1,143 @@
+/*
+ * Logical Partition State
+ *
+ * Copyright (c) 2005 Dave Boutcher
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+ */
+
+/*
+ * This file handles the pSeries firmware supported "ibm,suspend-me" RTAS
+ * call. It can't be done from userland, since all of the CPUs must be first
+ * synchronized using the H_JOIN hypervisor call prior to making the
+ * ibm,suspend-me RTAS call.
+ *
+ * Note that this is completely different than normal Linux "software suspend".
+ * This is a hypervisor function driven as part of logical partitioning to give
+ * the hypervisor a "sync point" across the operating system.
+ */
+
+#include <linux/kernel.h>
+#include <linux/string.h>
+#include <linux/errno.h>
+#include <linux/init.h>
+#include <linux/smp.h>
+#include <linux/proc_fs.h>
+#include <asm/uaccess.h>
+
+#include <asm/rtas.h>
+
+static int suspend_me_tok = RTAS_UNKNOWN_SERVICE;
+
+static void cpu_suspend(void *info)
+{
+ long rc;
+ long flags;
+ long *waiting = (long *)info;
+ int status;
+
+ /*
+ * We use "waiting" to indicate our state. As long
+ * as it is >0, we are still trying to all join up.
+ * If it goes to 0, we have successfully joined up and
+ * one thread got H_Continue. If any error happens,
+ * we set it to <0;
+ */
+ local_irq_save(flags);
+ do {
+ rc = plpar_hcall_norets(H_JOIN);
+ smp_rmb();
+ } while (rc == H_Success && *waiting > 0);
+ if (rc == H_Success)
+ goto out;
+
+ if (rc == H_Continue) {
+ *waiting = 0;
+ if (suspend_me_tok != RTAS_UNKNOWN_SERVICE) {
+ status = rtas_call(suspend_me_tok, 0, 1, NULL);
+ if (status != 0)
+ printk(KERN_ERR
+ "RTAS ibm,suspend-me failed: %i\n",
+ status);
+ }
+ } else {
+ *waiting = -EBUSY;
+ printk(KERN_ERR "Error on H_Join hypervisor call\n");
+ }
+
+out:
+ local_irq_restore(flags);
+ return;
+}
+
+static int do_suspend(void)
+{
+ int i;
+ long waiting = 1;
+
+ /* Call function on all other CPUs */
+ if (on_each_cpu(cpu_suspend, &waiting, 1, 0))
+ waiting = -EINVAL;
+
+ if (waiting != 0)
+ printk(KERN_ERR "Error doing global join!\n");
+
+ /* Prod each CPU. This won't hurt, and will wake
+ * anyone we successfully put to sleep with H_Join
+ */
+ for_each_cpu(i)
+ plpar_hcall_norets(H_PROD,i);
+
+ return 0;
+}
+
+static int write_proc(struct file *file, const char __user *buffer,
+ unsigned long count, void *data)
+{
+ char command[32];
+
+ if (count > sizeof(command))
+ count = sizeof(command);
+
+ if (copy_from_user(command, buffer, count))
+ return -EFAULT;
+
+ command[7] = '\0';
+
+ if (strcmp(command, "suspend") == 0) {
+ do_suspend();
+ }
+ return count;
+}
+
+static int __init sys_state_init(void)
+{
+ struct proc_dir_entry *entry;
+ suspend_me_tok = rtas_token("ibm,suspend-me");
+ if (suspend_me_tok == RTAS_UNKNOWN_SERVICE)
+ return 0;
+
+ entry = create_proc_entry("ppc64/rtas/platform_state", S_IWUSR, NULL);
+ if (!entry) {
+ printk(KERN_ERR "Failed to create platform_state proc entry\n");
+ return 0;
+ }
+
+ entry->write_proc = write_proc;
+
+ return 0;
+}
+
+arch_initcall(sys_state_init);
--
Dave Boutcher
More information about the Linuxppc64-dev
mailing list