1 // SPDX-License-Identifier: GPL-2.0+
3 * Copyright 2014 Broadcom Corporation.
10 #include <asm/iproc-common/timer.h>
11 #include <asm/iproc-common/sysmap.h>
13 static inline uint64_t timer_global_read(void)
20 count_h = readl(IPROC_PERIPH_GLB_TIM_REG_BASE +
22 count_l = readl(IPROC_PERIPH_GLB_TIM_REG_BASE +
23 TIMER_GLB_LOW_OFFSET);
24 cur_tick = readl(IPROC_PERIPH_GLB_TIM_REG_BASE +
26 } while (cur_tick != count_h);
28 return (cur_tick << 32) + count_l;
31 void timer_global_init(void)
33 writel(0, IPROC_PERIPH_GLB_TIM_REG_BASE + TIMER_GLB_CTRL_OFFSET);
34 writel(0, IPROC_PERIPH_GLB_TIM_REG_BASE + TIMER_GLB_LOW_OFFSET);
35 writel(0, IPROC_PERIPH_GLB_TIM_REG_BASE + TIMER_GLB_HI_OFFSET);
36 writel(TIMER_GLB_TIM_CTRL_TIM_EN,
37 IPROC_PERIPH_GLB_TIM_REG_BASE + TIMER_GLB_CTRL_OFFSET);
46 unsigned long get_timer(unsigned long base)
53 count = timer_global_read();
55 /* default arm clk is 1GHz, periph_clk=arm_clk/2, tick per msec */
57 tim_clk = lldiv(periph_clk,
58 (((readl(IPROC_PERIPH_GLB_TIM_REG_BASE +
59 TIMER_GLB_CTRL_OFFSET) &
60 TIMER_GLB_TIM_CTRL_PRESC_MASK) >> 8) + 1));
62 ret = lldiv(count, (uint32_t)tim_clk);
68 void __udelay(unsigned long usec)
70 uint64_t cur_tick, end_tick;
74 /* default arm clk is 1GHz, periph_clk=arm_clk/2, tick per usec */
77 tim_clk = lldiv(periph_clk,
78 (((readl(IPROC_PERIPH_GLB_TIM_REG_BASE +
79 TIMER_GLB_CTRL_OFFSET) &
80 TIMER_GLB_TIM_CTRL_PRESC_MASK) >> 8) + 1));
82 cur_tick = timer_global_read();
89 cur_tick = timer_global_read();
91 } while (cur_tick < end_tick);
94 void timer_systick_init(uint32_t tick_ms)
96 /* Disable timer and clear interrupt status*/
97 writel(0, IPROC_PERIPH_PVT_TIM_REG_BASE + TIMER_PVT_CTRL_OFFSET);
98 writel(TIMER_PVT_TIM_INT_STATUS_SET,
99 IPROC_PERIPH_PVT_TIM_REG_BASE + TIMER_PVT_STATUS_OFFSET);
100 writel((PLL_AXI_CLK/1000) * tick_ms,
101 IPROC_PERIPH_PVT_TIM_REG_BASE + TIMER_PVT_LOAD_OFFSET);
102 writel(TIMER_PVT_TIM_CTRL_INT_EN |
103 TIMER_PVT_TIM_CTRL_AUTO_RELD |
104 TIMER_PVT_TIM_CTRL_TIM_EN,
105 IPROC_PERIPH_PVT_TIM_REG_BASE + TIMER_PVT_CTRL_OFFSET);
108 void timer_systick_isr(void *data)
110 writel(TIMER_PVT_TIM_INT_STATUS_SET,
111 IPROC_PERIPH_PVT_TIM_REG_BASE + TIMER_PVT_STATUS_OFFSET);
115 * This function is derived from PowerPC code (read timebase as long long).
116 * On ARM it just returns the timer value in msec.
118 unsigned long long get_ticks(void)
124 * This is used in conjuction with get_ticks, which returns msec as ticks.
125 * Here we just return ticks/sec = msec/sec = 1000
127 ulong get_tbclk(void)