Skip to content

Commit

Permalink
ARM: smp_twd: add runtime registration support
Browse files Browse the repository at this point in the history
Add support for the new registration interface to smp_twd.
Platforms can populate a struct twd_local_timer with MMIO
and IRQ resources, and then call twd_local_timer_register()
to have the timer registered with the core.

Signed-off-by: Marc Zyngier <[email protected]>
  • Loading branch information
Marc Zyngier committed Mar 13, 2012
1 parent 0ef330e commit 81e46f7
Show file tree
Hide file tree
Showing 2 changed files with 65 additions and 4 deletions.
18 changes: 17 additions & 1 deletion arch/arm/include/asm/smp_twd.h
Original file line number Diff line number Diff line change
Expand Up @@ -18,10 +18,26 @@
#define TWD_TIMER_CONTROL_PERIODIC (1 << 1)
#define TWD_TIMER_CONTROL_IT_ENABLE (1 << 2)

#include <linux/ioport.h>

struct clock_event_device;

extern void __iomem *twd_base;

void twd_timer_setup(struct clock_event_device *);
int twd_timer_setup(struct clock_event_device *);

struct twd_local_timer {
struct resource res[2];
};

#define DEFINE_TWD_LOCAL_TIMER(name,base,irq) \
struct twd_local_timer name __initdata = { \
.res = { \
DEFINE_RES_MEM(base, 0x10), \
DEFINE_RES_IRQ(irq), \
}, \
};

int twd_local_timer_register(struct twd_local_timer *);

#endif
51 changes: 48 additions & 3 deletions arch/arm/kernel/smp_twd.c
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@ static struct clk *twd_clk;
static unsigned long twd_timer_rate;

static struct clock_event_device __percpu **twd_evt;
static int twd_ppi;

static void twd_set_mode(enum clock_event_mode mode,
struct clock_event_device *clk)
Expand Down Expand Up @@ -227,7 +228,7 @@ static struct clk *twd_get_clock(void)
/*
* Setup the local clock events for a CPU.
*/
void __cpuinit twd_timer_setup(struct clock_event_device *clk)
int __cpuinit twd_timer_setup(struct clock_event_device *clk)
{
struct clock_event_device **this_cpu_clk;

Expand All @@ -237,15 +238,15 @@ void __cpuinit twd_timer_setup(struct clock_event_device *clk)
twd_evt = alloc_percpu(struct clock_event_device *);
if (!twd_evt) {
pr_err("twd: can't allocate memory\n");
return;
return -ENOMEM;
}

err = request_percpu_irq(clk->irq, twd_handler,
"twd", twd_evt);
if (err) {
pr_err("twd: can't register interrupt %d (%d)\n",
clk->irq, err);
return;
return err;
}
}

Expand All @@ -265,11 +266,55 @@ void __cpuinit twd_timer_setup(struct clock_event_device *clk)
clk->rating = 350;
clk->set_mode = twd_set_mode;
clk->set_next_event = twd_set_next_event;
if (!clk->irq)
clk->irq = twd_ppi;

this_cpu_clk = __this_cpu_ptr(twd_evt);
*this_cpu_clk = clk;

clockevents_config_and_register(clk, twd_timer_rate,
0xf, 0xffffffff);
enable_percpu_irq(clk->irq, 0);

return 0;
}

static struct local_timer_ops twd_lt_ops __cpuinitdata = {
.setup = twd_timer_setup,
.stop = twd_timer_stop,
};

int __init twd_local_timer_register(struct twd_local_timer *tlt)
{
int err;

if (twd_base || twd_evt)
return -EBUSY;

twd_ppi = tlt->res[1].start;

twd_evt = alloc_percpu(struct clock_event_device *);
twd_base = ioremap(tlt->res[0].start, resource_size(&tlt->res[0]));
if (!twd_base || !twd_evt) {
err = -ENOMEM;
goto out;
}

err = request_percpu_irq(twd_ppi, twd_handler, "twd", twd_evt);
if (err) {
pr_err("twd: can't register interrupt %d (%d)\n", twd_ppi, err);
goto out;
}

err = local_timer_register(&twd_lt_ops);
if (err)
goto out;

return 0;

out:
iounmap(twd_base);
free_percpu(twd_evt);
twd_base = twd_evt = NULL;
return err;
}

0 comments on commit 81e46f7

Please sign in to comment.