2010-10-27 16:28:57 +00:00
|
|
|
/* MN10300 clockevents
|
|
|
|
*
|
|
|
|
* Copyright (C) 2010 Red Hat, Inc. All Rights Reserved.
|
|
|
|
* Written by Mark Salter (msalter@redhat.com)
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public Licence
|
|
|
|
* as published by the Free Software Foundation; either version
|
|
|
|
* 2 of the Licence, or (at your option) any later version.
|
|
|
|
*/
|
|
|
|
#include <linux/clockchips.h>
|
|
|
|
#include <linux/interrupt.h>
|
|
|
|
#include <linux/percpu.h>
|
|
|
|
#include <linux/smp.h>
|
|
|
|
#include <asm/timex.h>
|
|
|
|
#include "internal.h"
|
|
|
|
|
|
|
|
#ifdef CONFIG_SMP
|
|
|
|
#if (CONFIG_NR_CPUS > 2) && !defined(CONFIG_GEENERIC_CLOCKEVENTS_BROADCAST)
|
|
|
|
#error "This doesn't scale well! Need per-core local timers."
|
|
|
|
#endif
|
|
|
|
#else /* CONFIG_SMP */
|
|
|
|
#define stop_jiffies_counter1()
|
|
|
|
#define reload_jiffies_counter1(x)
|
|
|
|
#define TMJC1IRQ TMJCIRQ
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
static int next_event(unsigned long delta,
|
|
|
|
struct clock_event_device *evt)
|
|
|
|
{
|
|
|
|
unsigned int cpu = smp_processor_id();
|
|
|
|
|
|
|
|
if (cpu == 0) {
|
|
|
|
stop_jiffies_counter();
|
|
|
|
reload_jiffies_counter(delta - 1);
|
|
|
|
} else {
|
|
|
|
stop_jiffies_counter1();
|
|
|
|
reload_jiffies_counter1(delta - 1);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void set_clock_mode(enum clock_event_mode mode,
|
|
|
|
struct clock_event_device *evt)
|
|
|
|
{
|
|
|
|
/* Nothing to do ... */
|
|
|
|
}
|
|
|
|
|
|
|
|
static DEFINE_PER_CPU(struct clock_event_device, mn10300_clockevent_device);
|
|
|
|
static DEFINE_PER_CPU(struct irqaction, timer_irq);
|
|
|
|
|
|
|
|
static irqreturn_t timer_interrupt(int irq, void *dev_id)
|
|
|
|
{
|
|
|
|
struct clock_event_device *cd;
|
|
|
|
unsigned int cpu = smp_processor_id();
|
|
|
|
|
|
|
|
if (cpu == 0)
|
|
|
|
stop_jiffies_counter();
|
|
|
|
else
|
|
|
|
stop_jiffies_counter1();
|
|
|
|
|
|
|
|
cd = &per_cpu(mn10300_clockevent_device, cpu);
|
|
|
|
cd->event_handler(cd);
|
|
|
|
|
|
|
|
return IRQ_HANDLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void event_handler(struct clock_event_device *dev)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
mn10300: move setup_jiffies_interrupt() to cevt-mn10300.c
Move the static inline function setup_jiffies_interrupt() from
<asm/timex.h> to arch/mn10300/kernel/cevt-mn10300.c, which is its only
callsite.
This allows to remove the inclusion of <asm/hardirq.h> and <linux/irq.h>
from <asm/timex.h> and <unit/timex.h>, fixing include hell like:
include/linux/jiffies.h:260:31: warning: "CLOCK_TICK_RATE" is not defined [-Wundef]
include/linux/jiffies.h:260:31: warning: "CLOCK_TICK_RATE" is not defined [-Wundef]
include/linux/jiffies.h:46:42: error: division by zero in #if
...
make[4]: *** [arch/mn10300/kernel/asm-offsets.s] Error 1
and (after a quick hack for the above by defining CLOCK_TICK_RATE in
<linux/jiffies.h>):
In file included from include/linux/notifier.h:15:0,
from include/linux/memory_hotplug.h:6,
from include/linux/mmzone.h:718,
from include/linux/gfp.h:4,
from include/linux/irq.h:20,
from arch/mn10300/unit-asb2303/include/unit/timex.h:15,
from arch/mn10300/include/asm/timex.h:15,
from include/linux/timex.h:174,
from include/linux/jiffies.h:8,
from include/linux/ktime.h:25,
from include/linux/timer.h:5,
from include/linux/workqueue.h:8,
include/linux/srcu.h:55:22: error: field 'work' has incomplete type
As a consequence, we do need a few more inclusions of <asm/irq.h>, namely
in arch/mn10300/unit-asb2303/smc91111.c and
arch/mn10300/unit-asb2305/unit-init.c.
Signed-off-by: Geert Uytterhoeven <geert@linux-m68k.org>
Cc: David Howells <dhowells@redhat.com>
Cc: Koichi Yasutake <yasutake.koichi@jp.panasonic.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2012-07-11 21:01:56 +00:00
|
|
|
static inline void setup_jiffies_interrupt(int irq,
|
|
|
|
struct irqaction *action)
|
|
|
|
{
|
|
|
|
u16 tmp;
|
|
|
|
setup_irq(irq, action);
|
|
|
|
set_intr_level(irq, NUM2GxICR_LEVEL(CONFIG_TIMER_IRQ_LEVEL));
|
|
|
|
GxICR(irq) |= GxICR_ENABLE | GxICR_DETECT | GxICR_REQUEST;
|
|
|
|
tmp = GxICR(irq);
|
|
|
|
}
|
|
|
|
|
2010-10-27 16:28:57 +00:00
|
|
|
int __init init_clockevents(void)
|
|
|
|
{
|
|
|
|
struct clock_event_device *cd;
|
|
|
|
struct irqaction *iact;
|
|
|
|
unsigned int cpu = smp_processor_id();
|
|
|
|
|
|
|
|
cd = &per_cpu(mn10300_clockevent_device, cpu);
|
|
|
|
|
|
|
|
if (cpu == 0) {
|
|
|
|
stop_jiffies_counter();
|
|
|
|
cd->irq = TMJCIRQ;
|
|
|
|
} else {
|
|
|
|
stop_jiffies_counter1();
|
|
|
|
cd->irq = TMJC1IRQ;
|
|
|
|
}
|
|
|
|
|
|
|
|
cd->name = "Timestamp";
|
|
|
|
cd->features = CLOCK_EVT_FEAT_ONESHOT;
|
|
|
|
|
2011-03-18 16:52:50 +00:00
|
|
|
/* Calculate shift/mult. We want to spawn at least 1 second */
|
|
|
|
clockevents_calc_mult_shift(cd, MN10300_JCCLK, 1);
|
2010-10-27 16:28:57 +00:00
|
|
|
|
2011-03-18 16:52:50 +00:00
|
|
|
/* Calculate the min / max delta */
|
2010-10-27 16:28:57 +00:00
|
|
|
cd->max_delta_ns = clockevent_delta2ns(TMJCBR_MAX, cd);
|
|
|
|
cd->min_delta_ns = clockevent_delta2ns(100, cd);
|
|
|
|
|
|
|
|
cd->rating = 200;
|
|
|
|
cd->cpumask = cpumask_of(smp_processor_id());
|
|
|
|
cd->set_mode = set_clock_mode;
|
|
|
|
cd->event_handler = event_handler;
|
|
|
|
cd->set_next_event = next_event;
|
|
|
|
|
|
|
|
iact = &per_cpu(timer_irq, cpu);
|
2014-03-04 20:31:51 +00:00
|
|
|
iact->flags = IRQF_SHARED | IRQF_TIMER;
|
2010-10-27 16:28:57 +00:00
|
|
|
iact->handler = timer_interrupt;
|
|
|
|
|
|
|
|
clockevents_register_device(cd);
|
|
|
|
|
|
|
|
#if defined(CONFIG_SMP) && !defined(CONFIG_GENERIC_CLOCKEVENTS_BROADCAST)
|
|
|
|
/* setup timer irq affinity so it only runs on this cpu */
|
|
|
|
{
|
2011-03-18 16:52:53 +00:00
|
|
|
struct irq_data *data;
|
|
|
|
data = irq_get_irq_data(cd->irq);
|
|
|
|
cpumask_copy(data->affinity, cpumask_of(cpu));
|
2010-10-27 16:28:57 +00:00
|
|
|
iact->flags |= IRQF_NOBALANCING;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (cpu == 0) {
|
|
|
|
reload_jiffies_counter(MN10300_JC_PER_HZ - 1);
|
|
|
|
iact->name = "CPU0 Timer";
|
|
|
|
} else {
|
|
|
|
reload_jiffies_counter1(MN10300_JC_PER_HZ - 1);
|
|
|
|
iact->name = "CPU1 Timer";
|
|
|
|
}
|
|
|
|
|
|
|
|
setup_jiffies_interrupt(cd->irq, iact);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|