Subject: next68k softintr(9) patch
To: None <port-next68k@NetBSD.org>
From: Izumi Tsutsui <tsutsui@ceres.dti.ne.jp>
List: port-m68k
Date: 03/06/2007 00:56:02
Could anyone try or review attached patch?
There are several possible issues though:
- According to next68k/include/cpu.h, TIMER is at ipl 6
but current IPL_CLOCK is (PSL_S|PSL_IPL3).
It was chaned after DMA fixes were integrated, but
is it really required?
- NEXT_I_SOFTINT[01] are defined in cpu.h and it seems
that they can generate real hardware interrupts,
but current siron() doesn't touch any hardware and
it uses hp300 derived simulated sir to handle softintrs.
Are these bits are just taken from NEXTSTEP but unused?
- next68k seems to have its own interrupt control register,
so maybe it should have own spl control routines like x86
rather than traditional m68k splN() based implementation.
---
Izumi Tsutsui
Index: conf/files.next68k
===================================================================
RCS file: /cvsroot/src/sys/arch/next68k/conf/files.next68k,v
retrieving revision 1.36
diff -u -r1.36 files.next68k
--- conf/files.next68k 11 Dec 2005 12:18:25 -0000 1.36
+++ conf/files.next68k 5 Mar 2007 15:18:03 -0000
@@ -37,6 +37,7 @@
file arch/m68k/m68k/kgdb_machdep.c kgdb
file arch/m68k/m68k/pmap_motorola.c
file arch/m68k/m68k/procfs_machdep.c procfs
+file arch/m68k/m68k/softintr.c
file arch/m68k/m68k/sys_machdep.c
file arch/m68k/m68k/vm_machdep.c
Index: dev/zs.c
===================================================================
RCS file: /cvsroot/src/sys/arch/next68k/dev/zs.c,v
retrieving revision 1.27
diff -u -r1.27 zs.c
--- dev/zs.c 11 Dec 2005 12:18:25 -0000 1.27
+++ dev/zs.c 5 Mar 2007 15:18:03 -0000
@@ -176,7 +176,6 @@
/* Interrupt handlers. */
static int zshard(void *);
-static void zssoft(void *);
static int zs_get_speed(struct zs_chanstate *);
@@ -209,7 +208,7 @@
struct zsc_attach_args zsc_args;
volatile struct zschan *zc;
struct zs_chanstate *cs;
- int s, channel, sir;
+ int s, channel;
zs_attached = 1;
@@ -275,13 +274,10 @@
}
isrlink_autovec(zshard, NULL, NEXT_I_IPL(NEXT_I_SCC), 0, NULL);
+ zsc->zsc_softintr_cookie = softintr_establish(IPL_SOFTSERIAL,
+ (void (*)(void *))zsc_intr_soft, zsc);
INTR_ENABLE(NEXT_I_SCC);
- sir = allocate_sir(zssoft, zsc);
- if (sir != SIR_SERIAL) {
- panic("Unexpected zssoft sir");
- }
-
/*
* Set the master interrupt enable and interrupt vector.
* (common to both channels, do it on A)
@@ -319,12 +315,12 @@
zshard(void *arg)
{
struct zsc_softc *zsc;
- int unit, rr3, rval, softreq;
+ int unit, rr3, rval;
if (!INTR_OCCURRED(NEXT_I_SCC))
return 0;
- rval = softreq = 0;
+ rval = 0;
for (unit = 0; unit < zsc_cd.cd_ndevs; unit++) {
zsc = zsc_cd.cd_devs[unit];
if (zsc == NULL)
@@ -335,53 +331,15 @@
rval |= rr3;
zsc->zsc_intrcnt.ev_count++;
}
- softreq |= zsc->zsc_cs[0]->cs_softreq;
- softreq |= zsc->zsc_cs[1]->cs_softreq;
+ /* We are at splzs here, so no need to lock. */
+ if (zsc->zsc_cs[0]->cs_softreq || zsc->zsc_cs[1]->cs_softreq)
+ softintr_schedule(zsc->zsc_softintr_cookie);
}
- /* We are at splzs here, so no need to lock. */
- if (softreq && (zssoftpending == 0)) {
- zssoftpending = 1;
- setsoftserial();
- }
return(1);
}
/*
- * Similar scheme as for zshard (look at all of them)
- */
-static void
-zssoft(void *arg)
-{
- struct zsc_softc *zsc;
- int s, unit;
-
- /* This is not the only ISR on this IPL. */
- if (zssoftpending == 0)
- panic("zssoft not pending");
-
- /*
- * The soft intr. bit will be set by zshard only if
- * the variable zssoftpending is zero. The order of
- * these next two statements prevents our clearing
- * the soft intr bit just after zshard has set it.
- */
- /* ienab_bic(IE_ZSSOFT); */
- zssoftpending = 0;
-
- /* Make sure we call the tty layer at spltty. */
- s = spltty();
- for (unit = 0; unit < zsc_cd.cd_ndevs; unit++) {
- zsc = zsc_cd.cd_devs[unit];
- if (zsc == NULL)
- continue;
- (void)zsc_intr_soft(zsc);
- }
- splx(s);
-}
-
-
-/*
* Compute the current baud rate given a ZS channel.
*/
static int
Index: include/intr.h
===================================================================
RCS file: /cvsroot/src/sys/arch/next68k/include/intr.h,v
retrieving revision 1.17
diff -u -r1.17 intr.h
--- include/intr.h 16 Feb 2007 02:53:49 -0000 1.17
+++ include/intr.h 5 Mar 2007 15:18:03 -0000
@@ -42,72 +42,60 @@
/* watch out for side effects */
#define splx(s) ((s) & PSL_IPL ? _spl(s) : spl0())
+#define splsoft() splraise1()
+#define splsoftnet() splsoft()
+#define splsoftclock() splsoft()
+#define splsoftserial() splsoft()
+#define splbio() splraise3()
+#define splnet() splraise3()
+#define spltty() splraise3()
+#define splserial() splraise5()
+#define splvm() splraise6()
+#define splclock() splraise3() /* ??? */
+#define splstatclock() splclock()
+#define splhigh() spl7()
+#define splsched() spl7()
+#define spllock() spl7()
+
+#define spldma() splraise6()
+
/****************************************************************/
-#define IPL_HIGH (PSL_S|PSL_IPL7)
-#define IPL_SERIAL (PSL_S|PSL_IPL5)
-#define IPL_SCHED (PSL_S|PSL_IPL7)
-#define IPL_LOCK (PSL_S|PSL_IPL7)
-#define IPL_CLOCK (PSL_S|PSL_IPL3)
-#define IPL_STATCLOCK IPL_CLOCK
-#define IPL_VM (PSL_S|PSL_IPL6)
-#define IPL_TTY (PSL_S|PSL_IPL3)
-#define IPL_BIO (PSL_S|PSL_IPL3)
-#define IPL_NET (PSL_S|PSL_IPL3)
-#define IPL_SOFTNET (PSL_S|PSL_IPL2)
-#define IPL_SOFTCLOCK (PSL_S|PSL_IPL1)
#define IPL_NONE 0
+#define IPL_SOFTCLOCK 1
+#define IPL_SOFTNET 2
+#define IPL_SOFTSERIAL 3
+#define IPL_SOFT 4
+#define IPL_BIO 5
+#define IPL_NET 6
+#define IPL_TTY 7
+#define IPL_SERIAL 8
+#define IPL_VM 9
+#define IPL_CLOCK 10
+#define IPL_STATCLOCK IPL_CLOCK
+#define IPL_HIGH 11
+#define IPL_LOCK IPL_HIGH
+#define IPL_SCHED IPL_HIGH
+#define NIPL 12
typedef int ipl_t;
typedef struct {
- ipl_t _ipl;
+ ipl_t _psl;
} ipl_cookie_t;
-static inline ipl_cookie_t
-makeiplcookie(ipl_t ipl)
-{
+extern const int ipl2psl_table[NIPL];
- return (ipl_cookie_t){._ipl = ipl};
-}
+ipl_cookie_t makeiplcookie(ipl_t);
static inline int
splraiseipl(ipl_cookie_t icookie)
{
- return _splraise(icookie._ipl);
+ return _splraise(icookie._psl);
}
-#include <sys/spl.h>
-
-#define spldma() _splraise(PSL_S|PSL_IPL6)
-
/****************************************************************/
-/*
- * simulated software interrupt register
- */
-extern volatile u_int8_t ssir;
-
-#define SIR_NET 0x01
-#define SIR_CLOCK 0x02
-#define SIR_SERIAL 0x04
-#define SIR_DTMGR 0x08
-#define SIR_ADB 0x10
-
-#define siron(mask) \
- __asm volatile ( "orb %1,%0" : "=m" (ssir) : "i" (mask))
-#define siroff(mask) \
- __asm volatile ( "andb %1,%0" : "=m" (ssir) : "ir" (~(mask)));
-
-#define setsoftnet() siron(SIR_NET)
-#define setsoftclock() siron(SIR_CLOCK)
-#define setsoftserial() siron(SIR_SERIAL)
-#define setsoftdtmgr() siron(SIR_DTMGR)
-#define setsoftadb() siron(SIR_ADB)
-
-extern u_long allocate_sir(void (*)(void *),void *);
-extern void init_sir(void);
-
/* locore.s */
int spl0(void);
@@ -118,6 +106,8 @@
#define INTR_DISABLE(x) (*intrmask &= (~NEXT_I_BIT(x)))
#define INTR_OCCURRED(x) (*intrstat & NEXT_I_BIT(x))
+#include <m68k/softintr.h>
+
#endif /* _KERNEL */
#endif /* _NEXT68K_INTR_H_ */
Index: include/types.h
===================================================================
RCS file: /cvsroot/src/sys/arch/next68k/include/types.h,v
retrieving revision 1.2
diff -u -r1.2 types.h
--- include/types.h 11 Sep 2006 15:07:50 -0000 1.2
+++ include/types.h 5 Mar 2007 15:18:03 -0000
@@ -3,4 +3,5 @@
#include <m68k/types.h>
#define __HAVE_TIMECOUNTER
+#define __HAVE_GENERIC_SOFT_INTERRUPTS
#define __HAVE_GENERIC_TODR
Index: include/z8530var.h
===================================================================
RCS file: /cvsroot/src/sys/arch/next68k/include/z8530var.h,v
retrieving revision 1.5
diff -u -r1.5 z8530var.h
--- include/z8530var.h 11 Dec 2005 12:18:25 -0000 1.5
+++ include/z8530var.h 5 Mar 2007 15:18:03 -0000
@@ -50,6 +50,7 @@
/* Machine-dependent part follows... */
struct evcnt zsc_intrcnt; /* count interrupts */
struct zs_chanstate zsc_cs_store[2];
+ void *zsc_softintr_cookie;
};
/*
Index: next68k/autoconf.c
===================================================================
RCS file: /cvsroot/src/sys/arch/next68k/next68k/autoconf.c,v
retrieving revision 1.18
diff -u -r1.18 autoconf.c
--- next68k/autoconf.c 11 Dec 2005 12:18:29 -0000 1.18
+++ next68k/autoconf.c 5 Mar 2007 15:18:03 -0000
@@ -159,7 +159,7 @@
INTR_SETMASK(0);
- init_sir();
+ softintr_init();
if (config_rootfound("mainbus", NULL) == NULL)
panic("autoconfig failed, no root");
Index: next68k/isr.c
===================================================================
RCS file: /cvsroot/src/sys/arch/next68k/next68k/isr.c,v
retrieving revision 1.22
diff -u -r1.22 isr.c
--- next68k/isr.c 11 Dec 2005 12:18:29 -0000 1.22
+++ next68k/isr.c 5 Mar 2007 15:18:03 -0000
@@ -386,36 +386,6 @@
isr->isr_evcnt->ev_count++;
}
-/*
- * netisr junk...
- * should use an array of chars instead of
- * a bitmask to avoid atomicity locking issues.
- */
-
-void
-netintr(void)
-{
- int n, s;
-
- s = splhigh();
- n = netisr;
- netisr = 0;
- splx(s);
-
-#define DONETISR(bit, fn) do { \
- if (n & (1 << bit)) \
- fn(); \
- } while (0)
-
- s = splsoftnet();
-
-#include <net/netisr_dispatch.h>
-
-#undef DONETISR
-
- splx(s);
-}
-
#if 0
/* ARGSUSED */
static int
@@ -425,3 +395,25 @@
return (1);
}
#endif
+
+const int ipl2psl_table[NIPL] = {
+ [IPL_NONE] = PSL_IPL0,
+ [IPL_SOFTCLOCK] = PSL_IPL1,
+ [IPL_SOFTNET] = PSL_IPL1,
+ [IPL_SOFTSERIAL] = PSL_IPL1,
+ [IPL_SOFT] = PSL_IPL1,
+ [IPL_BIO] = PSL_IPL3,
+ [IPL_NET] = PSL_IPL3,
+ [IPL_TTY] = PSL_IPL3,
+ [IPL_SERIAL] = PSL_IPL5,
+ [IPL_VM] = PSL_IPL6,
+ [IPL_CLOCK] = PSL_IPL3, /* ??? */
+ [IPL_HIGH] = PSL_IPL7,
+};
+
+ipl_cookie_t
+makeiplcookie(ipl_t ipl)
+{
+
+ return (ipl_cookie_t){._psl = ipl2psl_table[ipl] | PSL_S};
+}
Index: next68k/locore.s
===================================================================
RCS file: /cvsroot/src/sys/arch/next68k/next68k/locore.s,v
retrieving revision 1.45
diff -u -r1.45 locore.s
--- next68k/locore.s 11 Dec 2005 12:18:29 -0000 1.45
+++ next68k/locore.s 5 Mar 2007 15:18:03 -0000
@@ -1018,8 +1018,6 @@
* necessitating a stack cleanup.
*/
-BSS(ssir,1)
-
ASENTRY_NOPROFILE(rei)
tstl _C_LABEL(astpending) | AST pending?
jeq Lchksir | no, go check for SIR
Index: next68k/trap.c
===================================================================
RCS file: /cvsroot/src/sys/arch/next68k/next68k/trap.c,v
retrieving revision 1.64
diff -u -r1.64 trap.c
--- next68k/trap.c 4 Mar 2007 06:00:27 -0000 1.64
+++ next68k/trap.c 5 Mar 2007 15:18:04 -0000
@@ -220,12 +220,6 @@
#define MDB_ISPID(p) ((p) == mmupid)
#endif
-
-#define NSIR 32
-void (*sir_routines[NSIR])(void *);
-void *sir_args[NSIR];
-int next_sir;
-
/*
* trap and syscall both need the following work done before returning
* to user mode.
@@ -315,7 +309,6 @@
ksiginfo_t ksi;
int s;
u_quad_t sticks = 0 /* XXX initialiser works around compiler bug */;
- int bit;
static int panicking = 0;
uvmexp.traps++;
@@ -585,13 +578,9 @@
case T_SSIR: /* software interrupt */
case T_SSIR|T_USER:
- while ((bit = ffs(ssir))) {
- --bit;
- ssir &= ~(1 << bit);
- uvmexp.softs++;
- if (sir_routines[bit])
- sir_routines[bit](sir_args[bit]);
- }
+
+ softintr_dispatch();
+
/*
* If this was not an AST trap, we are all done.
*/
@@ -1039,29 +1028,3 @@
}
#endif
#endif
-
-/*
- * Allocation routines for software interrupts.
- */
-u_long
-allocate_sir(void (*proc)(void *), void *arg)
-{
- int bit;
-
- if( next_sir >= NSIR )
- panic("allocate_sir: none left");
- bit = next_sir++;
- sir_routines[bit] = proc;
- sir_args[bit] = arg;
- return (1 << bit);
-}
-
-void
-init_sir(void)
-{
- extern void netintr(void);
-
- sir_routines[0] = (void (*)(void *))netintr;
- sir_routines[1] = softclock;
- next_sir = 2;
-}