Display the total time when kernel resumes normal from standby or suspend to mem mode. Signed-off-by: Sonic Zhang <sonic.zhang@analog.com> Signed-off-by: Bob Liu <lliubbo@gmail.com>
		
			
				
	
	
		
			320 lines
		
	
	
	
		
			5.6 KiB
			
		
	
	
	
		
			ArmAsm
		
	
	
	
	
	
			
		
		
	
	
			320 lines
		
	
	
	
		
			5.6 KiB
			
		
	
	
	
		
			ArmAsm
		
	
	
	
	
	
/*
 | 
						|
 * Copyright 2004-2008 Analog Devices Inc.
 | 
						|
 *
 | 
						|
 * Licensed under the GPL-2 or later.
 | 
						|
 */
 | 
						|
 | 
						|
#include <linux/linkage.h>
 | 
						|
#include <asm/blackfin.h>
 | 
						|
#include <mach/irq.h>
 | 
						|
#include <asm/dpmc.h>
 | 
						|
 | 
						|
.section .l1.text
 | 
						|
ENTRY(_sleep_mode)
 | 
						|
	[--SP] = (R7:4, P5:3);
 | 
						|
	[--SP] = RETS;
 | 
						|
 | 
						|
	call _set_sic_iwr;
 | 
						|
 | 
						|
	P0.H = hi(PLL_CTL);
 | 
						|
	P0.L = lo(PLL_CTL);
 | 
						|
	R1 = W[P0](z);
 | 
						|
	BITSET (R1, 3);
 | 
						|
	W[P0] = R1.L;
 | 
						|
 | 
						|
	CLI R2;
 | 
						|
	SSYNC;
 | 
						|
	IDLE;
 | 
						|
	STI R2;
 | 
						|
 | 
						|
	call _test_pll_locked;
 | 
						|
 | 
						|
	R0 = IWR_ENABLE(0);
 | 
						|
	R1 = IWR_DISABLE_ALL;
 | 
						|
	R2 = IWR_DISABLE_ALL;
 | 
						|
 | 
						|
	call _set_sic_iwr;
 | 
						|
 | 
						|
	P0.H = hi(PLL_CTL);
 | 
						|
	P0.L = lo(PLL_CTL);
 | 
						|
	R7 = w[p0](z);
 | 
						|
	BITCLR (R7, 3);
 | 
						|
	BITCLR (R7, 5);
 | 
						|
	w[p0] = R7.L;
 | 
						|
	IDLE;
 | 
						|
 | 
						|
	bfin_init_pm_bench_cycles;
 | 
						|
 | 
						|
	call _test_pll_locked;
 | 
						|
 | 
						|
	RETS = [SP++];
 | 
						|
	(R7:4, P5:3) = [SP++];
 | 
						|
	RTS;
 | 
						|
ENDPROC(_sleep_mode)
 | 
						|
 | 
						|
/*
 | 
						|
 * This func never returns as it puts the part into hibernate, and
 | 
						|
 * is only called from do_hibernate, so we don't bother saving or
 | 
						|
 * restoring any of the normal C runtime state.  When we wake up,
 | 
						|
 * the entry point will be in do_hibernate and not here.
 | 
						|
 *
 | 
						|
 * We accept just one argument -- the value to write to VR_CTL.
 | 
						|
 */
 | 
						|
 | 
						|
ENTRY(_hibernate_mode)
 | 
						|
	/* Save/setup the regs we need early for minor pipeline optimization */
 | 
						|
	R4 = R0;
 | 
						|
 | 
						|
	P3.H = hi(VR_CTL);
 | 
						|
	P3.L = lo(VR_CTL);
 | 
						|
	/* Disable all wakeup sources */
 | 
						|
	R0 = IWR_DISABLE_ALL;
 | 
						|
	R1 = IWR_DISABLE_ALL;
 | 
						|
	R2 = IWR_DISABLE_ALL;
 | 
						|
	call _set_sic_iwr;
 | 
						|
	call _set_dram_srfs;
 | 
						|
	SSYNC;
 | 
						|
 | 
						|
	/* Finally, we climb into our cave to hibernate */
 | 
						|
	W[P3] = R4.L;
 | 
						|
 | 
						|
	bfin_init_pm_bench_cycles;
 | 
						|
 | 
						|
	CLI R2;
 | 
						|
	IDLE;
 | 
						|
