static int tegra20_boot_secondary(unsigned int cpu, struct task_struct *idle) { cpu = cpu_logical_map(cpu); /* * Force the CPU into reset. The CPU must remain in reset when * the flow controller state is cleared (which will cause the * flow controller to stop driving reset if the CPU has been * power-gated via the flow controller). This will have no * effect on first boot of the CPU since it should already be * in reset. */ tegra_put_cpu_in_reset(cpu); /* * Unhalt the CPU. If the flow controller was used to * power-gate the CPU this will cause the flow controller to * stop driving reset. The CPU will remain in reset because the * clock and reset block is now driving reset. */ flowctrl_write_cpu_halt(cpu, 0); tegra_enable_cpu_clock(cpu); flowctrl_write_cpu_csr(cpu, 0); /* Clear flow controller CSR. */ tegra_cpu_out_of_reset(cpu); return 0; }
void flowctrl_cpu_suspend_exit(unsigned int cpuid) { unsigned int reg; /* Disable powergating via flow controller for CPU0 */ reg = flowctrl_read_cpu_csr(cpuid); switch (tegra_get_chip_id()) { case TEGRA20: /* clear wfe bitmap */ reg &= ~TEGRA20_FLOW_CTRL_CSR_WFE_BITMAP; /* clear wfi bitmap */ reg &= ~TEGRA20_FLOW_CTRL_CSR_WFI_BITMAP; break; case TEGRA30: case TEGRA114: case TEGRA124: /* clear wfe bitmap */ reg &= ~TEGRA30_FLOW_CTRL_CSR_WFE_BITMAP; /* clear wfi bitmap */ reg &= ~TEGRA30_FLOW_CTRL_CSR_WFI_BITMAP; break; } reg &= ~FLOW_CTRL_CSR_ENABLE; /* clear enable */ reg |= FLOW_CTRL_CSR_INTR_FLAG; /* clear intr */ reg |= FLOW_CTRL_CSR_EVENT_FLAG; /* clear event */ flowctrl_write_cpu_csr(cpuid, reg); }
static int tegra114_boot_secondary(unsigned int cpu, struct task_struct *idle) { int ret = 0; cpu = cpu_logical_map(cpu); if (cpumask_test_cpu(cpu, &tegra_cpu_init_mask)) { /* * Warm boot flow * The flow controller in charge of the power state and * control for each CPU. */ /* set SCLK as event trigger for flow controller */ flowctrl_write_cpu_csr(cpu, 1); flowctrl_write_cpu_halt(cpu, FLOW_CTRL_WAITEVENT | FLOW_CTRL_SCLK_RESUME); } else { /* * Cold boot flow * The CPU is powered up by toggling PMC directly. It will * also initial power state in flow controller. After that, * the CPU's power state is maintained by flow controller. */ ret = tegra_pmc_cpu_power_on(cpu); } return ret; }
void flowctrl_cpu_off(int cpu) { uint32_t val = FLOW_CTRL_CSR_INTR_FLAG | FLOW_CTRL_CSR_EVENT_FLAG | FLOW_CTRL_CSR_ENABLE | (FLOW_CTRL_CSR_WFI_CPU0 << cpu); flowctrl_write_cpu_csr(cpu, val); flowctrl_write_cpu_halt(cpu, FLOW_CTRL_WAITEVENT); flowctrl_write_cc4_ctrl(cpu, 0); }
static int tegra20_power_up_cpu(unsigned int cpu) { /* Enable the CPU clock. */ tegra_enable_cpu_clock(cpu); /* Clear flow controller CSR. */ flowctrl_write_cpu_csr(cpu, 0); return 0; }
void flowctrl_cpu_suspend_enter(unsigned int cpuid) { unsigned int reg; int i; reg = flowctrl_read_cpu_csr(cpuid); switch (tegra_get_chip_id()) { case TEGRA20: /* clear wfe bitmap */ reg &= ~TEGRA20_FLOW_CTRL_CSR_WFE_BITMAP; /* clear wfi bitmap */ reg &= ~TEGRA20_FLOW_CTRL_CSR_WFI_BITMAP; /* pwr gating on wfe */ reg |= TEGRA20_FLOW_CTRL_CSR_WFE_CPU0 << cpuid; break; case TEGRA30: case TEGRA114: case TEGRA124: /* clear wfe bitmap */ reg &= ~TEGRA30_FLOW_CTRL_CSR_WFE_BITMAP; /* clear wfi bitmap */ reg &= ~TEGRA30_FLOW_CTRL_CSR_WFI_BITMAP; /* pwr gating on wfi */ reg |= TEGRA30_FLOW_CTRL_CSR_WFI_CPU0 << cpuid; break; } reg |= FLOW_CTRL_CSR_INTR_FLAG; /* clear intr flag */ reg |= FLOW_CTRL_CSR_EVENT_FLAG; /* clear event flag */ reg |= FLOW_CTRL_CSR_ENABLE; /* pwr gating */ flowctrl_write_cpu_csr(cpuid, reg); for (i = 0; i < num_possible_cpus(); i++) { if (i == cpuid) continue; reg = flowctrl_read_cpu_csr(i); reg |= FLOW_CTRL_CSR_EVENT_FLAG; reg |= FLOW_CTRL_CSR_INTR_FLAG; flowctrl_write_cpu_csr(i, reg); } }
static void flowctrl_prepare_cpu_off(int cpu) { uint32_t reg; reg = flowctrl_read_cpu_csr(cpu); reg &= ~FLOW_CTRL_CSR_WFE_BITMAP; /* clear wfe bitmap */ reg &= ~FLOW_CTRL_CSR_WFI_BITMAP; /* clear wfi bitmap */ reg |= FLOW_CTRL_CSR_INTR_FLAG; /* clear intr flag */ reg |= FLOW_CTRL_CSR_EVENT_FLAG; /* clear event flag */ reg |= FLOW_CTRL_CSR_WFI_CPU0 << cpu; /* power gating on wfi */ reg |= FLOW_CTRL_CSR_ENABLE; /* enable power gating */ flowctrl_write_cpu_csr(cpu, reg); }
void flowctrl_cpu_suspend_exit(unsigned int cpuid) { unsigned int reg; /* Disable powergating via flow controller for CPU0 */ reg = flowctrl_read_cpu_csr(cpuid); reg &= ~TEGRA30_FLOW_CTRL_CSR_WFE_BITMAP; /* clear wfe bitmap */ reg &= ~TEGRA30_FLOW_CTRL_CSR_WFI_BITMAP; /* clear wfi bitmap */ reg &= ~FLOW_CTRL_CSR_ENABLE; /* clear enable */ reg |= FLOW_CTRL_CSR_INTR_FLAG; /* clear intr */ reg |= FLOW_CTRL_CSR_EVENT_FLAG; /* clear event */ flowctrl_write_cpu_csr(cpuid, reg); }
void flowctrl_cpu_suspend(int cpu) { uint32_t val; val = FLOW_CTRL_HALT_GIC_IRQ | FLOW_CTRL_HALT_GIC_FIQ | FLOW_CTRL_HALT_LIC_IRQ | FLOW_CTRL_HALT_LIC_FIQ | FLOW_CTRL_WAITEVENT; flowctrl_write_cpu_halt(cpu, val); val = FLOW_CTRL_CSR_INTR_FLAG | FLOW_CTRL_CSR_EVENT_FLAG | FLOW_CTRL_CSR_ENABLE | (FLOW_CTRL_CSR_WFI_CPU0 << cpu); flowctrl_write_cpu_csr(cpu, val); }
static int tegra20_power_up_cpu(unsigned int cpu) { u32 reg; /* Enable the CPU clock. */ reg = readl(CLK_RST_CONTROLLER_CLK_CPU_CMPLX); writel(reg & ~CPU_CLOCK(cpu), CLK_RST_CONTROLLER_CLK_CPU_CMPLX); barrier(); reg = readl(CLK_RST_CONTROLLER_CLK_CPU_CMPLX); /* Clear flow controller CSR. */ flowctrl_write_cpu_csr(cpu, 0); return 0; }
static int tegra30_power_up_cpu(unsigned int cpu) { u32 reg; int ret, pwrgateid; unsigned long timeout; pwrgateid = tegra_cpu_powergate_id(cpu); if (pwrgateid < 0) return pwrgateid; /* If this is the first boot, toggle powergates directly. */ if (!tegra_powergate_is_powered(pwrgateid)) { ret = tegra_powergate_power_on(pwrgateid); if (ret) return ret; /* Wait for the power to come up. */ timeout = jiffies + 10*HZ; while (tegra_powergate_is_powered(pwrgateid)) { if (time_after(jiffies, timeout)) return -ETIMEDOUT; udelay(10); } } /* CPU partition is powered. Enable the CPU clock. */ writel(CPU_CLOCK(cpu), CLK_RST_CONTROLLER_CLK_CPU_CMPLX_CLR); reg = readl(CLK_RST_CONTROLLER_CLK_CPU_CMPLX_CLR); udelay(10); /* Remove I/O clamps. */ ret = tegra_powergate_remove_clamping(pwrgateid); udelay(10); /* Clear flow controller CSR. */ flowctrl_write_cpu_csr(cpu, 0); return 0; }
void tegra_pmc_pm_set(enum tegra_suspend_mode mode) { u32 reg, csr_reg; unsigned long rate = 0; reg = tegra_pmc_readl(PMC_CTRL); reg |= TEGRA_POWER_CPU_PWRREQ_OE; reg &= ~TEGRA_POWER_EFFECT_LP0; switch (tegra_chip_id) { case TEGRA20: case TEGRA30: break; default: /* Turn off CRAIL */ csr_reg = flowctrl_read_cpu_csr(0); csr_reg &= ~FLOW_CTRL_CSR_ENABLE_EXT_MASK; csr_reg |= FLOW_CTRL_CSR_ENABLE_EXT_CRAIL; flowctrl_write_cpu_csr(0, csr_reg); break; } switch (mode) { case TEGRA_SUSPEND_LP1: rate = 32768; break; case TEGRA_SUSPEND_LP2: rate = clk_get_rate(tegra_pclk); break; default: break; } set_power_timers(pmc_pm_data.cpu_good_time, pmc_pm_data.cpu_off_time, rate); tegra_pmc_writel(reg, PMC_CTRL); }
static int tegra30_boot_secondary(unsigned int cpu, struct task_struct *idle) { int ret; unsigned long timeout; cpu = cpu_logical_map(cpu); tegra_put_cpu_in_reset(cpu); flowctrl_write_cpu_halt(cpu, 0); /* * The power up sequence of cold boot CPU and warm boot CPU * was different. * * For warm boot CPU that was resumed from CPU hotplug, the * power will be resumed automatically after un-halting the * flow controller of the warm boot CPU. We need to wait for * the confirmaiton that the CPU is powered then removing * the IO clamps. * For cold boot CPU, do not wait. After the cold boot CPU be * booted, it will run to tegra_secondary_init() and set * tegra_cpu_init_mask which influences what tegra30_boot_secondary() * next time around. */ if (cpumask_test_cpu(cpu, &tegra_cpu_init_mask)) { timeout = jiffies + msecs_to_jiffies(50); do { if (tegra_pmc_cpu_is_powered(cpu)) goto remove_clamps; udelay(10); } while (time_before(jiffies, timeout)); } /* * The power status of the cold boot CPU is power gated as * default. To power up the cold boot CPU, the power should * be un-gated by un-toggling the power gate register * manually. */ if (!tegra_pmc_cpu_is_powered(cpu)) { ret = tegra_pmc_cpu_power_on(cpu); if (ret) return ret; /* Wait for the power to come up. */ timeout = jiffies + msecs_to_jiffies(100); while (!tegra_pmc_cpu_is_powered(cpu)) { if (time_after(jiffies, timeout)) return -ETIMEDOUT; udelay(10); } } remove_clamps: /* CPU partition is powered. Enable the CPU clock. */ tegra_enable_cpu_clock(cpu); udelay(10); /* Remove I/O clamps. */ ret = tegra_pmc_cpu_remove_clamping(cpu); if (ret) return ret; udelay(10); flowctrl_write_cpu_csr(cpu, 0); /* Clear flow controller CSR. */ tegra_cpu_out_of_reset(cpu); return 0; }
void flowctrl_cpu_on(int cpu) { flowctrl_write_cpu_csr(cpu, FLOW_CTRL_CSR_ENABLE); flowctrl_write_cpu_halt(cpu, FLOW_CTRL_WAITEVENT | FLOW_CTRL_HALT_SCLK); }