2018-08-22 10:26:20 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
2009-05-01 14:51:00 +08:00
|
|
|
/*
|
|
|
|
* SuperH Timer Support - TMU
|
|
|
|
*
|
|
|
|
* Copyright (C) 2009 Magnus Damm
|
|
|
|
*/
|
|
|
|
|
2014-02-12 23:56:44 +08:00
|
|
|
#include <linux/clk.h>
|
|
|
|
#include <linux/clockchips.h>
|
|
|
|
#include <linux/clocksource.h>
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/err.h>
|
2009-05-01 14:51:00 +08:00
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/interrupt.h>
|
|
|
|
#include <linux/io.h>
|
2014-02-12 23:56:44 +08:00
|
|
|
#include <linux/ioport.h>
|
2009-05-01 14:51:00 +08:00
|
|
|
#include <linux/irq.h>
|
2011-07-04 01:36:22 +08:00
|
|
|
#include <linux/module.h>
|
2014-04-11 22:23:40 +08:00
|
|
|
#include <linux/of.h>
|
2014-02-12 23:56:44 +08:00
|
|
|
#include <linux/platform_device.h>
|
2012-03-14 05:40:00 +08:00
|
|
|
#include <linux/pm_domain.h>
|
2012-08-06 07:41:20 +08:00
|
|
|
#include <linux/pm_runtime.h>
|
2014-02-12 23:56:44 +08:00
|
|
|
#include <linux/sh_timer.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/spinlock.h>
|
2009-05-01 14:51:00 +08:00
|
|
|
|
2019-10-03 17:29:12 +08:00
|
|
|
#ifdef CONFIG_SUPERH
|
|
|
|
#include <asm/platform_early.h>
|
|
|
|
#endif
|
|
|
|
|
2014-01-28 19:36:48 +08:00
|
|
|
enum sh_tmu_model {
|
|
|
|
SH_TMU,
|
|
|
|
SH_TMU_SH3,
|
|
|
|
};
|
|
|
|
|
2014-01-28 05:04:17 +08:00
|
|
|
struct sh_tmu_device;
|
2014-01-27 22:29:19 +08:00
|
|
|
|
|
|
|
struct sh_tmu_channel {
|
2014-01-28 05:04:17 +08:00
|
|
|
struct sh_tmu_device *tmu;
|
2014-01-28 05:04:17 +08:00
|
|
|
unsigned int index;
|
2014-01-27 22:29:19 +08:00
|
|
|
|
2014-01-28 05:04:17 +08:00
|
|
|
void __iomem *base;
|
2014-02-17 18:27:49 +08:00
|
|
|
int irq;
|
2014-01-27 22:29:19 +08:00
|
|
|
|
2009-05-01 14:51:00 +08:00
|
|
|
unsigned long periodic;
|
|
|
|
struct clock_event_device ced;
|
|
|
|
struct clocksource cs;
|
2012-08-06 07:41:20 +08:00
|
|
|
bool cs_enabled;
|
2012-08-06 07:48:17 +08:00
|
|
|
unsigned int enable_count;
|
2009-05-01 14:51:00 +08:00
|
|
|
};
|
|
|
|
|
2014-01-28 05:04:17 +08:00
|
|
|
struct sh_tmu_device {
|
2014-01-27 22:29:19 +08:00
|
|
|
struct platform_device *pdev;
|
|
|
|
|
|
|
|
void __iomem *mapbase;
|
|
|
|
struct clk *clk;
|
clocksource: sh_tmu: Compute rate before registration again
With the upcoming NTP correction related rate adjustments to be implemented
in the clockevents core, the latter needs to get informed about every rate
change of a clockevent device made after its registration.
Currently, sh_tmu violates this requirement in that it registers its
clockevent device with a dummy rate and sets its final rate through
clockevents_config() called from its ->set_state_oneshot() and
->set_state_periodic() functions respectively.
This patch moves the setting of the clockevent device's rate to its
registration.
Note that there has been some back and forth regarding this question with
respect to the clocksource also provided by this driver:
commit 66f49121ffa4 ("clocksource: sh_tmu: compute mult and shift before
registration")
moves the rate determination from the clocksource's ->enable() function to
before its registration. OTOH, the later
commit 0aeac458d9eb ("clocksource: sh_tmu: __clocksource_updatefreq_hz()
update")
basically reverts this, saying
"Without this patch the old code uses clocksource_register() together
with a hack that assumes a never changing clock rate."
However, I checked all current sh_tmu users in arch/sh as well as in
arch/arm/mach-shmobile carefully and right now, none of them changes any
rate in any clock tree relevant to sh_tmu after their respective
time_init(). Since all sh_tmu instances are created after time_init(), none
of them should ever observe any clock rate changes.
What's more, both, a clocksource as well as a clockevent device, can
immediately get selected for use at their registration and thus, enabled
at this point already. So it's probably safer to assume a "never changing
clock rate" here.
- Move the struct sh_tmu_channel's ->rate member to struct sh_tmu_device:
it's a property of the underlying clock which is in turn specific to
the sh_tmu_device.
- Determine the ->rate value in sh_tmu_setup() at device probing rather
than at first usage.
- Set the clockevent device's rate at its registration.
- Although not strictly necessary for the upcoming clockevent core changes,
set the clocksource's rate at its registration for consistency.
Signed-off-by: Nicolai Stange <nicstange@gmail.com>
Signed-off-by: John Stultz <john.stultz@linaro.org>
2017-02-07 05:12:00 +08:00
|
|
|
unsigned long rate;
|
2014-01-27 22:29:19 +08:00
|
|
|
|
2014-01-28 19:36:48 +08:00
|
|
|
enum sh_tmu_model model;
|
|
|
|
|
2014-02-17 23:49:05 +08:00
|
|
|
raw_spinlock_t lock; /* Protect the shared start/stop register */
|
|
|
|
|
2014-01-28 05:04:17 +08:00
|
|
|
struct sh_tmu_channel *channels;
|
|
|
|
unsigned int num_channels;
|
2014-01-28 19:36:48 +08:00
|
|
|
|
|
|
|
bool has_clockevent;
|
|
|
|
bool has_clocksource;
|
2014-01-27 22:29:19 +08:00
|
|
|
};
|
|
|
|
|
2009-05-01 14:51:00 +08:00
|
|
|
#define TSTR -1 /* shared register */
|
|
|
|
#define TCOR 0 /* channel register */
|
|
|
|
#define TCNT 1 /* channel register */
|
|
|
|
#define TCR 2 /* channel register */
|
|
|
|
|
2014-01-29 07:33:08 +08:00
|
|
|
#define TCR_UNF (1 << 8)
|
|
|
|
#define TCR_UNIE (1 << 5)
|
|
|
|
#define TCR_TPSC_CLK4 (0 << 0)
|
|
|
|
#define TCR_TPSC_CLK16 (1 << 0)
|
|
|
|
#define TCR_TPSC_CLK64 (2 << 0)
|
|
|
|
#define TCR_TPSC_CLK256 (3 << 0)
|
|
|
|
#define TCR_TPSC_CLK1024 (4 << 0)
|
|
|
|
#define TCR_TPSC_MASK (7 << 0)
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
static inline unsigned long sh_tmu_read(struct sh_tmu_channel *ch, int reg_nr)
|
2009-05-01 14:51:00 +08:00
|
|
|
{
|
|
|
|
unsigned long offs;
|
|
|
|
|
2014-01-28 19:36:48 +08:00
|
|
|
if (reg_nr == TSTR) {
|
|
|
|
switch (ch->tmu->model) {
|
|
|
|
case SH_TMU_SH3:
|
|
|
|
return ioread8(ch->tmu->mapbase + 2);
|
|
|
|
case SH_TMU:
|
|
|
|
return ioread8(ch->tmu->mapbase + 4);
|
|
|
|
}
|
|
|
|
}
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
offs = reg_nr << 2;
|
|
|
|
|
|
|
|
if (reg_nr == TCR)
|
2014-01-28 05:04:17 +08:00
|
|
|
return ioread16(ch->base + offs);
|
2009-05-01 14:51:00 +08:00
|
|
|
else
|
2014-01-28 05:04:17 +08:00
|
|
|
return ioread32(ch->base + offs);
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
static inline void sh_tmu_write(struct sh_tmu_channel *ch, int reg_nr,
|
2009-05-01 14:51:00 +08:00
|
|
|
unsigned long value)
|
|
|
|
{
|
|
|
|
unsigned long offs;
|
|
|
|
|
|
|
|
if (reg_nr == TSTR) {
|
2014-01-28 19:36:48 +08:00
|
|
|
switch (ch->tmu->model) {
|
|
|
|
case SH_TMU_SH3:
|
|
|
|
return iowrite8(value, ch->tmu->mapbase + 2);
|
|
|
|
case SH_TMU:
|
|
|
|
return iowrite8(value, ch->tmu->mapbase + 4);
|
|
|
|
}
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
offs = reg_nr << 2;
|
|
|
|
|
|
|
|
if (reg_nr == TCR)
|
2014-01-28 05:04:17 +08:00
|
|
|
iowrite16(value, ch->base + offs);
|
2009-05-01 14:51:00 +08:00
|
|
|
else
|
2014-01-28 05:04:17 +08:00
|
|
|
iowrite32(value, ch->base + offs);
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
static void sh_tmu_start_stop_ch(struct sh_tmu_channel *ch, int start)
|
2009-05-01 14:51:00 +08:00
|
|
|
{
|
|
|
|
unsigned long flags, value;
|
|
|
|
|
|
|
|
/* start stop register shared by multiple timer channels */
|
2014-02-17 23:49:05 +08:00
|
|
|
raw_spin_lock_irqsave(&ch->tmu->lock, flags);
|
2014-01-27 22:29:19 +08:00
|
|
|
value = sh_tmu_read(ch, TSTR);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
if (start)
|
2014-01-28 05:04:17 +08:00
|
|
|
value |= 1 << ch->index;
|
2009-05-01 14:51:00 +08:00
|
|
|
else
|
2014-01-28 05:04:17 +08:00
|
|
|
value &= ~(1 << ch->index);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
sh_tmu_write(ch, TSTR, value);
|
2014-02-17 23:49:05 +08:00
|
|
|
raw_spin_unlock_irqrestore(&ch->tmu->lock, flags);
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
static int __sh_tmu_enable(struct sh_tmu_channel *ch)
|
2009-05-01 14:51:00 +08:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
2011-05-31 14:23:20 +08:00
|
|
|
/* enable clock */
|
2014-01-27 22:29:19 +08:00
|
|
|
ret = clk_enable(ch->tmu->clk);
|
2009-05-01 14:51:00 +08:00
|
|
|
if (ret) {
|
2014-01-28 05:04:17 +08:00
|
|
|
dev_err(&ch->tmu->pdev->dev, "ch%u: cannot enable clock\n",
|
|
|
|
ch->index);
|
2009-05-01 14:51:00 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* make sure channel is disabled */
|
2014-01-27 22:29:19 +08:00
|
|
|
sh_tmu_start_stop_ch(ch, 0);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
/* maximum timeout */
|
2014-01-27 22:29:19 +08:00
|
|
|
sh_tmu_write(ch, TCOR, 0xffffffff);
|
|
|
|
sh_tmu_write(ch, TCNT, 0xffffffff);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
/* configure channel to parent clock / 4, irq off */
|
2014-01-29 07:33:08 +08:00
|
|
|
sh_tmu_write(ch, TCR, TCR_TPSC_CLK4);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
/* enable channel */
|
2014-01-27 22:29:19 +08:00
|
|
|
sh_tmu_start_stop_ch(ch, 1);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
static int sh_tmu_enable(struct sh_tmu_channel *ch)
|
2012-08-06 07:48:17 +08:00
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
if (ch->enable_count++ > 0)
|
2012-08-06 07:48:17 +08:00
|
|
|
return 0;
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
pm_runtime_get_sync(&ch->tmu->pdev->dev);
|
|
|
|
dev_pm_syscore_device(&ch->tmu->pdev->dev, true);
|
2012-08-06 07:48:17 +08:00
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
return __sh_tmu_enable(ch);
|
2012-08-06 07:48:17 +08:00
|
|
|
}
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
static void __sh_tmu_disable(struct sh_tmu_channel *ch)
|
2009-05-01 14:51:00 +08:00
|
|
|
{
|
|
|
|
/* disable channel */
|
2014-01-27 22:29:19 +08:00
|
|
|
sh_tmu_start_stop_ch(ch, 0);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
2009-06-17 13:04:04 +08:00
|
|
|
/* disable interrupts in TMU block */
|
2014-01-29 07:33:08 +08:00
|
|
|
sh_tmu_write(ch, TCR, TCR_TPSC_CLK4);
|
2009-06-17 13:04:04 +08:00
|
|
|
|
2011-05-31 14:23:20 +08:00
|
|
|
/* stop clock */
|
2014-01-27 22:29:19 +08:00
|
|
|
clk_disable(ch->tmu->clk);
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
static void sh_tmu_disable(struct sh_tmu_channel *ch)
|
2012-08-06 07:48:17 +08:00
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
if (WARN_ON(ch->enable_count == 0))
|
2012-08-06 07:48:17 +08:00
|
|
|
return;
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
if (--ch->enable_count > 0)
|
2012-08-06 07:48:17 +08:00
|
|
|
return;
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
__sh_tmu_disable(ch);
|
2012-08-06 07:48:17 +08:00
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
dev_pm_syscore_device(&ch->tmu->pdev->dev, false);
|
|
|
|
pm_runtime_put(&ch->tmu->pdev->dev);
|
2012-08-06 07:48:17 +08:00
|
|
|
}
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
static void sh_tmu_set_next(struct sh_tmu_channel *ch, unsigned long delta,
|
2009-05-01 14:51:00 +08:00
|
|
|
int periodic)
|
|
|
|
{
|
|
|
|
/* stop timer */
|
2014-01-27 22:29:19 +08:00
|
|
|
sh_tmu_start_stop_ch(ch, 0);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
/* acknowledge interrupt */
|
2014-01-27 22:29:19 +08:00
|
|
|
sh_tmu_read(ch, TCR);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
/* enable interrupt */
|
2014-01-29 07:33:08 +08:00
|
|
|
sh_tmu_write(ch, TCR, TCR_UNIE | TCR_TPSC_CLK4);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
/* reload delta value in case of periodic timer */
|
|
|
|
if (periodic)
|
2014-01-27 22:29:19 +08:00
|
|
|
sh_tmu_write(ch, TCOR, delta);
|
2009-05-01 14:51:00 +08:00
|
|
|
else
|
2014-01-27 22:29:19 +08:00
|
|
|
sh_tmu_write(ch, TCOR, 0xffffffff);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
sh_tmu_write(ch, TCNT, delta);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
/* start timer */
|
2014-01-27 22:29:19 +08:00
|
|
|
sh_tmu_start_stop_ch(ch, 1);
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static irqreturn_t sh_tmu_interrupt(int irq, void *dev_id)
|
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
struct sh_tmu_channel *ch = dev_id;
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
/* disable or acknowledge interrupt */
|
2015-06-18 18:54:36 +08:00
|
|
|
if (clockevent_state_oneshot(&ch->ced))
|
2014-01-29 07:33:08 +08:00
|
|
|
sh_tmu_write(ch, TCR, TCR_TPSC_CLK4);
|
2009-05-01 14:51:00 +08:00
|
|
|
else
|
2014-01-29 07:33:08 +08:00
|
|
|
sh_tmu_write(ch, TCR, TCR_UNIE | TCR_TPSC_CLK4);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
/* notify clockevent layer */
|
2014-01-27 22:29:19 +08:00
|
|
|
ch->ced.event_handler(&ch->ced);
|
2009-05-01 14:51:00 +08:00
|
|
|
return IRQ_HANDLED;
|
|
|
|
}
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
static struct sh_tmu_channel *cs_to_sh_tmu(struct clocksource *cs)
|
2009-05-01 14:51:00 +08:00
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
return container_of(cs, struct sh_tmu_channel, cs);
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
|
|
|
|
2016-12-22 03:32:01 +08:00
|
|
|
static u64 sh_tmu_clocksource_read(struct clocksource *cs)
|
2009-05-01 14:51:00 +08:00
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
struct sh_tmu_channel *ch = cs_to_sh_tmu(cs);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
return sh_tmu_read(ch, TCNT) ^ 0xffffffff;
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int sh_tmu_clocksource_enable(struct clocksource *cs)
|
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
struct sh_tmu_channel *ch = cs_to_sh_tmu(cs);
|
2011-04-25 21:38:37 +08:00
|
|
|
int ret;
|
2009-05-01 14:51:00 +08:00
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
if (WARN_ON(ch->cs_enabled))
|
2012-08-06 07:48:17 +08:00
|
|
|
return 0;
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
ret = sh_tmu_enable(ch);
|
clocksource: sh_tmu: Compute rate before registration again
With the upcoming NTP correction related rate adjustments to be implemented
in the clockevents core, the latter needs to get informed about every rate
change of a clockevent device made after its registration.
Currently, sh_tmu violates this requirement in that it registers its
clockevent device with a dummy rate and sets its final rate through
clockevents_config() called from its ->set_state_oneshot() and
->set_state_periodic() functions respectively.
This patch moves the setting of the clockevent device's rate to its
registration.
Note that there has been some back and forth regarding this question with
respect to the clocksource also provided by this driver:
commit 66f49121ffa4 ("clocksource: sh_tmu: compute mult and shift before
registration")
moves the rate determination from the clocksource's ->enable() function to
before its registration. OTOH, the later
commit 0aeac458d9eb ("clocksource: sh_tmu: __clocksource_updatefreq_hz()
update")
basically reverts this, saying
"Without this patch the old code uses clocksource_register() together
with a hack that assumes a never changing clock rate."
However, I checked all current sh_tmu users in arch/sh as well as in
arch/arm/mach-shmobile carefully and right now, none of them changes any
rate in any clock tree relevant to sh_tmu after their respective
time_init(). Since all sh_tmu instances are created after time_init(), none
of them should ever observe any clock rate changes.
What's more, both, a clocksource as well as a clockevent device, can
immediately get selected for use at their registration and thus, enabled
at this point already. So it's probably safer to assume a "never changing
clock rate" here.
- Move the struct sh_tmu_channel's ->rate member to struct sh_tmu_device:
it's a property of the underlying clock which is in turn specific to
the sh_tmu_device.
- Determine the ->rate value in sh_tmu_setup() at device probing rather
than at first usage.
- Set the clockevent device's rate at its registration.
- Although not strictly necessary for the upcoming clockevent core changes,
set the clocksource's rate at its registration for consistency.
Signed-off-by: Nicolai Stange <nicstange@gmail.com>
Signed-off-by: John Stultz <john.stultz@linaro.org>
2017-02-07 05:12:00 +08:00
|
|
|
if (!ret)
|
2014-01-27 22:29:19 +08:00
|
|
|
ch->cs_enabled = true;
|
2012-08-06 07:48:17 +08:00
|
|
|
|
2011-04-25 21:38:37 +08:00
|
|
|
return ret;
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void sh_tmu_clocksource_disable(struct clocksource *cs)
|
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
struct sh_tmu_channel *ch = cs_to_sh_tmu(cs);
|
2012-08-06 07:41:20 +08:00
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
if (WARN_ON(!ch->cs_enabled))
|
2012-08-06 07:48:17 +08:00
|
|
|
return;
|
2012-08-06 07:41:20 +08:00
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
sh_tmu_disable(ch);
|
|
|
|
ch->cs_enabled = false;
|
2012-08-06 07:41:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void sh_tmu_clocksource_suspend(struct clocksource *cs)
|
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
struct sh_tmu_channel *ch = cs_to_sh_tmu(cs);
|
2012-08-06 07:41:20 +08:00
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
if (!ch->cs_enabled)
|
2012-08-06 07:48:17 +08:00
|
|
|
return;
|
2012-08-06 07:41:20 +08:00
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
if (--ch->enable_count == 0) {
|
|
|
|
__sh_tmu_disable(ch);
|
|
|
|
pm_genpd_syscore_poweroff(&ch->tmu->pdev->dev);
|
2012-08-06 07:48:17 +08:00
|
|
|
}
|
2012-08-06 07:41:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void sh_tmu_clocksource_resume(struct clocksource *cs)
|
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
struct sh_tmu_channel *ch = cs_to_sh_tmu(cs);
|
2012-08-06 07:41:20 +08:00
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
if (!ch->cs_enabled)
|
2012-08-06 07:48:17 +08:00
|
|
|
return;
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
if (ch->enable_count++ == 0) {
|
|
|
|
pm_genpd_syscore_poweron(&ch->tmu->pdev->dev);
|
|
|
|
__sh_tmu_enable(ch);
|
2012-08-06 07:48:17 +08:00
|
|
|
}
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
static int sh_tmu_register_clocksource(struct sh_tmu_channel *ch,
|
2014-02-20 00:00:31 +08:00
|
|
|
const char *name)
|
2009-05-01 14:51:00 +08:00
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
struct clocksource *cs = &ch->cs;
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
cs->name = name;
|
2014-02-20 00:00:31 +08:00
|
|
|
cs->rating = 200;
|
2009-05-01 14:51:00 +08:00
|
|
|
cs->read = sh_tmu_clocksource_read;
|
|
|
|
cs->enable = sh_tmu_clocksource_enable;
|
|
|
|
cs->disable = sh_tmu_clocksource_disable;
|
2012-08-06 07:41:20 +08:00
|
|
|
cs->suspend = sh_tmu_clocksource_suspend;
|
|
|
|
cs->resume = sh_tmu_clocksource_resume;
|
2009-05-01 14:51:00 +08:00
|
|
|
cs->mask = CLOCKSOURCE_MASK(32);
|
|
|
|
cs->flags = CLOCK_SOURCE_IS_CONTINUOUS;
|
2010-06-01 05:45:48 +08:00
|
|
|
|
2014-01-28 05:04:17 +08:00
|
|
|
dev_info(&ch->tmu->pdev->dev, "ch%u: used as clock source\n",
|
|
|
|
ch->index);
|
2011-04-25 21:38:37 +08:00
|
|
|
|
clocksource: sh_tmu: Compute rate before registration again
With the upcoming NTP correction related rate adjustments to be implemented
in the clockevents core, the latter needs to get informed about every rate
change of a clockevent device made after its registration.
Currently, sh_tmu violates this requirement in that it registers its
clockevent device with a dummy rate and sets its final rate through
clockevents_config() called from its ->set_state_oneshot() and
->set_state_periodic() functions respectively.
This patch moves the setting of the clockevent device's rate to its
registration.
Note that there has been some back and forth regarding this question with
respect to the clocksource also provided by this driver:
commit 66f49121ffa4 ("clocksource: sh_tmu: compute mult and shift before
registration")
moves the rate determination from the clocksource's ->enable() function to
before its registration. OTOH, the later
commit 0aeac458d9eb ("clocksource: sh_tmu: __clocksource_updatefreq_hz()
update")
basically reverts this, saying
"Without this patch the old code uses clocksource_register() together
with a hack that assumes a never changing clock rate."
However, I checked all current sh_tmu users in arch/sh as well as in
arch/arm/mach-shmobile carefully and right now, none of them changes any
rate in any clock tree relevant to sh_tmu after their respective
time_init(). Since all sh_tmu instances are created after time_init(), none
of them should ever observe any clock rate changes.
What's more, both, a clocksource as well as a clockevent device, can
immediately get selected for use at their registration and thus, enabled
at this point already. So it's probably safer to assume a "never changing
clock rate" here.
- Move the struct sh_tmu_channel's ->rate member to struct sh_tmu_device:
it's a property of the underlying clock which is in turn specific to
the sh_tmu_device.
- Determine the ->rate value in sh_tmu_setup() at device probing rather
than at first usage.
- Set the clockevent device's rate at its registration.
- Although not strictly necessary for the upcoming clockevent core changes,
set the clocksource's rate at its registration for consistency.
Signed-off-by: Nicolai Stange <nicstange@gmail.com>
Signed-off-by: John Stultz <john.stultz@linaro.org>
2017-02-07 05:12:00 +08:00
|
|
|
clocksource_register_hz(cs, ch->tmu->rate);
|
2009-05-01 14:51:00 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
static struct sh_tmu_channel *ced_to_sh_tmu(struct clock_event_device *ced)
|
2009-05-01 14:51:00 +08:00
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
return container_of(ced, struct sh_tmu_channel, ced);
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
static void sh_tmu_clock_event_start(struct sh_tmu_channel *ch, int periodic)
|
2009-05-01 14:51:00 +08:00
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
sh_tmu_enable(ch);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
if (periodic) {
|
clocksource: sh_tmu: Compute rate before registration again
With the upcoming NTP correction related rate adjustments to be implemented
in the clockevents core, the latter needs to get informed about every rate
change of a clockevent device made after its registration.
Currently, sh_tmu violates this requirement in that it registers its
clockevent device with a dummy rate and sets its final rate through
clockevents_config() called from its ->set_state_oneshot() and
->set_state_periodic() functions respectively.
This patch moves the setting of the clockevent device's rate to its
registration.
Note that there has been some back and forth regarding this question with
respect to the clocksource also provided by this driver:
commit 66f49121ffa4 ("clocksource: sh_tmu: compute mult and shift before
registration")
moves the rate determination from the clocksource's ->enable() function to
before its registration. OTOH, the later
commit 0aeac458d9eb ("clocksource: sh_tmu: __clocksource_updatefreq_hz()
update")
basically reverts this, saying
"Without this patch the old code uses clocksource_register() together
with a hack that assumes a never changing clock rate."
However, I checked all current sh_tmu users in arch/sh as well as in
arch/arm/mach-shmobile carefully and right now, none of them changes any
rate in any clock tree relevant to sh_tmu after their respective
time_init(). Since all sh_tmu instances are created after time_init(), none
of them should ever observe any clock rate changes.
What's more, both, a clocksource as well as a clockevent device, can
immediately get selected for use at their registration and thus, enabled
at this point already. So it's probably safer to assume a "never changing
clock rate" here.
- Move the struct sh_tmu_channel's ->rate member to struct sh_tmu_device:
it's a property of the underlying clock which is in turn specific to
the sh_tmu_device.
- Determine the ->rate value in sh_tmu_setup() at device probing rather
than at first usage.
- Set the clockevent device's rate at its registration.
- Although not strictly necessary for the upcoming clockevent core changes,
set the clocksource's rate at its registration for consistency.
Signed-off-by: Nicolai Stange <nicstange@gmail.com>
Signed-off-by: John Stultz <john.stultz@linaro.org>
2017-02-07 05:12:00 +08:00
|
|
|
ch->periodic = (ch->tmu->rate + HZ/2) / HZ;
|
2014-01-27 22:29:19 +08:00
|
|
|
sh_tmu_set_next(ch, ch->periodic, 1);
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-18 18:54:36 +08:00
|
|
|
static int sh_tmu_clock_event_shutdown(struct clock_event_device *ced)
|
|
|
|
{
|
|
|
|
struct sh_tmu_channel *ch = ced_to_sh_tmu(ced);
|
|
|
|
|
2015-07-21 10:31:14 +08:00
|
|
|
if (clockevent_state_oneshot(ced) || clockevent_state_periodic(ced))
|
|
|
|
sh_tmu_disable(ch);
|
2015-06-18 18:54:36 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int sh_tmu_clock_event_set_state(struct clock_event_device *ced,
|
|
|
|
int periodic)
|
2009-05-01 14:51:00 +08:00
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
struct sh_tmu_channel *ch = ced_to_sh_tmu(ced);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
/* deal with old setting first */
|
2015-06-18 18:54:36 +08:00
|
|
|
if (clockevent_state_oneshot(ced) || clockevent_state_periodic(ced))
|
2014-01-27 22:29:19 +08:00
|
|
|
sh_tmu_disable(ch);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
2015-06-18 18:54:36 +08:00
|
|
|
dev_info(&ch->tmu->pdev->dev, "ch%u: used for %s clock events\n",
|
|
|
|
ch->index, periodic ? "periodic" : "oneshot");
|
|
|
|
sh_tmu_clock_event_start(ch, periodic);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int sh_tmu_clock_event_set_oneshot(struct clock_event_device *ced)
|
|
|
|
{
|
|
|
|
return sh_tmu_clock_event_set_state(ced, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int sh_tmu_clock_event_set_periodic(struct clock_event_device *ced)
|
|
|
|
{
|
|
|
|
return sh_tmu_clock_event_set_state(ced, 1);
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int sh_tmu_clock_event_next(unsigned long delta,
|
|
|
|
struct clock_event_device *ced)
|
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
struct sh_tmu_channel *ch = ced_to_sh_tmu(ced);
|
2009-05-01 14:51:00 +08:00
|
|
|
|
2015-06-18 18:54:36 +08:00
|
|
|
BUG_ON(!clockevent_state_oneshot(ced));
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
/* program new delta value */
|
2014-01-27 22:29:19 +08:00
|
|
|
sh_tmu_set_next(ch, delta, 0);
|
2009-05-01 14:51:00 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-08-06 07:41:20 +08:00
|
|
|
static void sh_tmu_clock_event_suspend(struct clock_event_device *ced)
|
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
pm_genpd_syscore_poweroff(&ced_to_sh_tmu(ced)->tmu->pdev->dev);
|
2012-08-06 07:41:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void sh_tmu_clock_event_resume(struct clock_event_device *ced)
|
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
pm_genpd_syscore_poweron(&ced_to_sh_tmu(ced)->tmu->pdev->dev);
|
2012-08-06 07:41:20 +08:00
|
|
|
}
|
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
static void sh_tmu_register_clockevent(struct sh_tmu_channel *ch,
|
2014-02-20 00:00:31 +08:00
|
|
|
const char *name)
|
2009-05-01 14:51:00 +08:00
|
|
|
{
|
2014-01-27 22:29:19 +08:00
|
|
|
struct clock_event_device *ced = &ch->ced;
|
2009-05-01 14:51:00 +08:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
ced->name = name;
|
|
|
|
ced->features = CLOCK_EVT_FEAT_PERIODIC;
|
|
|
|
ced->features |= CLOCK_EVT_FEAT_ONESHOT;
|
2014-02-20 00:00:31 +08:00
|
|
|
ced->rating = 200;
|
2014-12-16 17:48:54 +08:00
|
|
|
ced->cpumask = cpu_possible_mask;
|
2009-05-01 14:51:00 +08:00
|
|
|
ced->set_next_event = sh_tmu_clock_event_next;
|
2015-06-18 18:54:36 +08:00
|
|
|
ced->set_state_shutdown = sh_tmu_clock_event_shutdown;
|
|
|
|
ced->set_state_periodic = sh_tmu_clock_event_set_periodic;
|
|
|
|
ced->set_state_oneshot = sh_tmu_clock_event_set_oneshot;
|
2012-08-06 07:41:20 +08:00
|
|
|
ced->suspend = sh_tmu_clock_event_suspend;
|
|
|
|
ced->resume = sh_tmu_clock_event_resume;
|
2009-05-01 14:51:00 +08:00
|
|
|
|
2014-01-28 05:04:17 +08:00
|
|
|
dev_info(&ch->tmu->pdev->dev, "ch%u: used for clock events\n",
|
|
|
|
ch->index);
|
2012-06-11 16:10:16 +08:00
|
|
|
|
clocksource: sh_tmu: Compute rate before registration again
With the upcoming NTP correction related rate adjustments to be implemented
in the clockevents core, the latter needs to get informed about every rate
change of a clockevent device made after its registration.
Currently, sh_tmu violates this requirement in that it registers its
clockevent device with a dummy rate and sets its final rate through
clockevents_config() called from its ->set_state_oneshot() and
->set_state_periodic() functions respectively.
This patch moves the setting of the clockevent device's rate to its
registration.
Note that there has been some back and forth regarding this question with
respect to the clocksource also provided by this driver:
commit 66f49121ffa4 ("clocksource: sh_tmu: compute mult and shift before
registration")
moves the rate determination from the clocksource's ->enable() function to
before its registration. OTOH, the later
commit 0aeac458d9eb ("clocksource: sh_tmu: __clocksource_updatefreq_hz()
update")
basically reverts this, saying
"Without this patch the old code uses clocksource_register() together
with a hack that assumes a never changing clock rate."
However, I checked all current sh_tmu users in arch/sh as well as in
arch/arm/mach-shmobile carefully and right now, none of them changes any
rate in any clock tree relevant to sh_tmu after their respective
time_init(). Since all sh_tmu instances are created after time_init(), none
of them should ever observe any clock rate changes.
What's more, both, a clocksource as well as a clockevent device, can
immediately get selected for use at their registration and thus, enabled
at this point already. So it's probably safer to assume a "never changing
clock rate" here.
- Move the struct sh_tmu_channel's ->rate member to struct sh_tmu_device:
it's a property of the underlying clock which is in turn specific to
the sh_tmu_device.
- Determine the ->rate value in sh_tmu_setup() at device probing rather
than at first usage.
- Set the clockevent device's rate at its registration.
- Although not strictly necessary for the upcoming clockevent core changes,
set the clocksource's rate at its registration for consistency.
Signed-off-by: Nicolai Stange <nicstange@gmail.com>
Signed-off-by: John Stultz <john.stultz@linaro.org>
2017-02-07 05:12:00 +08:00
|
|
|
clockevents_config_and_register(ced, ch->tmu->rate, 0x300, 0xffffffff);
|
2010-02-25 15:37:46 +08:00
|
|
|
|
2014-01-27 22:29:19 +08:00
|
|
|
ret = request_irq(ch->irq, sh_tmu_interrupt,
|
2014-02-17 18:27:49 +08:00
|
|
|
IRQF_TIMER | IRQF_IRQPOLL | IRQF_NOBALANCING,
|
2014-01-27 22:29:19 +08:00
|
|
|
dev_name(&ch->tmu->pdev->dev), ch);
|
2009-05-01 14:51:00 +08:00
|
|
|
if (ret) {
|
2014-01-28 05:04:17 +08:00
|
|
|
dev_err(&ch->tmu->pdev->dev, "ch%u: failed to request irq %d\n",
|
|
|
|
ch->index, ch->irq);
|
2009-05-01 14:51:00 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-17 23:04:16 +08:00
|
|
|
static int sh_tmu_register(struct sh_tmu_channel *ch, const char *name,
|
2014-02-20 00:00:31 +08:00
|
|
|
bool clockevent, bool clocksource)
|
2009-05-01 14:51:00 +08:00
|
|
|
{
|
2014-01-28 19:36:48 +08:00
|
|
|
if (clockevent) {
|
|
|
|
ch->tmu->has_clockevent = true;
|
2014-02-20 00:00:31 +08:00
|
|
|
sh_tmu_register_clockevent(ch, name);
|
2014-01-28 19:36:48 +08:00
|
|
|
} else if (clocksource) {
|
|
|
|
ch->tmu->has_clocksource = true;
|
2014-02-20 00:00:31 +08:00
|
|
|
sh_tmu_register_clocksource(ch, name);
|
2014-01-28 19:36:48 +08:00
|
|
|
}
|
2009-05-01 14:51:00 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-01-28 19:36:48 +08:00
|
|
|
static int sh_tmu_channel_setup(struct sh_tmu_channel *ch, unsigned int index,
|
|
|
|
bool clockevent, bool clocksource,
|
2014-01-28 05:04:17 +08:00
|
|
|
struct sh_tmu_device *tmu)
|
|
|
|
{
|
2014-01-28 19:36:48 +08:00
|
|
|
/* Skip unused channels. */
|
|
|
|
if (!clockevent && !clocksource)
|
|
|
|
return 0;
|
2014-01-28 05:04:17 +08:00
|
|
|
|
|
|
|
ch->tmu = tmu;
|
2014-01-28 22:52:46 +08:00
|
|
|
ch->index = index;
|
2014-01-28 05:04:17 +08:00
|
|
|
|
2014-01-28 22:52:46 +08:00
|
|
|
if (tmu->model == SH_TMU_SH3)
|
|
|
|
ch->base = tmu->mapbase + 4 + ch->index * 12;
|
|
|
|
else
|
|
|
|
ch->base = tmu->mapbase + 8 + ch->index * 12;
|
2014-01-28 05:04:17 +08:00
|
|
|
|
2014-05-16 20:44:23 +08:00
|
|
|
ch->irq = platform_get_irq(tmu->pdev, index);
|
2019-07-31 02:15:04 +08:00
|
|
|
if (ch->irq < 0)
|
2014-01-28 05:04:17 +08:00
|
|
|
return ch->irq;
|
|
|
|
|
|
|
|
ch->cs_enabled = false;
|
|
|
|
ch->enable_count = 0;
|
|
|
|
|
2014-02-17 23:04:16 +08:00
|
|
|
return sh_tmu_register(ch, dev_name(&tmu->pdev->dev),
|
2014-01-28 19:36:48 +08:00
|
|
|
clockevent, clocksource);
|
2014-01-28 05:04:17 +08:00
|
|
|
}
|
|
|
|
|
2014-01-28 19:36:48 +08:00
|
|
|
static int sh_tmu_map_memory(struct sh_tmu_device *tmu)
|
2009-05-01 14:51:00 +08:00
|
|
|
{
|
|
|
|
struct resource *res;
|
|
|
|
|
2014-01-28 05:04:17 +08:00
|
|
|
res = platform_get_resource(tmu->pdev, IORESOURCE_MEM, 0);
|
2009-05-01 14:51:00 +08:00
|
|
|
if (!res) {
|
2014-01-28 05:04:17 +08:00
|
|
|
dev_err(&tmu->pdev->dev, "failed to get I/O memory\n");
|
2014-01-28 19:36:48 +08:00
|
|
|
return -ENXIO;
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
|
|
|
|
2020-01-06 16:43:50 +08:00
|
|
|
tmu->mapbase = ioremap(res->start, resource_size(res));
|
2014-01-28 19:36:48 +08:00
|
|
|
if (tmu->mapbase == NULL)
|
|
|
|
return -ENXIO;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2014-01-28 05:04:17 +08:00
|
|
|
|
2014-04-11 22:23:40 +08:00
|
|
|
static int sh_tmu_parse_dt(struct sh_tmu_device *tmu)
|
|
|
|
{
|
|
|
|
struct device_node *np = tmu->pdev->dev.of_node;
|
|
|
|
|
|
|
|
tmu->model = SH_TMU;
|
|
|
|
tmu->num_channels = 3;
|
|
|
|
|
|
|
|
of_property_read_u32(np, "#renesas,channels", &tmu->num_channels);
|
|
|
|
|
|
|
|
if (tmu->num_channels != 2 && tmu->num_channels != 3) {
|
|
|
|
dev_err(&tmu->pdev->dev, "invalid number of channels %u\n",
|
|
|
|
tmu->num_channels);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-01-28 19:36:48 +08:00
|
|
|
static int sh_tmu_setup(struct sh_tmu_device *tmu, struct platform_device *pdev)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
tmu->pdev = pdev;
|
|
|
|
|
2014-02-17 23:49:05 +08:00
|
|
|
raw_spin_lock_init(&tmu->lock);
|
|
|
|
|
2014-04-11 22:23:40 +08:00
|
|
|
if (IS_ENABLED(CONFIG_OF) && pdev->dev.of_node) {
|
|
|
|
ret = sh_tmu_parse_dt(tmu);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
} else if (pdev->dev.platform_data) {
|
|
|
|
const struct platform_device_id *id = pdev->id_entry;
|
|
|
|
struct sh_timer_config *cfg = pdev->dev.platform_data;
|
|
|
|
|
|
|
|
tmu->model = id->driver_data;
|
|
|
|
tmu->num_channels = hweight8(cfg->channels_mask);
|
|
|
|
} else {
|
|
|
|
dev_err(&tmu->pdev->dev, "missing platform data\n");
|
|
|
|
return -ENXIO;
|
|
|
|
}
|
|
|
|
|
2014-01-28 19:36:48 +08:00
|
|
|
/* Get hold of clock. */
|
2014-01-28 22:52:46 +08:00
|
|
|
tmu->clk = clk_get(&tmu->pdev->dev, "fck");
|
2014-01-28 05:04:17 +08:00
|
|
|
if (IS_ERR(tmu->clk)) {
|
|
|
|
dev_err(&tmu->pdev->dev, "cannot get clock\n");
|
2014-01-28 19:36:48 +08:00
|
|
|
return PTR_ERR(tmu->clk);
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
2013-11-08 18:08:00 +08:00
|
|
|
|
2014-01-28 05:04:17 +08:00
|
|
|
ret = clk_prepare(tmu->clk);
|
2013-11-08 18:08:00 +08:00
|
|
|
if (ret < 0)
|
2014-01-28 19:36:48 +08:00
|
|
|
goto err_clk_put;
|
|
|
|
|
clocksource: sh_tmu: Compute rate before registration again
With the upcoming NTP correction related rate adjustments to be implemented
in the clockevents core, the latter needs to get informed about every rate
change of a clockevent device made after its registration.
Currently, sh_tmu violates this requirement in that it registers its
clockevent device with a dummy rate and sets its final rate through
clockevents_config() called from its ->set_state_oneshot() and
->set_state_periodic() functions respectively.
This patch moves the setting of the clockevent device's rate to its
registration.
Note that there has been some back and forth regarding this question with
respect to the clocksource also provided by this driver:
commit 66f49121ffa4 ("clocksource: sh_tmu: compute mult and shift before
registration")
moves the rate determination from the clocksource's ->enable() function to
before its registration. OTOH, the later
commit 0aeac458d9eb ("clocksource: sh_tmu: __clocksource_updatefreq_hz()
update")
basically reverts this, saying
"Without this patch the old code uses clocksource_register() together
with a hack that assumes a never changing clock rate."
However, I checked all current sh_tmu users in arch/sh as well as in
arch/arm/mach-shmobile carefully and right now, none of them changes any
rate in any clock tree relevant to sh_tmu after their respective
time_init(). Since all sh_tmu instances are created after time_init(), none
of them should ever observe any clock rate changes.
What's more, both, a clocksource as well as a clockevent device, can
immediately get selected for use at their registration and thus, enabled
at this point already. So it's probably safer to assume a "never changing
clock rate" here.
- Move the struct sh_tmu_channel's ->rate member to struct sh_tmu_device:
it's a property of the underlying clock which is in turn specific to
the sh_tmu_device.
- Determine the ->rate value in sh_tmu_setup() at device probing rather
than at first usage.
- Set the clockevent device's rate at its registration.
- Although not strictly necessary for the upcoming clockevent core changes,
set the clocksource's rate at its registration for consistency.
Signed-off-by: Nicolai Stange <nicstange@gmail.com>
Signed-off-by: John Stultz <john.stultz@linaro.org>
2017-02-07 05:12:00 +08:00
|
|
|
/* Determine clock rate. */
|
|
|
|
ret = clk_enable(tmu->clk);
|
|
|
|
if (ret < 0)
|
|
|
|
goto err_clk_unprepare;
|
|
|
|
|
|
|
|
tmu->rate = clk_get_rate(tmu->clk) / 4;
|
|
|
|
clk_disable(tmu->clk);
|
|
|
|
|
2014-01-28 19:36:48 +08:00
|
|
|
/* Map the memory resource. */
|
|
|
|
ret = sh_tmu_map_memory(tmu);
|
|
|
|
if (ret < 0) {
|
|
|
|
dev_err(&tmu->pdev->dev, "failed to remap I/O memory\n");
|
|
|
|
goto err_clk_unprepare;
|
|
|
|
}
|
2013-11-08 18:08:00 +08:00
|
|
|
|
2014-01-28 19:36:48 +08:00
|
|
|
/* Allocate and setup the channels. */
|
treewide: kzalloc() -> kcalloc()
The kzalloc() function has a 2-factor argument form, kcalloc(). This
patch replaces cases of:
kzalloc(a * b, gfp)
with:
kcalloc(a * b, gfp)
as well as handling cases of:
kzalloc(a * b * c, gfp)
with:
kzalloc(array3_size(a, b, c), gfp)
as it's slightly less ugly than:
kzalloc_array(array_size(a, b), c, gfp)
This does, however, attempt to ignore constant size factors like:
kzalloc(4 * 1024, gfp)
though any constants defined via macros get caught up in the conversion.
Any factors with a sizeof() of "unsigned char", "char", and "u8" were
dropped, since they're redundant.
The Coccinelle script used for this was:
// Fix redundant parens around sizeof().
@@
type TYPE;
expression THING, E;
@@
(
kzalloc(
- (sizeof(TYPE)) * E
+ sizeof(TYPE) * E
, ...)
|
kzalloc(
- (sizeof(THING)) * E
+ sizeof(THING) * E
, ...)
)
// Drop single-byte sizes and redundant parens.
@@
expression COUNT;
typedef u8;
typedef __u8;
@@
(
kzalloc(
- sizeof(u8) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(__u8) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(char) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(unsigned char) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(u8) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(__u8) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(char) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(unsigned char) * COUNT
+ COUNT
, ...)
)
// 2-factor product with sizeof(type/expression) and identifier or constant.
@@
type TYPE;
expression THING;
identifier COUNT_ID;
constant COUNT_CONST;
@@
(
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (COUNT_ID)
+ COUNT_ID, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * COUNT_ID
+ COUNT_ID, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (COUNT_CONST)
+ COUNT_CONST, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * COUNT_CONST
+ COUNT_CONST, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (COUNT_ID)
+ COUNT_ID, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * COUNT_ID
+ COUNT_ID, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (COUNT_CONST)
+ COUNT_CONST, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * COUNT_CONST
+ COUNT_CONST, sizeof(THING)
, ...)
)
// 2-factor product, only identifiers.
@@
identifier SIZE, COUNT;
@@
- kzalloc
+ kcalloc
(
- SIZE * COUNT
+ COUNT, SIZE
, ...)
// 3-factor product with 1 sizeof(type) or sizeof(expression), with
// redundant parens removed.
@@
expression THING;
identifier STRIDE, COUNT;
type TYPE;
@@
(
kzalloc(
- sizeof(TYPE) * (COUNT) * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * (COUNT) * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * COUNT * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * COUNT * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(THING) * (COUNT) * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * (COUNT) * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * COUNT * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * COUNT * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
)
// 3-factor product with 2 sizeof(variable), with redundant parens removed.
@@
expression THING1, THING2;
identifier COUNT;
type TYPE1, TYPE2;
@@
(
kzalloc(
- sizeof(TYPE1) * sizeof(TYPE2) * COUNT
+ array3_size(COUNT, sizeof(TYPE1), sizeof(TYPE2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(TYPE1), sizeof(TYPE2))
, ...)
|
kzalloc(
- sizeof(THING1) * sizeof(THING2) * COUNT
+ array3_size(COUNT, sizeof(THING1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(THING1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(THING1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * COUNT
+ array3_size(COUNT, sizeof(TYPE1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(TYPE1), sizeof(THING2))
, ...)
)
// 3-factor product, only identifiers, with redundant parens removed.
@@
identifier STRIDE, SIZE, COUNT;
@@
(
kzalloc(
- (COUNT) * STRIDE * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * (STRIDE) * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * STRIDE * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * (STRIDE) * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * (STRIDE) * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * STRIDE * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * (STRIDE) * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * STRIDE * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
)
// Any remaining multi-factor products, first at least 3-factor products,
// when they're not all constants...
@@
expression E1, E2, E3;
constant C1, C2, C3;
@@
(
kzalloc(C1 * C2 * C3, ...)
|
kzalloc(
- (E1) * E2 * E3
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- (E1) * (E2) * E3
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- (E1) * (E2) * (E3)
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- E1 * E2 * E3
+ array3_size(E1, E2, E3)
, ...)
)
// And then all remaining 2 factors products when they're not all constants,
// keeping sizeof() as the second factor argument.
@@
expression THING, E1, E2;
type TYPE;
constant C1, C2, C3;
@@
(
kzalloc(sizeof(THING) * C2, ...)
|
kzalloc(sizeof(TYPE) * C2, ...)
|
kzalloc(C1 * C2 * C3, ...)
|
kzalloc(C1 * C2, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (E2)
+ E2, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * E2
+ E2, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (E2)
+ E2, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * E2
+ E2, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- (E1) * E2
+ E1, E2
, ...)
|
- kzalloc
+ kcalloc
(
- (E1) * (E2)
+ E1, E2
, ...)
|
- kzalloc
+ kcalloc
(
- E1 * E2
+ E1, E2
, ...)
)
Signed-off-by: Kees Cook <keescook@chromium.org>
2018-06-13 05:03:40 +08:00
|
|
|
tmu->channels = kcalloc(tmu->num_channels, sizeof(*tmu->channels),
|
2014-01-28 19:36:48 +08:00
|
|
|
GFP_KERNEL);
|
2014-01-28 05:04:17 +08:00
|
|
|
if (tmu->channels == NULL) {
|
|
|
|
ret = -ENOMEM;
|
2014-01-28 19:36:48 +08:00
|
|
|
goto err_unmap;
|
2014-01-28 05:04:17 +08:00
|
|
|
}
|
|
|
|
|
2014-01-28 22:52:46 +08:00
|
|
|
/*
|
|
|
|
* Use the first channel as a clock event device and the second channel
|
|
|
|
* as a clock source.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < tmu->num_channels; ++i) {
|
|
|
|
ret = sh_tmu_channel_setup(&tmu->channels[i], i,
|
|
|
|
i == 0, i == 1, tmu);
|
2014-01-28 19:36:48 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto err_unmap;
|
|
|
|
}
|
2014-01-28 05:04:17 +08:00
|
|
|
|
2014-01-28 19:36:48 +08:00
|
|
|
platform_set_drvdata(pdev, tmu);
|
2013-11-08 18:07:59 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
2014-01-28 19:36:48 +08:00
|
|
|
err_unmap:
|
2014-01-28 05:04:17 +08:00
|
|
|
kfree(tmu->channels);
|
2014-01-28 22:52:46 +08:00
|
|
|
iounmap(tmu->mapbase);
|
2014-01-28 19:36:48 +08:00
|
|
|
err_clk_unprepare:
|
2014-01-28 05:04:17 +08:00
|
|
|
clk_unprepare(tmu->clk);
|
2014-01-28 19:36:48 +08:00
|
|
|
err_clk_put:
|
2014-01-28 05:04:17 +08:00
|
|
|
clk_put(tmu->clk);
|
2009-05-01 14:51:00 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-12-22 07:11:38 +08:00
|
|
|
static int sh_tmu_probe(struct platform_device *pdev)
|
2009-05-01 14:51:00 +08:00
|
|
|
{
|
2014-01-28 05:04:17 +08:00
|
|
|
struct sh_tmu_device *tmu = platform_get_drvdata(pdev);
|
2009-05-01 14:51:00 +08:00
|
|
|
int ret;
|
|
|
|
|
2019-10-03 17:29:13 +08:00
|
|
|
if (!is_sh_early_platform_device(pdev)) {
|
2012-08-06 07:48:17 +08:00
|
|
|
pm_runtime_set_active(&pdev->dev);
|
|
|
|
pm_runtime_enable(&pdev->dev);
|
2012-08-06 07:41:20 +08:00
|
|
|
}
|
2012-03-14 05:40:00 +08:00
|
|
|
|
2014-01-28 05:04:17 +08:00
|
|
|
if (tmu) {
|
2010-03-10 15:26:25 +08:00
|
|
|
dev_info(&pdev->dev, "kept as earlytimer\n");
|
2012-08-06 07:48:17 +08:00
|
|
|
goto out;
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
|
|
|
|
2014-01-28 05:04:17 +08:00
|
|
|
tmu = kzalloc(sizeof(*tmu), GFP_KERNEL);
|
2014-05-22 20:05:07 +08:00
|
|
|
if (tmu == NULL)
|
2009-05-01 14:51:00 +08:00
|
|
|
return -ENOMEM;
|
|
|
|
|
2014-01-28 05:04:17 +08:00
|
|
|
ret = sh_tmu_setup(tmu, pdev);
|
2009-05-01 14:51:00 +08:00
|
|
|
if (ret) {
|
2014-01-28 05:04:17 +08:00
|
|
|
kfree(tmu);
|
2012-08-06 07:48:17 +08:00
|
|
|
pm_runtime_idle(&pdev->dev);
|
|
|
|
return ret;
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
2019-10-03 17:29:12 +08:00
|
|
|
|
2019-10-03 17:29:13 +08:00
|
|
|
if (is_sh_early_platform_device(pdev))
|
2012-08-06 07:48:17 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
out:
|
2014-01-28 19:36:48 +08:00
|
|
|
if (tmu->has_clockevent || tmu->has_clocksource)
|
2012-08-06 07:48:17 +08:00
|
|
|
pm_runtime_irq_safe(&pdev->dev);
|
|
|
|
else
|
|
|
|
pm_runtime_idle(&pdev->dev);
|
|
|
|
|
|
|
|
return 0;
|
2009-05-01 14:51:00 +08:00
|
|
|
}
|
|
|
|
|
2012-12-22 07:11:38 +08:00
|
|
|
static int sh_tmu_remove(struct platform_device *pdev)
|
2009-05-01 14:51:00 +08:00
|
|
|
{
|
|
|
|
return -EBUSY; /* cannot unregister clockevent and clocksource */
|
|
|
|
}
|
|
|
|
|
2014-01-28 19:36:48 +08:00
|
|
|
static const struct platform_device_id sh_tmu_id_table[] = {
|
|
|
|
{ "sh-tmu", SH_TMU },
|
|
|
|
{ "sh-tmu-sh3", SH_TMU_SH3 },
|
|
|
|
{ }
|
|
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(platform, sh_tmu_id_table);
|
|
|
|
|
2014-04-11 22:23:40 +08:00
|
|
|
static const struct of_device_id sh_tmu_of_table[] __maybe_unused = {
|
|
|
|
{ .compatible = "renesas,tmu" },
|
|
|
|
{ }
|
|
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(of, sh_tmu_of_table);
|
|
|
|
|
2009-05-01 14:51:00 +08:00
|
|
|
static struct platform_driver sh_tmu_device_driver = {
|
|
|
|
.probe = sh_tmu_probe,
|
2012-12-22 07:11:38 +08:00
|
|
|
.remove = sh_tmu_remove,
|
2009-05-01 14:51:00 +08:00
|
|
|
.driver = {
|
|
|
|
.name = "sh_tmu",
|
2014-04-11 22:23:40 +08:00
|
|
|
.of_match_table = of_match_ptr(sh_tmu_of_table),
|
2014-01-28 19:36:48 +08:00
|
|
|
},
|
|
|
|
.id_table = sh_tmu_id_table,
|
2009-05-01 14:51:00 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
static int __init sh_tmu_init(void)
|
|
|
|
{
|
|
|
|
return platform_driver_register(&sh_tmu_device_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __exit sh_tmu_exit(void)
|
|
|
|
{
|
|
|
|
platform_driver_unregister(&sh_tmu_device_driver);
|
|
|
|
}
|
|
|
|
|
2019-10-03 17:29:12 +08:00
|
|
|
#ifdef CONFIG_SUPERH
|
2019-10-03 17:29:13 +08:00
|
|
|
sh_early_platform_init("earlytimer", &sh_tmu_device_driver);
|
2019-10-03 17:29:12 +08:00
|
|
|
#endif
|
|
|
|
|
2013-03-05 14:40:42 +08:00
|
|
|
subsys_initcall(sh_tmu_init);
|
2009-05-01 14:51:00 +08:00
|
|
|
module_exit(sh_tmu_exit);
|
|
|
|
|
|
|
|
MODULE_AUTHOR("Magnus Damm");
|
|
|
|
MODULE_DESCRIPTION("SuperH TMU Timer Driver");
|
|
|
|
MODULE_LICENSE("GPL v2");
|