| /* |
| * DaVinci timer subsystem |
| * |
| * Author: Kevin Hilman, MontaVista Software, Inc. <source@mvista.com> |
| * |
| * 2007 (c) MontaVista Software, Inc. This file is licensed under |
| * the terms of the GNU General Public License version 2. This program |
| * is licensed "as is" without any warranty of any kind, whether express |
| * or implied. |
| */ |
| #include <linux/kernel.h> |
| #include <linux/init.h> |
| #include <linux/types.h> |
| #include <linux/interrupt.h> |
| #include <linux/clocksource.h> |
| #include <linux/clockchips.h> |
| #include <linux/spinlock.h> |
| |
| #include <asm/io.h> |
| #include <asm/hardware.h> |
| #include <asm/system.h> |
| #include <asm/irq.h> |
| #include <asm/mach/irq.h> |
| #include <asm/mach/time.h> |
| #include <asm/errno.h> |
| #include <asm/arch/io.h> |
| |
| static struct clock_event_device clockevent_davinci; |
| |
| #define DAVINCI_TIMER0_BASE (IO_PHYS + 0x21400) |
| #define DAVINCI_TIMER1_BASE (IO_PHYS + 0x21800) |
| #define DAVINCI_WDOG_BASE (IO_PHYS + 0x21C00) |
| |
| enum { |
| T0_BOT = 0, T0_TOP, T1_BOT, T1_TOP, NUM_TIMERS, |
| }; |
| |
| #define IS_TIMER1(id) (id & 0x2) |
| #define IS_TIMER0(id) (!IS_TIMER1(id)) |
| #define IS_TIMER_TOP(id) ((id & 0x1)) |
| #define IS_TIMER_BOT(id) (!IS_TIMER_TOP(id)) |
| |
| static int timer_irqs[NUM_TIMERS] = { |
| IRQ_TINT0_TINT12, |
| IRQ_TINT0_TINT34, |
| IRQ_TINT1_TINT12, |
| IRQ_TINT1_TINT34, |
| }; |
| |
| /* |
| * This driver configures the 2 64-bit count-up timers as 4 independent |
| * 32-bit count-up timers used as follows: |
| * |
| * T0_BOT: Timer 0, bottom: clockevent source for hrtimers |
| * T0_TOP: Timer 0, top : clocksource for generic timekeeping |
| * T1_BOT: Timer 1, bottom: (used by DSP in TI DSPLink code) |
| * T1_TOP: Timer 1, top : <unused> |
| */ |
| #define TID_CLOCKEVENT T0_BOT |
| #define TID_CLOCKSOURCE T0_TOP |
| |
| /* Timer register offsets */ |
| #define PID12 0x0 |
| #define TIM12 0x10 |
| #define TIM34 0x14 |
| #define PRD12 0x18 |
| #define PRD34 0x1c |
| #define TCR 0x20 |
| #define TGCR 0x24 |
| #define WDTCR 0x28 |
| |
| /* Timer register bitfields */ |
| #define TCR_ENAMODE_DISABLE 0x0 |
| #define TCR_ENAMODE_ONESHOT 0x1 |
| #define TCR_ENAMODE_PERIODIC 0x2 |
| #define TCR_ENAMODE_MASK 0x3 |
| |
| #define TGCR_TIMMODE_SHIFT 2 |
| #define TGCR_TIMMODE_64BIT_GP 0x0 |
| #define TGCR_TIMMODE_32BIT_UNCHAINED 0x1 |
| #define TGCR_TIMMODE_64BIT_WDOG 0x2 |
| #define TGCR_TIMMODE_32BIT_CHAINED 0x3 |
| |
| #define TGCR_TIM12RS_SHIFT 0 |
| #define TGCR_TIM34RS_SHIFT 1 |
| #define TGCR_RESET 0x0 |
| #define TGCR_UNRESET 0x1 |
| #define TGCR_RESET_MASK 0x3 |
| |
| #define WDTCR_WDEN_SHIFT 14 |
| #define WDTCR_WDEN_DISABLE 0x0 |
| #define WDTCR_WDEN_ENABLE 0x1 |
| #define WDTCR_WDKEY_SHIFT 16 |
| #define WDTCR_WDKEY_SEQ0 0xa5c6 |
| #define WDTCR_WDKEY_SEQ1 0xda7e |
| |
| struct timer_s { |
| char *name; |
| unsigned int id; |
| unsigned long period; |
| unsigned long opts; |
| unsigned long reg_base; |
| unsigned long tim_reg; |
| unsigned long prd_reg; |
| unsigned long enamode_shift; |
| struct irqaction irqaction; |
| }; |
| static struct timer_s timers[]; |
| |
| /* values for 'opts' field of struct timer_s */ |
| #define TIMER_OPTS_DISABLED 0x00 |
| #define TIMER_OPTS_ONESHOT 0x01 |
| #define TIMER_OPTS_PERIODIC 0x02 |
| |
| static int timer32_config(struct timer_s *t) |
| { |
| u32 tcr = davinci_readl(t->reg_base + TCR); |
| |
| /* disable timer */ |
| tcr &= ~(TCR_ENAMODE_MASK << t->enamode_shift); |
| davinci_writel(tcr, t->reg_base + TCR); |
| |
| /* reset counter to zero, set new period */ |
| davinci_writel(0, t->tim_reg); |
| davinci_writel(t->period, t->prd_reg); |
| |
| /* Set enable mode */ |
| if (t->opts & TIMER_OPTS_ONESHOT) { |
| tcr |= TCR_ENAMODE_ONESHOT << t->enamode_shift; |
| } else if (t->opts & TIMER_OPTS_PERIODIC) { |
| tcr |= TCR_ENAMODE_PERIODIC << t->enamode_shift; |
| } |
| |
| davinci_writel(tcr, t->reg_base + TCR); |
| return 0; |
| } |
| |
| static inline u32 timer32_read(struct timer_s *t) |
| { |
| return davinci_readl(t->tim_reg); |
| } |
| |
| static irqreturn_t timer_interrupt(int irq, void *dev_id) |
| { |
| struct clock_event_device *evt = &clockevent_davinci; |
| |
| evt->event_handler(evt); |
| return IRQ_HANDLED; |
| } |
| |
| /* called when 32-bit counter wraps */ |
| static irqreturn_t freerun_interrupt(int irq, void *dev_id) |
| { |
| return IRQ_HANDLED; |
| } |
| |
| static struct timer_s timers[] = { |
| [TID_CLOCKEVENT] = { |
| .name = "clockevent", |
| .opts = TIMER_OPTS_DISABLED, |
| .irqaction = { |
| .flags = IRQF_DISABLED | IRQF_TIMER, |
| .handler = timer_interrupt, |
| } |
| }, |
| [TID_CLOCKSOURCE] = { |
| .name = "free-run counter", |
| .period = ~0, |
| .opts = TIMER_OPTS_PERIODIC, |
| .irqaction = { |
| .flags = IRQF_DISABLED | IRQF_TIMER, |
| .handler = freerun_interrupt, |
| } |
| }, |
| }; |
| |
| static void __init timer_init(void) |
| { |
| u32 bases[] = {DAVINCI_TIMER0_BASE, DAVINCI_TIMER1_BASE}; |
| int i; |
| |
| /* Global init of each 64-bit timer as a whole */ |
| for(i=0; i<2; i++) { |
| u32 tgcr, base = bases[i]; |
| |
| /* Disabled, Internal clock source */ |
| davinci_writel(0, base + TCR); |
| |
| /* reset both timers, no pre-scaler for timer34 */ |
| tgcr = 0; |
| davinci_writel(tgcr, base + TGCR); |
| |
| /* Set both timers to unchained 32-bit */ |
| tgcr = TGCR_TIMMODE_32BIT_UNCHAINED << TGCR_TIMMODE_SHIFT; |
| davinci_writel(tgcr, base + TGCR); |
| |
| /* Unreset timers */ |
| tgcr |= (TGCR_UNRESET << TGCR_TIM12RS_SHIFT) | |
| (TGCR_UNRESET << TGCR_TIM34RS_SHIFT); |
| davinci_writel(tgcr, base + TGCR); |
| |
| /* Init both counters to zero */ |
| davinci_writel(0, base + TIM12); |
| davinci_writel(0, base + TIM34); |
| } |
| |
| /* Init of each timer as a 32-bit timer */ |
| for (i=0; i< ARRAY_SIZE(timers); i++) { |
| struct timer_s *t = &timers[i]; |
| |
| if (t->name) { |
| t->id = i; |
| t->reg_base = (IS_TIMER1(t->id) ? |
| DAVINCI_TIMER1_BASE : DAVINCI_TIMER0_BASE); |
| |
| if (IS_TIMER_BOT(t->id)) { |
| t->enamode_shift = 6; |
| t->tim_reg = t->reg_base + TIM12; |
| t->prd_reg = t->reg_base + PRD12; |
| } else { |
| t->enamode_shift = 22; |
| t->tim_reg = t->reg_base + TIM34; |
| t->prd_reg = t->reg_base + PRD34; |
| } |
| |
| /* Register interrupt */ |
| t->irqaction.name = t->name; |
| t->irqaction.dev_id = (void *)t; |
| if (t->irqaction.handler != NULL) { |
| setup_irq(timer_irqs[t->id], &t->irqaction); |
| } |
| |
| timer32_config(&timers[i]); |
| } |
| } |
| } |
| |
| /* |
| * clocksource |
| */ |
| static cycle_t read_cycles(void) |
| { |
| struct timer_s *t = &timers[TID_CLOCKSOURCE]; |
| |
| return (cycles_t)timer32_read(t); |
| } |
| |
| static struct clocksource clocksource_davinci = { |
| .name = "timer0_1", |
| .rating = 300, |
| .read = read_cycles, |
| .mask = CLOCKSOURCE_MASK(32), |
| .shift = 24, |
| .flags = CLOCK_SOURCE_IS_CONTINUOUS, |
| }; |
| |
| /* |
| * clockevent |
| */ |
| static int davinci_set_next_event(unsigned long cycles, |
| struct clock_event_device *evt) |
| { |
| struct timer_s *t = &timers[TID_CLOCKEVENT]; |
| |
| t->period = cycles; |
| timer32_config(t); |
| return 0; |
| } |
| |
| static void davinci_set_mode(enum clock_event_mode mode, |
| struct clock_event_device *evt) |
| { |
| struct timer_s *t = &timers[TID_CLOCKEVENT]; |
| |
| switch (mode) { |
| case CLOCK_EVT_MODE_PERIODIC: |
| t->period = CLOCK_TICK_RATE / (HZ); |
| t->opts = TIMER_OPTS_PERIODIC; |
| timer32_config(t); |
| break; |
| case CLOCK_EVT_MODE_ONESHOT: |
| t->opts = TIMER_OPTS_ONESHOT; |
| break; |
| case CLOCK_EVT_MODE_UNUSED: |
| case CLOCK_EVT_MODE_SHUTDOWN: |
| t->opts = TIMER_OPTS_DISABLED; |
| break; |
| } |
| } |
| |
| static struct clock_event_device clockevent_davinci = { |
| .name = "timer0_0", |
| .features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT, |
| .shift = 32, |
| .set_next_event = davinci_set_next_event, |
| .set_mode = davinci_set_mode, |
| }; |
| |
| |
| static void __init davinci_timer_init(void) |
| { |
| static char err[] __initdata = KERN_ERR |
| "%s: can't register clocksource!\n"; |
| |
| /* init timer hw */ |
| timer_init(); |
| |
| /* setup clocksource */ |
| clocksource_davinci.mult = |
| clocksource_khz2mult(CLOCK_TICK_RATE/1000, |
| clocksource_davinci.shift); |
| if (clocksource_register(&clocksource_davinci)) |
| printk(err, clocksource_davinci.name); |
| |
| /* setup clockevent */ |
| clockevent_davinci.mult = div_sc(CLOCK_TICK_RATE, NSEC_PER_SEC, |
| clockevent_davinci.shift); |
| clockevent_davinci.max_delta_ns = |
| clockevent_delta2ns(0xfffffffe, &clockevent_davinci); |
| clockevent_davinci.min_delta_ns = |
| clockevent_delta2ns(1, &clockevent_davinci); |
| |
| clockevent_davinci.cpumask = cpumask_of_cpu(0); |
| clockevents_register_device(&clockevent_davinci); |
| } |
| |
| struct sys_timer davinci_timer = { |
| .init = davinci_timer_init, |
| }; |
| |
| |
| /* reset board using watchdog timer */ |
| void davinci_watchdog_reset(void) { |
| u32 tgcr, wdtcr, base = DAVINCI_WDOG_BASE; |
| |
| /* disable, internal clock source */ |
| davinci_writel(0, base + TCR); |
| |
| /* reset timer, set mode to 64-bit watchdog, and unreset */ |
| tgcr = 0; |
| davinci_writel(tgcr, base + TCR); |
| tgcr = TGCR_TIMMODE_64BIT_WDOG << TGCR_TIMMODE_SHIFT; |
| tgcr |= (TGCR_UNRESET << TGCR_TIM12RS_SHIFT) | |
| (TGCR_UNRESET << TGCR_TIM34RS_SHIFT); |
| davinci_writel(tgcr, base + TCR); |
| |
| /* clear counter and period regs */ |
| davinci_writel(0, base + TIM12); |
| davinci_writel(0, base + TIM34); |
| davinci_writel(0, base + PRD12); |
| davinci_writel(0, base + PRD34); |
| |
| /* enable */ |
| wdtcr = davinci_readl(base + WDTCR); |
| wdtcr |= WDTCR_WDEN_ENABLE << WDTCR_WDEN_SHIFT; |
| davinci_writel(wdtcr, base + WDTCR); |
| |
| /* put watchdog in pre-active state */ |
| wdtcr = (WDTCR_WDKEY_SEQ0 << WDTCR_WDKEY_SHIFT) | |
| (WDTCR_WDEN_ENABLE << WDTCR_WDEN_SHIFT); |
| davinci_writel(wdtcr, base + WDTCR); |
| |
| /* put watchdog in active state */ |
| wdtcr = (WDTCR_WDKEY_SEQ1 << WDTCR_WDKEY_SHIFT) | |
| (WDTCR_WDEN_ENABLE << WDTCR_WDEN_SHIFT); |
| davinci_writel(wdtcr, base + WDTCR); |
| |
| /* write an invalid value to the WDKEY field to trigger |
| * a watchdog reset */ |
| wdtcr = 0x00004000; |
| davinci_writel(wdtcr, base + WDTCR); |
| } |