tegrakernel/kernel/kernel-4.9/drivers/clocksource/tegra210_timer.c

301 lines
7.6 KiB
C
Raw Normal View History

2022-02-16 09:13:02 -06:00
/*
* Copyright (c) 2014-2017, NVIDIA CORPORATION. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify it
* under the terms and conditions of the GNU General Public License,
* version 2, as published by the Free Software Foundation.
*
* This program is distributed in the hope it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
* more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include <linux/clk.h>
#include <linux/clockchips.h>
#include <linux/cpu.h>
#include <linux/cpumask.h>
#include <linux/err.h>
#include <linux/interrupt.h>
#include <linux/io.h>
#include <linux/of_address.h>
#include <linux/of_irq.h>
#include <linux/percpu.h>
#include <linux/syscore_ops.h>
#include <linux/version.h>
static u32 tegra210_timer_freq;
static void __iomem *tegra210_timer_reg_base;
static u32 usec_config;
static u32 timer_us_mult, timer_us_shift;
#define TIMER_PTV 0x0
#define TIMER_PTV_EN BIT(31)
#define TIMER_PTV_PER BIT(30)
#define TIMER_PCR 0x4
#define TIMER_PCR_INTR_CLR BIT(30)
#define TIMERUS_CNTR_1US 0x10
#define TIMERUS_USEC_CFG 0x14
#define TIMER10_OFFSET 0x90
#define TIMER_FOR_CPU(cpu) (TIMER10_OFFSET + (cpu) * 8)
struct tegra210_clockevent {
struct clock_event_device evt;
char name[20];
void __iomem *reg_base;
struct irqaction irq_action;
bool irq_requested;
};
#define to_tegra_cevt(p) (container_of(p, struct tegra210_clockevent, evt))
static DEFINE_PER_CPU(struct tegra210_clockevent, tegra210_evt);
static int tegra210_timer_set_next_event(unsigned long cycles,
struct clock_event_device *evt)
{
struct tegra210_clockevent *tevt;
tevt = to_tegra_cevt(evt);
writel(TIMER_PTV_EN |
((cycles > 1) ? (cycles - 1) : 0), /* n+1 scheme */
tevt->reg_base + TIMER_PTV);
return 0;
}
static inline void timer_shutdown(struct tegra210_clockevent *tevt)
{
writel(0, tevt->reg_base + TIMER_PTV);
}
static int tegra210_timer_shutdown(struct clock_event_device *evt)
{
struct tegra210_clockevent *tevt;
tevt = to_tegra_cevt(evt);
timer_shutdown(tevt);
return 0;
}
static int tegra210_timer_set_periodic(struct clock_event_device *evt)
{
struct tegra210_clockevent *tevt;
tevt = to_tegra_cevt(evt);
writel(TIMER_PTV_EN | TIMER_PTV_PER | ((tegra210_timer_freq / HZ) - 1),
tevt->reg_base + TIMER_PTV);
return 0;
}
static irqreturn_t tegra210_timer_isr(int irq, void *dev_id)
{
struct tegra210_clockevent *tevt;
tevt = dev_id;
writel(TIMER_PCR_INTR_CLR, tevt->reg_base + TIMER_PCR);
tevt->evt.event_handler(&tevt->evt);
return IRQ_HANDLED;
}
static int tegra210_timer_setup(unsigned int cpu)
{
struct tegra210_clockevent *tevt = &per_cpu(tegra210_evt, cpu);
int ret;
if (!tevt->irq_requested) {
ret = setup_irq(tevt->evt.irq, &tevt->irq_action);
if (ret) {
pr_err("%s: cannot setup irq %d for CPU%d\n",
__func__, tevt->evt.irq, cpu);
return -EPERM;
}
irq_force_affinity(tevt->evt.irq, cpumask_of(cpu));
tevt->irq_requested = true;
} else {
irq_force_affinity(tevt->evt.irq, cpumask_of(cpu));
enable_irq(tevt->evt.irq);
}
clockevents_config_and_register(&tevt->evt, tegra210_timer_freq,
1, /* min */
0x1fffffff); /* 29 bits */
return 0;
}
static int tegra210_timer_stop(unsigned int cpu)
{
struct tegra210_clockevent *tevt = &per_cpu(tegra210_evt, cpu);
tevt->evt.set_state_shutdown(&tevt->evt);
disable_irq_nosync(tevt->evt.irq);
return 0;
}
#if LINUX_VERSION_CODE < KERNEL_VERSION(4, 7, 0)
static int tegra210_timer_cpu_notify(struct notifier_block *self,
unsigned long action, void *hcpu)
{
switch (action & ~CPU_TASKS_FROZEN) {
case CPU_STARTING:
tegra210_timer_setup(smp_processor_id());
break;
case CPU_DYING:
tegra210_timer_stop(smp_processor_id());
break;
}
return NOTIFY_OK;
}
static struct notifier_block tegra210_timer_cpu_nb = {
.notifier_call = tegra210_timer_cpu_notify,
};
#endif
static int tegra_timer_suspend(void)
{
int cpu;
for_each_possible_cpu(cpu) {
void __iomem *reg_base = tegra210_timer_reg_base +
TIMER_FOR_CPU(cpu);
writel(TIMER_PCR_INTR_CLR, reg_base + TIMER_PCR);
}
return 0;
}
static void tegra_timer_resume(void)
{
writel(usec_config, tegra210_timer_reg_base + TIMERUS_USEC_CFG);
}
static struct syscore_ops tegra_timer_syscore_ops = {
.suspend = tegra_timer_suspend,
.resume = tegra_timer_resume,
};
static void __init tegra210_timer_init(struct device_node *np)
{
int cpu;
struct tegra210_clockevent *tevt;
struct clk *clk;
tegra210_timer_reg_base = of_iomap(np, 0);
if (!tegra210_timer_reg_base) {
WARN(1, "Can't map timer registers\n");
return;
}
clk = of_clk_get(np, 0);
if (IS_ERR(clk)) {
WARN(1, "Unable to get timer clock.\n");
return;
}
clk_prepare_enable(clk);
tegra210_timer_freq = clk_get_rate(clk);
for_each_possible_cpu(cpu) {
tevt = &per_cpu(tegra210_evt, cpu);
tevt->reg_base = tegra210_timer_reg_base + TIMER_FOR_CPU(cpu);
tevt->evt.irq = irq_of_parse_and_map(np, cpu);
if (!tevt->evt.irq) {
pr_err("%s: can't map IRQ for CPU%d\n",
__func__, cpu);
BUG();
}
snprintf(tevt->name, ARRAY_SIZE(tevt->name),
"tegra210_timer%d", cpu);
tevt->evt.name = tevt->name;
tevt->evt.cpumask = cpumask_of(cpu);
tevt->evt.set_next_event = tegra210_timer_set_next_event;
tevt->evt.set_state_shutdown = tegra210_timer_shutdown;
tevt->evt.set_state_periodic = tegra210_timer_set_periodic;
tevt->evt.set_state_oneshot = tegra210_timer_shutdown;
tevt->evt.tick_resume = tegra210_timer_shutdown;
tevt->evt.features = CLOCK_EVT_FEAT_PERIODIC |
CLOCK_EVT_FEAT_ONESHOT;
/* want to be preferred over arch timers */
tevt->evt.rating = 460;
tevt->irq_action.name = tevt->evt.name;
tevt->irq_action.flags = IRQF_TIMER | IRQF_NOBALANCING;
tevt->irq_action.handler = tegra210_timer_isr;
tevt->irq_action.dev_id = tevt;
}
/*
* Configure microsecond timers to have 1MHz clock
* Config register is 0xqqww, where qq is "dividend", ww is "divisor"
* Uses n+1 scheme
*/
switch (tegra210_timer_freq) {
case 12000000:
usec_config = 0x000b; /* (11+1)/(0+1) */
break;
case 12800000:
usec_config = 0x043f; /* (63+1)/(4+1) */
break;
case 13000000:
usec_config = 0x000c; /* (12+1)/(0+1) */
break;
case 16800000:
usec_config = 0x0453; /* (83+1)/(4+1) */
break;
case 19200000:
usec_config = 0x045f; /* (95+1)/(4+1) */
break;
case 26000000:
usec_config = 0x0019; /* (25+1)/(0+1) */
break;
case 38400000:
usec_config = 0x04bf; /* (191+1)/(4+1) */
break;
case 48000000:
usec_config = 0x002f; /* (47+1)/(0+1) */
break;
default:
BUG();
}
writel(usec_config, tegra210_timer_reg_base + TIMERUS_USEC_CFG);
clocks_calc_mult_shift(&timer_us_mult, &timer_us_shift,
tegra210_timer_freq, USEC_PER_SEC, 0);
#if LINUX_VERSION_CODE < KERNEL_VERSION(4, 7, 0)
/* boot cpu is online */
tegra210_timer_setup(0);
if (register_cpu_notifier(&tegra210_timer_cpu_nb)) {
WARN(1, "Cannot setup CPU notifier\n");
return;
}
#else
cpuhp_setup_state(CPUHP_AP_TEGRA_TIMER_STARTING,
"AP_TEGRA_TIMER_STARTING", tegra210_timer_setup,
tegra210_timer_stop);
#endif
register_syscore_ops(&tegra_timer_syscore_ops);
}
#if LINUX_VERSION_CODE < KERNEL_VERSION(4, 7, 0)
#define tegra210_timer_init_func tegra210_timer_init
#else
static int __init tegra210_timer_init_ret(struct device_node *np)
{
tegra210_timer_init(np);
return 0;
}
#define tegra210_timer_init_func tegra210_timer_init_ret
#endif
CLOCKSOURCE_OF_DECLARE(tegra210_timer, "nvidia,tegra210-timer",
tegra210_timer_init_func);