mirror of
				https://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson
				synced 2025-10-25 17:24:48 +00:00 
			
		
		
		
	 c75f902b93
			
		
	
	
		c75f902b93
		
	
	
	
	
		
			
			It's possibly that we get an reset requestion when interrupts are disabled. (For example an oops in an interrupt handler). Therefor, we can't call ioremap in the reset function. Moving the ioremap of the registers we need access to an arch_initcall helps the problem. However we still have a window between boot and the arch_initcall in which the register pointer will not be setup and thus we spin if the reset function is called. If one needs to ensure even this case is covered, look at use of the watchdog provided on 83xx to reset the processor. Signed-off-by: Kumar Gala <galak@kernel.crashing.org>
		
			
				
	
	
		
			68 lines
		
	
	
		
			1.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			68 lines
		
	
	
		
			1.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * misc setup functions for MPC83xx
 | |
|  *
 | |
|  * Maintainer: Kumar Gala <galak@kernel.crashing.org>
 | |
|  *
 | |
|  * 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.
 | |
|  */
 | |
| 
 | |
| #include <linux/stddef.h>
 | |
| #include <linux/kernel.h>
 | |
| 
 | |
| #include <asm/io.h>
 | |
| #include <asm/hw_irq.h>
 | |
| #include <sysdev/fsl_soc.h>
 | |
| 
 | |
| #include "mpc83xx.h"
 | |
| 
 | |
| static __be32 __iomem *restart_reg_base;
 | |
| 
 | |
| static int __init mpc83xx_restart_init(void)
 | |
| {
 | |
| 	/* map reset restart_reg_baseister space */
 | |
| 	restart_reg_base = ioremap(get_immrbase() + 0x900, 0xff);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| arch_initcall(mpc83xx_restart_init);
 | |
| 
 | |
| void mpc83xx_restart(char *cmd)
 | |
| {
 | |
| #define RST_OFFSET	0x00000900
 | |
| #define RST_PROT_REG	0x00000018
 | |
| #define RST_CTRL_REG	0x0000001c
 | |
| 
 | |
| 	local_irq_disable();
 | |
| 
 | |
| 	if (restart_reg_base) {
 | |
| 		/* enable software reset "RSTE" */
 | |
| 		out_be32(restart_reg_base + (RST_PROT_REG >> 2), 0x52535445);
 | |
| 
 | |
| 		/* set software hard reset */
 | |
| 		out_be32(restart_reg_base + (RST_CTRL_REG >> 2), 0x2);
 | |
| 	} else {
 | |
| 		printk (KERN_EMERG "Error: Restart registers not mapped, spinning!\n");
 | |
| 	}
 | |
| 
 | |
| 	for (;;) ;
 | |
| }
 | |
| 
 | |
| long __init mpc83xx_time_init(void)
 | |
| {
 | |
| #define SPCR_OFFSET	0x00000110
 | |
| #define SPCR_TBEN	0x00400000
 | |
| 	__be32 __iomem *spcr = ioremap(get_immrbase() + SPCR_OFFSET, 4);
 | |
| 	__be32 tmp;
 | |
| 
 | |
| 	tmp = in_be32(spcr);
 | |
| 	out_be32(spcr, tmp | SPCR_TBEN);
 | |
| 
 | |
| 	iounmap(spcr);
 | |
| 
 | |
| 	return 0;
 | |
| }
 |