| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  |  * This file contains driver for the Cadence Triple Timer Counter Rev 06 | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  |  *  Copyright (C) 2011-2013 Xilinx | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  |  * | 
					
						
							|  |  |  |  * based on arch/mips/kernel/time.c timer driver | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This software is licensed under the terms of the GNU General Public | 
					
						
							|  |  |  |  * License version 2, as published by the Free Software Foundation, and | 
					
						
							|  |  |  |  * may be copied, distributed, and modified under those terms. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that 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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | #include <linux/clk.h>
 | 
					
						
							| 
									
										
										
										
											2014-02-19 15:14:42 -08:00
										 |  |  | #include <linux/clk-provider.h>
 | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | #include <linux/interrupt.h>
 | 
					
						
							|  |  |  | #include <linux/clockchips.h>
 | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | #include <linux/of_address.h>
 | 
					
						
							|  |  |  | #include <linux/of_irq.h>
 | 
					
						
							|  |  |  | #include <linux/slab.h>
 | 
					
						
							| 
									
										
										
										
											2013-07-08 09:51:38 -07:00
										 |  |  | #include <linux/sched_clock.h>
 | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * This driver configures the 2 16-bit count-up timers as follows: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * T1: Timer 1, clocksource for generic timekeeping | 
					
						
							|  |  |  |  * T2: Timer 2, clockevent source for hrtimers | 
					
						
							|  |  |  |  * T3: Timer 3, <unused> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The input frequency to the timer module for emulation is 2.5MHz which is | 
					
						
							|  |  |  |  * common to all the timer channels (T1, T2, and T3). With a pre-scaler of 32, | 
					
						
							|  |  |  |  * the timers are clocked at 78.125KHz (12.8 us resolution). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  * The input frequency to the timer module in silicon is configurable and | 
					
						
							|  |  |  |  * obtained from device tree. The pre-scaler of 32 is used. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Timer Register Offset Definitions of Timer 1, Increment base address by 4 | 
					
						
							|  |  |  |  * and use same offsets for Timer 2 | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | #define TTC_CLK_CNTRL_OFFSET		0x00 /* Clock Control Reg, RW */
 | 
					
						
							|  |  |  | #define TTC_CNT_CNTRL_OFFSET		0x0C /* Counter Control Reg, RW */
 | 
					
						
							|  |  |  | #define TTC_COUNT_VAL_OFFSET		0x18 /* Counter Value Reg, RO */
 | 
					
						
							|  |  |  | #define TTC_INTR_VAL_OFFSET		0x24 /* Interval Count Reg, RW */
 | 
					
						
							|  |  |  | #define TTC_ISR_OFFSET		0x54 /* Interrupt Status Reg, RO */
 | 
					
						
							|  |  |  | #define TTC_IER_OFFSET		0x60 /* Interrupt Enable Reg, RW */
 | 
					
						
							| 
									
										
										
										
											2012-12-19 10:18:36 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | #define TTC_CNT_CNTRL_DISABLE_MASK	0x1
 | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-13 10:46:38 -07:00
										 |  |  | #define TTC_CLK_CNTRL_CSRC_MASK		(1 << 5)	/* clock source */
 | 
					
						
							| 
									
										
										
										
											2014-02-19 15:14:42 -08:00
										 |  |  | #define TTC_CLK_CNTRL_PSV_MASK		0x1e
 | 
					
						
							|  |  |  | #define TTC_CLK_CNTRL_PSV_SHIFT		1
 | 
					
						
							| 
									
										
										
										
											2013-05-13 10:46:38 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-19 10:18:41 -08:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Setup the timers to use pre-scaling, using a fixed value for now that will | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  |  * work across most input frequency, but it may need to be more dynamic | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #define PRESCALE_EXPONENT	11	/* 2 ^ PRESCALE_EXPONENT = PRESCALE */
 | 
					
						
							|  |  |  | #define PRESCALE		2048	/* The exponent must match this */
 | 
					
						
							|  |  |  | #define CLK_CNTRL_PRESCALE	((PRESCALE_EXPONENT - 1) << 1)
 | 
					
						
							|  |  |  | #define CLK_CNTRL_PRESCALE_EN	1
 | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | #define CNT_CNTRL_RESET		(1 << 4)
 | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-19 15:14:42 -08:00
										 |  |  | #define MAX_F_ERR 50
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  |  * struct ttc_timer - This definition defines local timer structure | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  |  * | 
					
						
							|  |  |  |  * @base_addr:	Base address of timer | 
					
						
							| 
									
										
										
										
											2013-11-26 17:04:50 -08:00
										 |  |  |  * @freq:	Timer input clock frequency | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  |  * @clk:	Associated clock source | 
					
						
							|  |  |  |  * @clk_rate_change_nb	Notifier block for clock rate changes | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | struct ttc_timer { | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 	void __iomem *base_addr; | 
					
						
							| 
									
										
										
										
											2013-11-26 17:04:50 -08:00
										 |  |  | 	unsigned long freq; | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 	struct clk *clk; | 
					
						
							|  |  |  | 	struct notifier_block clk_rate_change_nb; | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | #define to_ttc_timer(x) \
 | 
					
						
							|  |  |  | 		container_of(x, struct ttc_timer, clk_rate_change_nb) | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | struct ttc_timer_clocksource { | 
					
						
							| 
									
										
										
										
											2014-02-19 15:14:42 -08:00
										 |  |  | 	u32			scale_clk_ctrl_reg_old; | 
					
						
							|  |  |  | 	u32			scale_clk_ctrl_reg_new; | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	struct ttc_timer	ttc; | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 	struct clocksource	cs; | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | #define to_ttc_timer_clksrc(x) \
 | 
					
						
							|  |  |  | 		container_of(x, struct ttc_timer_clocksource, cs) | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | struct ttc_timer_clockevent { | 
					
						
							|  |  |  | 	struct ttc_timer		ttc; | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 	struct clock_event_device	ce; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | #define to_ttc_timer_clkevent(x) \
 | 
					
						
							|  |  |  | 		container_of(x, struct ttc_timer_clockevent, ce) | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-08 09:51:38 -07:00
										 |  |  | static void __iomem *ttc_sched_clock_val_reg; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  |  * ttc_set_interval - Set the timer interval value | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  |  * | 
					
						
							|  |  |  |  * @timer:	Pointer to the timer instance | 
					
						
							|  |  |  |  * @cycles:	Timer interval ticks | 
					
						
							|  |  |  |  **/ | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | static void ttc_set_interval(struct ttc_timer *timer, | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 					unsigned long cycles) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	u32 ctrl_reg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Disable the counter, set the counter value  and re-enable counter */ | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 	ctrl_reg = readl_relaxed(timer->base_addr + TTC_CNT_CNTRL_OFFSET); | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	ctrl_reg |= TTC_CNT_CNTRL_DISABLE_MASK; | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 	writel_relaxed(ctrl_reg, timer->base_addr + TTC_CNT_CNTRL_OFFSET); | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 	writel_relaxed(cycles, timer->base_addr + TTC_INTR_VAL_OFFSET); | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-19 10:18:41 -08:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * Reset the counter (0x10) so that it starts from 0, one-shot | 
					
						
							|  |  |  | 	 * mode makes this needed for timing to be right. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 	ctrl_reg |= CNT_CNTRL_RESET; | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	ctrl_reg &= ~TTC_CNT_CNTRL_DISABLE_MASK; | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 	writel_relaxed(ctrl_reg, timer->base_addr + TTC_CNT_CNTRL_OFFSET); | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  |  * ttc_clock_event_interrupt - Clock event timer interrupt handler | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  |  * | 
					
						
							|  |  |  |  * @irq:	IRQ number of the Timer | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  |  * @dev_id:	void pointer to the ttc_timer instance | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  |  * | 
					
						
							|  |  |  |  * returns: Always IRQ_HANDLED - success | 
					
						
							|  |  |  |  **/ | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | static irqreturn_t ttc_clock_event_interrupt(int irq, void *dev_id) | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	struct ttc_timer_clockevent *ttce = dev_id; | 
					
						
							|  |  |  | 	struct ttc_timer *timer = &ttce->ttc; | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Acknowledge the interrupt and call event handler */ | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 	readl_relaxed(timer->base_addr + TTC_ISR_OFFSET); | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	ttce->ce.event_handler(&ttce->ce); | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return IRQ_HANDLED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  |  * __ttc_clocksource_read - Reads the timer counter register | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  |  * | 
					
						
							|  |  |  |  * returns: Current timer counter register value | 
					
						
							|  |  |  |  **/ | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | static cycle_t __ttc_clocksource_read(struct clocksource *cs) | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	struct ttc_timer *timer = &to_ttc_timer_clksrc(cs)->ttc; | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 	return (cycle_t)readl_relaxed(timer->base_addr + | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 				TTC_COUNT_VAL_OFFSET); | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-20 00:47:32 +01:00
										 |  |  | static u64 notrace ttc_sched_clock_read(void) | 
					
						
							| 
									
										
										
										
											2013-07-08 09:51:38 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 	return readl_relaxed(ttc_sched_clock_val_reg); | 
					
						
							| 
									
										
										
										
											2013-07-08 09:51:38 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  |  * ttc_set_next_event - Sets the time interval for next event | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  |  * | 
					
						
							|  |  |  |  * @cycles:	Timer interval ticks | 
					
						
							|  |  |  |  * @evt:	Address of clock event instance | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * returns: Always 0 - success | 
					
						
							|  |  |  |  **/ | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | static int ttc_set_next_event(unsigned long cycles, | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 					struct clock_event_device *evt) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	struct ttc_timer_clockevent *ttce = to_ttc_timer_clkevent(evt); | 
					
						
							|  |  |  | 	struct ttc_timer *timer = &ttce->ttc; | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	ttc_set_interval(timer, cycles); | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  |  * ttc_set_mode - Sets the mode of timer | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  |  * | 
					
						
							|  |  |  |  * @mode:	Mode to be set | 
					
						
							|  |  |  |  * @evt:	Address of clock event instance | 
					
						
							|  |  |  |  **/ | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | static void ttc_set_mode(enum clock_event_mode mode, | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 					struct clock_event_device *evt) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	struct ttc_timer_clockevent *ttce = to_ttc_timer_clkevent(evt); | 
					
						
							|  |  |  | 	struct ttc_timer *timer = &ttce->ttc; | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 	u32 ctrl_reg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (mode) { | 
					
						
							|  |  |  | 	case CLOCK_EVT_MODE_PERIODIC: | 
					
						
							| 
									
										
										
										
											2013-11-26 17:04:50 -08:00
										 |  |  | 		ttc_set_interval(timer, DIV_ROUND_CLOSEST(ttce->ttc.freq, | 
					
						
							|  |  |  | 						PRESCALE * HZ)); | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 	case CLOCK_EVT_MODE_ONESHOT: | 
					
						
							|  |  |  | 	case CLOCK_EVT_MODE_UNUSED: | 
					
						
							|  |  |  | 	case CLOCK_EVT_MODE_SHUTDOWN: | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 		ctrl_reg = readl_relaxed(timer->base_addr + | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 					TTC_CNT_CNTRL_OFFSET); | 
					
						
							|  |  |  | 		ctrl_reg |= TTC_CNT_CNTRL_DISABLE_MASK; | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 		writel_relaxed(ctrl_reg, | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 				timer->base_addr + TTC_CNT_CNTRL_OFFSET); | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 	case CLOCK_EVT_MODE_RESUME: | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 		ctrl_reg = readl_relaxed(timer->base_addr + | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 					TTC_CNT_CNTRL_OFFSET); | 
					
						
							|  |  |  | 		ctrl_reg &= ~TTC_CNT_CNTRL_DISABLE_MASK; | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 		writel_relaxed(ctrl_reg, | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 				timer->base_addr + TTC_CNT_CNTRL_OFFSET); | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | static int ttc_rate_change_clocksource_cb(struct notifier_block *nb, | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 		unsigned long event, void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct clk_notifier_data *ndata = data; | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	struct ttc_timer *ttc = to_ttc_timer(nb); | 
					
						
							|  |  |  | 	struct ttc_timer_clocksource *ttccs = container_of(ttc, | 
					
						
							|  |  |  | 			struct ttc_timer_clocksource, ttc); | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	switch (event) { | 
					
						
							| 
									
										
										
										
											2014-02-19 15:14:42 -08:00
										 |  |  | 	case PRE_RATE_CHANGE: | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		u32 psv; | 
					
						
							|  |  |  | 		unsigned long factor, rate_low, rate_high; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (ndata->new_rate > ndata->old_rate) { | 
					
						
							|  |  |  | 			factor = DIV_ROUND_CLOSEST(ndata->new_rate, | 
					
						
							|  |  |  | 					ndata->old_rate); | 
					
						
							|  |  |  | 			rate_low = ndata->old_rate; | 
					
						
							|  |  |  | 			rate_high = ndata->new_rate; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			factor = DIV_ROUND_CLOSEST(ndata->old_rate, | 
					
						
							|  |  |  | 					ndata->new_rate); | 
					
						
							|  |  |  | 			rate_low = ndata->new_rate; | 
					
						
							|  |  |  | 			rate_high = ndata->old_rate; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (!is_power_of_2(factor)) | 
					
						
							|  |  |  | 				return NOTIFY_BAD; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (abs(rate_high - (factor * rate_low)) > MAX_F_ERR) | 
					
						
							|  |  |  | 			return NOTIFY_BAD; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		factor = __ilog2_u32(factor); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 		/*
 | 
					
						
							| 
									
										
										
										
											2014-02-19 15:14:42 -08:00
										 |  |  | 		 * store timer clock ctrl register so we can restore it in case | 
					
						
							|  |  |  | 		 * of an abort. | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2014-02-19 15:14:42 -08:00
										 |  |  | 		ttccs->scale_clk_ctrl_reg_old = | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 			readl_relaxed(ttccs->ttc.base_addr + | 
					
						
							|  |  |  | 			TTC_CLK_CNTRL_OFFSET); | 
					
						
							| 
									
										
										
										
											2014-02-19 15:14:42 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		psv = (ttccs->scale_clk_ctrl_reg_old & | 
					
						
							|  |  |  | 				TTC_CLK_CNTRL_PSV_MASK) >> | 
					
						
							|  |  |  | 				TTC_CLK_CNTRL_PSV_SHIFT; | 
					
						
							|  |  |  | 		if (ndata->new_rate < ndata->old_rate) | 
					
						
							|  |  |  | 			psv -= factor; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			psv += factor; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* prescaler within legal range? */ | 
					
						
							|  |  |  | 		if (psv & ~(TTC_CLK_CNTRL_PSV_MASK >> TTC_CLK_CNTRL_PSV_SHIFT)) | 
					
						
							|  |  |  | 			return NOTIFY_BAD; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		ttccs->scale_clk_ctrl_reg_new = ttccs->scale_clk_ctrl_reg_old & | 
					
						
							|  |  |  | 			~TTC_CLK_CNTRL_PSV_MASK; | 
					
						
							|  |  |  | 		ttccs->scale_clk_ctrl_reg_new |= psv << TTC_CLK_CNTRL_PSV_SHIFT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* scale down: adjust divider in post-change notification */ | 
					
						
							|  |  |  | 		if (ndata->new_rate < ndata->old_rate) | 
					
						
							|  |  |  | 			return NOTIFY_DONE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* scale up: adjust divider now - before frequency change */ | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 		writel_relaxed(ttccs->scale_clk_ctrl_reg_new, | 
					
						
							|  |  |  | 			       ttccs->ttc.base_addr + TTC_CLK_CNTRL_OFFSET); | 
					
						
							| 
									
										
										
										
											2014-02-19 15:14:42 -08:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	case POST_RATE_CHANGE: | 
					
						
							|  |  |  | 		/* scale up: pre-change notification did the adjustment */ | 
					
						
							|  |  |  | 		if (ndata->new_rate > ndata->old_rate) | 
					
						
							|  |  |  | 			return NOTIFY_OK; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* scale down: adjust divider now - after frequency change */ | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 		writel_relaxed(ttccs->scale_clk_ctrl_reg_new, | 
					
						
							|  |  |  | 			       ttccs->ttc.base_addr + TTC_CLK_CNTRL_OFFSET); | 
					
						
							| 
									
										
										
										
											2014-02-19 15:14:42 -08:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 	case ABORT_RATE_CHANGE: | 
					
						
							| 
									
										
										
										
											2014-02-19 15:14:42 -08:00
										 |  |  | 		/* we have to undo the adjustment in case we scale up */ | 
					
						
							|  |  |  | 		if (ndata->new_rate < ndata->old_rate) | 
					
						
							|  |  |  | 			return NOTIFY_OK; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* restore original register value */ | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 		writel_relaxed(ttccs->scale_clk_ctrl_reg_old, | 
					
						
							|  |  |  | 			       ttccs->ttc.base_addr + TTC_CLK_CNTRL_OFFSET); | 
					
						
							| 
									
										
										
										
											2014-02-19 15:14:42 -08:00
										 |  |  | 		/* fall through */ | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 	default: | 
					
						
							|  |  |  | 		return NOTIFY_DONE; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-02-19 15:14:42 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return NOTIFY_DONE; | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | static void __init ttc_setup_clocksource(struct clk *clk, void __iomem *base) | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	struct ttc_timer_clocksource *ttccs; | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 	int err; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ttccs = kzalloc(sizeof(*ttccs), GFP_KERNEL); | 
					
						
							|  |  |  | 	if (WARN_ON(!ttccs)) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	ttccs->ttc.clk = clk; | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	err = clk_prepare_enable(ttccs->ttc.clk); | 
					
						
							| 
									
										
										
										
											2013-03-20 10:24:59 +01:00
										 |  |  | 	if (WARN_ON(err)) { | 
					
						
							|  |  |  | 		kfree(ttccs); | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2013-03-20 10:24:59 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-26 17:04:50 -08:00
										 |  |  | 	ttccs->ttc.freq = clk_get_rate(ttccs->ttc.clk); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	ttccs->ttc.clk_rate_change_nb.notifier_call = | 
					
						
							|  |  |  | 		ttc_rate_change_clocksource_cb; | 
					
						
							|  |  |  | 	ttccs->ttc.clk_rate_change_nb.next = NULL; | 
					
						
							|  |  |  | 	if (clk_notifier_register(ttccs->ttc.clk, | 
					
						
							|  |  |  | 				&ttccs->ttc.clk_rate_change_nb)) | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 		pr_warn("Unable to register clock notifier.\n"); | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	ttccs->ttc.base_addr = base; | 
					
						
							|  |  |  | 	ttccs->cs.name = "ttc_clocksource"; | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 	ttccs->cs.rating = 200; | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	ttccs->cs.read = __ttc_clocksource_read; | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 	ttccs->cs.mask = CLOCKSOURCE_MASK(16); | 
					
						
							|  |  |  | 	ttccs->cs.flags = CLOCK_SOURCE_IS_CONTINUOUS; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * Setup the clock source counter to be an incrementing counter | 
					
						
							|  |  |  | 	 * with no interrupt and it rolls over at 0xFFFF. Pre-scale | 
					
						
							|  |  |  | 	 * it by 32 also. Let it start running now. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 	writel_relaxed(0x0,  ttccs->ttc.base_addr + TTC_IER_OFFSET); | 
					
						
							|  |  |  | 	writel_relaxed(CLK_CNTRL_PRESCALE | CLK_CNTRL_PRESCALE_EN, | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 		     ttccs->ttc.base_addr + TTC_CLK_CNTRL_OFFSET); | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 	writel_relaxed(CNT_CNTRL_RESET, | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 		     ttccs->ttc.base_addr + TTC_CNT_CNTRL_OFFSET); | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-26 17:04:50 -08:00
										 |  |  | 	err = clocksource_register_hz(&ttccs->cs, ttccs->ttc.freq / PRESCALE); | 
					
						
							| 
									
										
										
										
											2013-03-20 10:24:59 +01:00
										 |  |  | 	if (WARN_ON(err)) { | 
					
						
							|  |  |  | 		kfree(ttccs); | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2013-03-20 10:24:59 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-07-08 09:51:38 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ttc_sched_clock_val_reg = base + TTC_COUNT_VAL_OFFSET; | 
					
						
							| 
									
										
										
										
											2014-01-20 11:34:26 -08:00
										 |  |  | 	sched_clock_register(ttc_sched_clock_read, 16, ttccs->ttc.freq / PRESCALE); | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | static int ttc_rate_change_clockevent_cb(struct notifier_block *nb, | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 		unsigned long event, void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct clk_notifier_data *ndata = data; | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	struct ttc_timer *ttc = to_ttc_timer(nb); | 
					
						
							|  |  |  | 	struct ttc_timer_clockevent *ttcce = container_of(ttc, | 
					
						
							|  |  |  | 			struct ttc_timer_clockevent, ttc); | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	switch (event) { | 
					
						
							|  |  |  | 	case POST_RATE_CHANGE: | 
					
						
							| 
									
										
										
										
											2013-11-26 17:04:50 -08:00
										 |  |  | 		/* update cached frequency */ | 
					
						
							|  |  |  | 		ttc->freq = ndata->new_rate; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-19 15:14:41 -08:00
										 |  |  | 		clockevents_update_freq(&ttcce->ce, ndata->new_rate / PRESCALE); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 		/* fall through */ | 
					
						
							|  |  |  | 	case PRE_RATE_CHANGE: | 
					
						
							|  |  |  | 	case ABORT_RATE_CHANGE: | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return NOTIFY_DONE; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | static void __init ttc_setup_clockevent(struct clk *clk, | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 						void __iomem *base, u32 irq) | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	struct ttc_timer_clockevent *ttcce; | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 	int err; | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ttcce = kzalloc(sizeof(*ttcce), GFP_KERNEL); | 
					
						
							|  |  |  | 	if (WARN_ON(!ttcce)) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	ttcce->ttc.clk = clk; | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	err = clk_prepare_enable(ttcce->ttc.clk); | 
					
						
							| 
									
										
										
										
											2013-03-20 10:24:59 +01:00
										 |  |  | 	if (WARN_ON(err)) { | 
					
						
							|  |  |  | 		kfree(ttcce); | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2013-03-20 10:24:59 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	ttcce->ttc.clk_rate_change_nb.notifier_call = | 
					
						
							|  |  |  | 		ttc_rate_change_clockevent_cb; | 
					
						
							|  |  |  | 	ttcce->ttc.clk_rate_change_nb.next = NULL; | 
					
						
							|  |  |  | 	if (clk_notifier_register(ttcce->ttc.clk, | 
					
						
							|  |  |  | 				&ttcce->ttc.clk_rate_change_nb)) | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 		pr_warn("Unable to register clock notifier.\n"); | 
					
						
							| 
									
										
										
										
											2013-11-26 17:04:50 -08:00
										 |  |  | 	ttcce->ttc.freq = clk_get_rate(ttcce->ttc.clk); | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	ttcce->ttc.base_addr = base; | 
					
						
							|  |  |  | 	ttcce->ce.name = "ttc_clockevent"; | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 	ttcce->ce.features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT; | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	ttcce->ce.set_next_event = ttc_set_next_event; | 
					
						
							|  |  |  | 	ttcce->ce.set_mode = ttc_set_mode; | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 	ttcce->ce.rating = 200; | 
					
						
							|  |  |  | 	ttcce->ce.irq = irq; | 
					
						
							| 
									
										
										
										
											2012-12-19 10:18:42 -08:00
										 |  |  | 	ttcce->ce.cpumask = cpu_possible_mask; | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * Setup the clock event timer to be an interval timer which | 
					
						
							|  |  |  | 	 * is prescaled by 32 using the interval interrupt. Leave it | 
					
						
							|  |  |  | 	 * disabled for now. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 	writel_relaxed(0x23, ttcce->ttc.base_addr + TTC_CNT_CNTRL_OFFSET); | 
					
						
							|  |  |  | 	writel_relaxed(CLK_CNTRL_PRESCALE | CLK_CNTRL_PRESCALE_EN, | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 		     ttcce->ttc.base_addr + TTC_CLK_CNTRL_OFFSET); | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 	writel_relaxed(0x1,  ttcce->ttc.base_addr + TTC_IER_OFFSET); | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | 	err = request_irq(irq, ttc_clock_event_interrupt, | 
					
						
							| 
									
										
										
										
											2013-12-09 10:12:10 +01:00
										 |  |  | 			  IRQF_TIMER, ttcce->ce.name, ttcce); | 
					
						
							| 
									
										
										
										
											2013-03-20 10:24:59 +01:00
										 |  |  | 	if (WARN_ON(err)) { | 
					
						
							|  |  |  | 		kfree(ttcce); | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2013-03-20 10:24:59 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	clockevents_config_and_register(&ttcce->ce, | 
					
						
							| 
									
										
										
										
											2013-11-26 17:04:50 -08:00
										 |  |  | 			ttcce->ttc.freq / PRESCALE, 1, 0xfffe); | 
					
						
							| 
									
										
										
										
											2012-10-31 13:56:14 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  |  * ttc_timer_init - Initialize the timer | 
					
						
							| 
									
										
										
										
											2011-06-20 11:47:27 -06:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Initializes the timer hardware and register the clock source and clock event | 
					
						
							|  |  |  |  * timers with Linux kernal timer framework | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | static void __init ttc_timer_init(struct device_node *timer) | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | { | 
					
						
							|  |  |  | 	unsigned int irq; | 
					
						
							|  |  |  | 	void __iomem *timer_baseaddr; | 
					
						
							| 
									
										
										
										
											2013-05-13 10:46:38 -07:00
										 |  |  | 	struct clk *clk_cs, *clk_ce; | 
					
						
							| 
									
										
										
										
											2013-03-20 10:24:59 +01:00
										 |  |  | 	static int initialized; | 
					
						
							| 
									
										
										
										
											2013-05-13 10:46:38 -07:00
										 |  |  | 	int clksel; | 
					
						
							| 
									
										
										
										
											2013-03-20 10:24:59 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (initialized) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	initialized = 1; | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * Get the 1st Triple Timer Counter (TTC) block from the device tree | 
					
						
							|  |  |  | 	 * and use it. Note that the event timer uses the interrupt and it's the | 
					
						
							|  |  |  | 	 * 2nd TTC hence the irq_of_parse_and_map(,1) | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	timer_baseaddr = of_iomap(timer, 0); | 
					
						
							|  |  |  | 	if (!timer_baseaddr) { | 
					
						
							|  |  |  | 		pr_err("ERROR: invalid timer base address\n"); | 
					
						
							|  |  |  | 		BUG(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	irq = irq_of_parse_and_map(timer, 1); | 
					
						
							|  |  |  | 	if (irq <= 0) { | 
					
						
							|  |  |  | 		pr_err("ERROR: invalid interrupt number\n"); | 
					
						
							|  |  |  | 		BUG(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 	clksel = readl_relaxed(timer_baseaddr + TTC_CLK_CNTRL_OFFSET); | 
					
						
							| 
									
										
										
										
											2013-05-13 10:46:38 -07:00
										 |  |  | 	clksel = !!(clksel & TTC_CLK_CNTRL_CSRC_MASK); | 
					
						
							|  |  |  | 	clk_cs = of_clk_get(timer, clksel); | 
					
						
							|  |  |  | 	if (IS_ERR(clk_cs)) { | 
					
						
							|  |  |  | 		pr_err("ERROR: timer input clock not found\n"); | 
					
						
							|  |  |  | 		BUG(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-11 15:39:29 +02:00
										 |  |  | 	clksel = readl_relaxed(timer_baseaddr + 4 + TTC_CLK_CNTRL_OFFSET); | 
					
						
							| 
									
										
										
										
											2013-05-13 10:46:38 -07:00
										 |  |  | 	clksel = !!(clksel & TTC_CLK_CNTRL_CSRC_MASK); | 
					
						
							|  |  |  | 	clk_ce = of_clk_get(timer, clksel); | 
					
						
							|  |  |  | 	if (IS_ERR(clk_ce)) { | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 		pr_err("ERROR: timer input clock not found\n"); | 
					
						
							|  |  |  | 		BUG(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-13 10:46:38 -07:00
										 |  |  | 	ttc_setup_clocksource(clk_cs, timer_baseaddr); | 
					
						
							|  |  |  | 	ttc_setup_clockevent(clk_ce, timer_baseaddr + 4, irq); | 
					
						
							| 
									
										
										
										
											2013-03-20 10:15:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	pr_info("%s #0 at %p, irq=%d\n", timer->name, timer_baseaddr, irq); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 12:05:28 +01:00
										 |  |  | CLOCKSOURCE_OF_DECLARE(ttc, "cdns,ttc", ttc_timer_init); |