Release 4.11 arch/blackfin/mach-common/pm.c
/*
* Blackfin power management
*
* Copyright 2006-2009 Analog Devices Inc.
*
* Licensed under the GPL-2
* based on arm/mach-omap/pm.c
* Copyright 2001, Cliff Brake <cbrake@accelent.com> and others
*/
#include <linux/suspend.h>
#include <linux/sched.h>
#include <linux/proc_fs.h>
#include <linux/slab.h>
#include <linux/io.h>
#include <linux/irq.h>
#include <linux/delay.h>
#include <linux/gpio.h>
#include <asm/cplb.h>
#include <asm/dma.h>
#include <asm/dpmc.h>
#include <asm/pm.h>
#ifdef CONFIG_BF60x
struct bfin_cpu_pm_fns *bfin_cpu_pm;
#endif
void bfin_pm_suspend_standby_enter(void)
{
#if !BFIN_GPIO_PINT
bfin_pm_standby_setup();
#endif
#ifdef CONFIG_BF60x
bfin_cpu_pm->enter(PM_SUSPEND_STANDBY);
#else
# ifdef CONFIG_PM_BFIN_SLEEP_DEEPER
sleep_deeper(bfin_sic_iwr[0], bfin_sic_iwr[1], bfin_sic_iwr[2]);
# else
sleep_mode(bfin_sic_iwr[0], bfin_sic_iwr[1], bfin_sic_iwr[2]);
# endif
#endif
#if !BFIN_GPIO_PINT
bfin_pm_standby_restore();
#endif
#ifndef CONFIG_BF60x
#ifdef SIC_IWR0
bfin_write_SIC_IWR0(IWR_DISABLE_ALL);
# ifdef SIC_IWR1
/* BF52x system reset does not properly reset SIC_IWR1 which
* will screw up the bootrom as it relies on MDMA0/1 waking it
* up from IDLE instructions. See this report for more info:
* http://blackfin.uclinux.org/gf/tracker/4323
*/
if (ANOMALY_05000435)
bfin_write_SIC_IWR1(IWR_ENABLE(10) | IWR_ENABLE(11));
else
bfin_write_SIC_IWR1(IWR_DISABLE_ALL);
# endif
# ifdef SIC_IWR2
bfin_write_SIC_IWR2(IWR_DISABLE_ALL);
# endif
#else
bfin_write_SIC_IWR(IWR_DISABLE_ALL);
#endif
#endif
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Michael Hennerich | 53 | 37.06% | 4 | 40.00% |
Sonic Zhang | 32 | 22.38% | 2 | 20.00% |
Steven Miao | 23 | 16.08% | 1 | 10.00% |
Bryan Wu | 18 | 12.59% | 1 | 10.00% |
Mike Frysinger | 17 | 11.89% | 2 | 20.00% |
Total | 143 | 100.00% | 10 | 100.00% |
int bf53x_suspend_l1_mem(unsigned char *memptr)
{
dma_memcpy_nocache(memptr, (const void *) L1_CODE_START,
L1_CODE_LENGTH);
dma_memcpy_nocache(memptr + L1_CODE_LENGTH,
(const void *) L1_DATA_A_START, L1_DATA_A_LENGTH);
dma_memcpy_nocache(memptr + L1_CODE_LENGTH + L1_DATA_A_LENGTH,
(const void *) L1_DATA_B_START, L1_DATA_B_LENGTH);
memcpy(memptr + L1_CODE_LENGTH + L1_DATA_A_LENGTH +
L1_DATA_B_LENGTH, (const void *) L1_SCRATCH_START,
L1_SCRATCH_LENGTH);
return 0;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Michael Hennerich | 81 | 100.00% | 2 | 100.00% |
Total | 81 | 100.00% | 2 | 100.00% |
int bf53x_resume_l1_mem(unsigned char *memptr)
{
dma_memcpy_nocache((void *) L1_CODE_START, memptr, L1_CODE_LENGTH);
dma_memcpy_nocache((void *) L1_DATA_A_START, memptr + L1_CODE_LENGTH,
L1_DATA_A_LENGTH);
dma_memcpy_nocache((void *) L1_DATA_B_START, memptr + L1_CODE_LENGTH +
L1_DATA_A_LENGTH, L1_DATA_B_LENGTH);
memcpy((void *) L1_SCRATCH_START, memptr + L1_CODE_LENGTH +
L1_DATA_A_LENGTH + L1_DATA_B_LENGTH, L1_SCRATCH_LENGTH);
return 0;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Michael Hennerich | 77 | 100.00% | 2 | 100.00% |
Total | 77 | 100.00% | 2 | 100.00% |
#if defined(CONFIG_BFIN_EXTMEM_WRITEBACK) || defined(CONFIG_BFIN_L2_WRITEBACK)
# ifdef CONFIG_BF60x
__attribute__((l1_text))
# endif
static void flushinv_all_dcache(void)
{
register u32 way, bank, subbank, set;
register u32 status, addr;
u32 dmem_ctl = bfin_read_DMEM_CONTROL();
for (bank = 0; bank < 2; ++bank) {
if (!(dmem_ctl & (1 << (DMC1_P - bank))))
continue;
for (way = 0; way < 2; ++way)
for (subbank = 0; subbank < 4; ++subbank)
for (set = 0; set < 64; ++set) {
bfin_write_DTEST_COMMAND(
way << 26 |
bank << 23 |
subbank << 16 |
set << 5
);
CSYNC();
status = bfin_read_DTEST_DATA0();
/* only worry about valid/dirty entries */
if ((status & 0x3) != 0x3)
continue;
/* construct the address using the tag */
addr = (status & 0xFFFFC800) | (subbank << 12) | (set << 5);
/* flush it */
__asm__ __volatile__("FLUSHINV[%0];" : : "a"(addr));
}
}
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Michael Hennerich | 166 | 98.81% | 1 | 50.00% |
Steven Miao | 2 | 1.19% | 1 | 50.00% |
Total | 168 | 100.00% | 2 | 100.00% |
#endif
int bfin_pm_suspend_mem_enter(void)
{
int ret;
#ifndef CONFIG_BF60x
int wakeup;
#endif
unsigned char *memptr = kmalloc(L1_CODE_LENGTH + L1_DATA_A_LENGTH
+ L1_DATA_B_LENGTH + L1_SCRATCH_LENGTH,
GFP_ATOMIC);
if (memptr == NULL) {
panic("bf53x_suspend_l1_mem malloc failed");
return -ENOMEM;
}
#ifndef CONFIG_BF60x
wakeup = bfin_read_VR_CTL() & ~FREQ;
wakeup |= SCKELOW;
#ifdef CONFIG_PM_BFIN_WAKE_PH6
wakeup |= PHYWE;
#endif
#ifdef CONFIG_PM_BFIN_WAKE_GP
wakeup |= GPWE;
#endif
#endif
ret = blackfin_dma_suspend();
if (ret) {
kfree(memptr);
return ret;
}
#ifdef CONFIG_GPIO_ADI
bfin_gpio_pm_hibernate_suspend();
#endif
#if defined(CONFIG_BFIN_EXTMEM_WRITEBACK) || defined(CONFIG_BFIN_L2_WRITEBACK)
flushinv_all_dcache();
udelay(1);
#endif
_disable_dcplb();
_disable_icplb();
bf53x_suspend_l1_mem(memptr);
#ifndef CONFIG_BF60x
do_hibernate(wakeup | vr_wakeup); /* See you later! */
#else
bfin_cpu_pm->enter(PM_SUSPEND_MEM);
#endif
bf53x_resume_l1_mem(memptr);
_enable_icplb();
_enable_dcplb();
#ifdef CONFIG_GPIO_ADI
bfin_gpio_pm_hibernate_restore();
#endif
blackfin_dma_resume();
kfree(memptr);
return 0;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Michael Hennerich | 133 | 66.83% | 3 | 25.00% |
Steven Miao | 33 | 16.58% | 5 | 41.67% |
Yi Li | 20 | 10.05% | 1 | 8.33% |
Sonic Zhang | 7 | 3.52% | 1 | 8.33% |
Aaron Wu | 5 | 2.51% | 1 | 8.33% |
Scott Jiang | 1 | 0.50% | 1 | 8.33% |
Total | 199 | 100.00% | 12 | 100.00% |
/*
* bfin_pm_valid - Tell the PM core that we only support the standby sleep
* state
* @state: suspend state we're checking.
*
*/
static int bfin_pm_valid(suspend_state_t state)
{
return (state == PM_SUSPEND_STANDBY
#if !(defined(BF533_FAMILY) || defined(CONFIG_BF561))
/*
* On BF533/2/1:
* If we enter Hibernate the SCKE Pin is driven Low,
* so that the SDRAM enters Self Refresh Mode.
* However when the reset sequence that follows hibernate
* state is executed, SCKE is driven High, taking the
* SDRAM out of Self Refresh.
*
* If you reconfigure and access the SDRAM "very quickly",
* you are likely to avoid errors, otherwise the SDRAM
* start losing its contents.
* An external HW workaround is possible using logic gates.
*/
|| state == PM_SUSPEND_MEM
#endif
);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Michael Hennerich | 21 | 56.76% | 2 | 50.00% |
Bryan Wu | 11 | 29.73% | 1 | 25.00% |
Rafael J. Wysocki | 5 | 13.51% | 1 | 25.00% |
Total | 37 | 100.00% | 4 | 100.00% |
/*
* bfin_pm_enter - Actually enter a sleep state.
* @state: State we're entering.
*
*/
static int bfin_pm_enter(suspend_state_t state)
{
switch (state) {
case PM_SUSPEND_STANDBY:
bfin_pm_suspend_standby_enter();
break;
case PM_SUSPEND_MEM:
bfin_pm_suspend_mem_enter();
break;
default:
return -EINVAL;
}
return 0;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Bryan Wu | 33 | 89.19% | 1 | 50.00% |
Michael Hennerich | 4 | 10.81% | 1 | 50.00% |
Total | 37 | 100.00% | 2 | 100.00% |
#ifdef CONFIG_BFIN_PM_WAKEUP_TIME_BENCH
void bfin_pm_end(void)
{
u32 cycle, cycle2;
u64 usec64;
u32 usec;
__asm__ __volatile__ (
"1: %0 = CYCLES2\n"
"%1 = CYCLES\n"
"%2 = CYCLES2\n"
"CC = %2 == %0\n"
"if ! CC jump 1b\n"
: "=d,a" (cycle2), "=d,a" (cycle), "=d,a" (usec) : : "CC"
);
usec64 = ((u64)cycle2 << 32) + cycle;
do_div(usec64, get_cclk() / USEC_PER_SEC);
usec = usec64;
if (usec == 0)
usec = 1;
pr_info("PM: resume of kernel completes after %ld msec %03ld usec\n",
usec / USEC_PER_MSEC, usec % USEC_PER_MSEC);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Sonic Zhang | 71 | 100.00% | 1 | 100.00% |
Total | 71 | 100.00% | 1 | 100.00% |
#endif
static const struct platform_suspend_ops bfin_pm_ops = {
.enter = bfin_pm_enter,
.valid = bfin_pm_valid,
#ifdef CONFIG_BFIN_PM_WAKEUP_TIME_BENCH
.end = bfin_pm_end,
#endif
};
static int __init bfin_pm_init(void)
{
suspend_set_ops(&bfin_pm_ops);
return 0;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Bryan Wu | 17 | 94.44% | 1 | 50.00% |
Rafael J. Wysocki | 1 | 5.56% | 1 | 50.00% |
Total | 18 | 100.00% | 2 | 100.00% |
__initcall(bfin_pm_init);
Overall Contributors
Person | Tokens | Prop | Commits | CommitProp |
Michael Hennerich | 546 | 58.02% | 9 | 28.12% |
Sonic Zhang | 127 | 13.50% | 3 | 9.38% |
Bryan Wu | 110 | 11.69% | 1 | 3.12% |
Steven Miao | 80 | 8.50% | 5 | 15.62% |
Yi Li | 22 | 2.34% | 1 | 3.12% |
Mike Frysinger | 19 | 2.02% | 3 | 9.38% |
Jie Zhang | 11 | 1.17% | 1 | 3.12% |
Rafael J. Wysocki | 9 | 0.96% | 3 | 9.38% |
Aaron Wu | 8 | 0.85% | 1 | 3.12% |
Tejun Heo | 3 | 0.32% | 1 | 3.12% |
Lionel Debroux | 2 | 0.21% | 1 | 3.12% |
Björn Helgaas | 2 | 0.21% | 1 | 3.12% |
Robin Getz | 1 | 0.11% | 1 | 3.12% |
Scott Jiang | 1 | 0.11% | 1 | 3.12% |
Total | 941 | 100.00% | 32 | 100.00% |
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.