Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/arch/evbmips/ingenic add timecounter, timer interrupt an...
details: https://anonhg.NetBSD.org/src/rev/3a9bf6a236d3
branches: trunk
changeset: 334774:3a9bf6a236d3
user: macallan <macallan%NetBSD.org@localhost>
date: Sat Dec 06 14:24:58 2014 +0000
description:
add timecounter, timer interrupt and plenty of debugging goop
very much work in progress
diffstat:
sys/arch/evbmips/ingenic/clock.c | 133 +++++++++++++++++++++++++++++++++++++-
1 files changed, 129 insertions(+), 4 deletions(-)
diffs (164 lines):
diff -r 938aa5495ea5 -r 3a9bf6a236d3 sys/arch/evbmips/ingenic/clock.c
--- a/sys/arch/evbmips/ingenic/clock.c Sat Dec 06 12:24:22 2014 +0000
+++ b/sys/arch/evbmips/ingenic/clock.c Sat Dec 06 14:24:58 2014 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: clock.c,v 1.1 2014/11/22 15:17:02 macallan Exp $ */
+/* $NetBSD: clock.c,v 1.2 2014/12/06 14:24:58 macallan Exp $ */
/*-
* Copyright (c) 2014 Michael Lorenz
@@ -27,20 +27,104 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: clock.c,v 1.1 2014/11/22 15:17:02 macallan Exp $");
+__KERNEL_RCSID(0, "$NetBSD: clock.c,v 1.2 2014/12/06 14:24:58 macallan Exp $");
#include <sys/param.h>
#include <sys/cpu.h>
#include <sys/device.h>
#include <sys/kernel.h>
#include <sys/systm.h>
+#include <sys/timetc.h>
+
#include <mips/ingenic/ingenic_regs.h>
+extern void ingenic_puts(const char *);
+
+void ingenic_clockintr(uint32_t);
+
+struct clockframe cf;
+
+static u_int
+ingenic_count_read(struct timecounter *tc)
+{
+ return readreg(JZ_OST_CNT_LO);
+}
+
void
cpu_initclocks(void)
{
+ struct cpu_info * const ci = curcpu();
+ uint32_t cnt;
- /* timecounter setup and such */
+ static struct timecounter tc = {
+ ingenic_count_read, /* get_timecount */
+ 0, /* no poll_pps */
+ ~0u, /* counter_mask */
+ 12000000, /* frequency */
+ "Ingenic OS timer", /* name */
+ 100, /* quality */
+ };
+
+ curcpu()->ci_cctr_freq = tc.tc_frequency;
+
+ tc_init(&tc);
+
+ printf("starting timer interrupt...\n");
+ /* start the timer interrupt */
+ cnt = readreg(JZ_OST_CNT_LO);
+ ci->ci_next_cp0_clk_intr = cnt + ci->ci_cycles_per_hz;
+ writereg(JZ_OST_DATA, ci->ci_next_cp0_clk_intr);
+ writereg(JZ_TC_TFCR, TFR_OSTFLAG);
+ /*
+ * XXX
+ * We can use OST or one of the regular timers to generate the 100hz
+ * interrupt. OST interrupts need to be rescheduled every time and by
+ * only one core, the regular timer can be programmed to fire every
+ * 10ms without rescheduling and we'd still use the OST as time base.
+ * OST is supposed to fire on INT2 although I haven't been able to get
+ * that to work yet ( all I get is INT0 which is for hardware interrupts
+ * in general )
+ * So if we can get OST to fire on INT2 we can just block INT0 on core1
+ * and have a timer interrupt on both cores, if not the regular timer
+ * would be more convenient but we'd have to shoot an IPI to core1 on
+ * every tick.
+ * For now, use OST and hope we'll figure out how to make it fire on
+ * INT2.
+ */
+#if 1
+ writereg(JZ_TC_TMCR, TFR_OSTFLAG);
+#else
+ writereg(JZ_TC_TECR, TESR_TCST5); /* disable timer 5 */
+ writereg(JZ_TC_TCNT(5), 0);
+ writereg(JZ_TC_TDFR(5), 30000); /* 10ms at 48MHz / 16 */
+ writereg(JZ_TC_TDHR(5), 60000); /* not reached */
+ writereg(JZ_TC_TCSR(5), TCSR_EXT_EN| TCSR_DIV_16);
+ writereg(JZ_TC_TMCR, TFR_FFLAG5);
+ writereg(JZ_TC_TFCR, TFR_FFLAG5);
+ writereg(JZ_TC_TESR, TESR_TCST5); /* enable timer 5 */
+#endif
+ printf("INTC %08x %08x\n", readreg(JZ_ICSR0), readreg(JZ_ICSR1));
+ writereg(JZ_ICMCR0, 0x0c000000); /* TCU2, OST */
+ spl0();
+ printf("TFR: %08x\n", readreg(JZ_TC_TFR));
+ printf("TMR: %08x\n", readreg(JZ_TC_TMR));
+ printf("cnt5: %08x\n", readreg(JZ_TC_TCNT(5)));
+ printf("CR: %08x\n", MFC0(MIPS_COP_0_CAUSE, 0));
+ printf("SR: %08x\n", MFC0(MIPS_COP_0_STATUS, 0));
+ delay(100000);
+ printf("TFR: %08x\n", readreg(JZ_TC_TFR));
+ printf("TMR: %08x\n", readreg(JZ_TC_TMR));
+ printf("cnt5: %08x\n", readreg(JZ_TC_TCNT(5)));
+ printf("CR: %08x\n", MFC0(MIPS_COP_0_CAUSE, 0));
+ printf("SR: %08x\n", MFC0(MIPS_COP_0_STATUS, 0));
+ printf("TFR: %08x\n", readreg(JZ_TC_TFR));
+ printf("TMR: %08x\n", readreg(JZ_TC_TMR));
+ printf("cnt5: %08x\n", readreg(JZ_TC_TCNT(5)));
+ printf("CR: %08x\n", MFC0(MIPS_COP_0_CAUSE, 0));
+ printf("SR: %08x\n", MFC0(MIPS_COP_0_STATUS, 0));
+
+ printf("INTC %08x %08x\n", readreg(JZ_ICSR0), readreg(JZ_ICSR1));
+ delay(3000000);
}
/* shamelessly stolen from mips3_clock.c */
@@ -89,5 +173,46 @@
void
setstatclockrate(int r)
{
- /* nothing to see here */
+ /* we could just use another timer channel here */
}
+
+int cnt = 99;
+
+
+void
+ingenic_clockintr(uint32_t id)
+{
+ struct cpu_info * const ci = curcpu();
+ uint32_t new_cnt;
+
+ ci->ci_ev_count_compare.ev_count++;
+
+ /* clear flags */
+ writereg(JZ_TC_TFCR, TFR_OSTFLAG);
+
+ KASSERT((ci->ci_cycles_per_hz & ~(0xffffffff)) == 0);
+ ci->ci_next_cp0_clk_intr += (uint32_t)(ci->ci_cycles_per_hz & 0xffffffff);
+ writereg(JZ_OST_DATA, ci->ci_next_cp0_clk_intr);
+
+ /* Check for lost clock interrupts */
+ new_cnt = readreg(JZ_OST_CNT_LO);
+
+ /*
+ * Missed one or more clock interrupts, so let's start
+ * counting again from the current value.
+ */
+ if ((ci->ci_next_cp0_clk_intr - new_cnt) & 0x80000000) {
+
+ ci->ci_next_cp0_clk_intr = new_cnt + curcpu()->ci_cycles_per_hz;
+ writereg(JZ_OST_DATA, ci->ci_next_cp0_clk_intr);
+ curcpu()->ci_ev_count_compare_missed.ev_count++;
+ }
+
+ cnt++;
+ if (cnt == 100) {
+ cnt = 0;
+ ingenic_puts("+");
+ }
+
+ hardclock(&cf);
+}
Home |
Main Index |
Thread Index |
Old Index