cregit-Linux how code gets into the kernel

Release 4.11 arch/powerpc/sysdev/fsl_rcpm.c

/*
 * RCPM(Run Control/Power Management) support
 *
 * Copyright 2012-2015 Freescale Semiconductor Inc.
 *
 * Author: Chenhui Zhao <chenhui.zhao@freescale.com>
 *
 * 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.
 */


#define pr_fmt(fmt) "%s: " fmt, __func__

#include <linux/types.h>
#include <linux/errno.h>
#include <linux/of_address.h>
#include <linux/export.h>

#include <asm/io.h>
#include <linux/fsl/guts.h>
#include <asm/cputhreads.h>
#include <asm/fsl_pm.h>
#include <asm/smp.h>


static struct ccsr_rcpm_v1 __iomem *rcpm_v1_regs;

static struct ccsr_rcpm_v2 __iomem *rcpm_v2_regs;

static unsigned int fsl_supported_pm_modes;


static void rcpm_v1_irq_mask(int cpu) { int hw_cpu = get_hard_smp_processor_id(cpu); unsigned int mask = 1 << hw_cpu; setbits32(&rcpm_v1_regs->cpmimr, mask); setbits32(&rcpm_v1_regs->cpmcimr, mask); setbits32(&rcpm_v1_regs->cpmmcmr, mask); setbits32(&rcpm_v1_regs->cpmnmimr, mask); }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao65100.00%1100.00%
Total65100.00%1100.00%


static void rcpm_v2_irq_mask(int cpu) { int hw_cpu = get_hard_smp_processor_id(cpu); unsigned int mask = 1 << hw_cpu; setbits32(&rcpm_v2_regs->tpmimr0, mask); setbits32(&rcpm_v2_regs->tpmcimr0, mask); setbits32(&rcpm_v2_regs->tpmmcmr0, mask); setbits32(&rcpm_v2_regs->tpmnmimr0, mask); }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao65100.00%1100.00%
Total65100.00%1100.00%


static void rcpm_v1_irq_unmask(int cpu) { int hw_cpu = get_hard_smp_processor_id(cpu); unsigned int mask = 1 << hw_cpu; clrbits32(&rcpm_v1_regs->cpmimr, mask); clrbits32(&rcpm_v1_regs->cpmcimr, mask); clrbits32(&rcpm_v1_regs->cpmmcmr, mask); clrbits32(&rcpm_v1_regs->cpmnmimr, mask); }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao65100.00%1100.00%
Total65100.00%1100.00%


static void rcpm_v2_irq_unmask(int cpu) { int hw_cpu = get_hard_smp_processor_id(cpu); unsigned int mask = 1 << hw_cpu; clrbits32(&rcpm_v2_regs->tpmimr0, mask); clrbits32(&rcpm_v2_regs->tpmcimr0, mask); clrbits32(&rcpm_v2_regs->tpmmcmr0, mask); clrbits32(&rcpm_v2_regs->tpmnmimr0, mask); }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao65100.00%1100.00%
Total65100.00%1100.00%


static void rcpm_v1_set_ip_power(bool enable, u32 mask) { if (enable) setbits32(&rcpm_v1_regs->ippdexpcr, mask); else clrbits32(&rcpm_v1_regs->ippdexpcr, mask); }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao37100.00%1100.00%
Total37100.00%1100.00%


static void rcpm_v2_set_ip_power(bool enable, u32 mask) { if (enable) setbits32(&rcpm_v2_regs->ippdexpcr[0], mask); else clrbits32(&rcpm_v2_regs->ippdexpcr[0], mask); }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao43100.00%1100.00%
Total43100.00%1100.00%


static void rcpm_v1_cpu_enter_state(int cpu, int state) { int hw_cpu = get_hard_smp_processor_id(cpu); unsigned int mask = 1 << hw_cpu; switch (state) { case E500_PM_PH10: setbits32(&rcpm_v1_regs->cdozcr, mask); break; case E500_PM_PH15: setbits32(&rcpm_v1_regs->cnapcr, mask); break; default: pr_warn("Unknown cpu PM state (%d)\n", state); break; } }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao71100.00%1100.00%
Total71100.00%1100.00%


