Message ID | 550F6EDA.30709@adapteva.com (mailing list archive) |
---|---|
State | New, archived |
Headers | show |
Hi, On 03/23/2015 02:39 AM, Ola Jeppsson wrote: > As mentioned in this commit: > arm: zynq: Don't use arm_global_timer with cpufreq > 61f1fc7e9258a169ac8afb5ddf657a181e60d052 > > arm_global_timer depends on the CPU frequency. With cpufreq altering the > CPU frequency arm_global_timer will not maintain a stable time base. So > arm_global_timer must not be the clocksource when cpufreq is enabled. > > The above commit tries to solve this at build time by only selecting > CONFIG_ARM_GLOBAL_TIMER if CONFIG_CPU_FREQ is disabled. This is not > always sufficient because other machs can also enable > CONFIG_ARM_GLOBAL_TIMER. > > Therefore: If built with CONFIG_CPU_FREQ and CONFIG_ARM_GLOBAL_TIMER, > disable (on Zynq) the arm_global_timer devicetree node at boot before > clock sources are initialized. This ensures that arm_global_timer will > not be selected clocksource. > > Signed-off-by: Ola Jeppsson <ola@adapteva.com> > --- > arch/arm/mach-zynq/common.c | 26 ++++++++++++++++++++++++++ > 1 file changed, 26 insertions(+) > > diff --git a/arch/arm/mach-zynq/common.c b/arch/arm/mach-zynq/common.c > index c887196..a4666d4 100644 > --- a/arch/arm/mach-zynq/common.c > +++ b/arch/arm/mach-zynq/common.c > @@ -148,10 +148,36 @@ out: > platform_device_register_full(&devinfo); > } > > +#if defined(CONFIG_CPU_FREQ) && defined(CONFIG_ARM_GLOBAL_TIMER) > +static struct property zynq_disable_arm_global_timer_prop = { > + .name = "status", > + .length = sizeof("disabled"), > + .value = "disabled" > +}; > + > +static void __init zynq_disable_arm_global_timer(void) > +{ > + struct device_node *np; > + > + np = of_find_compatible_node(NULL, NULL, "arm,cortex-a9-global-timer"); > + if (!np) > + return; > + > + pr_info("%s: disabling arm_global_timer node\n", __func__); > + > + if (of_update_property(np, &zynq_disable_arm_global_timer_prop)) > + pr_warn("%s: could not disable arm_global_timer node\n", > + __func__); > +} > +#endif > + > static void __init zynq_timer_init(void) > { > zynq_early_slcr_init(); > > +#if defined(CONFIG_CPU_FREQ) && defined(CONFIG_ARM_GLOBAL_TIMER) > + zynq_disable_arm_global_timer(); > +#endif > zynq_clock_init(); > of_clk_init(NULL); > clocksource_of_init(); > Arnd: Waiting for your thoughts on this one? It is some sort of arch/arm/mach-mvebu/board-v7.c quirk code. I don't think it can be done via Kconfig because it will affect others. Thanks, Michal
On Monday 23 March 2015 08:05:45 Michal Simek wrote: > Hi, > > On 03/23/2015 02:39 AM, Ola Jeppsson wrote: > > As mentioned in this commit: > > arm: zynq: Don't use arm_global_timer with cpufreq > > 61f1fc7e9258a169ac8afb5ddf657a181e60d052 > > > > arm_global_timer depends on the CPU frequency. With cpufreq altering the > > CPU frequency arm_global_timer will not maintain a stable time base. So > > arm_global_timer must not be the clocksource when cpufreq is enabled. > > > > The above commit tries to solve this at build time by only selecting > > CONFIG_ARM_GLOBAL_TIMER if CONFIG_CPU_FREQ is disabled. This is not > > always sufficient because other machs can also enable > > CONFIG_ARM_GLOBAL_TIMER. > > > > Therefore: If built with CONFIG_CPU_FREQ and CONFIG_ARM_GLOBAL_TIMER, > > disable (on Zynq) the arm_global_timer devicetree node at boot before > > clock sources are initialized. This ensures that arm_global_timer will > > not be selected clocksource. > > > > Signed-off-by: Ola Jeppsson <ola@adapteva.com> > > Arnd: Waiting for your thoughts on this one? > It is some sort of arch/arm/mach-mvebu/board-v7.c quirk code. > > I don't think it can be done via Kconfig because it will affect others. (catching up with old email) I think it's ok to work around the problem like this in principle. However, I have one concern about the way this condition is detected. At the moment, you assume that all zynq variants suffer from this problem and no other chip has it. How would you handle the situation if a future zynq variant fixes the problem? What if it turns out to be a more common problem and we actually want to work around it in the arm global timer implementation by dynamically adapting to CPU frequency changes? I believe some other drivers (x86 tsc?) do this. Would we be able to detect this case on zynq? Arnd
On 2015-04-10 15:44, Arnd Bergmann wrote: > On Monday 23 March 2015 08:05:45 Michal Simek wrote: >> Hi, >> >> On 03/23/2015 02:39 AM, Ola Jeppsson wrote: >>> As mentioned in this commit: >>> arm: zynq: Don't use arm_global_timer with cpufreq >>> 61f1fc7e9258a169ac8afb5ddf657a181e60d052 >>> >>> arm_global_timer depends on the CPU frequency. With cpufreq altering the >>> CPU frequency arm_global_timer will not maintain a stable time base. So >>> arm_global_timer must not be the clocksource when cpufreq is enabled. >>> >>> The above commit tries to solve this at build time by only selecting >>> CONFIG_ARM_GLOBAL_TIMER if CONFIG_CPU_FREQ is disabled. This is not >>> always sufficient because other machs can also enable >>> CONFIG_ARM_GLOBAL_TIMER. >>> >>> Therefore: If built with CONFIG_CPU_FREQ and CONFIG_ARM_GLOBAL_TIMER, >>> disable (on Zynq) the arm_global_timer devicetree node at boot before >>> clock sources are initialized. This ensures that arm_global_timer will >>> not be selected clocksource. >>> >>> Signed-off-by: Ola Jeppsson <ola@adapteva.com> > >> Arnd: Waiting for your thoughts on this one? >> It is some sort of arch/arm/mach-mvebu/board-v7.c quirk code. >> >> I don't think it can be done via Kconfig because it will affect others. > (catching up with old email) > > I think it's ok to work around the problem like this in principle. > > However, I have one concern about the way this condition is detected. > At the moment, you assume that all zynq variants suffer from this > problem and no other chip has it. > > How would you handle the situation if a future zynq variant fixes > the problem? The check is only applicable on Cortex A9 zynqs. Assume for now that it is broken on all those variants. If it gets fixed on some variant add a of_machine_is_compatible() check? > What if it turns out to be a more common problem and we actually > want to work around it in the arm global timer implementation > by dynamically adapting to CPU frequency changes? I believe some > other drivers (x86 tsc?) do this. Would we be able to detect this case > on zynq? > > Arnd If the arm global timer implementation gets cpufreq scaling support, this check isn't needed and should be removed? The only thing that happens if it is left in is that you end up with an inferior, but correct, clock source (which is what you already had before cpufreq support in arm-global-timer). Thanks, Ola
On Mon, 2015-03-23 at 02:39AM +0100, Ola Jeppsson wrote: > As mentioned in this commit: > arm: zynq: Don't use arm_global_timer with cpufreq > 61f1fc7e9258a169ac8afb5ddf657a181e60d052 > > arm_global_timer depends on the CPU frequency. With cpufreq altering the > CPU frequency arm_global_timer will not maintain a stable time base. So > arm_global_timer must not be the clocksource when cpufreq is enabled. > > The above commit tries to solve this at build time by only selecting > CONFIG_ARM_GLOBAL_TIMER if CONFIG_CPU_FREQ is disabled. This is not > always sufficient because other machs can also enable > CONFIG_ARM_GLOBAL_TIMER. > > Therefore: If built with CONFIG_CPU_FREQ and CONFIG_ARM_GLOBAL_TIMER, > disable (on Zynq) the arm_global_timer devicetree node at boot before > clock sources are initialized. This ensures that arm_global_timer will > not be selected clocksource. > > Signed-off-by: Ola Jeppsson <ola@adapteva.com> > --- > arch/arm/mach-zynq/common.c | 26 ++++++++++++++++++++++++++ > 1 file changed, 26 insertions(+) > > diff --git a/arch/arm/mach-zynq/common.c b/arch/arm/mach-zynq/common.c > index c887196..a4666d4 100644 > --- a/arch/arm/mach-zynq/common.c > +++ b/arch/arm/mach-zynq/common.c > @@ -148,10 +148,36 @@ out: > platform_device_register_full(&devinfo); > } > > +#if defined(CONFIG_CPU_FREQ) && defined(CONFIG_ARM_GLOBAL_TIMER) > +static struct property zynq_disable_arm_global_timer_prop = { > + .name = "status", > + .length = sizeof("disabled"), > + .value = "disabled" > +}; > + > +static void __init zynq_disable_arm_global_timer(void) > +{ > + struct device_node *np; > + > + np = of_find_compatible_node(NULL, NULL, "arm,cortex-a9-global-timer"); > + if (!np) > + return; > + > + pr_info("%s: disabling arm_global_timer node\n", __func__); > + > + if (of_update_property(np, &zynq_disable_arm_global_timer_prop)) > + pr_warn("%s: could not disable arm_global_timer node\n", > + __func__); > +} > +#endif can we get an empty implementation of the function here and... > + > static void __init zynq_timer_init(void) > { > zynq_early_slcr_init(); > > +#if defined(CONFIG_CPU_FREQ) && defined(CONFIG_ARM_GLOBAL_TIMER) > + zynq_disable_arm_global_timer(); > +#endif remove #ifdefs down here, please? > zynq_clock_init(); > of_clk_init(NULL); > clocksource_of_init(); > -- > 2.3.3 > Sören
On Friday 10 April 2015 17:09:01 Ola Jeppsson wrote: > On 2015-04-10 15:44, Arnd Bergmann wrote: > > On Monday 23 March 2015 08:05:45 Michal Simek wrote: > >> Hi, > >> > >> On 03/23/2015 02:39 AM, Ola Jeppsson wrote: > >>> As mentioned in this commit: > >>> arm: zynq: Don't use arm_global_timer with cpufreq > >>> 61f1fc7e9258a169ac8afb5ddf657a181e60d052 > >>> > >>> arm_global_timer depends on the CPU frequency. With cpufreq altering the > >>> CPU frequency arm_global_timer will not maintain a stable time base. So > >>> arm_global_timer must not be the clocksource when cpufreq is enabled. > >>> > >>> The above commit tries to solve this at build time by only selecting > >>> CONFIG_ARM_GLOBAL_TIMER if CONFIG_CPU_FREQ is disabled. This is not > >>> always sufficient because other machs can also enable > >>> CONFIG_ARM_GLOBAL_TIMER. > >>> > >>> Therefore: If built with CONFIG_CPU_FREQ and CONFIG_ARM_GLOBAL_TIMER, > >>> disable (on Zynq) the arm_global_timer devicetree node at boot before > >>> clock sources are initialized. This ensures that arm_global_timer will > >>> not be selected clocksource. > >>> > >>> Signed-off-by: Ola Jeppsson <ola@adapteva.com> > > > >> Arnd: Waiting for your thoughts on this one? > >> It is some sort of arch/arm/mach-mvebu/board-v7.c quirk code. > >> > >> I don't think it can be done via Kconfig because it will affect others. > > (catching up with old email) > > > > I think it's ok to work around the problem like this in principle. > > > > However, I have one concern about the way this condition is detected. > > At the moment, you assume that all zynq variants suffer from this > > problem and no other chip has it. > > > > How would you handle the situation if a future zynq variant fixes > > the problem? > The check is only applicable on Cortex A9 zynqs. > Assume for now that it is broken on all those variants. If it gets fixed on some variant add a > of_machine_is_compatible() check? I think it would be better to avoid those if we can. > > What if it turns out to be a more common problem and we actually > > want to work around it in the arm global timer implementation > > by dynamically adapting to CPU frequency changes? I believe some > > other drivers (x86 tsc?) do this. Would we be able to detect this case > > on zynq? > > > If the arm global timer implementation gets cpufreq scaling support, > this check isn't needed and should be removed? The timer code however would still need to find out whether to scale the frequence along with the CPU or not. > The only thing that happens if it is left in is that you end up with > an inferior, but correct, clock source (which is what you already had > before cpufreq support in arm-global-timer). I've just had another idea: how about introducing a new compatible string for the global timer that gets used for timers that have their frequency tied to the CPU (alternatively a bool property in that node). This can be checked by the clocksource driver, which will then be able to either skip the device if cpufreq is in use, or implement a more sophisticated workaround. This would keep the hack outside of platform code and make it immediately available to any platform with the same problem. The only zynq specific hack you need is to modify that property on old dtb files that you want to keep supporting with new kernels (if any). Arnd
On Fri, 2015-04-10 at 10:04PM +0200, Arnd Bergmann wrote: > On Friday 10 April 2015 17:09:01 Ola Jeppsson wrote: > > On 2015-04-10 15:44, Arnd Bergmann wrote: > > > On Monday 23 March 2015 08:05:45 Michal Simek wrote: > > >> Hi, > > >> > > >> On 03/23/2015 02:39 AM, Ola Jeppsson wrote: > > >>> As mentioned in this commit: > > >>> arm: zynq: Don't use arm_global_timer with cpufreq > > >>> 61f1fc7e9258a169ac8afb5ddf657a181e60d052 > > >>> > > >>> arm_global_timer depends on the CPU frequency. With cpufreq altering the > > >>> CPU frequency arm_global_timer will not maintain a stable time base. So > > >>> arm_global_timer must not be the clocksource when cpufreq is enabled. > > >>> > > >>> The above commit tries to solve this at build time by only selecting > > >>> CONFIG_ARM_GLOBAL_TIMER if CONFIG_CPU_FREQ is disabled. This is not > > >>> always sufficient because other machs can also enable > > >>> CONFIG_ARM_GLOBAL_TIMER. > > >>> > > >>> Therefore: If built with CONFIG_CPU_FREQ and CONFIG_ARM_GLOBAL_TIMER, > > >>> disable (on Zynq) the arm_global_timer devicetree node at boot before > > >>> clock sources are initialized. This ensures that arm_global_timer will > > >>> not be selected clocksource. > > >>> > > >>> Signed-off-by: Ola Jeppsson <ola@adapteva.com> > > > > > >> Arnd: Waiting for your thoughts on this one? > > >> It is some sort of arch/arm/mach-mvebu/board-v7.c quirk code. > > >> > > >> I don't think it can be done via Kconfig because it will affect others. > > > (catching up with old email) > > > > > > I think it's ok to work around the problem like this in principle. > > > > > > However, I have one concern about the way this condition is detected. > > > At the moment, you assume that all zynq variants suffer from this > > > problem and no other chip has it. > > > > > > How would you handle the situation if a future zynq variant fixes > > > the problem? > > The check is only applicable on Cortex A9 zynqs. > > Assume for now that it is broken on all those variants. If it gets fixed on some variant add a > > of_machine_is_compatible() check? > > I think it would be better to avoid those if we can. > > > > What if it turns out to be a more common problem and we actually > > > want to work around it in the arm global timer implementation > > > by dynamically adapting to CPU frequency changes? I believe some > > > other drivers (x86 tsc?) do this. Would we be able to detect this case > > > on zynq? > > > > > If the arm global timer implementation gets cpufreq scaling support, > > this check isn't needed and should be removed? > > The timer code however would still need to find out whether to scale the > frequence along with the CPU or not. > > > The only thing that happens if it is left in is that you end up with > > an inferior, but correct, clock source (which is what you already had > > before cpufreq support in arm-global-timer). > > I've just had another idea: how about introducing a new compatible string > for the global timer that gets used for timers that have their frequency > tied to the CPU (alternatively a bool property in that node). This can > be checked by the clocksource driver, which will then be able to either > skip the device if cpufreq is in use, or implement a more sophisticated > workaround. If this is the only available clocksource, you'd still want to use it though. A bad clocksource is still better than none. But I guess that would just mean to lie in the DT. Sören
On Friday 10 April 2015 13:41:34 Sören Brinkmann wrote: > On Fri, 2015-04-10 at 10:04PM +0200, Arnd Bergmann wrote: > > I've just had another idea: how about introducing a new compatible string > > for the global timer that gets used for timers that have their frequency > > tied to the CPU (alternatively a bool property in that node). This can > > be checked by the clocksource driver, which will then be able to either > > skip the device if cpufreq is in use, or implement a more sophisticated > > workaround. > > If this is the only available clocksource, you'd still want to use it > though. A bad clocksource is still better than none. But I guess that > would just mean to lie in the DT. On zynq, you always have a local time, right? If we get a platform that has no reliable clocksource, it's probably time to implement scaling in the global timer driver. Arnd
On Fri, 2015-04-10 at 10:59PM +0200, Arnd Bergmann wrote: > On Friday 10 April 2015 13:41:34 Sören Brinkmann wrote: > > On Fri, 2015-04-10 at 10:04PM +0200, Arnd Bergmann wrote: > > > I've just had another idea: how about introducing a new compatible string > > > for the global timer that gets used for timers that have their frequency > > > tied to the CPU (alternatively a bool property in that node). This can > > > be checked by the clocksource driver, which will then be able to either > > > skip the device if cpufreq is in use, or implement a more sophisticated > > > workaround. > > > > If this is the only available clocksource, you'd still want to use it > > though. A bad clocksource is still better than none. But I guess that > > would just mean to lie in the DT. > > On zynq, you always have a local time, right? Yes, we have the cadence_ttc as clocksource. > If we get a platform that has no reliable clocksource, it's probably > time to implement scaling in the global timer driver. IIRC, the person submitting the driver for the arm_gt said something like that those days: https://lkml.org/lkml/2013/5/8/250 No clue what the current status of that is today though. Sören
On Friday 10 April 2015 15:15:11 Sören Brinkmann wrote: > On Fri, 2015-04-10 at 10:59PM +0200, Arnd Bergmann wrote: > > On Friday 10 April 2015 13:41:34 Sören Brinkmann wrote: > > > On Fri, 2015-04-10 at 10:04PM +0200, Arnd Bergmann wrote: > > > > I've just had another idea: how about introducing a new compatible string > > > > for the global timer that gets used for timers that have their frequency > > > > tied to the CPU (alternatively a bool property in that node). This can > > > > be checked by the clocksource driver, which will then be able to either > > > > skip the device if cpufreq is in use, or implement a more sophisticated > > > > workaround. > > > > > > If this is the only available clocksource, you'd still want to use it > > > though. A bad clocksource is still better than none. But I guess that > > > would just mean to lie in the DT. > > > > On zynq, you always have a local time, right? > > Yes, we have the cadence_ttc as clocksource. I was thinking of the generic ARM timer (drivers/clocksource/arm_arch_timer.c), but I forgot that this was only introduced with Cortex-A7/A15 while Zynq is based on the older Cortex-A9. > > If we get a platform that has no reliable clocksource, it's probably > > time to implement scaling in the global timer driver. > > IIRC, the person submitting the driver for the arm_gt said something > like that those days: https://lkml.org/lkml/2013/5/8/250 > No clue what the current status of that is today though. I see. Well, I'm not making it your problem, since you have another timer to use, but I guess it will get fixed eventually. This kind of bug tends to get duplicated across chips from multiple vendors. Arnd
diff --git a/arch/arm/mach-zynq/common.c b/arch/arm/mach-zynq/common.c index c887196..a4666d4 100644 --- a/arch/arm/mach-zynq/common.c +++ b/arch/arm/mach-zynq/common.c @@ -148,10 +148,36 @@ out: platform_device_register_full(&devinfo); } +#if defined(CONFIG_CPU_FREQ) && defined(CONFIG_ARM_GLOBAL_TIMER) +static struct property zynq_disable_arm_global_timer_prop = { + .name = "status", + .length = sizeof("disabled"), + .value = "disabled" +}; + +static void __init zynq_disable_arm_global_timer(void) +{ + struct device_node *np; + + np = of_find_compatible_node(NULL, NULL, "arm,cortex-a9-global-timer"); + if (!np) + return; + + pr_info("%s: disabling arm_global_timer node\n", __func__); + + if (of_update_property(np, &zynq_disable_arm_global_timer_prop)) + pr_warn("%s: could not disable arm_global_timer node\n", + __func__); +} +#endif + static void __init zynq_timer_init(void) { zynq_early_slcr_init(); +#if defined(CONFIG_CPU_FREQ) && defined(CONFIG_ARM_GLOBAL_TIMER) + zynq_disable_arm_global_timer(); +#endif zynq_clock_init(); of_clk_init(NULL); clocksource_of_init();
As mentioned in this commit: arm: zynq: Don't use arm_global_timer with cpufreq 61f1fc7e9258a169ac8afb5ddf657a181e60d052 arm_global_timer depends on the CPU frequency. With cpufreq altering the CPU frequency arm_global_timer will not maintain a stable time base. So arm_global_timer must not be the clocksource when cpufreq is enabled. The above commit tries to solve this at build time by only selecting CONFIG_ARM_GLOBAL_TIMER if CONFIG_CPU_FREQ is disabled. This is not always sufficient because other machs can also enable CONFIG_ARM_GLOBAL_TIMER. Therefore: If built with CONFIG_CPU_FREQ and CONFIG_ARM_GLOBAL_TIMER, disable (on Zynq) the arm_global_timer devicetree node at boot before clock sources are initialized. This ensures that arm_global_timer will not be selected clocksource. Signed-off-by: Ola Jeppsson <ola@adapteva.com> --- arch/arm/mach-zynq/common.c | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+)