[PATCH] clocksource: register persistent clock for arm arch_timer

From: Lei Wen
Date: Wed Apr 02 2014 - 07:05:31 EST


Since arm's arch_timer's counter would keep accumulated even in the
low power mode, including suspend state, it is very suitable to be
the persistent clock instead of RTC.

While read_persistent_clock calling place shall be rare, like only
suspend/resume place? So we shall don't care for its performance
very much, so use direclty divided by frequency should be accepted
for this reason. Actually archtimer's counter read performance already
be very good, since it is directly access from core's bus, not from
soc, so this is another reason why we choose use divide here.

Final reason for why we don't use multi+shift way is for we may not
call read_persistent_clock for long time, like system long time
not enter into suspend, so that the accumulated cycle difference value
may larger than we used for calculate the multi+shift, thus precise
would be highly affected in such corner case.

Signed-off-by: Lei Wen <leiwen@xxxxxxxxxxx>
---

I am not sure whether it is good to add something like
generic_persistent_clock_read in the new added kernel/time/sched_clock.c?
Since from arch timer's perspective, all it need to do is to pick
the suspend period from the place where sched_clock being stopped/restarted.

Any idea for make the persistent clock reading as one generic function,
like current sched_clock do?


drivers/clocksource/arm_arch_timer.c | 31 +++++++++++++++++++++++++++++++
1 file changed, 31 insertions(+)

diff --git a/drivers/clocksource/arm_arch_timer.c b/drivers/clocksource/arm_arch_timer.c
index 57e823c..5eaa34a 100644
--- a/drivers/clocksource/arm_arch_timer.c
+++ b/drivers/clocksource/arm_arch_timer.c
@@ -23,6 +23,7 @@
#include <linux/sched_clock.h>

#include <asm/arch_timer.h>
+#include <asm/mach/time.h>
#include <asm/virt.h>

#include <clocksource/arm_arch_timer.h>
@@ -52,6 +53,8 @@ struct arch_timer {
#define to_arch_timer(e) container_of(e, struct arch_timer, evt)

static u32 arch_timer_rate;
+static u32 persistent_multi;
+static u32 persistent_div;

enum ppi_nr {
PHYS_SECURE_PPI,
@@ -68,6 +71,31 @@ static struct clock_event_device __percpu *arch_timer_evt;
static bool arch_timer_use_virtual = true;
static bool arch_timer_mem_use_virtual;

+static void get_persistent_clock_multi_div(void)
+{
+ u32 i, tmp;
+
+ tmp = arch_timer_rate;
+ persistent_multi = NSEC_PER_SEC;
+ for (i = 0; i < 9; i ++) {
+ if (tmp % 10)
+ break;
+ tmp /= 10;
+ persistent_multi /= 10;
+ }
+
+ persistent_div = tmp;
+}
+
+static void arch_timer_persistent_clock_read(struct timespec *ts)
+{
+ u64 ns;
+
+ ns = arch_timer_read_counter() * persistent_multi;
+ do_div(ns, persistent_div);
+ *ts = ns_to_timespec(ns);
+}
+
/*
* Architected system timer support.
*/
@@ -631,6 +659,9 @@ static void __init arch_timer_common_init(void)
arch_timer_banner(arch_timers_present);
arch_counter_register(arch_timers_present);
arch_timer_arch_init();
+
+ get_persistent_clock_multi_div();
+ register_persistent_clock(NULL, arch_timer_persistent_clock_read);
}

static void __init arch_timer_init(struct device_node *np)
--
1.8.3.2

--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/