blob: 80c458fc4a30d3045cc3f630198c77fd18d09fa6 [file] [log] [blame]
Marc Zyngier022c03a2012-01-11 17:25:17 +00001/*
2 * linux/arch/arm/kernel/arch_timer.c
3 *
4 * Copyright (C) 2011 ARM Ltd.
5 * All Rights Reserved
6 *
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License version 2 as
9 * published by the Free Software Foundation.
10 */
11#include <linux/init.h>
Mark Rutland8a4da6e2012-11-12 14:33:44 +000012#include <linux/types.h>
Arnd Bergmannfe7dc722013-02-14 17:45:25 +010013#include <linux/errno.h>
Marc Zyngier022c03a2012-01-11 17:25:17 +000014
Jonathan Austin56942fe2012-09-21 18:51:44 +010015#include <asm/delay.h>
Marc Zyngier3f61c802011-01-14 15:32:36 +000016#include <asm/sched_clock.h>
Marc Zyngier022c03a2012-01-11 17:25:17 +000017
Mark Rutland8a4da6e2012-11-12 14:33:44 +000018#include <clocksource/arm_arch_timer.h>
Marc Zyngierf48b5f12012-09-07 18:09:57 +010019
Mark Rutland8a4da6e2012-11-12 14:33:44 +000020static unsigned long arch_timer_read_counter_long(void)
21{
22 return arch_timer_read_counter();
23}
Marc Zyngierf48b5f12012-09-07 18:09:57 +010024
Rob Herring023796b2013-03-11 16:23:46 -050025static u32 sched_clock_mult __read_mostly;
26
27static unsigned long long notrace arch_timer_sched_clock(void)
Mark Rutland8a4da6e2012-11-12 14:33:44 +000028{
Rob Herring023796b2013-03-11 16:23:46 -050029 return arch_timer_read_counter() * sched_clock_mult;
Mark Rutland8a4da6e2012-11-12 14:33:44 +000030}
Marc Zyngier022c03a2012-01-11 17:25:17 +000031
Jonathan Austin56942fe2012-09-21 18:51:44 +010032static struct delay_timer arch_delay_timer;
Will Deacond0a533b2012-07-06 15:47:17 +010033
Mark Rutland8a4da6e2012-11-12 14:33:44 +000034static void __init arch_timer_delay_timer_register(void)
Marc Zyngierf48b5f12012-09-07 18:09:57 +010035{
Jonathan Austin56942fe2012-09-21 18:51:44 +010036 /* Use the architected timer for the delay loop. */
Mark Rutland8a4da6e2012-11-12 14:33:44 +000037 arch_delay_timer.read_current_timer = arch_timer_read_counter_long;
38 arch_delay_timer.freq = arch_timer_get_rate();
Jonathan Austin56942fe2012-09-21 18:51:44 +010039 register_current_timer_delay(&arch_delay_timer);
Marc Zyngier022c03a2012-01-11 17:25:17 +000040}
Marc Zyngier3f61c802011-01-14 15:32:36 +000041
Marc Zyngier00752422012-01-19 13:53:50 +000042int __init arch_timer_of_register(void)
43{
Mark Rutland8a4da6e2012-11-12 14:33:44 +000044 int ret;
Marc Zyngier00752422012-01-19 13:53:50 +000045
Mark Rutland8a4da6e2012-11-12 14:33:44 +000046 ret = arch_timer_init();
47 if (ret)
48 return ret;
Marc Zyngier00752422012-01-19 13:53:50 +000049
Mark Rutland8a4da6e2012-11-12 14:33:44 +000050 arch_timer_delay_timer_register();
Marc Zyngier00752422012-01-19 13:53:50 +000051
Mark Rutland8a4da6e2012-11-12 14:33:44 +000052 return 0;
Marc Zyngier00752422012-01-19 13:53:50 +000053}
Marc Zyngier00752422012-01-19 13:53:50 +000054
Marc Zyngier3f61c802011-01-14 15:32:36 +000055int __init arch_timer_sched_clock_init(void)
56{
Rob Herring023796b2013-03-11 16:23:46 -050057 u32 arch_timer_rate = arch_timer_get_rate();
58
59 if (arch_timer_rate == 0)
Mark Rutland8a4da6e2012-11-12 14:33:44 +000060 return -ENXIO;
Marc Zyngier3f61c802011-01-14 15:32:36 +000061
Rob Herring023796b2013-03-11 16:23:46 -050062 /* Cache the sched_clock multiplier to save a divide in the hot path. */
63 sched_clock_mult = NSEC_PER_SEC / arch_timer_rate;
64 sched_clock_func = arch_timer_sched_clock;
65 pr_info("sched_clock: ARM arch timer >56 bits at %ukHz, resolution %uns\n",
66 arch_timer_rate / 1000, sched_clock_mult);
67
Marc Zyngier3f61c802011-01-14 15:32:36 +000068 return 0;
69}