/* * This file is subject to the terms and conditions of the GNU General Public * License. See the file "COPYING" in the main directory of this archive * for more details. * * Copyright (C) 2001, 06 by Ralf Baechle (ralf@linux-mips.org) * Copyright (C) 2001 MIPS Technologies, Inc.
*/ #include <linux/kernel.h> #include <linux/export.h> #include <linux/pm.h> #include <linux/types.h> #include <linux/reboot.h> #include <linux/delay.h>
/* * Urgs ... Too many MIPS machines to handle this in a generic way. * So handle all using function pointers to machine specific * functions.
*/ void (*_machine_restart)(char *command); void (*_machine_halt)(void); void (*pm_power_off)(void);
EXPORT_SYMBOL(pm_power_off);
staticvoid machine_hang(void)
{ /* * We're hanging the system so we don't want to be interrupted anymore. * Any interrupt handlers that ran would at best be useless & at worst * go awry because the system isn't in a functional state.
*/
local_irq_disable();
/* * Mask all interrupts, giving us a better chance of remaining in the * low power wait state.
*/
clear_c0_status(ST0_IM);
while (true) { if (cpu_has_mips_r) { /* * We know that the wait instruction is supported so * make use of it directly, leaving interrupts * disabled.
*/ asmvolatile( ".set push\n\t" ".set " MIPS_ISA_ARCH_LEVEL "\n\t" "wait\n\t" ".set pop");
} elseif (cpu_wait) { /* * Try the cpu_wait() callback. This isn't ideal since * it'll re-enable interrupts, but that ought to be * harmless given that they're all masked.
*/
cpu_wait();
local_irq_disable();
} else { /* * We're going to burn some power running round the * loop, but we don't really have a choice. This isn't * a path we should expect to run for long during * typical use anyway.
*/
}
/* * In most modern MIPS CPUs interrupts will cause the wait * instruction to graduate even when disabled, and in some * cases even when masked. In order to prevent a timer * interrupt from continuously taking us out of the low power * wait state, we clear any pending timer interrupt here.
*/ if (cpu_has_counter)
write_c0_compare(0);
}
}
void machine_restart(char *command)
{ if (_machine_restart)
_machine_restart(command);
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.