Message ID | 20130225134041.GA22785@e106331-lin.cambridge.arm.com (mailing list archive) |
---|---|
State | New, archived |
Headers | show |
On 02/25/13 05:40, Mark Rutland wrote: > I've had a quick go at writing a generic timer driver. I've not had a chance to > test it, and there are a couple of things that are up for discussion (e.g. what > should the rating be) but I think we want something very close to this. > This looks good to me. I only have some minor comments. What's the plan for merging? Get tglx to take this and provide a stable branch and then base my patches off that and get these patches taken through arm-soc? > diff --git a/drivers/clocksource/dummy_timer.c b/drivers/clocksource/dummy_timer.c > new file mode 100644 > index 0000000..bdaba34 > --- /dev/null > +++ b/drivers/clocksource/dummy_timer.c > @@ -0,0 +1,67 @@ > +/* > + * linux/drivers/clocksource/dummy_timer.c > + * > + * Copyright (C) 2013 ARM Ltd. > + * All Rights Reserved > + * > + * This program is free software; you can redistribute it and/or modify > + * it under the terms of the GNU General Public License version 2 as > + * published by the Free Software Foundation. > + */ > +#include <linux/clockchips.h> > +#include <linux/cpu.h> > +#include <linux/init.h> > +#include <linux/percpu.h> > + > +DEFINE_PER_CPU(struct clock_event_device, dummy_evt); static? > + > +static void dummy_set_mode(enum clock_event_mode mode, > + struct clock_event_device *evt) > +{ > + /* > + * Core clockevents code will call this when exchanging timer devices. > + * We don't need to do anything here. > + */ > +} > + > +static void __cpuinit dummy_setup(void) > +{ > + int cpu = smp_processor_id(); > + struct clock_event_device *evt = &per_cpu(dummy_evt, cpu); Can we use __this_cpu_ptr()? I wonder if that makes the code generation better or worse. I didn't do it in my 8/8 patch because I wanted the code to be the same before and after to show code movement. > + > + evt->name = "dummy timer"; > + evt->features = CLOCK_EVT_FEAT_PERIODIC | > + CLOCK_EVT_FEAT_ONESHOT | > + CLOCK_EVT_FEAT_DUMMY; > + evt->rating = 100; > + evt->set_mode = dummy_set_mode; > + evt->cpumask = cpumask_of(cpu); > + > + clockevents_register_device(evt); > +} > + > +static int __cpuinit dummy_cpu_notify(struct notifier_block *self, > + unsigned long action, void *hcpu) > +{ > + if ((action & ~CPU_TASKS_FROZEN) == CPU_STARTING) > + dummy_setup(); There are already two dummy_setup() functions. Perhaps we can s/dummy/dummy_broadcast/ throughout this file?
On 03/04/13 15:50, Stephen Boyd wrote: > >> + >> + evt->name = "dummy timer"; >> + evt->features = CLOCK_EVT_FEAT_PERIODIC | >> + CLOCK_EVT_FEAT_ONESHOT | >> + CLOCK_EVT_FEAT_DUMMY; >> + evt->rating = 100; >> + evt->set_mode = dummy_set_mode; >> + evt->cpumask = cpumask_of(cpu); >> + >> + clockevents_register_device(evt); >> +} >> + >> +static int __cpuinit dummy_cpu_notify(struct notifier_block *self, >> + unsigned long action, void *hcpu) >> +{ >> + if ((action & ~CPU_TASKS_FROZEN) == CPU_STARTING) >> + dummy_setup(); > There are already two dummy_setup() functions. Perhaps we can > s/dummy/dummy_broadcast/ throughout this file? > Sorry, meant to say dummy_timer not dummy_broadcast.
On Mon, Mar 04, 2013 at 11:50:22PM +0000, Stephen Boyd wrote: > On 02/25/13 05:40, Mark Rutland wrote: > > I've had a quick go at writing a generic timer driver. I've not had a chance to > > test it, and there are a couple of things that are up for discussion (e.g. what > > should the rating be) but I think we want something very close to this. > > > > This looks good to me. I only have some minor comments. What's the plan > for merging? Get tglx to take this and provide a stable branch and then > base my patches off that and get these patches taken through arm-soc? Great. That sounds about right, I don't really know what the best way would be. > > > diff --git a/drivers/clocksource/dummy_timer.c b/drivers/clocksource/dummy_timer.c > > new file mode 100644 > > index 0000000..bdaba34 > > --- /dev/null > > +++ b/drivers/clocksource/dummy_timer.c > > @@ -0,0 +1,67 @@ > > +/* > > + * linux/drivers/clocksource/dummy_timer.c > > + * > > + * Copyright (C) 2013 ARM Ltd. > > + * All Rights Reserved > > + * > > + * This program is free software; you can redistribute it and/or modify > > + * it under the terms of the GNU General Public License version 2 as > > + * published by the Free Software Foundation. > > + */ > > +#include <linux/clockchips.h> > > +#include <linux/cpu.h> > > +#include <linux/init.h> > > +#include <linux/percpu.h> > > + > > +DEFINE_PER_CPU(struct clock_event_device, dummy_evt); > > static? Oops. Added. > > > + > > +static void dummy_set_mode(enum clock_event_mode mode, > > + struct clock_event_device *evt) > > +{ > > + /* > > + * Core clockevents code will call this when exchanging timer devices. > > + * We don't need to do anything here. > > + */ > > +} > > + > > +static void __cpuinit dummy_setup(void) > > +{ > > + int cpu = smp_processor_id(); > > + struct clock_event_device *evt = &per_cpu(dummy_evt, cpu); > > Can we use __this_cpu_ptr()? I wonder if that makes the code generation > better or worse. I didn't do it in my 8/8 patch because I wanted the > code to be the same before and after to show code movement. I did that originally, but thought as I needed the cpu value for the mask anyway that there wasn't much point. I'm not that good at reading generated assembly, so I can't really say if either's better. > > > + > > + evt->name = "dummy timer"; > > + evt->features = CLOCK_EVT_FEAT_PERIODIC | > > + CLOCK_EVT_FEAT_ONESHOT | > > + CLOCK_EVT_FEAT_DUMMY; > > + evt->rating = 100; > > + evt->set_mode = dummy_set_mode; > > + evt->cpumask = cpumask_of(cpu); > > + > > + clockevents_register_device(evt); > > +} > > + > > +static int __cpuinit dummy_cpu_notify(struct notifier_block *self, > > + unsigned long action, void *hcpu) > > +{ > > + if ((action & ~CPU_TASKS_FROZEN) == CPU_STARTING) > > + dummy_setup(); > > There are already two dummy_setup() functions. Perhaps we can > s/dummy/dummy_broadcast/ throughout this file? I've done s/dummy/dummy_timer/ as suggested in your other reply. Thanks, Mark.
On 03/05/13 03:02, Mark Rutland wrote: > >>> + >>> +static void dummy_set_mode(enum clock_event_mode mode, >>> + struct clock_event_device *evt) >>> +{ >>> + /* >>> + * Core clockevents code will call this when exchanging timer devices. >>> + * We don't need to do anything here. >>> + */ >>> +} >>> + >>> +static void __cpuinit dummy_setup(void) >>> +{ >>> + int cpu = smp_processor_id(); >>> + struct clock_event_device *evt = &per_cpu(dummy_evt, cpu); >> Can we use __this_cpu_ptr()? I wonder if that makes the code generation >> better or worse. I didn't do it in my 8/8 patch because I wanted the >> code to be the same before and after to show code movement. > I did that originally, but thought as I needed the cpu value for the mask > anyway that there wasn't much point. I'm not that good at reading generated > assembly, so I can't really say if either's better. It looks to be two instructions shorter.
diff --git a/drivers/clocksource/Makefile b/drivers/clocksource/Makefile index 7d671b8..8f2b13f 100644 --- a/drivers/clocksource/Makefile +++ b/drivers/clocksource/Makefile @@ -21,3 +21,4 @@ obj-$(CONFIG_ARCH_TEGRA) += tegra20_timer.o obj-$(CONFIG_VT8500_TIMER) += vt8500_timer.o obj-$(CONFIG_ARM_ARCH_TIMER) += arm_arch_timer.o +obj-$(CONFIG_ARCH_HAS_TICK_BROADCAST) += dummy_timer.o diff --git a/drivers/clocksource/dummy_timer.c b/drivers/clocksource/dummy_timer.c new file mode 100644 index 0000000..bdaba34 --- /dev/null +++ b/drivers/clocksource/dummy_timer.c @@ -0,0 +1,67 @@ +/* + * linux/drivers/clocksource/dummy_timer.c + * + * Copyright (C) 2013 ARM Ltd. + * All Rights Reserved + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ +#include <linux/clockchips.h> +#include <linux/cpu.h> +#include <linux/init.h> +#include <linux/percpu.h> + +DEFINE_PER_CPU(struct clock_event_device, dummy_evt); + +static void dummy_set_mode(enum clock_event_mode mode, + struct clock_event_device *evt) +{ + /* + * Core clockevents code will call this when exchanging timer devices. + * We don't need to do anything here. + */ +} + +static void __cpuinit dummy_setup(void) +{ + int cpu = smp_processor_id(); + struct clock_event_device *evt = &per_cpu(dummy_evt, cpu); + + evt->name = "dummy timer"; + evt->features = CLOCK_EVT_FEAT_PERIODIC | + CLOCK_EVT_FEAT_ONESHOT | + CLOCK_EVT_FEAT_DUMMY; + evt->rating = 100; + evt->set_mode = dummy_set_mode; + evt->cpumask = cpumask_of(cpu); + + clockevents_register_device(evt); +} + +static int __cpuinit dummy_cpu_notify(struct notifier_block *self, + unsigned long action, void *hcpu) +{ + if ((action & ~CPU_TASKS_FROZEN) == CPU_STARTING) + dummy_setup(); + + return NOTIFY_OK; +} + +static struct notifier_block dummy_cpu_nb __cpuinitdata = { + .notifier_call = dummy_cpu_notify, +}; + +static int __init dummy_register(void) +{ + int err = register_cpu_notifier(&dummy_cpu_nb); + if (err) + return err; + + /* We won't get a call on the boot CPU, so register immediately */ + dummy_setup(); + + return 0; +} +device_initcall(dummy_register);