diff options
author | Theo de Raadt <deraadt@cvs.openbsd.org> | 1995-11-13 03:53:53 +0000 |
---|---|---|
committer | Theo de Raadt <deraadt@cvs.openbsd.org> | 1995-11-13 03:53:53 +0000 |
commit | be6610d2f34f2369c6132c9aea7abe125de4448c (patch) | |
tree | fd76f3aea962540b09d8c099b326b80018e89940 /sys/arch/amiga | |
parent | 5c7a8b804cd7c4371a60d2319a767fbb86d4d6b5 (diff) |
update to netbsd
Diffstat (limited to 'sys/arch/amiga')
-rw-r--r-- | sys/arch/amiga/amiga/amiga_init.c | 11 | ||||
-rw-r--r-- | sys/arch/amiga/amiga/pmap.c | 17 | ||||
-rw-r--r-- | sys/arch/amiga/amiga/sys_machdep.c | 6 | ||||
-rw-r--r-- | sys/arch/amiga/amiga/trap.c | 6 | ||||
-rw-r--r-- | sys/arch/amiga/conf/Makefile.amiga | 15 | ||||
-rw-r--r-- | sys/arch/amiga/conf/files.amiga | 5 | ||||
-rw-r--r-- | sys/arch/amiga/dev/grf_cv.c | 144 | ||||
-rw-r--r-- | sys/arch/amiga/dev/grf_cvreg.h | 39 | ||||
-rw-r--r-- | sys/arch/amiga/dev/grf_ul.c | 31 | ||||
-rw-r--r-- | sys/arch/amiga/dev/grf_ultms.g (renamed from sys/arch/amiga/dev/grf_ul.gsp) | 0 | ||||
-rw-r--r-- | sys/arch/amiga/dev/if_qn.c | 481 | ||||
-rw-r--r-- | sys/arch/amiga/dev/if_qnreg.h | 8 |
12 files changed, 397 insertions, 366 deletions
diff --git a/sys/arch/amiga/amiga/amiga_init.c b/sys/arch/amiga/amiga/amiga_init.c index 2d99fa05107..336da5a5b72 100644 --- a/sys/arch/amiga/amiga/amiga_init.c +++ b/sys/arch/amiga/amiga/amiga_init.c @@ -1,4 +1,4 @@ -/* $NetBSD: amiga_init.c,v 1.33 1995/10/05 12:40:48 chopps Exp $ */ +/* $NetBSD: amiga_init.c,v 1.33.2.1 1995/11/10 16:09:54 chopps Exp $ */ /* * Copyright (c) 1994 Michael L. Hitch @@ -63,6 +63,9 @@ extern u_int lowram; extern u_int Sysptmap, Sysptsize, Sysseg, Umap, proc0paddr; extern u_int Sysseg_pa; extern u_int virtual_avail; +#ifdef M68040 +extern int protostfree; +#endif extern char *esym; @@ -346,7 +349,7 @@ start_c(id, fphystart, fphysize, cphysize, esym_addr, flags) * Initialize level 2 descriptors (which immediately * follow the level 1 table). We need: * NPTEPG / SG4_LEV3SIZE - * level 2 descriptors to map eachof the nptpages + 1 + * level 2 descriptors to map each of the nptpages + 1 * pages of PTEs. Note that we set the "used" bit * now to save the HW the expense of doing it. */ @@ -363,8 +366,10 @@ start_c(id, fphystart, fphysize, cphysize, esym_addr, flags) * roundup(num, SG4_LEV2SIZE) / SG4_LEVEL2SIZE * level 1 descriptors to map the 'num' level 2's. */ + i = roundup(i, SG4_LEV2SIZE) / SG4_LEV2SIZE; + protostfree = (-1 << (i + 1)) /* & ~(-1 << MAXKL2SIZE) */; sg = (u_int *) Sysseg_pa; - esg = &sg[roundup(i, SG4_LEV2SIZE) / SG4_LEV2SIZE]; + esg = &sg[i]; sg_proto = (u_int)&sg[SG4_LEV1SIZE] | SG_U | SG_RW |SG_V; while (sg < esg) { *sg++ = sg_proto; diff --git a/sys/arch/amiga/amiga/pmap.c b/sys/arch/amiga/amiga/pmap.c index 860f320f30f..60697284291 100644 --- a/sys/arch/amiga/amiga/pmap.c +++ b/sys/arch/amiga/amiga/pmap.c @@ -1,4 +1,4 @@ -/* $NetBSD: pmap.c,v 1.27 1995/10/09 04:34:02 chopps Exp $ */ +/* $NetBSD: pmap.c,v 1.27.2.1 1995/11/10 16:10:00 chopps Exp $ */ /* * Copyright (c) 1991 Regents of the University of California. @@ -237,6 +237,7 @@ boolean_t pmap_initialized = FALSE; /* Has pmap_init completed? */ char *pmap_attributes; /* reference and modify bits */ #ifdef M68040 static int pmap_ishift; /* segment table index shift */ +int protostfree; /* prototype (default) free ST map */ #endif #ifdef MACHINE_NONCONTIG @@ -363,7 +364,7 @@ pmap_bootstrap(firstaddr, loadaddr) #ifdef M68040 if (mmutype == MMU_68040) { pmap_ishift = SG4_SHIFT1; - pmap_kernel()->pm_stfree = 0xfffffff8; /* XXXX */ + pmap_kernel()->pm_stfree = protostfree; } else pmap_ishift = SG_ISHIFT; #endif @@ -574,6 +575,12 @@ pmap_init(phys_start, phys_end) printf("pmap_init: pt_map [%x - %x)\n", addr, addr2); #endif +#ifdef M68040 + if (mmutype == MMU_68040) { + protostfree = ~1 & ~(-1 << MAXUL2SIZE); + } +#endif + /* * Now it is safe to enable pv_table recording. */ @@ -734,7 +741,7 @@ pmap_pinit(pmap) pmap->pm_stpa = Segtabzeropa; #ifdef M68040 if (mmutype == MMU_68040) - pmap->pm_stfree = 0x0000fffe; /* XXXX */ + pmap->pm_stfree = protostfree; #endif pmap->pm_stchanged = TRUE; pmap->pm_count = 1; @@ -1011,7 +1018,7 @@ printf ("pmap_remove: PA %08x index %d\n", pa, pa_index(pa)); ptpmap->pm_stpa = Segtabzeropa; #ifdef M68040 if (mmutype == MMU_68040) - ptpmap->pm_stfree = 0x0000fffe; /* XXXX */ + ptpmap->pm_stfree = protostfree; #endif ptpmap->pm_stchanged = TRUE; /* @@ -2046,7 +2053,7 @@ pmap_enter_ptpage(pmap, va) pmap_kernel(), (vm_offset_t)pmap->pm_stab); #ifdef M68040 if (mmutype == MMU_68040) { - pmap->pm_stfree = 0x0000fffe; /* XXXX */ + pmap->pm_stfree = protostfree; } #endif pmap->pm_stchanged = TRUE; diff --git a/sys/arch/amiga/amiga/sys_machdep.c b/sys/arch/amiga/amiga/sys_machdep.c index a7f098b472e..bf4f9b66e4c 100644 --- a/sys/arch/amiga/amiga/sys_machdep.c +++ b/sys/arch/amiga/amiga/sys_machdep.c @@ -1,4 +1,4 @@ -/* $NetBSD: sys_machdep.c,v 1.12 1995/10/09 04:34:05 chopps Exp $ */ +/* $NetBSD: sys_machdep.c,v 1.12.2.1 1995/11/10 16:13:41 chopps Exp $ */ /* * Copyright (c) 1982, 1986 Regents of the University of California. @@ -60,7 +60,7 @@ sys_vtrace(p, v, retval) void *v; register_t *retval; { - register struct vtrace_args /* { + register struct sys_vtrace_args /* { syscallarg(int) request; syscallarg(int) value; } */ *uap = v; @@ -276,7 +276,7 @@ sys_sysarch(p, v, retval) void *v; register_t *retval; { - struct sysarch_args /* { + struct sys_sysarch_args /* { syscallarg(int) op; syscallarg(char *) parms; } */ *uap = v; diff --git a/sys/arch/amiga/amiga/trap.c b/sys/arch/amiga/amiga/trap.c index 90fd14b22a0..ee104684fde 100644 --- a/sys/arch/amiga/amiga/trap.c +++ b/sys/arch/amiga/amiga/trap.c @@ -1,4 +1,4 @@ -/* $NetBSD: trap.c,v 1.42 1995/10/09 04:34:08 chopps Exp $ */ +/* $NetBSD: trap.c,v 1.42.2.1 1995/11/10 19:32:59 chopps Exp $ */ /* * Copyright (c) 1988 University of Utah. @@ -280,8 +280,10 @@ trapmmufault(type, code, v, fp, p, sticks) /* * Print out some data about the fault */ +#ifdef DEBUG_PAGE0 if (v < NBPG) /* XXX PAGE0 */ mmudebug |= 0x100; /* XXX PAGE0 */ +#endif if (mmudebug && mmutype == MMU_68040) { printf ("68040 access error: pc %x, code %x," " ea %x, fa %x\n", fp->f_pc, code, fp->f_fmt7.f_ea, v); @@ -294,8 +296,10 @@ trapmmufault(type, code, v, fp, p, sticks) Debugger(); /* XXX PAGE0 */ #endif /* XXX PAGE0 */ } +#ifdef DEBUG_PAGE0 mmudebug &= ~0x100; /* XXX PAGE0 */ #endif +#endif if (p) vm = p->p_vmspace; diff --git a/sys/arch/amiga/conf/Makefile.amiga b/sys/arch/amiga/conf/Makefile.amiga index f3820465088..b00139375aa 100644 --- a/sys/arch/amiga/conf/Makefile.amiga +++ b/sys/arch/amiga/conf/Makefile.amiga @@ -1,4 +1,4 @@ -# $NetBSD: Makefile.amiga,v 1.28 1995/10/09 15:23:25 chopps Exp $ +# $NetBSD: Makefile.amiga,v 1.28.2.1 1995/11/10 19:29:46 chopps Exp $ # @(#)Makefile.hp300 7.10 (Berkeley) 6/27/91 # @@ -75,6 +75,12 @@ PROFILE_C= ${CC} -S -c ${COPTS} $<; \ NORMAL_S= ${CPP} ${COPTS} $< | ${AS} -o $@ NORMAL_S_C= ${CPP} ${COPTS} ${PARAM} $< | ${AS} -o $@ +.SUFFIXES: .o .g + +.g.o: ; echo This is a dummy rule and should not be executed. + +NORMAL_G= gspa < $< | gspahextoc > $*.c; ${CC} -c ${CFLAGS} ${PROF} $*.c + %OBJS %CFILES @@ -142,7 +148,7 @@ genassym: ${CC} -static ${INCLUDES} ${IDENT} ${PARAM} -Dmc68020 -Damiga \ -o genassym ${AMIGA}/amiga/genassym.c -depend: assym.s param.c vnode_if.h grf_ultmscode.h +depend: assym.s param.c vnode_if.h mkdep ${COPTS} ${CFILES} ioconf.c param.c mkdep -a -p ${INCLUDES} ${IDENT} ${PARAM} ${AMIGA}/amiga/genassym.c @@ -169,9 +175,4 @@ param.o: param.c Makefile vnode_if.c vnode_if.h: $S/kern/vnode_if.sh $S/kern/vnode_if.src AWK="${AWK}" sh $S/kern/vnode_if.sh $S/kern/vnode_if.src -grf_ultmscode.h: ${AMIGA}/dev/grf_ul.gsp - gspa < ${AMIGA}/dev/grf_ul.gsp | gspahextoc > grf_ultmscode.h - -grf_ul.o: grf_ultmscode.h - %RULES diff --git a/sys/arch/amiga/conf/files.amiga b/sys/arch/amiga/conf/files.amiga index 35bc2520975..29471264bd3 100644 --- a/sys/arch/amiga/conf/files.amiga +++ b/sys/arch/amiga/conf/files.amiga @@ -1,4 +1,4 @@ -# $NetBSD: files.amiga,v 1.31.2.1 1995/10/20 11:01:01 chopps Exp $ +# $NetBSD: files.amiga,v 1.31.2.2 1995/11/10 19:29:49 chopps Exp $ # maxpartitions must be first item in files.${ARCH}.newconf maxpartitions 16 # NOTE THAT AMIGA IS SPECIAL! @@ -82,7 +82,8 @@ file arch/amiga/dev/grf_cv.c grfcv needs-flag # A2410 grf device grful at zbus: grfbus file arch/amiga/dev/grf_ul.c grful needs-flag -file arch/amiga/dev/ite_ul.c grful ite +file arch/amiga/dev/ite_ul.c grful ite +file arch/amiga/dev/grf_ultms.g grful # retina ZIII grf device grfrh at zbus: grfbus diff --git a/sys/arch/amiga/dev/grf_cv.c b/sys/arch/amiga/dev/grf_cv.c index 83480b8cb04..ec5a54aca7b 100644 --- a/sys/arch/amiga/dev/grf_cv.c +++ b/sys/arch/amiga/dev/grf_cv.c @@ -1,4 +1,4 @@ -/* $NetBSD: grf_cv.c,v 1.3.2.2 1995/10/21 15:15:41 chopps Exp $ */ +/* $NetBSD: grf_cv.c,v 1.3.2.3 1995/11/10 16:27:10 chopps Exp $ */ /* * Copyright (c) 1995 Michael Teske @@ -74,24 +74,26 @@ #include <amiga/dev/grf_cvreg.h> #include <amiga/dev/zbusvar.h> -int cv_mondefok __P((struct grfvideo_mode *)); -void cv_boardinit(); +int grfcvmatch __P((struct device *, struct cfdata *, void *)); +void grfcvattach __P((struct device *, struct device *, void *)); +int grfcvprint __P((void *, char *)); + +static int cv_has_4mb __P((volatile char *)); static unsigned short compute_clock __P((unsigned long)); -int cv_getvmode __P((struct grf_softc *, struct grfvideo_mode *)); -int cv_setvmode __P((struct grf_softc *, unsigned int)); -int cv_setmonitor __P((struct grf_softc *, struct grfvideo_mode *)); -int cv_toggle __P((struct grf_softc *,unsigned short)); -int cv_getcmap __P((struct grf_softc *, struct grf_colormap *)); -int cv_putcmap __P((struct grf_softc *, struct grf_colormap *)); -void cv_off __P((struct grf_softc *)); -int cv_blank __P((struct grf_softc *, int *)); -void cv_inittextmode __P((struct grf_softc *)); - -int cv_ioctl __P((register struct grf_softc *gp, int cmd, void *data)); -void grfcvattach __P((struct device *, struct device *, void *)); -int grfcvprint __P((void *, char *)); -int grfcvmatch __P((struct device *, struct cfdata *, void *)); -void cv_memset __P((unsigned char *, unsigned char, int)); +void cv_boardinit __P((struct grf_softc *)); +int cv_getvmode __P((struct grf_softc *, struct grfvideo_mode *)); +int cv_setvmode __P((struct grf_softc *, unsigned int)); +int cv_blank __P((struct grf_softc *, int *)); +int cv_mode __P((register struct grf_softc *, int, void *, int, int)); +int cv_ioctl __P((register struct grf_softc *gp, int cmd, void *data)); +int cv_setmonitor __P((struct grf_softc *, struct grfvideo_mode *)); +int cv_getcmap __P((struct grf_softc *, struct grf_colormap *)); +int cv_putcmap __P((struct grf_softc *, struct grf_colormap *)); +int cv_toggle __P((struct grf_softc *)); +int cv_mondefok __P((struct grfvideo_mode *)); +int cv_load_mon __P((struct grf_softc *, struct grfcvtext_mode *)); +void cv_inittextmode __P((struct grf_softc *)); +void cv_memset __P((unsigned char *, unsigned char, int)); #ifdef CV64CONSOLE extern void grfcv_iteinit __P((struct grf_softc *)); @@ -126,18 +128,15 @@ struct grfcvtext_mode cvconsole_mode = { S3FONTX, S3FONTY, 80, 506/S3FONTY, S3FONT, 32, 255 }; - /* Console colors */ unsigned char cvconscolors[3][3] = { /* background, foreground, hilite */ {0,0x40,0x50}, {152,152,152}, {255,255,255} }; -unsigned char pass_toggle; /* passthru status tracker */ - /* Board Address of CV64 */ -static void *cv_boardaddr; +static volatile caddr_t cv_boardaddr; static int cv_fbsize; int @@ -227,59 +226,36 @@ grfcvattach(pdp, dp, auxp) struct device *pdp, *dp; void *auxp; { - static struct grf_softc congrf; struct zbus_args *zap; struct grf_softc *gp; - static char attachflag = 0; zap = auxp; printf("\n"); - /* make sure id's have matched */ - if (!cv_boardaddr) - return; + gp = (struct grf_softc *)dp; - /* do all that messy console/grf stuff */ - if (dp == NULL) - gp = &congrf; - else - gp = (struct grf_softc *)dp; - - if (dp != NULL && congrf.g_regkva != 0) { - /* - * inited earlier, just copy (not device struct) - */ - bcopy(&congrf.g_display, &gp->g_display, - (char *)&gp[1] - (char *)&gp->g_display); - } else { - gp->g_regkva = (volatile caddr_t)cv_boardaddr + READ_OFFSET; - gp->g_fbkva = (volatile caddr_t)cv_boardaddr + 0x01400000; + gp->g_regkva = (volatile caddr_t)cv_boardaddr + READ_OFFSET; + gp->g_fbkva = (volatile caddr_t)cv_boardaddr + 0x01400000; - gp->g_unit = GRF_CV64_UNIT; - gp->g_mode = cv_mode; - gp->g_conpri = grfcv_cnprobe(); - gp->g_flags = GF_ALIVE; + gp->g_unit = GRF_CV64_UNIT; + gp->g_mode = cv_mode; + gp->g_conpri = grfcv_cnprobe(); + gp->g_flags = GF_ALIVE; - /* wakeup the board */ - cv_boardinit(gp); + /* wakeup the board */ + cv_boardinit(gp); #ifdef CV64CONSOLE - grfcv_iteinit(gp); - (void)cv_load_mon(gp, &cvconsole_mode); + grfcv_iteinit(gp); + (void)cv_load_mon(gp, &cvconsole_mode); #endif - } /* - * attach grf (once) + * attach grf */ - if (amiga_config_found(cfdata, &gp->g_device, gp, grfcvprint)) { - attachflag = 1; + if (amiga_config_found(cfdata, &gp->g_device, gp, grfcvprint)) printf("grfcv: CyberVision64 with %dMB being used\n", cv_fbsize/0x100000); - } else { - if (!attachflag) - printf("grfcv unattached!!\n"); - } } int @@ -353,7 +329,7 @@ void cv_boardinit(gp) struct grf_softc *gp; { - unsigned char *ba = gp->g_regkva; + volatile caddr_t ba = gp->g_regkva; unsigned char test; unsigned int clockpar; int i; @@ -606,7 +582,7 @@ cv_setvmode(gp, mode) unsigned mode; { if (!mode || (mode > monitor_def_max) || - monitor_def[mode-1].mode_num == 0) + monitor_def[mode - 1].mode_num == 0) return (EINVAL); monitor_current = monitor_def + (mode - 1); @@ -614,25 +590,13 @@ cv_setvmode(gp, mode) return (0); } -void -cv_off(gp) - struct grf_softc *gp; -{ - char *ba = gp->g_regkva - READ_OFFSET; - - /* we'll put the pass-through on for cc ite and set Full Bandwidth - * bit on just in case it didn't work...but then it doesn't matter - * does it? =) - */ - cvscreen(1, ba); -} int cv_blank(gp, on) struct grf_softc *gp; int *on; { - char *ba = gp->g_regkva; + volatile caddr_t ba = gp->g_regkva; gfx_on_off(*on ? 1 : 0, ba); return (0); @@ -660,7 +624,7 @@ cv_mode(gp, cmd, arg, a2, a3) case GM_GRFOFF: #ifndef CV64CONSOLE - cv_off(gp); + (void)cv_toggle(gp); #else cv_load_mon(gp, &cvconsole_mode); #endif @@ -717,7 +681,7 @@ cv_ioctl (gp, cmd, data) break; case GRFTOGGLE: - return (cv_toggle (gp, 0)); + return (cv_toggle (gp)); case GRFIOCSETMON: return (cv_setmonitor (gp, (struct grfvideo_mode *)data)); @@ -775,7 +739,7 @@ cv_getcmap(gfp, cmap) struct grf_softc *gfp; struct grf_colormap *cmap; { - volatile unsigned char *ba; + volatile caddr_t ba; u_char red[256], green[256], blue[256], *rp, *gp, *bp; short x; int error; @@ -814,7 +778,7 @@ cv_putcmap(gfp, cmap) struct grf_softc *gfp; struct grf_colormap *cmap; { - volatile unsigned char *ba; + volatile caddr_t ba; u_char red[256], green[256], blue[256], *rp, *gp, *bp; short x; int error; @@ -849,19 +813,14 @@ cv_putcmap(gfp, cmap) int -cv_toggle(gp,wopp) +cv_toggle(gp) struct grf_softc *gp; - unsigned short wopp; /* don't need that one yet, ill */ { - volatile unsigned char *ba; + volatile caddr_t ba; - ba = gp->g_regkva - READ_OFFSET; + ba = gp->g_regkva; + cvscreen(1, ba - READ_OFFSET); - if (pass_toggle) { - cvscreen(0, ba); - } else { - cvscreen(1, ba); - } return (0); } @@ -908,8 +867,7 @@ cv_load_mon(gp, md) { struct grfvideo_mode *gv; struct grfinfo *gi; - volatile unsigned char *ba; - volatile unsigned char *fb; + volatile caddr_t ba, fb; unsigned short mnr; unsigned short HT, HDE, HBS, HBE, HSS, HSE, VDE, VBS, VBE, VSS, VSE, VT; @@ -957,12 +915,12 @@ cv_load_mon(gp, md) HBE = gv->hblank_stop; HSS = gv->hsync_start; HSE = gv->hsync_stop; - HT = gv->htotal; - VBS = gv->vblank_start; + HT = gv->htotal - 5; + VBS = gv->vblank_start - 1; VSS = gv->vsync_start; VSE = gv->vsync_stop; VBE = gv->vblank_stop; - VT = gv->vtotal; + VT = gv->vtotal - 2; if (TEXT) HDE = ((gv->disp_width + md->fx - 1) / md->fx) - 1; @@ -1198,8 +1156,8 @@ cv_inittextmode(gp) struct grf_softc *gp; { struct grfcvtext_mode *tm = (struct grfcvtext_mode *)gp->g_data; - volatile unsigned char *ba = gp->g_regkva; - unsigned char *fb = gp->g_fbkva; + volatile caddr_t ba = gp->g_regkva; + volatile caddr_t fb = gp->g_fbkva; unsigned char *c, *f, y; unsigned short z; diff --git a/sys/arch/amiga/dev/grf_cvreg.h b/sys/arch/amiga/dev/grf_cvreg.h index dbdd657a62d..a38dcbea6b9 100644 --- a/sys/arch/amiga/dev/grf_cvreg.h +++ b/sys/arch/amiga/dev/grf_cvreg.h @@ -1,4 +1,4 @@ -/* $NetBSD: grf_cvreg.h,v 1.1.2.1 1995/10/21 15:15:44 chopps Exp $ */ +/* $NetBSD: grf_cvreg.h,v 1.1.2.2 1995/11/10 16:27:13 chopps Exp $ */ /* * Copyright (c) 1995 Michael Teske @@ -57,11 +57,11 @@ struct grfcvtext_mode { /* read VGA register */ -#define vgar(ba, reg) (*(((volatile unsigned char *)ba)+reg)) +#define vgar(ba, reg) (*(((volatile caddr_t)ba)+reg)) /* write VGA register */ #define vgaw(ba, reg, val) \ - *(((volatile unsigned char *)ba)+reg) = ((val) & 0xff) + *(((volatile caddr_t)ba)+reg) = ((val) & 0xff) /* * defines for the used register addresses (mw) @@ -339,7 +339,7 @@ struct grfcvtext_mode { static inline unsigned char RAttr(ba, idx) - volatile void *ba; + volatile caddr_t ba; short idx; { @@ -350,7 +350,7 @@ RAttr(ba, idx) static inline unsigned char RSeq(ba, idx) - volatile void *ba; + volatile caddr_t ba; short idx; { vgaw(ba, SEQ_ADDRESS, idx); @@ -359,7 +359,7 @@ RSeq(ba, idx) static inline unsigned char RCrt(ba, idx) - volatile void *ba; + volatile caddr_t ba; short idx; { vgaw(ba, CRT_ADDRESS, idx); @@ -368,7 +368,7 @@ RCrt(ba, idx) static inline unsigned char RGfx(ba, idx) - volatile void *ba; + volatile caddr_t ba; short idx; { vgaw(ba, GCT_ADDRESS, idx); @@ -379,10 +379,10 @@ RGfx(ba, idx) static inline void cv_write_port(bits, BoardAddr) unsigned short bits; - volatile unsigned char *BoardAddr; + volatile caddr_t BoardAddr; { volatile char *addr; - static unsigned char CVPortBits=0; /* mirror port bits here */ + static unsigned char CVPortBits = 0; /* mirror port bits here */ addr = BoardAddr + 0x40001; if (bits & 0x8000) @@ -409,8 +409,8 @@ cv_write_port(bits, BoardAddr) static inline void cvscreen(toggle, ba) - char toggle; - volatile unsigned char *ba; + char *toggle; + volatile caddr_t ba; { if (toggle) @@ -423,23 +423,22 @@ cvscreen(toggle, ba) /* ba= registerbase */ static inline void gfx_on_off(toggle, ba) - char toggle; - volatile unsigned char *ba; + int toggle; + volatile caddr_t ba; { - unsigned char r; + int r; toggle &= 0x1; toggle = toggle << 5; - r = RSeq(ba, 0x1); - r &= 0xDF; /* Bit 5 auf 0!!! */ - WSeq(ba, 1, r | toggle); + r = RSeq(ba, SEQ_ID_CLOCKING_MODE); + r &= 0xdf; /* set Bit 5 to 0 */ + + WSeq(ba, SEQ_ID_CLOCKING_MODE, r | toggle); } -int cv_mode __P((register struct grf_softc *gp, int cmd, void *arg, int a2, int a3)); -int cv_load_mon __P((struct grf_softc *gp, struct grfcvtext_mode *gv)); -int grfcv_cnprobe __P((void)); #if 0 +int grfcv_cnprobe __P((void)); void grfcv_iteinit __P((struct grf_softc *gp)); #endif diff --git a/sys/arch/amiga/dev/grf_ul.c b/sys/arch/amiga/dev/grf_ul.c index 1ca64fdf12a..6e6c36401d7 100644 --- a/sys/arch/amiga/dev/grf_ul.c +++ b/sys/arch/amiga/dev/grf_ul.c @@ -1,4 +1,4 @@ -/* $NetBSD: grf_ul.c,v 1.7 1995/10/09 02:14:46 chopps Exp $ */ +/* $NetBSD: grf_ul.c,v 1.7.2.1 1995/11/10 19:29:54 chopps Exp $ */ #define UL_DEBUG /* @@ -53,7 +53,7 @@ #include <amiga/dev/grfvar.h> #include <amiga/dev/grf_ulreg.h> -#include <grf_ultmscode.h> +extern u_int16_t tmscode[]; int ul_ioctl __P((struct grf_softc *, u_long, void *, dev_t)); int ul_getcmap __P((struct grf_softc *, struct grf_colormap *, dev_t)); @@ -81,12 +81,11 @@ struct grfvideo_mode ul_monitor_defs[] = { /* * Horizontal values are given in TMS units, that is, for the * A2410 board, units of 16 pixels. The ioctl multiplies (when - * exporting) or divides (when importing) them by 2 to conform to - * the other grfs. + * exporting) or divides (when importing) them by 16 to conform to. * - * XXX This should have been in units of pixels, IMHO. If you change - * this, you must also change amiga/stand/grfconfig/grfconfig.c, - * grf_{rt,rh,cl}.c and egsgrfconfig (the latter to generate the + * XXX This used to be in units of 8 pixel times. We + * must also change amiga/stand/grfconfig/grfconfig.c, + * grf_{rt,rh,cl,cv}.c and egsgrfconfig (the latter to generate the * horizontal timings in units of pixels instead of 8 pixels. * You will also have to write warnings in BIG BOLD RED BLINKING * LETTERS all over the docs, and still people will fry their monitors. @@ -556,19 +555,11 @@ ul_getvmode (gp, vm) vm->disp_height = md->disp_height; vm->depth = md->depth; - /* XXX should use 16 instead of 2, but others use 1 instead of 8 */ - - /* - * XXX another idea: maybe we should transform the timings to - * have blank stop as point of reference (like other grfs) instead - * of sync start? - */ - - vm->hblank_start = (md->hblank_start - md->hblank_stop) * 2; - vm->hblank_stop = (md->htotal - 1) * 2; - vm->hsync_start = (md->hsync_start - md->hblank_stop) * 2; - vm->hsync_stop = (md->hsync_stop + md->htotal - md->hblank_stop) * 2; - vm->htotal = md->htotal * 2; + vm->hblank_start = (md->hblank_start - md->hblank_stop) * 16; + vm->hblank_stop = (md->htotal - 1) * 16; + vm->hsync_start = (md->hsync_start - md->hblank_stop) * 16; + vm->hsync_stop = (md->hsync_stop + md->htotal - md->hblank_stop) * 16; + vm->htotal = md->htotal * 16; vm->vblank_start = md->vblank_start - md->vblank_stop; vm->vblank_stop = md->vtotal - 1; diff --git a/sys/arch/amiga/dev/grf_ul.gsp b/sys/arch/amiga/dev/grf_ultms.g index 502af7b3323..502af7b3323 100644 --- a/sys/arch/amiga/dev/grf_ul.gsp +++ b/sys/arch/amiga/dev/grf_ultms.g diff --git a/sys/arch/amiga/dev/if_qn.c b/sys/arch/amiga/dev/if_qn.c index b4ad5e22efe..6f6de6be0fc 100644 --- a/sys/arch/amiga/dev/if_qn.c +++ b/sys/arch/amiga/dev/if_qn.c @@ -1,4 +1,4 @@ -/* $NetBSD: if_qn.c,v 1.1 1995/10/07 18:04:27 chopps Exp $ */ +/* $NetBSD: if_qn.c,v 1.1.2.1 1995/11/10 16:39:12 chopps Exp $ */ /* * Copyright (c) 1995 Mika Kortelainen @@ -35,7 +35,8 @@ * me with the necessary 'inside' information to write the driver. * * This is partly based on other code: - * - if_ed.c: basic function structure for Ethernet driver + * - if_ed.c: basic function structure for Ethernet driver and now also + * qn_put() is done similarly, i.e. no extra packet buffers. * * Device driver for National Semiconductor DS8390/WD83C690 based ethernet * adapters. @@ -54,22 +55,27 @@ * Copyright (c) 1995 Michael L. Hitch * All rights reserved. * + * - if_fe.c: some ideas for error handling for qn_rint() which might + * have fixed those random lock ups, too. + * + * All Rights Reserved, Copyright (C) Fujitsu Limited 1995 + * * * TODO: * - add multicast support - * - try to find out what is the reason for random lock-ups happening - * when (or after) getting data */ #include "qn.h" #if NQN > 0 #define QN_DEBUG +#define QN_DEBUG1_no /* hides some old tests */ #include "bpfilter.h" /* - * Fujitsu MB86950 Ethernet Controller (as used in QuickNet QN2000 Ethernet card) + * Fujitsu MB86950 Ethernet Controller (as used in the QuickNet QN2000 + * Ethernet card) */ #include <sys/param.h> @@ -108,10 +114,12 @@ #include <amiga/dev/if_qnreg.h> -#define ETHER_MIN_LEN 64 -#define ETHER_MAX_LEN 1518 -#define ETHER_ADDR_LEN 6 - +#define ETHER_MIN_LEN 60 +#define ETHER_MAX_LEN 1514 +#define ETHER_HDR_SIZE 14 +#define NIC_R_MASK (R_INT_PKT_RDY | R_INT_ALG_ERR |\ + R_INT_CRC_ERR | R_INT_OVR_FLO) +#define MAX_PACKETS 30 /* max number of packets read per interrupt */ /* * Ethernet software status per interface @@ -157,12 +165,17 @@ void qnreset __P((struct qn_softc *)); void qninit __P((struct qn_softc *)); void qnstop __P((struct qn_softc *)); static u_short qn_put __P((u_short volatile *, struct mbuf *)); +static void qn_rint __P((struct qn_softc *, u_short)); static void qn_flush __P((struct qn_softc *)); +#ifdef QN_DEBUG1 +static void qn_dump __P((struct qn_softc *)); +#endif struct cfdriver qncd = { NULL, "qn", qnmatch, qnattach, DV_IFNET, sizeof(struct qn_softc) }; + int qnmatch(parent, match, aux) struct device *parent; @@ -261,20 +274,10 @@ qninit(sc) { struct ifnet *ifp = &sc->sc_arpcom.ac_if; u_short i; - int s; - - s = splimp(); + static retry = 0; - /* Initialize NIC */ - *sc->nic_reset = DISABLE_DLC; - *sc->nic_t_status = CLEAR_T_ERR; - *sc->nic_t_mask = CLEAR_T_MASK; - *sc->nic_r_status = CLEAR_R_ERR; - *sc->nic_r_mask = R_INT_PKT_RDY; - *sc->nic_t_mode = NO_LOOPBACK; - - /* Turn DMA off */ - *((u_short volatile *)(sc->sc_nic_base + NIC_BMPR4)) = (u_short)0x0000; + *sc->nic_r_mask = NIC_R_MASK; + *sc->nic_t_mode = NO_LOOPBACK; if (sc->sc_arpcom.ac_if.if_flags & IFF_PROMISC) { *sc->nic_r_mode = PROMISCUOUS_MODE; @@ -284,27 +287,30 @@ qninit(sc) /* Set physical ethernet address. */ for (i = 0; i < ETHER_ADDR_LEN; i++) - *((u_short volatile *)(sc->sc_nic_base+QNET_HARDWARE_ADDRESS+2*i)) = + *((u_short volatile *)(sc->sc_nic_base+ + QNET_HARDWARE_ADDRESS+2*i)) = ((((u_short)sc->sc_arpcom.ac_enaddr[i]) << 8) | sc->sc_arpcom.ac_enaddr[i]); - sc->transmit_pending = 0; ifp->if_flags |= IFF_RUNNING; ifp->if_flags &= ~IFF_OACTIVE; + sc->transmit_pending = 0; qn_flush(sc); + /* QuickNet magic. Done ONLY once, otherwise a lockup occurs. */ + if (retry == 0) { + *((u_short volatile *)(sc->sc_nic_base + QNET_MAGIC)) = 0; + retry = 1; + } + /* Enable data link controller. */ - *((u_short volatile *)(sc->sc_nic_base + QNET_MAGIC)) = (u_short)0x0000; *sc->nic_reset = ENABLE_DLC; /* Attempt to start output, if any. */ qnstart(ifp); - - splx(s); } - /* * Device timeout/watchdog routine. Entered if the device neglects to * generate an interrupt after a transmit has been started on it. @@ -315,41 +321,35 @@ qnwatchdog(unit) { struct qn_softc *sc = qncd.cd_devs[unit]; - log(LOG_ERR, "%s: device timeout (watchdog)\n", sc->sc_dev.dv_xname); + log(LOG_INFO, "qn: device timeout (watchdog)\n"); ++sc->sc_arpcom.ac_if.if_oerrors; qnreset(sc); } - /* - * Flush card's buffer RAM + * Flush card's buffer RAM. */ static void qn_flush(sc) struct qn_softc *sc; { -#ifdef QN_DEBUG1 - int cnt = 0; -#endif +#if 1 /* Read data until bus read error (i.e. buffer empty). */ - while (!(*sc->nic_r_status & R_BUS_RD_ERR)) { + while (!(*sc->nic_r_status & R_BUS_RD_ERR)) (void)(*sc->nic_fifo); -#ifdef QN_DEBUG1 - cnt++; -#endif - } -#ifdef QN_DEBUG1 - log(LOG_INFO, "Flushed %d words\n", cnt); +#else + /* Read data twice to clear some internal pipelines. */ + (void)(*sc->nic_fifo); + (void)(*sc->nic_fifo); #endif /* Clear bus read error. */ *sc->nic_r_status = R_BUS_RD_ERR; } - /* - * Reset the interface... + * Reset the interface. * */ void @@ -364,9 +364,8 @@ qnreset(sc) splx(s); } - /* - * Take interface offline + * Take interface offline. */ void qnstop(sc) @@ -374,15 +373,23 @@ qnstop(sc) { /* Stop the interface. */ - *sc->nic_reset = DISABLE_DLC; + *sc->nic_reset = DISABLE_DLC; + delay(200); + *sc->nic_t_status = CLEAR_T_ERR; + *sc->nic_t_mask = CLEAR_T_MASK; + *sc->nic_r_status = CLEAR_R_ERR; + *sc->nic_r_mask = CLEAR_R_MASK; - *sc->nic_t_mask = CLEAR_T_MASK; - *sc->nic_r_mask = CLEAR_R_MASK; + /* Turn DMA off */ + *((u_short volatile *)(sc->sc_nic_base + NIC_BMPR4)) = 0; + + /* Accept no packets. */ + *sc->nic_r_mode = 0; + *sc->nic_t_mode = 0; qn_flush(sc); } - /* * Start output on interface. Get another datagram to send * off the interface queue, and copy it to the @@ -398,10 +405,10 @@ qnstart(ifp) struct qn_softc *sc = qncd.cd_devs[ifp->if_unit]; struct mbuf *m; u_short len; - int timout = 50000; + int timout = 60000; + - if ((sc->sc_arpcom.ac_if.if_flags & (IFF_RUNNING | IFF_OACTIVE)) != - IFF_RUNNING) + if ((ifp->if_flags & (IFF_RUNNING | IFF_OACTIVE)) != IFF_RUNNING) return; IF_DEQUEUE(&sc->sc_arpcom.ac_if.if_snd, m); @@ -422,22 +429,26 @@ qnstart(ifp) bpf_mtap(sc->sc_bpf, m); #endif len = qn_put(sc->nic_fifo, m); + m_freem(m); /* - * Really transmit the packet + * Really transmit the packet. */ - /* set packet length (byte-swapped) */ + /* Set packet length (byte-swapped). */ len = ((len >> 8) & 0x0007) | TRANSMIT_START | ((len & 0x00ff) << 8); *sc->nic_len = len; - /* Wait for the packet to really leave */ - while (!(*sc->nic_t_status & T_TMT_OK) && --timout) - ; + /* Wait for the packet to really leave. */ + while (!(*sc->nic_t_status & T_TMT_OK) && --timout) { + if ((timout % 10000) == 0) + log(LOG_INFO, "qn: timout...\n"); + } + if (timout == 0) /* Maybe we should try to recover from this one? */ - /* But for now on, let's just fall thru and hope the best... */ - log(LOG_INFO, "qn:timout\n"); + /* But now, let's just fall thru and hope the best... */ + log(LOG_INFO, "qn: transmit timout (fatal?)\n"); sc->transmit_pending = 1; *sc->nic_t_mask = INT_TMT_OK | INT_SIXTEEN_COL; @@ -446,7 +457,6 @@ qnstart(ifp) ifp->if_timer = 2; } - /* * Memory copy, copies word at a time */ @@ -472,7 +482,6 @@ word_copy_to_card(a, card, len) *card = *a++; } - /* * Copy packet from mbuf to the board memory * @@ -485,53 +494,59 @@ qn_put(addr, m) u_short volatile *addr; struct mbuf *m; { - struct mbuf *mp; - register u_short len, tlen; - static u_short packet_buf[1536/2]; - register u_char *p; + u_char *data, savebyte[2]; + int len, wantbyte; + u_short totlen; + + totlen = wantbyte = 0; + + for (; m != NULL; m = m->m_next) { + data = mtod(m, u_char *); + len = m->m_len; + totlen += len; + if (len > 0) { + /* Finish the last word. */ + if (wantbyte) { + savebyte[1] = *data; + *addr = *((u_short *)savebyte); + data++; + len--; + wantbyte = 0; + } + /* Output contiguous words. */ + if (len > 1) { + word_copy_to_card(data, addr, len); + data += len & ~1; + len &= 1; + } + /* Save last byte, if necessary. */ + if (len == 1) { + savebyte[0] = *data; + wantbyte = 1; + } + } + } - /* - * If buffer is bzeroed, we might copy max(60, tlen) bytes and get - * rid of extra zero fill-ins. - */ - bzero(packet_buf, 1536); + if (wantbyte) { + savebyte[1] = 0; + *addr = *((u_short *)savebyte); + } - /* - * The whole packet in one mbuf? - */ - if (m->m_next == NULL) { - tlen = m->m_len; - bcopy(mtod(m, u_char *), (u_char *)packet_buf, tlen); - /*word_copy_to_card(mtod(m, u_short *), addr, tlen + 1);*/ - word_copy_to_card(packet_buf, - addr, - max(tlen + 1, ETHER_MIN_LEN-4)); - } else { - /* No it wasn't, let's start copying */ - for (p = (u_char *)packet_buf, tlen = 0, mp = m; - mp; - mp = mp->m_next) { - if ((len = mp->m_len) == 0) - continue; - tlen += len; - bcopy(mtod(mp, u_char *), p, len); - p += len; - } - /* word_copy_to_card(packet_buf, addr, tlen + 1); */ - word_copy_to_card(packet_buf, - addr, - max(tlen + 1, ETHER_MIN_LEN-4)); + if(totlen < ETHER_MIN_LEN) { + /* + * Fill the rest of the packet with zeros. + * N.B.: This is required! Otherwise MB86950 fails. + */ + for(len = totlen + 1; len < ETHER_MIN_LEN; len += 2) + *addr = (u_short)0x0000; + totlen = ETHER_MIN_LEN; } - m_freem(m); - if (tlen < ETHER_MIN_LEN - 4) - /* We have copied ETHER_MIN_LEN-4 bytes. */ - tlen = ETHER_MIN_LEN - 4; - return (tlen); + return (totlen); } /* - * Copy packet from board RAM + * Copy packet from board RAM. * * Trailers not supported. * @@ -543,15 +558,21 @@ qn_get_packet(sc, len) { register u_short volatile *nic_fifo_ptr = sc->nic_fifo; struct ether_header *eh; - struct mbuf *m, *dst, *head = 0; + struct mbuf *m, *dst, *head = NULL; register u_short len1; u_short amount; /* Allocate header mbuf. */ MGETHDR(m, M_DONTWAIT, MT_DATA); - if (m == 0) + if (m == NULL) goto bad; + /* + * Round len to even value. + */ + if (len & 1) + len++; + m->m_pkthdr.rcvif = &sc->sc_arpcom.ac_if; m->m_pkthdr.len = len; m->m_len = 0; @@ -571,10 +592,10 @@ qn_get_packet(sc, len) amount = M_TRAILINGSPACE(m); if (amount == 0) { - /* allocate another mbuf */ + /* Allocate another mbuf. */ dst = m; MGET(m, M_DONTWAIT, MT_DATA); - if (m == 0) + if (m == NULL) goto bad; if (len1 >= MINCLSIZE) @@ -586,14 +607,12 @@ qn_get_packet(sc, len) amount = M_TRAILINGSPACE(m); } - if (amount < len1) { - if (amount & 1) - log(LOG_INFO, "Room for odd-length packet\n"); + if (amount < len1) len1 = amount; - } + word_copy_from_card(nic_fifo_ptr, (u_short *)(mtod(m, caddr_t) + m->m_len), - len1 + 1); + len1); m->m_len += len1; len -= len1; } @@ -624,94 +643,122 @@ qn_get_packet(sc, len) return; bad: - if (head) + if (head) { m_freem(head); + log(LOG_INFO, "qn_get_packet: mbuf alloc failed\n"); + } } - /* * Ethernet interface receiver interrupt. */ static void -qn_rint(sc) +qn_rint(sc, rstat) struct qn_softc *sc; + u_short rstat; { int i; u_short len, status; + /* Clear the status register. */ + *sc->nic_r_status = CLEAR_R_ERR; + /* - * Read at most 25 packets per interrupt + * Was there some error? + * Some of them are senseless because they are masked off. + * XXX */ - for (i = 0; i < 25; i++) { - if (*sc->nic_r_mode & RM_BUF_EMP) { - /* - * Buffer empty - */ - if ((status = *sc->nic_r_status)) { - /* It was some error: let's first clear - * the register. - */ - *sc->nic_r_status = CLEAR_R_ERR; - if (status & 0x0101) { + if (rstat & 0x0101) { #ifdef QN_DEBUG - log(LOG_INFO, "Overflow\n"); + log(LOG_INFO, "Overflow\n"); #endif - ++sc->sc_arpcom.ac_if.if_ierrors; - } else if (status & 0x0202) { + ++sc->sc_arpcom.ac_if.if_ierrors; + } + if (rstat & 0x0202) { #ifdef QN_DEBUG - log(LOG_INFO, "CRC Error\n"); + log(LOG_INFO, "CRC Error\n"); #endif - ++sc->sc_arpcom.ac_if.if_ierrors; - } else if (status & 0x0404) { + ++sc->sc_arpcom.ac_if.if_ierrors; + } + if (rstat & 0x0404) { #ifdef QN_DEBUG - log(LOG_INFO, "Alignment error\n"); + log(LOG_INFO, "Alignment error\n"); #endif - ++sc->sc_arpcom.ac_if.if_ierrors; - } else if (status & 0x0808) { - /* Short packet (these may occur and are - * no reason to worry about - or maybe - * they are?). - */ + ++sc->sc_arpcom.ac_if.if_ierrors; + } + if (rstat & 0x0808) { + /* Short packet (these may occur and are + * no reason to worry about - or maybe + * they are?). + */ #ifdef QN_DEBUG - log(LOG_INFO, "Short packet\n"); + log(LOG_INFO, "Short packet\n"); #endif - ++sc->sc_arpcom.ac_if.if_ierrors; - /* qnreset(sc); */ - } else if (status & 0x4040) { + ++sc->sc_arpcom.ac_if.if_ierrors; + } + if (rstat & 0x4040) { #ifdef QN_DEBUG - log(LOG_INFO, "Bus read error\n"); + log(LOG_INFO, "Bus read error\n"); #endif - ++sc->sc_arpcom.ac_if.if_ierrors; - } else { - /* There are some registers which are - * not meaningful when read, so don't - * care... - */ - } - } - return; - } else { - /* At least one valid packet. Clear the whole register. */ - *sc->nic_r_status = CLEAR_R_ERR; + ++sc->sc_arpcom.ac_if.if_ierrors; + qnreset(sc); + } - /* Read one word of garbage. */ - (void)(*sc->nic_fifo); + /* + * Read at most MAX_PACKETS packets per interrupt + */ + for (i = 0; i < MAX_PACKETS; i++) { + if (*sc->nic_r_mode & RM_BUF_EMP) + /* Buffer empty. */ + break; - /* Read packet length. */ - len = *sc->nic_fifo; - len = ((len << 8) & 0xff00) | ((len >> 8) & 0x00ff); + /* + * Read the first word: upper byte contains useful + * information. + */ + status = *sc->nic_fifo; + if ((status & 0x7000) != 0x2000) { + log(LOG_INFO, "qn: ERROR: status=%04x\n", status); + continue; + } + + /* + * Read packet length (byte-swapped). + * CRC is stripped off by the NIC. + */ + len = *sc->nic_fifo; + len = ((len << 8) & 0xff00) | ((len >> 8) & 0x00ff); - /* Read the packet. */ - qn_get_packet(sc, len); - ++sc->sc_arpcom.ac_if.if_ipackets; +#ifdef QN_DEBUG + if (len > ETHER_MAX_LEN || len < ETHER_HDR_SIZE) { + log(LOG_WARNING, + "%s: received a %s packet? (%u bytes)\n", + sc->sc_dev.dv_xname, + len < ETHER_HDR_SIZE ? "partial" : "big", len); + ++sc->sc_arpcom.ac_if.if_ierrors; + continue; } +#endif +#ifdef QN_DEBUG + if (len < ETHER_MIN_LEN) + log(LOG_WARNING, + "%s: received a short packet? (%u bytes)\n", + sc->sc_dev.dv_xname, len); +#endif + + /* Read the packet. */ + qn_get_packet(sc, len); + + ++sc->sc_arpcom.ac_if.if_ipackets; } - if (i == 25) - log(LOG_INFO, "used all the 25 loops\n"); +#ifdef QN_DEBUG + /* This print just to see whether MAX_PACKETS is large enough. */ + if (i == MAX_PACKETS) + log(LOG_INFO, "used all the %d loops\n", MAX_PACKETS); +#endif } - /* * Our interrupt routine */ @@ -719,34 +766,36 @@ int qnintr(sc) struct qn_softc *sc; { - u_short tint, rint, tintmask, rintmask; + u_short tint, rint, tintmask; + char return_tintmask = 0; /* * If the driver has not been initialized, just return immediately. * This also happens if there is no QuickNet board present. */ - if (sc->sc_base == NULL) return (0); /* Get interrupt statuses and masks. */ - rint = *sc->nic_r_status; - tint = *sc->nic_t_status; - rintmask = *sc->nic_r_mask /* 0x8f8f */; + rint = (*sc->nic_r_status) & NIC_R_MASK; tintmask = *sc->nic_t_mask; - if (!(tint&tintmask) && !(rint&rintmask)) + tint = (*sc->nic_t_status) & tintmask; + if (tint == 0 && rint == 0) return (0); - /* Disable receive interrupts so that we won't miss anything. */ + /* Disable interrupts so that we won't miss anything. */ *sc->nic_r_mask = CLEAR_R_MASK; + *sc->nic_t_mask = CLEAR_T_MASK; /* * Handle transmitter interrupts. Some of them are not asked for * but do happen, anyway. */ - if (tint) { - *sc->nic_t_mask = CLEAR_T_MASK; + + if (tint != 0) { + /* Clear transmit interrupt status. */ *sc->nic_t_status = CLEAR_T_ERR; + if (sc->transmit_pending && (tint & T_TMT_OK)) { sc->transmit_pending = 0; /* @@ -761,55 +810,41 @@ qnintr(sc) * 16 collision (i.e., packet lost). */ log(LOG_INFO, "qn: 16 collision - packet lost\n"); +#ifdef QN_DEBUG1 + qn_dump(sc); +#endif sc->sc_arpcom.ac_if.if_oerrors++; sc->sc_arpcom.ac_if.if_collisions += 16; sc->transmit_pending = 0; } - if (tint & T_COL && !(tint & T_TMT_OK)) { - /* - * Normal collision - */ - log(LOG_INFO, "qn:collision (shouldn't hurt)\n"); - sc->transmit_pending = 1; - sc->sc_arpcom.ac_if.if_oerrors++; - sc->sc_arpcom.ac_if.if_collisions++; - } - - if (tint & BUS_WRITE_ERROR) { - /* One bus write error occurs at start up, at least on my - * card. So I don't care about this one. - */ - sc->transmit_pending = 0; - } - - if (tint & T_UNDERFLOW) { - log(LOG_INFO, "qn:underflow\n"); - } - if (sc->transmit_pending) { log(LOG_INFO, "qn:still pending...\n"); - /* Return transmission interrupt mask. */ - *sc->nic_t_mask = tintmask; + + /* Must return transmission interrupt mask. */ + return_tintmask = 1; } else { sc->sc_arpcom.ac_if.if_flags &= ~IFF_OACTIVE; /* Clear watchdog timer. */ sc->sc_arpcom.ac_if.if_timer = 0; } - } + } else + return_tintmask = 1; /* * Handle receiver interrupts. */ - if (rint & rintmask) - qn_rint(sc); - - /* Set receive interrupt mask back. */ - *sc->nic_r_mask = rintmask; + if (rint != 0) + qn_rint(sc, rint); if ((sc->sc_arpcom.ac_if.if_flags & IFF_OACTIVE) == 0) qnstart(&sc->sc_arpcom.ac_if); + else if (return_tintmask == 1) + *sc->nic_t_mask = tintmask; + + /* Set receive interrupt mask back. */ + *sc->nic_r_mask = NIC_R_MASK; return (1); } @@ -841,6 +876,7 @@ qnioctl(ifp, command, data) switch (ifa->ifa_addr->sa_family) { #ifdef INET case AF_INET: + qnstop(sc); qninit(sc); arp_ifinit(&sc->sc_arpcom, ifa); break; @@ -857,41 +893,46 @@ qnioctl(ifp, command, data) bcopy(ina->x_host.c_host, sc->sc_arpcom.ac_enaddr, sizeof(sc->sc_arpcom.ac_enaddr)); + qnstop(sc); qninit(sc); break; } #endif default: log(LOG_INFO, "qn:sa_family:default (not tested)\n"); + qnstop(sc); qninit(sc); break; } break; case SIOCSIFFLAGS: - /* - * If interface is marked down and it is running, then stop it. - */ if ((ifp->if_flags & IFF_UP) == 0 && (ifp->if_flags & IFF_RUNNING) != 0) { /* * If interface is marked down and it is running, then * stop it. */ +#ifdef QN_DEBUG1 + qn_dump(sc); +#endif qnstop(sc); ifp->if_flags &= ~IFF_RUNNING; } else if ((ifp->if_flags & IFF_UP) != 0 && - (ifp->if_flags & IFF_RUNNING) == 0) + (ifp->if_flags & IFF_RUNNING) == 0) { /* * If interface is marked up and it is stopped, then * start it. */ qninit(sc); - else { + } else { /* * Something else... we won't do anything so we won't * break anything (hope so). */ +#ifdef QN_DEBUG1 + log(LOG_INFO, "Else branch...\n"); +#endif } break; @@ -921,8 +962,28 @@ qnioctl(ifp, command, data) error = EINVAL; } - (void)splx(s); + splx(s); return (error); } +/* + * Dump some register information. + */ +#ifdef QN_DEBUG1 +static void +qn_dump(sc) + struct qn_softc *sc; +{ + + log(LOG_INFO, "t_status : %04x\n", *sc->nic_t_status); + log(LOG_INFO, "t_mask : %04x\n", *sc->nic_t_mask); + log(LOG_INFO, "t_mode : %04x\n", *sc->nic_t_mode); + log(LOG_INFO, "r_status : %04x\n", *sc->nic_r_status); + log(LOG_INFO, "r_mask : %04x\n", *sc->nic_r_mask); + log(LOG_INFO, "r_mode : %04x\n", *sc->nic_r_mode); + log(LOG_INFO, "pending : %02x\n", sc->transmit_pending); + log(LOG_INFO, "if_flags : %04x\n", sc->sc_arpcom.ac_if.if_flags); +} +#endif + #endif /* NQN > 0 */ diff --git a/sys/arch/amiga/dev/if_qnreg.h b/sys/arch/amiga/dev/if_qnreg.h index 57f42866377..2094db7d925 100644 --- a/sys/arch/amiga/dev/if_qnreg.h +++ b/sys/arch/amiga/dev/if_qnreg.h @@ -1,4 +1,4 @@ -/* $NetBSD: if_qnreg.h,v 1.1 1995/10/07 18:04:31 chopps Exp $ */ +/* $NetBSD: if_qnreg.h,v 1.1.2.1 1995/11/10 16:39:14 chopps Exp $ */ /* * Copyright (c) 1995 Mika Kortelainen @@ -94,6 +94,10 @@ #define CLEAR_R_ERR 0xcfcf /* Clear receive errors */ /* DLCR3 - Receive Interrupt Masks */ +#define R_INT_OVR_FLO 0x0101 /* Receive buf overflow */ +#define R_INT_CRC_ERR 0x0202 /* CRC error */ +#define R_INT_ALG_ERR 0x0404 /* Alignment error */ +#define R_INT_SRT_PKT 0x0808 /* Short packet */ #define R_INT_PKT_RDY 0x8080 /* Packet ready */ #define CLEAR_R_MASK 0x0000 /* Clear receive intr masks */ @@ -101,7 +105,7 @@ #define NO_LOOPBACK 0x0202 /* Loopback control */ /* DLCR5 - Receive Mode */ -/* Accept physical address, multicast group addresses +/* Normal mode: accept physical address, multicast group addresses * which match the 1st three bytes and broadcast address. */ #define NORMAL_MODE 0x0101 |