// SPDX-License-Identifier: GPL-2.0-only /* * at91sam926x_time.c - Periodic Interval Timer (PIT) for at91sam926x * * Copyright (C) 2005-2006 M. Amine SAYA, ATMEL Rousset, France * Revision 2005 M. Nicolas Diremdjian, ATMEL Rousset, France * Converted to ClockSource/ClockEvents by David Brownell.
*/
/* * Clocksource: just a monotonic counter of MCK/16 cycles. * We don't care whether or not PIT irqs are enabled.
*/ static u64 read_pit_clk(struct clocksource *cs)
{ struct pit_data *data = clksrc_to_pit_data(cs); unsignedlong flags;
u32 elapsed;
u32 t;
raw_local_irq_save(flags);
elapsed = data->cnt;
t = pit_read(data->base, AT91_PIT_PIIR);
raw_local_irq_restore(flags);
/* * IRQ handler for the timer.
*/ static irqreturn_t at91sam926x_pit_interrupt(int irq, void *dev_id)
{ struct pit_data *data = dev_id;
/* The PIT interrupt may be disabled, and is shared */ if (clockevent_state_periodic(&data->clkevt) &&
(pit_read(data->base, AT91_PIT_SR) & AT91_PIT_PITS)) { /* Get number of ticks performed before irq, and ack it */
data->cnt += data->cycle * PIT_PICNT(pit_read(data->base,
AT91_PIT_PIVR));
data->clkevt.event_handler(&data->clkevt);
return IRQ_HANDLED;
}
return IRQ_NONE;
}
/* * Set up both clocksource and clockevent support.
*/ staticint __init at91sam926x_pit_dt_init(struct device_node *node)
{ unsignedlong pit_rate; unsigned bits; int ret; struct pit_data *data;
data = kzalloc(sizeof(*data), GFP_KERNEL); if (!data) return -ENOMEM;
data->base = of_iomap(node, 0); if (!data->base) {
pr_err("Could not map PIT address\n");
ret = -ENXIO; gotoexit;
}
data->mck = of_clk_get(node, 0); if (IS_ERR(data->mck)) {
pr_err("Unable to get mck clk\n");
ret = PTR_ERR(data->mck); gotoexit;
}
ret = clk_prepare_enable(data->mck); if (ret) {
pr_err("Unable to enable mck\n"); gotoexit;
}
/* Get the interrupts property */
data->irq = irq_of_parse_and_map(node, 0); if (!data->irq) {
pr_err("Unable to get IRQ from DT\n");
ret = -EINVAL; gotoexit;
}
/* * Use our actual MCK to figure out how many MCK/16 ticks per * 1/HZ period (instead of a compile-time constant LATCH).
*/
pit_rate = clk_get_rate(data->mck) / 16;
data->cycle = DIV_ROUND_CLOSEST(pit_rate, HZ);
WARN_ON(((data->cycle - 1) & ~AT91_PIT_PIV) != 0);
/* Initialize and enable the timer */
at91sam926x_pit_reset(data);
/* * Register clocksource. The high order bits of PIV are unused, * so this isn't a 32-bit counter unless we get clockevent irqs.
*/
bits = 12 /* PICNT */ + ilog2(data->cycle) /* PIV */;
data->clksrc.mask = CLOCKSOURCE_MASK(bits);
data->clksrc.name = "pit";
data->clksrc.rating = 175;
data->clksrc.read = read_pit_clk;
data->clksrc.flags = CLOCK_SOURCE_IS_CONTINUOUS;
ret = clocksource_register_hz(&data->clksrc, pit_rate); if (ret) {
pr_err("Failed to register clocksource\n"); gotoexit;
}
/* Set up irq handler */
ret = request_irq(data->irq, at91sam926x_pit_interrupt,
IRQF_SHARED | IRQF_TIMER | IRQF_IRQPOLL, "at91_tick", data); if (ret) {
pr_err("Unable to setup IRQ\n");
clocksource_unregister(&data->clksrc); gotoexit;
}
/* Set up and register clockevents */
data->clkevt.name = "pit";
data->clkevt.features = CLOCK_EVT_FEAT_PERIODIC;
data->clkevt.shift = 32;
data->clkevt.mult = div_sc(pit_rate, NSEC_PER_SEC, data->clkevt.shift);
data->clkevt.rating = 100;
data->clkevt.cpumask = cpumask_of(0);
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.