diff options
author | Anson Huang <Anson.Huang@nxp.com> | 2016-11-07 23:44:39 +0800 |
---|---|---|
committer | Leonard Crestez <leonard.crestez@nxp.com> | 2018-08-24 12:41:33 +0300 |
commit | 63f7b1bd764afbcf04f32fc471f2022e06779cd5 (patch) | |
tree | 68eb6604459dde4673415257abf20d2687e79fd3 /drivers/clocksource | |
parent | 3231db15c6b20e2e2f0ed3d8672483842bafaf47 (diff) |
MLK-13441-4 clocksource: tpm: add i.mx tpm driver
Add i.MX TPM support for clock source.
Signed-off-by: Anson Huang <Anson.Huang@nxp.com>
Signed-off-by: Bai Ping <ping.bai@nxp.com>
Diffstat (limited to 'drivers/clocksource')
-rw-r--r-- | drivers/clocksource/Kconfig | 5 | ||||
-rw-r--r-- | drivers/clocksource/Makefile | 1 | ||||
-rw-r--r-- | drivers/clocksource/timer-imx-tpm.c | 185 |
3 files changed, 191 insertions, 0 deletions
diff --git a/drivers/clocksource/Kconfig b/drivers/clocksource/Kconfig index e2c6e43cf8ca..daab8b22a9de 100644 --- a/drivers/clocksource/Kconfig +++ b/drivers/clocksource/Kconfig @@ -532,6 +532,11 @@ config CLKSRC_IMX_GPT depends on ARM && CLKDEV_LOOKUP select CLKSRC_MMIO +config CLKSRC_IMX_TPM + bool "Clocksource using i.MX TPM" if COMPILE_TEST + depends on ARM && CLKDEV_LOOKUP && GENERIC_CLOCKEVENTS + select CLKSRC_MMIO + config CLKSRC_ST_LPC bool "Low power clocksource found in the LPC" if COMPILE_TEST select CLKSRC_OF if OF diff --git a/drivers/clocksource/Makefile b/drivers/clocksource/Makefile index cf87f407f1ad..8015232448d7 100644 --- a/drivers/clocksource/Makefile +++ b/drivers/clocksource/Makefile @@ -63,6 +63,7 @@ obj-$(CONFIG_CLKSRC_VERSATILE) += versatile.o obj-$(CONFIG_CLKSRC_MIPS_GIC) += mips-gic-timer.o obj-$(CONFIG_CLKSRC_TANGO_XTAL) += tango_xtal.o obj-$(CONFIG_CLKSRC_IMX_GPT) += timer-imx-gpt.o +obj-$(CONFIG_CLKSRC_IMX_TPM) += timer-imx-tpm.o obj-$(CONFIG_ASM9260_TIMER) += asm9260_timer.o obj-$(CONFIG_H8300_TMR8) += h8300_timer8.o obj-$(CONFIG_H8300_TMR16) += h8300_timer16.o diff --git a/drivers/clocksource/timer-imx-tpm.c b/drivers/clocksource/timer-imx-tpm.c new file mode 100644 index 000000000000..ffebed83bff7 --- /dev/null +++ b/drivers/clocksource/timer-imx-tpm.c @@ -0,0 +1,185 @@ +/* + * Copyright 2016 Freescale Semiconductor, Inc. + * + * 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. + */ + +#include <linux/interrupt.h> +#include <linux/clk.h> +#include <linux/clockchips.h> +#include <linux/clocksource.h> +#include <linux/delay.h> +#include <linux/of_address.h> +#include <linux/of_irq.h> +#include <linux/sched_clock.h> + +#define TPM_GLOBAL 0x8 + +#define TPM_SC 0x10 +#define TPM_CNT 0x14 +#define TPM_MOD 0x18 +#define TPM_STATUS 0x1c +#define TPM_C0SC 0x20 +#define TPM_C0V 0x24 + +#define TPM_STATUS_CH0F 0x1 + +#define TPM_C0SC_MSA 0x4 + +static void __iomem *timer_base; +static struct clock_event_device clockevent_tpm; + +static inline void tpm_timer_disable(void) +{ + unsigned int val; + + val = __raw_readl(timer_base + TPM_C0SC); + val &= ~(0x5 << TPM_C0SC_MSA); + __raw_writel(val, timer_base + TPM_C0SC); +} + +static inline void tpm_timer_enable(void) +{ + unsigned int val; + + val = __raw_readl(timer_base + TPM_C0SC); + val |= (0x5 << TPM_C0SC_MSA); + __raw_writel(val, timer_base + TPM_C0SC); +} + +static inline void tpm_irq_acknowledge(void) +{ + __raw_writel(1, timer_base + TPM_STATUS); +} + +static struct delay_timer tpm_delay_timer; + +static unsigned long tpm_read_current_timer(void) +{ + return __raw_readl(timer_base + TPM_CNT); +} + +static u64 notrace tpm_read_sched_clock(void) +{ + return __raw_readl(timer_base + TPM_CNT); +} + +static int __init tpm_clocksource_init(unsigned long rate) +{ + tpm_delay_timer.read_current_timer = &tpm_read_current_timer; + tpm_delay_timer.freq = rate; + register_current_timer_delay(&tpm_delay_timer); + + sched_clock_register(tpm_read_sched_clock, 32, rate); + return clocksource_mmio_init(timer_base + TPM_CNT, "imx-tpm", + rate, 200, 32, clocksource_mmio_readl_up); +} + +static int tpm_set_next_event(unsigned long delta, + struct clock_event_device *evt) +{ + unsigned long tmp; + + tmp = __raw_readl(timer_base + TPM_CNT) + delta; + __raw_writel(tmp, timer_base + TPM_C0V); + + return 0; +} + +static int tpm_set_state_oneshot(struct clock_event_device *evt) +{ + /* enable timer */ + tpm_timer_enable(); + + return 0; +} + +static int tpm_set_state_shutdown(struct clock_event_device *evt) +{ + /* disable timer */ + tpm_timer_disable(); + + return 0; +} + +static irqreturn_t tpm_timer_interrupt(int irq, void *dev_id) +{ + struct clock_event_device *evt = &clockevent_tpm; + + tpm_irq_acknowledge(); + + evt->event_handler(evt); + + return IRQ_HANDLED; +} + +static struct clock_event_device clockevent_tpm = { + .name = "i.MX7ULP tpm timer", + .features = CLOCK_EVT_FEAT_ONESHOT, + .set_state_oneshot = tpm_set_state_oneshot, + .set_next_event = tpm_set_next_event, + .set_state_shutdown = tpm_set_state_shutdown, + .rating = 200, +}; + +static struct irqaction tpm_timer_irq = { + .name = "i.MX7ULP tpm timer", + .flags = IRQF_TIMER | IRQF_IRQPOLL, + .handler = tpm_timer_interrupt, + .dev_id = &clockevent_tpm, +}; + +static int __init tpm_clockevent_init(unsigned long rate, int irq) +{ + /* init the channel */ + setup_irq(irq, &tpm_timer_irq); + + clockevent_tpm.cpumask = cpumask_of(0); + clockevent_tpm.irq = irq; + clockevents_config_and_register(&clockevent_tpm, + rate, 2, 0xfffffffe); + + return 0; +} + +static int __init tpm_timer_init(struct device_node *np) +{ + struct clk *clk; + uint32_t val; + int irq; + + timer_base = of_iomap(np, 0); + BUG_ON(!timer_base); + + irq = irq_of_parse_and_map(np, 0); + + clk = of_clk_get(np, 0); + if (IS_ERR(clk)) + return PTR_ERR(clk); + + /* clock shoube be enabled before access to the timer registers */ + clk_prepare_enable(clk); + + /* Initialize tpm module to a known state(counter disabled). */ + __raw_writel(0, timer_base + TPM_SC); + __raw_writel(0, timer_base + TPM_CNT); + __raw_writel(0, timer_base + TPM_C0SC); + + /* set the prescale div, div by 16 */ + __raw_writel(0xc, timer_base + TPM_SC); + + /* set the MOD register to 0xffffffff for free running counter */ + __raw_writel(0xffffffff, timer_base + TPM_MOD); + + tpm_clocksource_init(clk_get_rate(clk) / 16); + tpm_clockevent_init(clk_get_rate(clk) / 16, irq); + + val = __raw_readl(timer_base); + + return 0; +} +CLOCKSOURCE_OF_DECLARE(imx7ulp, "fsl,imx7ulp-tpm", tpm_timer_init); + |