2  * PXA250/210 Power Management Routines
 
   4  * Original code for the SA11x0:
 
   5  * Copyright (c) 2001 Cliff Brake <cbrake@accelent.com>
 
   7  * Modified for the PXA250 by Nicolas Pitre:
 
   8  * Copyright (c) 2002 Monta Vista Software, Inc.
 
  10  * This program is free software; you can redistribute it and/or
 
  11  * modify it under the terms of the GNU General Public License.
 
  13 #include <linux/init.h>
 
  14 #include <linux/module.h>
 
  15 #include <linux/suspend.h>
 
  16 #include <linux/errno.h>
 
  17 #include <linux/time.h>
 
  19 #include <mach/hardware.h>
 
  20 #include <asm/memory.h>
 
  21 #include <asm/system.h>
 
  23 #include <mach/pxa-regs.h>
 
  24 #include <mach/lubbock.h>
 
  25 #include <asm/mach/time.h>
 
  27 struct pxa_cpu_pm_fns *pxa_cpu_pm_fns;
 
  28 static unsigned long *sleep_save;
 
  30 int pxa_pm_enter(suspend_state_t state)
 
  32         unsigned long sleep_save_checksum = 0, checksum = 0;
 
  36         /* force any iWMMXt context to ram **/
 
  37         if (elf_hwcap & HWCAP_IWMMXT)
 
  38                 iwmmxt_task_disable(NULL);
 
  41         /* skip registers saving for standby */
 
  42         if (state != PM_SUSPEND_STANDBY) {
 
  43                 pxa_cpu_pm_fns->save(sleep_save);
 
  44                 /* before sleeping, calculate and save a checksum */
 
  45                 for (i = 0; i < pxa_cpu_pm_fns->save_count - 1; i++)
 
  46                         sleep_save_checksum += sleep_save[i];
 
  50         pxa_cpu_pm_fns->enter(state);
 
  53         if (state != PM_SUSPEND_STANDBY) {
 
  54                 /* after sleeping, validate the checksum */
 
  55                 for (i = 0; i < pxa_cpu_pm_fns->save_count - 1; i++)
 
  56                         checksum += sleep_save[i];
 
  58                 /* if invalid, display message and wait for a hardware reset */
 
  59                 if (checksum != sleep_save_checksum) {
 
  60 #ifdef CONFIG_ARCH_LUBBOCK
 
  61                         LUB_HEXLED = 0xbadbadc5;
 
  64                                 pxa_cpu_pm_fns->enter(state);
 
  66                 pxa_cpu_pm_fns->restore(sleep_save);
 
  69         pr_debug("*** made it back from resume\n");
 
  74 EXPORT_SYMBOL_GPL(pxa_pm_enter);
 
  76 unsigned long sleep_phys_sp(void *sp)
 
  78         return virt_to_phys(sp);
 
  81 static int pxa_pm_valid(suspend_state_t state)
 
  84                 return pxa_cpu_pm_fns->valid(state);
 
  89 static struct platform_suspend_ops pxa_pm_ops = {
 
  90         .valid          = pxa_pm_valid,
 
  91         .enter          = pxa_pm_enter,
 
  94 static int __init pxa_pm_init(void)
 
  96         if (!pxa_cpu_pm_fns) {
 
  97                 printk(KERN_ERR "no valid pxa_cpu_pm_fns defined\n");
 
 101         sleep_save = kmalloc(pxa_cpu_pm_fns->save_count * sizeof(unsigned long),
 
 104                 printk(KERN_ERR "failed to alloc memory for pm save\n");
 
 108         suspend_set_ops(&pxa_pm_ops);
 
 112 device_initcall(pxa_pm_init);