1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * Copyright 2012-2013 Freescale Semiconductor, Inc.
6 #include <linux/interrupt.h>
7 #include <linux/clockchips.h>
9 #include <linux/of_address.h>
10 #include <linux/of_irq.h>
11 #include <linux/sched_clock.h>
14 * Each pit takes 0x10 Bytes register space
17 #define PIT0_OFFSET 0x100
18 #define PITn_OFFSET(n) (PIT0_OFFSET + 0x10 * (n))
24 #define PITMCR_MDIS (0x1 << 1)
26 #define PITTCTRL_TEN (0x1 << 0)
27 #define PITTCTRL_TIE (0x1 << 1)
28 #define PITCTRL_CHN (0x1 << 2)
30 #define PITTFLG_TIF 0x1
32 static void __iomem
*clksrc_base
;
33 static void __iomem
*clkevt_base
;
34 static unsigned long cycle_per_jiffy
;
36 static inline void pit_timer_enable(void)
38 __raw_writel(PITTCTRL_TEN
| PITTCTRL_TIE
, clkevt_base
+ PITTCTRL
);
41 static inline void pit_timer_disable(void)
43 __raw_writel(0, clkevt_base
+ PITTCTRL
);
46 static inline void pit_irq_acknowledge(void)
48 __raw_writel(PITTFLG_TIF
, clkevt_base
+ PITTFLG
);
51 static u64 notrace
pit_read_sched_clock(void)
53 return ~__raw_readl(clksrc_base
+ PITCVAL
);
56 static int __init
pit_clocksource_init(unsigned long rate
)
58 /* set the max load value and start the clock source counter */
59 __raw_writel(0, clksrc_base
+ PITTCTRL
);
60 __raw_writel(~0UL, clksrc_base
+ PITLDVAL
);
61 __raw_writel(PITTCTRL_TEN
, clksrc_base
+ PITTCTRL
);
63 sched_clock_register(pit_read_sched_clock
, 32, rate
);
64 return clocksource_mmio_init(clksrc_base
+ PITCVAL
, "vf-pit", rate
,
65 300, 32, clocksource_mmio_readl_down
);
68 static int pit_set_next_event(unsigned long delta
,
69 struct clock_event_device
*unused
)
72 * set a new value to PITLDVAL register will not restart the timer,
73 * to abort the current cycle and start a timer period with the new
74 * value, the timer must be disabled and enabled again.
75 * and the PITLAVAL should be set to delta minus one according to pit
76 * hardware requirement.
79 __raw_writel(delta
- 1, clkevt_base
+ PITLDVAL
);
85 static int pit_shutdown(struct clock_event_device
*evt
)
91 static int pit_set_periodic(struct clock_event_device
*evt
)
93 pit_set_next_event(cycle_per_jiffy
, evt
);
97 static irqreturn_t
pit_timer_interrupt(int irq
, void *dev_id
)
99 struct clock_event_device
*evt
= dev_id
;
101 pit_irq_acknowledge();
104 * pit hardware doesn't support oneshot, it will generate an interrupt
105 * and reload the counter value from PITLDVAL when PITCVAL reach zero,
106 * and start the counter again. So software need to disable the timer
107 * to stop the counter loop in ONESHOT mode.
109 if (likely(clockevent_state_oneshot(evt
)))
112 evt
->event_handler(evt
);
117 static struct clock_event_device clockevent_pit
= {
118 .name
= "VF pit timer",
119 .features
= CLOCK_EVT_FEAT_PERIODIC
| CLOCK_EVT_FEAT_ONESHOT
,
120 .set_state_shutdown
= pit_shutdown
,
121 .set_state_periodic
= pit_set_periodic
,
122 .set_next_event
= pit_set_next_event
,
126 static int __init
pit_clockevent_init(unsigned long rate
, int irq
)
128 __raw_writel(0, clkevt_base
+ PITTCTRL
);
129 __raw_writel(PITTFLG_TIF
, clkevt_base
+ PITTFLG
);
131 BUG_ON(request_irq(irq
, pit_timer_interrupt
, IRQF_TIMER
| IRQF_IRQPOLL
,
132 "VF pit timer", &clockevent_pit
));
134 clockevent_pit
.cpumask
= cpumask_of(0);
135 clockevent_pit
.irq
= irq
;
137 * The value for the LDVAL register trigger is calculated as:
138 * LDVAL trigger = (period / clock period) - 1
139 * The pit is a 32-bit down count timer, when the conter value
140 * reaches 0, it will generate an interrupt, thus the minimal
141 * LDVAL trigger value is 1. And then the min_delta is
142 * minimal LDVAL trigger value + 1, and the max_delta is full 32-bit.
144 clockevents_config_and_register(&clockevent_pit
, rate
, 2, 0xffffffff);
149 static int __init
pit_timer_init(struct device_node
*np
)
152 void __iomem
*timer_base
;
153 unsigned long clk_rate
;
156 timer_base
= of_iomap(np
, 0);
158 pr_err("Failed to iomap\n");
163 * PIT0 and PIT1 can be chained to build a 64-bit timer,
164 * so choose PIT2 as clocksource, PIT3 as clockevent device,
165 * and leave PIT0 and PIT1 unused for anyone else who needs them.
167 clksrc_base
= timer_base
+ PITn_OFFSET(2);
168 clkevt_base
= timer_base
+ PITn_OFFSET(3);
170 irq
= irq_of_parse_and_map(np
, 0);
174 pit_clk
= of_clk_get(np
, 0);
176 return PTR_ERR(pit_clk
);
178 ret
= clk_prepare_enable(pit_clk
);
182 clk_rate
= clk_get_rate(pit_clk
);
183 cycle_per_jiffy
= clk_rate
/ (HZ
);
185 /* enable the pit module */
186 __raw_writel(~PITMCR_MDIS
, timer_base
+ PITMCR
);
188 ret
= pit_clocksource_init(clk_rate
);
192 return pit_clockevent_init(clk_rate
, irq
);
194 TIMER_OF_DECLARE(vf610
, "fsl,vf610-pit", pit_timer_init
);