static void rcpm_v2_cpu_enter_state(int cpu, int state) { int hw_cpu = get_hard_smp_processor_id(cpu); u32 mask = 1 << cpu_core_index_of_thread(cpu); switch (state) { case E500_PM_PH10: /* one bit corresponds to one thread for PH10 of 6500 */ setbits32(&rcpm_v2_regs->tph10setr0, 1 << hw_cpu); break; case E500_PM_PH15: setbits32(&rcpm_v2_regs->pcph15setr, mask); break; case E500_PM_PH20: setbits32(&rcpm_v2_regs->pcph20setr, mask); break; case E500_PM_PH30: setbits32(&rcpm_v2_regs->pcph30setr, mask); break; default: pr_warn("Unknown cpu PM state (%d)\n", state); } }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao103100.00%1100.00%
Total103100.00%1100.00%


static void rcpm_v1_cpu_die(int cpu) { rcpm_v1_cpu_enter_state(cpu, E500_PM_PH15); }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao16100.00%1100.00%
Total16100.00%1100.00%

#ifdef CONFIG_PPC64
static void qoriq_disable_thread(int cpu) { int thread = cpu_thread_in_core(cpu); book3e_stop_thread(thread); }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao22100.00%1100.00%
Total22100.00%1100.00%

#endif
static void rcpm_v2_cpu_die(int cpu) { #ifdef CONFIG_PPC64 int primary; if (threads_per_core == 2) { primary = cpu_first_thread_sibling(cpu); if (cpu_is_offline(primary) && cpu_is_offline(primary + 1)) { /* if both threads are offline, put the cpu in PH20 */ rcpm_v2_cpu_enter_state(cpu, E500_PM_PH20); } else { /* if only one thread is offline, disable the thread */ qoriq_disable_thread(cpu); } } #endif if (threads_per_core == 1) rcpm_v2_cpu_enter_state(cpu, E500_PM_PH20); }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao78100.00%1100.00%
Total78100.00%1100.00%


static void rcpm_v1_cpu_exit_state(int cpu, int state) { int hw_cpu = get_hard_smp_processor_id(cpu); unsigned int mask = 1 << hw_cpu; switch (state) { case E500_PM_PH10: clrbits32(&rcpm_v1_regs->cdozcr, mask); break; case E500_PM_PH15: clrbits32(&rcpm_v1_regs->cnapcr, mask); break; default: pr_warn("Unknown cpu PM state (%d)\n", state); break; } }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao71100.00%1100.00%
Total71100.00%1100.00%


static void rcpm_v1_cpu_up_prepare(int cpu) { rcpm_v1_cpu_exit_state(cpu, E500_PM_PH15); rcpm_v1_irq_unmask(cpu); }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao21100.00%1100.00%
Total21100.00%1100.00%


static void rcpm_v2_cpu_exit_state(int cpu, int state) { int hw_cpu = get_hard_smp_processor_id(cpu); u32 mask = 1 << cpu_core_index_of_thread(cpu); switch (state) { case E500_PM_PH10: setbits32(&rcpm_v2_regs->tph10clrr0, 1 << hw_cpu); break; case E500_PM_PH15: setbits32(&rcpm_v2_regs->pcph15clrr, mask); break; case E500_PM_PH20: setbits32(&rcpm_v2_regs->pcph20clrr, mask); break; case E500_PM_PH30: setbits32(&rcpm_v2_regs->pcph30clrr, mask); break; default: pr_warn("Unknown cpu PM state (%d)\n", state); } }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao102100.00%1100.00%
Total102100.00%1100.00%


static void rcpm_v2_cpu_up_prepare(int cpu) { rcpm_v2_cpu_exit_state(cpu, E500_PM_PH20); rcpm_v2_irq_unmask(cpu); }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao21100.00%1100.00%
Total21100.00%1100.00%


static int rcpm_v1_plat_enter_state(int state) { u32 *pmcsr_reg = &rcpm_v1_regs->powmgtcsr; int ret = 0; int result; switch (state) { case PLAT_PM_SLEEP: setbits32(pmcsr_reg, RCPM_POWMGTCSR_SLP); /* Upon resume, wait for RCPM_POWMGTCSR_SLP bit to be clear. */ result = spin_event_timeout( !(in_be32(pmcsr_reg) & RCPM_POWMGTCSR_SLP), 10000, 10); if (!result) { pr_err("timeout waiting for SLP bit to be cleared\n"); ret = -ETIMEDOUT; } break; default: pr_warn("Unknown platform PM state (%d)", state); ret = -EINVAL; } return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao96100.00%1100.00%
Total96100.00%1100.00%


static int rcpm_v2_plat_enter_state(int state) { u32 *pmcsr_reg = &rcpm_v2_regs->powmgtcsr; int ret = 0; int result; switch (state) { case PLAT_PM_LPM20: /* clear previous LPM20 status */ setbits32(pmcsr_reg, RCPM_POWMGTCSR_P_LPM20_ST); /* enter LPM20 status */ setbits32(pmcsr_reg, RCPM_POWMGTCSR_LPM20_RQ); /* At this point, the device is in LPM20 status. */ /* resume ... */ result = spin_event_timeout( !(in_be32(pmcsr_reg) & RCPM_POWMGTCSR_LPM20_ST), 10000, 10); if (!result) { pr_err("timeout waiting for LPM20 bit to be cleared\n"); ret = -ETIMEDOUT; } break; default: pr_warn("Unknown platform PM state (%d)\n", state); ret = -EINVAL; } return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao106100.00%1100.00%
Total106100.00%1100.00%


static int rcpm_v1_plat_enter_sleep(void) { return rcpm_v1_plat_enter_state(PLAT_PM_SLEEP); }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao14100.00%1100.00%
Total14100.00%1100.00%


static int rcpm_v2_plat_enter_sleep(void) { return rcpm_v2_plat_enter_state(PLAT_PM_LPM20); }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao14100.00%1100.00%
Total14100.00%1100.00%


static void rcpm_common_freeze_time_base(u32 *tben_reg, int freeze) { static u32 mask; if (freeze) { mask = in_be32(tben_reg); clrbits32(tben_reg, mask); } else { setbits32(tben_reg, mask); } /* read back to push the previous write */ in_be32(tben_reg); }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao53100.00%1100.00%
Total53100.00%1100.00%


static void rcpm_v1_freeze_time_base(bool freeze) { rcpm_common_freeze_time_base(&rcpm_v1_regs->ctbenr, freeze); }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao19100.00%1100.00%
Total19100.00%1100.00%


static void rcpm_v2_freeze_time_base(bool freeze) { rcpm_common_freeze_time_base(&rcpm_v2_regs->pctbenr, freeze); }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao19100.00%1100.00%
Total19100.00%1100.00%


static unsigned int rcpm_get_pm_modes(void) { return fsl_supported_pm_modes; }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao12100.00%1100.00%
Total12100.00%1100.00%

static const struct fsl_pm_ops qoriq_rcpm_v1_ops = { .irq_mask = rcpm_v1_irq_mask, .irq_unmask = rcpm_v1_irq_unmask, .cpu_enter_state = rcpm_v1_cpu_enter_state, .cpu_exit_state = rcpm_v1_cpu_exit_state, .cpu_up_prepare = rcpm_v1_cpu_up_prepare, .cpu_die = rcpm_v1_cpu_die, .plat_enter_sleep = rcpm_v1_plat_enter_sleep, .set_ip_power = rcpm_v1_set_ip_power, .freeze_time_base = rcpm_v1_freeze_time_base, .get_pm_modes = rcpm_get_pm_modes, }; static const struct fsl_pm_ops qoriq_rcpm_v2_ops = { .irq_mask = rcpm_v2_irq_mask, .irq_unmask = rcpm_v2_irq_unmask, .cpu_enter_state = rcpm_v2_cpu_enter_state, .cpu_exit_state = rcpm_v2_cpu_exit_state, .cpu_up_prepare = rcpm_v2_cpu_up_prepare, .cpu_die = rcpm_v2_cpu_die, .plat_enter_sleep = rcpm_v2_plat_enter_sleep, .set_ip_power = rcpm_v2_set_ip_power, .freeze_time_base = rcpm_v2_freeze_time_base, .get_pm_modes = rcpm_get_pm_modes, }; static const struct of_device_id rcpm_matches[] = { { .compatible = "fsl,qoriq-rcpm-1.0", .data = &qoriq_rcpm_v1_ops, }, { .compatible = "fsl,qoriq-rcpm-2.0", .data = &qoriq_rcpm_v2_ops, }, { .compatible = "fsl,qoriq-rcpm-2.1", .data = &qoriq_rcpm_v2_ops, }, {}, };
int __init fsl_rcpm_init(void) { struct device_node *np; const struct of_device_id *match; void __iomem *base; np = of_find_matching_node_and_match(NULL, rcpm_matches, &match); if (!np) return 0; base = of_iomap(np, 0); of_node_put(np); if (!base) { pr_err("of_iomap() error.\n"); return -ENOMEM; } rcpm_v1_regs = base; rcpm_v2_regs = base; /* support sleep by default */ fsl_supported_pm_modes = FSL_PM_SLEEP; qoriq_pm_ops = match->data; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao96100.00%1100.00%
Total96100.00%1100.00%


Overall Contributors

PersonTokensPropCommitsCommitProp
Chen-Hui Zhao149699.80%150.00%
Michael Ellerman30.20%150.00%
Total1499100.00%2100.00%
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.