Message ID | 20090528091936.GA27545@linux-sh.org (mailing list archive) |
---|---|
State | Changes Requested |
Headers | show |
On Thu, 2009-05-28 at 18:19 +0900, Paul Mundt wrote: > Ok, there were some ordering problems with the early platform code, but > I've played with this a bit more and got it to the point where this now > also works. I can live with this over the v3 version if people prefer > this approach instead. > > -- > @@ -38,8 +39,7 @@ > */ > unsigned long long __attribute__((weak)) sched_clock(void) > { > - return (unsigned long long)(jiffies - INITIAL_JIFFIES) > - * (NSEC_PER_SEC / HZ); > + return cyc2ns(sched_clocksource, clocksource_read(sched_clocksource)); > } > > static __read_mostly int sched_clock_running; > @@ -362,6 +363,9 @@ static struct clocksource *select_clocksource(void) > if (next == curr_clocksource) > return NULL; > > + if (next->flags & CLOCK_SOURCE_USE_FOR_SCHED_CLOCK) > + sched_clocksource = next; > + > return next; > } > That's a single assignment, vs two reads on use. Should we be worried about the SMP race where we observe two different sched_clocksource pointers on read? I would suggest we write it as: u64 __weak sched_clock(void) { struct clocksource *clock = ACCESS_ONCE(sched_clocksource); return cyc2ns(clock, clocksource_read(clock)); } -- To unsubscribe from this list: send the line "unsubscribe linux-sh" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
diff --git a/include/linux/clocksource.h b/include/linux/clocksource.h index c56457c..2109940 100644 --- a/include/linux/clocksource.h +++ b/include/linux/clocksource.h @@ -202,7 +202,8 @@ struct clocksource { #endif }; -extern struct clocksource *clock; /* current clocksource */ +extern struct clocksource *clock; /* current clocksource */ +extern struct clocksource *sched_clocksource; /* sched_clock() clocksource */ /* * Clock source flags bits:: @@ -212,6 +213,7 @@ extern struct clocksource *clock; /* current clocksource */ #define CLOCK_SOURCE_WATCHDOG 0x10 #define CLOCK_SOURCE_VALID_FOR_HRES 0x20 +#define CLOCK_SOURCE_USE_FOR_SCHED_CLOCK 0x40 /* simplify initialization of mask field */ #define CLOCKSOURCE_MASK(bits) (cycle_t)((bits) < 64 ? ((1ULL<<(bits))-1) : -1) diff --git a/kernel/sched_clock.c b/kernel/sched_clock.c index e1d16c9..c06c285 100644 --- a/kernel/sched_clock.c +++ b/kernel/sched_clock.c @@ -30,6 +30,7 @@ #include <linux/percpu.h> #include <linux/ktime.h> #include <linux/sched.h> +#include <linux/clocksource.h> /* * Scheduler clock - returns current time in nanosec units. @@ -38,8 +39,7 @@ */ unsigned long long __attribute__((weak)) sched_clock(void) { - return (unsigned long long)(jiffies - INITIAL_JIFFIES) - * (NSEC_PER_SEC / HZ); + return cyc2ns(sched_clocksource, clocksource_read(sched_clocksource)); } static __read_mostly int sched_clock_running; diff --git a/kernel/time/clocksource.c b/kernel/time/clocksource.c index 80189f6..d148a75 100644 --- a/kernel/time/clocksource.c +++ b/kernel/time/clocksource.c @@ -109,6 +109,7 @@ EXPORT_SYMBOL(timecounter_cyc2time); /* XXX - Would like a better way for initializing curr_clocksource */ extern struct clocksource clocksource_jiffies; +struct clocksource *sched_clocksource = &clocksource_jiffies; /*[Clocksource internal variables]--------- * curr_clocksource: @@ -362,6 +363,9 @@ static struct clocksource *select_clocksource(void) if (next == curr_clocksource) return NULL; + if (next->flags & CLOCK_SOURCE_USE_FOR_SCHED_CLOCK) + sched_clocksource = next; + return next; } diff --git a/kernel/time/jiffies.c b/kernel/time/jiffies.c index c3f6c30..727d881 100644 --- a/kernel/time/jiffies.c +++ b/kernel/time/jiffies.c @@ -52,7 +52,7 @@ static cycle_t jiffies_read(struct clocksource *cs) { - return (cycle_t) jiffies; + return (cycle_t) (jiffies - INITIAL_JIFFIES); } struct clocksource clocksource_jiffies = {