.Lforever:
 | 
						|
	jump .Lforever;
 | 
						|
ENDPROC(_hibernate_mode)
 | 
						|
 | 
						|
ENTRY(_sleep_deeper)
 | 
						|
	[--SP] = (R7:4, P5:3);
 | 
						|
	[--SP] = RETS;
 | 
						|
 | 
						|
	CLI R4;
 | 
						|
 | 
						|
	P3 = R0;
 | 
						|
	P4 = R1;
 | 
						|
	P5 = R2;
 | 
						|
 | 
						|
	R0 = IWR_ENABLE(0);
 | 
						|
	R1 = IWR_DISABLE_ALL;
 | 
						|
	R2 = IWR_DISABLE_ALL;
 | 
						|
 | 
						|
	call _set_sic_iwr;
 | 
						|
	call _set_dram_srfs;	/* Set SDRAM Self Refresh */
 | 
						|
 | 
						|
	P0.H = hi(PLL_DIV);
 | 
						|
	P0.L = lo(PLL_DIV);
 | 
						|
	R6 = W[P0](z);
 | 
						|
	R0.L = 0xF;
 | 
						|
	W[P0] = R0.l;		/* Set Max VCO to SCLK divider */
 | 
						|
 | 
						|
	P0.H = hi(PLL_CTL);
 | 
						|
	P0.L = lo(PLL_CTL);
 | 
						|
	R5 = W[P0](z);
 | 
						|
	R0.L = (CONFIG_MIN_VCO_HZ/CONFIG_CLKIN_HZ) << 9;
 | 
						|
	W[P0] = R0.l;		/* Set Min CLKIN to VCO multiplier */
 | 
						|
 | 
						|
	SSYNC;
 | 
						|
	IDLE;
 | 
						|
 | 
						|
	call _test_pll_locked;
 | 
						|
 | 
						|
	P0.H = hi(VR_CTL);
 | 
						|
	P0.L = lo(VR_CTL);
 | 
						|
	R7 = W[P0](z);
 | 
						|
	R1 = 0x6;
 | 
						|
	R1 <<= 16;
 | 
						|
	R2 = 0x0404(Z);
 | 
						|
	R1 = R1|R2;
 | 
						|
 | 
						|
	R2 = DEPOSIT(R7, R1);
 | 
						|
	W[P0] = R2;		/* Set Min Core Voltage */
 | 
						|
 | 
						|
	SSYNC;
 | 
						|
	IDLE;
 | 
						|
 | 
						|
	call _test_pll_locked;
 | 
						|
 | 
						|
	R0 = P3;
 | 
						|
	R1 = P4;
 | 
						|
	R3 = P5;
 | 
						|
	call _set_sic_iwr;	/* Set Awake from IDLE */
 | 
						|
 | 
						|
	P0.H = hi(PLL_CTL);
 | 
						|
	P0.L = lo(PLL_CTL);
 | 
						|
	R0 = W[P0](z);
 | 
						|
	BITSET (R0, 3);
 | 
						|
	W[P0] = R0.L;		/* Turn CCLK OFF */
 | 
						|
	SSYNC;
 | 
						|
	IDLE;
 | 
						|
 | 
						|
	call _test_pll_locked;
 | 
						|
 | 
						|
	R0 = IWR_ENABLE(0);
 | 
						|
	R1 = IWR_DISABLE_ALL;
 | 
						|
	R2 = IWR_DISABLE_ALL;
 | 
						|
 | 
						|
	call _set_sic_iwr;	/* Set Awake from IDLE PLL */
 | 
						|
 | 
						|
	P0.H = hi(VR_CTL);
 | 
						|
	P0.L = lo(VR_CTL);
 | 
						|
	W[P0]= R7;
 | 
						|
 | 
						|
	SSYNC;
 | 
						|
	IDLE;
 | 
						|
 | 
						|
	bfin_init_pm_bench_cycles;
 | 
						|
 | 
						|
	call _test_pll_locked;
 | 
						|
 | 
						|
	P0.H = hi(PLL_DIV);
 | 
						|
	P0.L = lo(PLL_DIV);
 | 
						|
	W[P0]= R6;		/* Restore CCLK and SCLK divider */
 | 
						|
 | 
						|
	P0.H = hi(PLL_CTL);
 | 
						|
	P0.L = lo(PLL_CTL);
 | 
						|
	w[p0] = R5;		/* Restore VCO multiplier */
 | 
						|
	IDLE;
 | 
						|
	call _test_pll_locked;
 | 
						|
 | 
						|
	call _unset_dram_srfs;	/* SDRAM Self Refresh Off */
 | 
						|
 | 
						|
	STI R4;
 | 
						|
 | 
						|
	RETS = [SP++];
 | 
						|
	(R7:4, P5:3) = [SP++];
 | 
						|
	RTS;
 | 
						|
