| /* | 
 |  * Keystone broadcast clock-event | 
 |  * | 
 |  * Copyright 2013 Texas Instruments, Inc. | 
 |  * | 
 |  * Author: Ivan Khoronzhuk <ivan.khoronzhuk@ti.com> | 
 |  * | 
 |  * This program is free software; you can redistribute it and/or modify | 
 |  * it under the terms of the GNU General Public License version 2 as | 
 |  * published by the Free Software Foundation. | 
 |  * | 
 |  */ | 
 |  | 
 | #include <linux/clk.h> | 
 | #include <linux/clockchips.h> | 
 | #include <linux/clocksource.h> | 
 | #include <linux/interrupt.h> | 
 | #include <linux/of_address.h> | 
 | #include <linux/of_irq.h> | 
 |  | 
 | #define TIMER_NAME			"timer-keystone" | 
 |  | 
 | /* Timer register offsets */ | 
 | #define TIM12				0x10 | 
 | #define TIM34				0x14 | 
 | #define PRD12				0x18 | 
 | #define PRD34				0x1c | 
 | #define TCR				0x20 | 
 | #define TGCR				0x24 | 
 | #define INTCTLSTAT			0x44 | 
 |  | 
 | /* Timer register bitfields */ | 
 | #define TCR_ENAMODE_MASK		0xC0 | 
 | #define TCR_ENAMODE_ONESHOT_MASK	0x40 | 
 | #define TCR_ENAMODE_PERIODIC_MASK	0x80 | 
 |  | 
 | #define TGCR_TIM_UNRESET_MASK		0x03 | 
 | #define INTCTLSTAT_ENINT_MASK		0x01 | 
 |  | 
 | /** | 
 |  * struct keystone_timer: holds timer's data | 
 |  * @base: timer memory base address | 
 |  * @hz_period: cycles per HZ period | 
 |  * @event_dev: event device based on timer | 
 |  */ | 
 | static struct keystone_timer { | 
 | 	void __iomem *base; | 
 | 	unsigned long hz_period; | 
 | 	struct clock_event_device event_dev; | 
 | } timer; | 
 |  | 
 | static inline u32 keystone_timer_readl(unsigned long rg) | 
 | { | 
 | 	return readl_relaxed(timer.base + rg); | 
 | } | 
 |  | 
 | static inline void keystone_timer_writel(u32 val, unsigned long rg) | 
 | { | 
 | 	writel_relaxed(val, timer.base + rg); | 
 | } | 
 |  | 
 | /** | 
 |  * keystone_timer_barrier: write memory barrier | 
 |  * use explicit barrier to avoid using readl/writel non relaxed function | 
 |  * variants, because in our case non relaxed variants hide the true places | 
 |  * where barrier is needed. | 
 |  */ | 
 | static inline void keystone_timer_barrier(void) | 
 | { | 
 | 	__iowmb(); | 
 | } | 
 |  | 
 | /** | 
 |  * keystone_timer_config: configures timer to work in oneshot/periodic modes. | 
 |  * @ mode: mode to configure | 
 |  * @ period: cycles number to configure for | 
 |  */ | 
 | static int keystone_timer_config(u64 period, enum clock_event_mode mode) | 
 | { | 
 | 	u32 tcr; | 
 | 	u32 off; | 
 |  | 
 | 	tcr = keystone_timer_readl(TCR); | 
 | 	off = tcr & ~(TCR_ENAMODE_MASK); | 
 |  | 
 | 	/* set enable mode */ | 
 | 	switch (mode) { | 
 | 	case CLOCK_EVT_MODE_ONESHOT: | 
 | 		tcr |= TCR_ENAMODE_ONESHOT_MASK; | 
 | 		break; | 
 | 	case CLOCK_EVT_MODE_PERIODIC: | 
 | 		tcr |= TCR_ENAMODE_PERIODIC_MASK; | 
 | 		break; | 
 | 	default: | 
 | 		return -1; | 
 | 	} | 
 |  | 
 | 	/* disable timer */ | 
 | 	keystone_timer_writel(off, TCR); | 
 | 	/* here we have to be sure the timer has been disabled */ | 
 | 	keystone_timer_barrier(); | 
 |  | 
 | 	/* reset counter to zero, set new period */ | 
 | 	keystone_timer_writel(0, TIM12); | 
 | 	keystone_timer_writel(0, TIM34); | 
 | 	keystone_timer_writel(period & 0xffffffff, PRD12); | 
 | 	keystone_timer_writel(period >> 32, PRD34); | 
 |  | 
 | 	/* | 
 | 	 * enable timer | 
 | 	 * here we have to be sure that CNTLO, CNTHI, PRDLO, PRDHI registers | 
 | 	 * have been written. | 
 | 	 */ | 
 | 	keystone_timer_barrier(); | 
 | 	keystone_timer_writel(tcr, TCR); | 
 | 	return 0; | 
 | } | 
 |  | 
 | static void keystone_timer_disable(void) | 
 | { | 
 | 	u32 tcr; | 
 |  | 
 | 	tcr = keystone_timer_readl(TCR); | 
 |  | 
 | 	/* disable timer */ | 
 | 	tcr &= ~(TCR_ENAMODE_MASK); | 
 | 	keystone_timer_writel(tcr, TCR); | 
 | } | 
 |  | 
 | static irqreturn_t keystone_timer_interrupt(int irq, void *dev_id) | 
 | { | 
 | 	struct clock_event_device *evt = dev_id; | 
 |  | 
 | 	evt->event_handler(evt); | 
 | 	return IRQ_HANDLED; | 
 | } | 
 |  | 
 | static int keystone_set_next_event(unsigned long cycles, | 
 | 				  struct clock_event_device *evt) | 
 | { | 
 | 	return keystone_timer_config(cycles, evt->mode); | 
 | } | 
 |  | 
 | static void keystone_set_mode(enum clock_event_mode mode, | 
 | 			     struct clock_event_device *evt) | 
 | { | 
 | 	switch (mode) { | 
 | 	case CLOCK_EVT_MODE_PERIODIC: | 
 | 		keystone_timer_config(timer.hz_period, CLOCK_EVT_MODE_PERIODIC); | 
 | 		break; | 
 | 	case CLOCK_EVT_MODE_UNUSED: | 
 | 	case CLOCK_EVT_MODE_SHUTDOWN: | 
 | 	case CLOCK_EVT_MODE_ONESHOT: | 
 | 		keystone_timer_disable(); | 
 | 		break; | 
 | 	default: | 
 | 		break; | 
 | 	} | 
 | } | 
 |  | 
 | static void __init keystone_timer_init(struct device_node *np) | 
 | { | 
 | 	struct clock_event_device *event_dev = &timer.event_dev; | 
 | 	unsigned long rate; | 
 | 	struct clk *clk; | 
 | 	int irq, error; | 
 |  | 
 | 	irq  = irq_of_parse_and_map(np, 0); | 
 | 	if (irq == NO_IRQ) { | 
 | 		pr_err("%s: failed to map interrupts\n", __func__); | 
 | 		return; | 
 | 	} | 
 |  | 
 | 	timer.base = of_iomap(np, 0); | 
 | 	if (!timer.base) { | 
 | 		pr_err("%s: failed to map registers\n", __func__); | 
 | 		return; | 
 | 	} | 
 |  | 
 | 	clk = of_clk_get(np, 0); | 
 | 	if (IS_ERR(clk)) { | 
 | 		pr_err("%s: failed to get clock\n", __func__); | 
 | 		iounmap(timer.base); | 
 | 		return; | 
 | 	} | 
 |  | 
 | 	error = clk_prepare_enable(clk); | 
 | 	if (error) { | 
 | 		pr_err("%s: failed to enable clock\n", __func__); | 
 | 		goto err; | 
 | 	} | 
 |  | 
 | 	rate = clk_get_rate(clk); | 
 |  | 
 | 	/* disable, use internal clock source */ | 
 | 	keystone_timer_writel(0, TCR); | 
 | 	/* here we have to be sure the timer has been disabled */ | 
 | 	keystone_timer_barrier(); | 
 |  | 
 | 	/* reset timer as 64-bit, no pre-scaler, plus features are disabled */ | 
 | 	keystone_timer_writel(0, TGCR); | 
 |  | 
 | 	/* unreset timer */ | 
 | 	keystone_timer_writel(TGCR_TIM_UNRESET_MASK, TGCR); | 
 |  | 
 | 	/* init counter to zero */ | 
 | 	keystone_timer_writel(0, TIM12); | 
 | 	keystone_timer_writel(0, TIM34); | 
 |  | 
 | 	timer.hz_period = DIV_ROUND_UP(rate, HZ); | 
 |  | 
 | 	/* enable timer interrupts */ | 
 | 	keystone_timer_writel(INTCTLSTAT_ENINT_MASK, INTCTLSTAT); | 
 |  | 
 | 	error = request_irq(irq, keystone_timer_interrupt, IRQF_TIMER, | 
 | 			    TIMER_NAME, event_dev); | 
 | 	if (error) { | 
 | 		pr_err("%s: failed to setup irq\n", __func__); | 
 | 		goto err; | 
 | 	} | 
 |  | 
 | 	/* setup clockevent */ | 
 | 	event_dev->features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT; | 
 | 	event_dev->set_next_event = keystone_set_next_event; | 
 | 	event_dev->set_mode = keystone_set_mode; | 
 | 	event_dev->cpumask = cpu_all_mask; | 
 | 	event_dev->owner = THIS_MODULE; | 
 | 	event_dev->name = TIMER_NAME; | 
 | 	event_dev->irq = irq; | 
 |  | 
 | 	clockevents_config_and_register(event_dev, rate, 1, ULONG_MAX); | 
 |  | 
 | 	pr_info("keystone timer clock @%lu Hz\n", rate); | 
 | 	return; | 
 | err: | 
 | 	clk_put(clk); | 
 | 	iounmap(timer.base); | 
 | } | 
 |  | 
 | CLOCKSOURCE_OF_DECLARE(keystone_timer, "ti,keystone-timer", | 
 | 					keystone_timer_init); |