ENDPROC(_sleep_deeper)
 | 
						|
 | 
						|
ENTRY(_set_dram_srfs)
 | 
						|
	/*  set the dram to self refresh mode */
 | 
						|
	SSYNC;
 | 
						|
#if defined(EBIU_RSTCTL)	/* DDR */
 | 
						|
	P0.H = hi(EBIU_RSTCTL);
 | 
						|
	P0.L = lo(EBIU_RSTCTL);
 | 
						|
	R2 = [P0];
 | 
						|
	BITSET(R2, 3); /* SRREQ enter self-refresh mode */
 | 
						|
	[P0] = R2;
 | 
						|
	SSYNC;
 | 
						|
1:
 | 
						|
	R2 = [P0];
 | 
						|
	CC = BITTST(R2, 4);
 | 
						|
	if !CC JUMP 1b;
 | 
						|
#else 				/* SDRAM */
 | 
						|
	P0.L = lo(EBIU_SDGCTL);
 | 
						|
	P0.H = hi(EBIU_SDGCTL);
 | 
						|
	P1.L = lo(EBIU_SDSTAT);
 | 
						|
	P1.H = hi(EBIU_SDSTAT);
 | 
						|
 | 
						|
	R2 = [P0];
 | 
						|
	BITSET(R2, 24); /* SRFS enter self-refresh mode */
 | 
						|
	[P0] = R2;
 | 
						|
	SSYNC;
 | 
						|
 | 
						|
1:
 | 
						|
	R2 = w[P1];
 | 
						|
	SSYNC;
 | 
						|
	cc = BITTST(R2, 1); /* SDSRA poll self-refresh status */
 | 
						|
	if !cc jump 1b;
 | 
						|
 | 
						|
	R2 = [P0];
 | 
						|
	BITCLR(R2, 0); /* SCTLE disable CLKOUT */
 | 
						|
	[P0] = R2;
 | 
						|
#endif
 | 
						|
	RTS;
 | 
						|
ENDPROC(_set_dram_srfs)
 | 
						|
 | 
						|
ENTRY(_unset_dram_srfs)
 | 
						|
	/*  set the dram out of self refresh mode */
 | 
						|
 | 
						|
#if defined(EBIU_RSTCTL)	/* DDR */
 | 
						|
	P0.H = hi(EBIU_RSTCTL);
 | 
						|
	P0.L = lo(EBIU_RSTCTL);
 | 
						|
	R2 = [P0];
 | 
						|
	BITCLR(R2, 3); /* clear SRREQ bit */
 | 
						|
	[P0] = R2;
 | 
						|
#elif defined(EBIU_SDGCTL)	/* SDRAM */
 | 
						|
	/* release CLKOUT from self-refresh */
 | 
						|
	P0.L = lo(EBIU_SDGCTL);
 | 
						|
	P0.H = hi(EBIU_SDGCTL);
 | 
						|
 | 
						|
	R2 = [P0];
 | 
						|
	BITSET(R2, 0); /* SCTLE enable CLKOUT */
 | 
						|
	[P0] = R2
 | 
						|
	SSYNC;
 | 
						|
 | 
						|
	/* release SDRAM from self-refresh */
 | 
						|
	R2 = [P0];
 | 
						|
	BITCLR(R2, 24); /* clear SRFS bit */
 | 
						|
	[P0] = R2
 | 
						|
#endif
 | 
						|
 | 
						|
	SSYNC;
 | 
						|
	RTS;
 | 
						|
ENDPROC(_unset_dram_srfs)
 | 
						|
 | 
						|
ENTRY(_set_sic_iwr)
 | 
						|
#ifdef SIC_IWR0
 | 
						|
	P0.H = hi(SYSMMR_BASE);
 | 
						|
	P0.L = lo(SYSMMR_BASE);
 | 
						|
	[P0 + (SIC_IWR0 - SYSMMR_BASE)] = R0;
 | 
						|
	[P0 + (SIC_IWR1 - SYSMMR_BASE)] = R1;
 | 
						|
# ifdef SIC_IWR2
 | 
						|
	[P0 + (SIC_IWR2 - SYSMMR_BASE)] = R2;
 | 
						|
# endif
 | 
						|
#else
 | 
						|
	P0.H = hi(SIC_IWR);
 | 
						|
	P0.L = lo(SIC_IWR);
 | 
						|
	[P0] = R0;
 | 
						|
#endif
 | 
						|
 | 
						|
	SSYNC;
 | 
						|
	RTS;
 | 
						|
ENDPROC(_set_sic_iwr)
 | 
						|
 | 
						|
ENTRY(_test_pll_locked)
 | 
						|
	P0.H = hi(PLL_STAT);
 | 
						|
	P0.L = lo(PLL_STAT);
 | 
						|
1:
 | 
						|
	R0 = W[P0] (Z);
 | 
						|
	CC = BITTST(R0,5);
 | 
						|
	IF !CC JUMP 1b;
 | 
						|
	RTS;
 | 
						|
ENDPROC(_test_pll_locked)
 | 
						|
 | 
						|
.section .text
 | 
						|
ENTRY(_do_hibernate)
 | 
						|
	bfin_cpu_reg_save;
 | 
						|
	bfin_sys_mmr_save;
 | 
						|
	bfin_core_mmr_save;
 | 
						|
 | 
						|
	/* Setup args to hibernate mode early for pipeline optimization */
 | 
						|
	R0 = M3;
 | 
						|
	P1.H = _hibernate_mode;
 | 
						|
	P1.L = _hibernate_mode;
 | 
						|
 | 
						|
	/* Save Magic, return address and Stack Pointer */
 | 
						|
	P0 = 0;
 | 
						|
	R1.H = 0xDEAD;	/* Hibernate Magic */
 | 
						|
	R1.L = 0xBEEF;
 | 
						|
	R2.H = .Lpm_resume_here;
 | 
						|
	R2.L = .Lpm_resume_here;
 | 
						|
	[P0++] = R1;	/* Store Hibernate Magic */
 | 
						|
	[P0++] = R2;	/* Save Return Address */
 | 
						|
	[P0++] = SP;	/* Save Stack Pointer */
 | 
						|
 | 
						|
	/* Must use an indirect call as we need to jump to L1 */
 | 
						|
	call (P1); /* Goodbye */
 | 
						|
 | 
						|
.Lpm_resume_here:
 | 
						|
 | 
						|
	bfin_core_mmr_restore;
 | 
						|
	bfin_sys_mmr_restore;
 | 
						|
	bfin_cpu_reg_restore;
 | 
						|
 | 
						|
	[--sp] = RETI;	/* Clear Global Interrupt Disable */
 | 
						|
	SP += 4;
 | 
						|
 | 
						|
	RTS;
 | 
						|
ENDPROC(_do_hibernate)
 |