summaryrefslogtreecommitdiff
path: root/sys/arch/sparc
diff options
context:
space:
mode:
authorTheo de Raadt <deraadt@cvs.openbsd.org>1996-08-11 05:35:37 +0000
committerTheo de Raadt <deraadt@cvs.openbsd.org>1996-08-11 05:35:37 +0000
commite511a52691ddb2290f5d3c9ca6e9f0fc3b320289 (patch)
treef68510b774a71e2a0879cb013e3abd7a15665abc /sys/arch/sparc
parentfc98d7c9a0046f274410f903f952ef6224f68e10 (diff)
netbsd port, now we merge our changes back in
Diffstat (limited to 'sys/arch/sparc')
-rw-r--r--sys/arch/sparc/Makefile14
-rw-r--r--sys/arch/sparc/conf/GENERIC337
-rw-r--r--sys/arch/sparc/conf/GENERIC_SCSI3338
-rw-r--r--sys/arch/sparc/conf/Makefile.sparc169
-rw-r--r--sys/arch/sparc/conf/files.sparc176
-rw-r--r--sys/arch/sparc/dev/amd7930.c69
-rw-r--r--sys/arch/sparc/dev/amd7930var.h6
-rw-r--r--sys/arch/sparc/dev/bt_subr.c5
-rw-r--r--sys/arch/sparc/dev/btreg.h62
-rw-r--r--sys/arch/sparc/dev/bwtwo.c380
-rw-r--r--sys/arch/sparc/dev/bwtworeg.h17
-rw-r--r--sys/arch/sparc/dev/cgeight.c383
-rw-r--r--sys/arch/sparc/dev/cgfour.c346
-rw-r--r--sys/arch/sparc/dev/cgsix.c216
-rw-r--r--sys/arch/sparc/dev/cgsixreg.h5
-rw-r--r--sys/arch/sparc/dev/cgthree.c143
-rw-r--r--sys/arch/sparc/dev/cgthreereg.h6
-rw-r--r--sys/arch/sparc/dev/cgtwo.c118
-rw-r--r--sys/arch/sparc/dev/cons.c226
-rw-r--r--sys/arch/sparc/dev/dma.c628
-rw-r--r--sys/arch/sparc/dev/dmareg.h104
-rw-r--r--sys/arch/sparc/dev/dmavar.h93
-rw-r--r--sys/arch/sparc/dev/esp.c2560
-rw-r--r--sys/arch/sparc/dev/espreg.h255
-rw-r--r--sys/arch/sparc/dev/espvar.h335
-rw-r--r--sys/arch/sparc/dev/fb.c275
-rw-r--r--sys/arch/sparc/dev/fd.c367
-rw-r--r--sys/arch/sparc/dev/fdreg.h4
-rw-r--r--sys/arch/sparc/dev/fdvar.h10
-rw-r--r--sys/arch/sparc/dev/if_ie.c191
-rw-r--r--sys/arch/sparc/dev/if_le.c157
-rw-r--r--sys/arch/sparc/dev/if_levar.h39
-rw-r--r--sys/arch/sparc/dev/kbd.c99
-rw-r--r--sys/arch/sparc/dev/ms.c21
-rw-r--r--sys/arch/sparc/dev/obio.c361
-rw-r--r--sys/arch/sparc/dev/pfourreg.h21
-rw-r--r--sys/arch/sparc/dev/power.c103
-rw-r--r--sys/arch/sparc/dev/power.h60
-rw-r--r--sys/arch/sparc/dev/rcons_font.h2
-rw-r--r--sys/arch/sparc/dev/sbus.c132
-rw-r--r--sys/arch/sparc/dev/sbusvar.h7
-rw-r--r--sys/arch/sparc/dev/si.c160
-rw-r--r--sys/arch/sparc/dev/xd.c176
-rw-r--r--sys/arch/sparc/dev/xdreg.h10
-rw-r--r--sys/arch/sparc/dev/xdvar.h20
-rw-r--r--sys/arch/sparc/dev/xio.h6
-rw-r--r--sys/arch/sparc/dev/xy.c245
-rw-r--r--sys/arch/sparc/dev/xyreg.h6
-rw-r--r--sys/arch/sparc/dev/xyvar.h8
-rw-r--r--sys/arch/sparc/dev/zs.c274
-rw-r--r--sys/arch/sparc/dev/zsvar.h28
-rw-r--r--sys/arch/sparc/fpu/fpu.c5
-rw-r--r--sys/arch/sparc/fpu/fpu_add.c7
-rw-r--r--sys/arch/sparc/fpu/fpu_explode.c4
-rw-r--r--sys/arch/sparc/fpu/fpu_extern.h76
-rw-r--r--sys/arch/sparc/fpu/fpu_implode.c9
-rw-r--r--sys/arch/sparc/fpu/fpu_subr.c7
-rw-r--r--sys/arch/sparc/include/ansi.h5
-rw-r--r--sys/arch/sparc/include/autoconf.h107
-rw-r--r--sys/arch/sparc/include/bsd_openprom.h60
-rw-r--r--sys/arch/sparc/include/cdefs.h2
-rw-r--r--sys/arch/sparc/include/conf.h81
-rw-r--r--sys/arch/sparc/include/cpu.h88
-rw-r--r--sys/arch/sparc/include/ctlreg.h246
-rw-r--r--sys/arch/sparc/include/db_machdep.h22
-rw-r--r--sys/arch/sparc/include/eeprom.h450
-rw-r--r--sys/arch/sparc/include/endian.h10
-rw-r--r--sys/arch/sparc/include/exec.h7
-rw-r--r--sys/arch/sparc/include/fbio.h10
-rw-r--r--sys/arch/sparc/include/fbvar.h16
-rw-r--r--sys/arch/sparc/include/ieeefp.h2
-rw-r--r--sys/arch/sparc/include/kbd.h11
-rw-r--r--sys/arch/sparc/include/kbio.h18
-rw-r--r--sys/arch/sparc/include/oldmon.h135
-rw-r--r--sys/arch/sparc/include/param.h106
-rw-r--r--sys/arch/sparc/include/pmap.h230
-rw-r--r--sys/arch/sparc/include/profile.h31
-rw-r--r--sys/arch/sparc/include/psl.h76
-rw-r--r--sys/arch/sparc/include/pte.h213
-rw-r--r--sys/arch/sparc/include/signal.h10
-rw-r--r--sys/arch/sparc/include/sun_disklabel.h10
-rw-r--r--sys/arch/sparc/include/svr4_machdep.h10
-rw-r--r--sys/arch/sparc/include/trap.h10
-rw-r--r--sys/arch/sparc/include/types.h6
-rw-r--r--sys/arch/sparc/include/vmparam.h6
-rw-r--r--sys/arch/sparc/sparc/amd7930intr.s13
-rw-r--r--sys/arch/sparc/sparc/autoconf.c1018
-rw-r--r--sys/arch/sparc/sparc/auxreg.c33
-rw-r--r--sys/arch/sparc/sparc/auxreg.h6
-rw-r--r--sys/arch/sparc/sparc/bsd_fdintr.s9
-rw-r--r--sys/arch/sparc/sparc/cache.c239
-rw-r--r--sys/arch/sparc/sparc/cache.h82
-rw-r--r--sys/arch/sparc/sparc/clock.c760
-rw-r--r--sys/arch/sparc/sparc/conf.c144
-rw-r--r--sys/arch/sparc/sparc/cpu.c228
-rw-r--r--sys/arch/sparc/sparc/db_disasm.c75
-rw-r--r--sys/arch/sparc/sparc/db_interface.c112
-rw-r--r--sys/arch/sparc/sparc/db_trace.c37
-rw-r--r--sys/arch/sparc/sparc/disksubr.c107
-rw-r--r--sys/arch/sparc/sparc/genassym.c26
-rw-r--r--sys/arch/sparc/sparc/in_cksum.c143
-rw-r--r--sys/arch/sparc/sparc/intersil7170.h66
-rw-r--r--sys/arch/sparc/sparc/intr.c66
-rw-r--r--sys/arch/sparc/sparc/intreg.h39
-rw-r--r--sys/arch/sparc/sparc/iommu.c352
-rw-r--r--sys/arch/sparc/sparc/iommureg.h93
-rw-r--r--sys/arch/sparc/sparc/kgdb_proto.h10
-rw-r--r--sys/arch/sparc/sparc/kgdb_stub.c131
-rw-r--r--sys/arch/sparc/sparc/locore.s1382
-rw-r--r--sys/arch/sparc/sparc/locore2.c6
-rw-r--r--sys/arch/sparc/sparc/machdep.c161
-rw-r--r--sys/arch/sparc/sparc/mem.c12
-rw-r--r--sys/arch/sparc/sparc/memreg.c196
-rw-r--r--sys/arch/sparc/sparc/memreg.h12
-rw-r--r--sys/arch/sparc/sparc/openprom.c17
-rw-r--r--sys/arch/sparc/sparc/pmap.c3676
-rw-r--r--sys/arch/sparc/sparc/process_machdep.c3
-rw-r--r--sys/arch/sparc/sparc/svr4_machdep.c221
-rw-r--r--sys/arch/sparc/sparc/swapgeneric.c4
-rw-r--r--sys/arch/sparc/sparc/sys_machdep.c5
-rw-r--r--sys/arch/sparc/sparc/timerreg.h57
-rw-r--r--sys/arch/sparc/sparc/trap.c447
-rw-r--r--sys/arch/sparc/sparc/vaddrs.h52
-rw-r--r--sys/arch/sparc/sparc/vm_machdep.c171
-rw-r--r--sys/arch/sparc/stand/Makefile.inc4
-rw-r--r--sys/arch/sparc/stand/binstall.sh6
-rw-r--r--sys/arch/sparc/stand/boot.c6
-rw-r--r--sys/arch/sparc/stand/bootxx/Makefile2
-rw-r--r--sys/arch/sparc/stand/installboot.c2
-rw-r--r--sys/arch/sparc/stand/net.c6
-rw-r--r--sys/arch/sparc/stand/promdev.c2
-rw-r--r--sys/arch/sparc/stand/srt0.S9
-rw-r--r--sys/arch/sparc/stand/version.c2
133 files changed, 15391 insertions, 6718 deletions
diff --git a/sys/arch/sparc/Makefile b/sys/arch/sparc/Makefile
index 2dd71ec7454..88ba39ae50c 100644
--- a/sys/arch/sparc/Makefile
+++ b/sys/arch/sparc/Makefile
@@ -1,23 +1,24 @@
-# $NetBSD: Makefile,v 1.3 1994/11/20 20:51:32 deraadt Exp $
+# $NetBSD: Makefile,v 1.4 1996/03/25 22:48:09 mrg Exp $
# from: @(#)Makefile 8.1 (Berkeley) 6/11/93
#
# Makefile for sparc links, tags file
+SUBDIR= stand
+
NOPROG= noprog
NOMAN= noman
NOOBJ= noobj
-SUBDIR= stand
-
-DIRS= conf dev fpu include sparc
+DIRS= conf dev fpu include rcons sbus sparc
links::
-for i in ${DIRS}; do \
(cd $$i && { rm -f tags; ln -s ${SYSTAGS} tags; }) done
SPARC= /sys/arch/sparc/dev/*.[ch] /sys/arch/sparc/fpu/*.[ch] \
- /sys/arch/sparc/include/*.[ch] /sys/arch/sparc/sparc/*.[ch]
+ /sys/arch/sparc/include/*.[ch] /sys/arch/sparc/rcons/*.[ch] \
+ /sys/arch/sparc/sbus/*.[ch] /sys/arch/sparc/sparc/*.[ch]
ASPARC= /sys/arch/sparc/sparc/*.s
tags:
@@ -27,6 +28,5 @@ tags:
>> tags
sort -o tags tags
-obj: _SUBDIRUSE
-
.include <bsd.prog.mk>
+.include <bsd.subdir.mk>
diff --git a/sys/arch/sparc/conf/GENERIC b/sys/arch/sparc/conf/GENERIC
index 61af08d2ad3..47b14c74271 100644
--- a/sys/arch/sparc/conf/GENERIC
+++ b/sys/arch/sparc/conf/GENERIC
@@ -1,32 +1,55 @@
-# $NetBSD: GENERIC,v 1.19 1995/10/08 11:45:39 pk Exp $
+# $NetBSD: GENERIC,v 1.28.2.1 1996/07/02 23:55:22 jtc Exp $
+# Machine architecture; required by config(8)
machine sparc
-options SUN4, SUN4C, SUN4M
-#options MMU_3L
-#options DDB,DEBUG,DIAGNOSTIC
-# obsolete timezone spec
-options TIMEZONE=0, DST=0
+# Options for variants of the Sun SPARC architecure.
+# At least one is required.
+options SUN4 # sun4/100, sun4/200, sun4/300
+options SUN4C # sun4c - SS1, 1+, 2, ELC, SLC, IPC, IPX, etc.
+options SUN4M # sun4m - SS10, SS20, Classic, etc.
+
+#options MMU_3L # 3-level MMU on sun4/400
+
+# obsolete timezone spec; optional
+#options TIMEZONE=0, DST=0
# Standard system options
-options SWAPPAGER, VNODEPAGER, DEVPAGER # paging
-#options DEBUG, DIAGNOSTIC # extra kernel debugging
-options KTRACE # system call tracing support
-#options KGDB # support for kernel gdb
-#options KGDBDEV=0xc01, KGDBRATE=38400 # device & baud rate
-options RASTERCONSOLE # fast rasterop console
-options SYSVMSG,SYSVSEM,SYSVSHM
-options COMPAT_09, COMPAT_10
+options SWAPPAGER # swap and anonymous memory; required
+options VNODEPAGER # mapped files; required
+options DEVPAGER # mapped devices; required
+options KTRACE # system call tracing
+options SYSVMSG # System V message queues
+options SYSVSEM # System V semaphores
+options SYSVSHM # System V shared memory
+#options SHMMAXPGS=1024 # 1024 pages is the default
+options LKM # loadable kernel modules
+#options INSECURE # disable kernel security level
+#options UCONSOLE # allow anyone to steal the virtual console
+
+# Debugging options
+#options DDB # kernel dynamic debugger
+#options DEBUG # kernel debugging code
+#options DIAGNOSTIC # extra kernel sanity checking
+#options KGDB # support for kernel gdb
+#options KGDBDEV=0xc01 # kgdb device number (dev_t)
+#options KGDBRATE=38400 # baud rate
+
+# Compatibility options
+options COMPAT_43 # 4.3BSD system interfaces
+options COMPAT_10 # NetBSD 1.0 binary compatibility
+options COMPAT_11 # NetBSD 1.1 binary compatibility
+options COMPAT_SUNOS # SunOS 4.x binary compatibility
+options COMPAT_SVR4 # SunOS 5.x binary compatibility
# Filesystem options
-options FFS
-options NFSSERVER # Sun NFS-compatible filesystem
-options NFSCLIENT # Sun NFS-compatible filesystem
+options FFS # Berkeley Fast Filesystem
+options QUOTA # FFS quotas
+options NFSSERVER # Sun NFS-compatible filesystem server
+options NFSCLIENT # Sun NFS-compatible filesystem client
options KERNFS # kernel data-structure filesystem
-options FIFO # POSIX fifo support (in all filesystems)
-options QUOTA # fast filesystem with user and group quotas
+options NULLFS # NULL layered filesystem
options MFS # memory-based filesystem
-options LOFS # Loop-back filesystem
options FDESC # user file descriptor filesystem
options UMAPFS # uid/gid remapping filesystem
options LFS # Log-based filesystem (still experimental)
@@ -34,131 +57,229 @@ options PORTAL # portal filesystem (still experimental)
options PROCFS # /proc
options CD9660 # ISO 9660 + Rock Ridge file system
options UNION # union file system
+options FIFO # POSIX fifo support (in all filesystems)
# Networking options
-options INET
+options INET # IP stack
options TCP_COMPAT_42 # compatibility with 4.2BSD TCP/IP
#options GATEWAY # IP packet forwarding
-#options ISO # OSI networking
-#options TPIP
-#options EON
-options COMPAT_43
-
-options LKM
+#options ISO,TPIP # OSI networking
+#options EON # OSI tunneling over IP
+#options CCITT,LLC,HDLC # X.25
# Options for SPARCstation hardware
-options COMPAT_SUNOS # compatibility with SunOS binaries
-options COMPAT_SVR4 # compatibility with SVR4 binaries
+options RASTERCONSOLE # fast rasterop console
-config bsd swap generic
+# Generic swap; second partition of root disk or network.
+config netbsd swap generic
+# Main bus and CPU .. all systems.
mainbus0 at root
cpu0 at mainbus0
-sbus0 at mainbus0
-obio0 at mainbus0
-vmes0 at mainbus0
-vmel0 at mainbus0
-
-audio0 at mainbus0
-auxreg0 at mainbus0
-clock0 at mainbus0
-clock0 at obio0 addr 0xf2000000
-oclock0 at obio0 addr 0xf3000000
-oclock0 at obio0 addr 0x03000000 # 4/100
-memreg0 at mainbus0
-memreg0 at obio0 addr 0xf4000000
-memreg0 at obio0 addr 0x04000000 # 4/100
-timer0 at mainbus0
-timer0 at obio0 addr 0xef000000
-eeprom0 at obio0 addr 0xf2000000
-eeprom0 at obio0 addr 0x02000000 # 4/100
-
-zs0 at mainbus0
-zs0 at obio0 addr 0xf1000000 level 12 flags 0x103
-zs0 at obio0 addr 0x01000000 level 12 flags 0x103 # 4/100
-zs1 at mainbus0
-zs1 at obio0 addr 0xf0000000 level 12 flags 0x103
-zs1 at obio0 addr 0x00000000 level 12 flags 0x103 # 4/100
-zs2 at obio0 addr 0xe0000000 level 12 flags 0x103
-
-# FSBE/S or sun4m
-espdma0 at sbus0 slot ? offset ?
-espdma* at sbus0 slot ? offset ?
-esp0 at espdma? slot ? offset ?
-esp* at espdma? slot ? offset ?
-
-# old sun4c SCSI
-dma0 at sbus0 slot ? offset ?
-dma* at sbus0 slot ? offset ?
-dma0 at obio0 addr 0xfa001000 level 4
-esp0 at sbus0 slot ? offset ?
-esp* at sbus0 slot ? offset ?
-esp0 at obio0 addr 0xfa000000 level 4
-
-# old sun4c
-le0 at sbus? slot ? offset ?
+# Bus types found on SPARC systems.
+sbus0 at mainbus0 # sun4c
+obio0 at mainbus0 # sun4 and sun4m
+vmes0 at mainbus0 # sun4
+vmel0 at mainbus0 # sun4
+iommu0 at mainbus0 # sun4m
+sbus0 at iommu0 # sun4m
+
+audio0 at mainbus0 # sun4c
+audio0 at obio0 # sun4m
+
+auxreg0 at mainbus0 # sun4c
+auxreg0 at obio0 # sun4m
+
+# Power status and control register found on Sun4m systems
+power0 at obio0
+
+# Mostek clock found on 4/300, sun4c, and sun4m systems.
+# The Mostek clock NVRAM is the "eeprom" on sun4/300 systems.
+clock0 at mainbus0 # sun4c
+clock0 at obio0 # sun4m
+clock0 at obio0 addr 0xf2000000 # sun4/300
+
+# Intersil clock found on 4/100 and 4/200 systems.
+oclock0 at obio0 addr 0xf3000000 # sun4/200
+oclock0 at obio0 addr 0x03000000 # sun4/100
+
+# Memory error registers.
+memreg0 at mainbus0 # sun4c
+memreg0 at obio0 # sun4m
+memreg0 at obio0 addr 0xf4000000 # sun4/200 and sun4/300
+memreg0 at obio0 addr 0x04000000 # sun4/100
+
+# Timer chip found on 4/300, sun4c, and sun4m systems.
+timer0 at mainbus0 # sun4c
+timer0 at obio0 # sun4m
+timer0 at obio0 addr 0xef000000 # sun4/300
+
+# EEPROM found on 4/100 and 4/200 systems. Note that the 4/300
+# doesn't use this driver; the `EEPROM' is in the NVRAM on the
+# Mostek clock chip on 4/300 systems.
+eeprom0 at obio0 addr 0xf2000000 # sun4/200
+eeprom0 at obio0 addr 0x02000000 # sun4/100
+
+# Zilog 8530 serial chips. Each has two-channels.
+# zs0 is ttya and ttyb. zs1 is the keyboard and mouse.
+zs0 at mainbus0 # sun4c
+zs0 at obio0 # sun4m
+zs0 at obio0 addr 0xf1000000 level 12 flags 0x103 # sun4/200 and sun4/300
+zs0 at obio0 addr 0x01000000 level 12 flags 0x103 # sun4/100
+zs1 at mainbus0 # sun4c
+zs1 at obio0 # sun4m
+zs1 at obio0 addr 0xf0000000 level 12 flags 0x103 # sun4/200 and sun4/300
+zs1 at obio0 addr 0x00000000 level 12 flags 0x103 # sun4/100
+zs2 at obio0 addr 0xe0000000 level 12 flags 0x103 # sun4/300
+
+#
+# Note the flags on the esp entries below, that work around
+# deficiencies in the current driver:
+# bits 0-7: disable disconnect/reselect for the corresponding target
+# bits 8-15: disable synch negotiation for target [bit-8]
+# Note: targets 4-7 have disconnect/reselect enabled on the premise
+# that tape devices normally have one of these targets. Tape
+# devices should be allowed to disconnect for the SCSI bus
+# to operate acceptably.
+#
+
+# sun4/300 SCSI - an NCR53c94 or equivalent behind
+# an LSI Logic DMA controller
+dma0 at obio0 addr 0xfa001000 level 4 # sun4/300
+esp0 at obio0 addr 0xfa000000 level 4 flags 0xff0f #
+
+# sun4c or sun4m SCSI - an NCR53c94 or equivalent behind
+# specialized DMA glue
+dma0 at sbus0 slot ? offset ? # on-board SCSI
+esp0 at sbus0 slot ? offset ? flags 0xff0f # sun4c
+esp0 at dma0 flags 0xff0f # sun4m
+
+# FSBE/S SCSI - an NCR53c94 or equivalent behind
+dma* at sbus? slot ? offset ? # SBus SCSI
+esp* at sbus? slot ? offset ? flags 0xff0f # two flavours
+esp* at dma? flags 0xff0f # depending on model
+
+# sun4m Ethernet - an AMD 7990 LANCE behind
+# specialized DMA glue
+ledma0 at sbus0 slot ? offset ? # sun4m on-board
+le0 at ledma0 #
+
+# Additional SBus LANCE devices - glued on by lebuffer (not yet implemented)
+#lebuffer* at sbus0 slot ? offset ? # sun4m SBus
+#le* at lebuffer? #
+
+# sun4/300 and sun4c Ethernet - an AMD 7990 LANCE
+le0 at sbus0 slot ? offset ? # sun4c on-board
le* at sbus? slot ? offset ?
-le0 at obio0 addr 0xf9000000 level 6
-ie0 at obio0 addr 0xf6000000 level 6
-ie0 at obio0 addr 0x06000000 level 6 # 4/100
+le0 at obio0 addr 0xf9000000 level 6 # sun4/300
+
+# sun4/100 and sun4/200 Ethernet - an Intel 82586 on-board
+# or on a Multibus/VME card.
+ie0 at obio0 addr 0xf6000000 level 6 # sun4/200 on-board
+ie0 at obio0 addr 0x06000000 level 6 # sun4/100 on-board
+
ie1 at vmes0 addr 0xffe88000 level 5 vect 0x75
ie2 at vmes0 addr 0xff31ff02 level 5 vect 0x76
ie3 at vmes0 addr 0xff35ff02 level 5 vect 0x77
ie4 at vmes0 addr 0xff2dff02 level 5 vect 0x7c
+# Xylogics 753 or 7053 VME SMD disk controllers and disks, found
+# on sun4 systems.
xdc0 at vmel0 addr 0xffffee80 level 3 vect 0x44
xdc1 at vmel0 addr 0xffffee90 level 3 vect 0x45
xdc2 at vmel0 addr 0xffffeea0 level 3 vect 0x46
xdc3 at vmel0 addr 0xffffeeb0 level 3 vect 0x47
xd* at xdc? drive ?
+# Xylogics 451 or 451 VME SMD disk controllers and disks, found
+# on sun4 systems.
xyc0 at vmes0 addr 0xffffee40 level 3 vect 0x48
xyc1 at vmes0 addr 0xffffee48 level 3 vect 0x49
xy* at xyc? drive ?
+# NCR5380-based "Sun SCSI 3" VME SCSI controller.
+# This driver has several flags which may be enabled by OR'ing
+# the values and using the "flags" directive.
+# Valid flags are:
+#
+# 0x01 Use DMA (may be polled)
+# 0x02 Use DMA completion interrupts
+# 0x04 Allow disconnect/reselect
+#
+# E.g. the following would enable DMA, interrupts, and reselect:
+# si0 at vmes0 addr 0xff200000 level 3 vect 0x40 flags 0x07
+#
+# By default, DMA is enabled in the driver.
si0 at vmes0 addr 0xff200000 level 3 vect 0x40
-si1 at vmes0 addr 0xff204000 level 3 vect 0x41
+
+# NCR5380-based "SCSI Weird" on-board SCSI interface found
+# on sun4/100 systems. The flags are the same as the "si"
+# controller. Note, while DMA is enabled by default, only
+# polled DMA works at this time, and reselects do not work
+# on this particular controller.
sw0 at obio0 addr 0x0a000000 level 3
-bwtwo0 at sbus? slot ? offset ?
-bwtwo* at sbus? slot ? offset ?
-bwtwo0 at obio0 addr 0xfd000000 level 4 # 4/260
-bwtwo0 at obio0 addr 0xfb300000 level 4 # 4/{300/400} ??
+# Sun "bwtwo" black and white framebuffer, found on sun4, sun4c, and sun4m
+# systems. If your sun4 system has a cgfour installed in the P4 slot,
+# the P4 entries for "bwtwo" will attach to the overlay plane of the
+# "cgfour".
+bwtwo0 at sbus0 slot ? offset ? # sun4c on-board
+bwtwo* at sbus? slot ? offset ? # sun4c and sun4m
+bwtwo0 at obio0 addr 0xfd000000 level 4 # sun4/200
+bwtwo0 at obio0 addr 0xfb300000 level 4 # sun4/300 in P4 slot
+bwtwo0 at obio0 addr 0x0b300000 level 4 # sun4/100 in P4 slot
+
+# Sun "cgtwo" VME color framebuffer
cgtwo0 at vmes0 addr 0xff400000 level 4 vect 0xa8
-#cgnine0 at vmel0 addr 0x08000000 level 4
+
+# Sun "cgthree" Sbus color framebuffer
cgthree0 at sbus? slot ? offset ?
cgthree* at sbus? slot ? offset ?
-cgsix0 at sbus? slot ? offset ?
-cgsix* at sbus? slot ? offset ?
+#cgthree0 at obio? slot ? offset ? # sun4m
+#cgthree* at obio? slot ? offset ? # sun4m
-pfour0 at obio0 addr 0xfb000000
-pfour0 at obio0 addr 0x0b000000 # 4/100
-bwtwo0 at pfour0
-cgfour0 at pfour0
-cgeight0 at pfour0
-cgsix0 at pfour0
+# Sun "cgfour" color framebuffer with overlay plane. See above comment
+# regarding overlay plane.
+cgfour0 at obio0 addr 0xfb300000 level 4 # sun4/300 P4
+cgfour0 at obio0 addr 0x0b300000 level 4 # sun4/100 P4
+# Sun "cgsix" accelerated color framebuffer.
+cgsix0 at sbus? slot ? offset ?
+cgsix* at sbus? slot ? offset ?
+cgsix0 at obio0 addr 0xfb000000 level 4 # sun4/300 P4
+cgsix0 at obio0 addr 0x0b000000 level 4 # sun4/100 P4
+
+# Sun "cgeight" 24-bit framebuffer
+cgeight0 at obio0 addr 0xfb300000 level 4 # sun4/300 P4
+cgeight0 at obio0 addr 0x0b300000 level 4 # sun4/100 P4
+
+# SCSI bus layer. SCSI devices attach to the SCSI bus, which attaches
+# to the underlying hardware controller.
scsibus* at esp?
scsibus* at si?
scsibus* at sw?
-sd* at scsibus? target ? lun ?
-st* at scsibus? target ? lun ?
-cd* at scsibus? target ? lun ?
-ch* at scsibus? target ? lun ?
-ss* at scsibus? target ? lun ?
-uk* at scsibus? target ? lun ?
-
-fdc0 at mainbus0
-fd* at fdc0
-
-pseudo-device loop
-pseudo-device pty 32
-pseudo-device sl 2
-pseudo-device kbd
-pseudo-device ppp 2
-pseudo-device tun 4
-pseudo-device vnd 3
-pseudo-device bpfilter 16
+# These entries find devices on all SCSI busses and assign
+# unit numers dynamically.
+sd* at scsibus? target ? lun ? # SCSI disks
+st* at scsibus? target ? lun ? # SCSI tapes
+cd* at scsibus? target ? lun ? # SCSI CD-ROMs
+ch* at scsibus? target ? lun ? # SCSI changer devices
+
+# Floppy controller and drive found on SPARCstations.
+fdc0 at mainbus0 # sun4c controller
+fdc0 at obio0 # sun4m controller
+fd* at fdc0 # the drive itself
+
+pseudo-device loop # loopback interface; required
+pseudo-device pty 32 # pseudo-ttys (for network, etc.)
+pseudo-device sl 2 # SLIP interfaces
+pseudo-device kbd # Sun keyboard
+pseudo-device ppp 2 # PPP interfaces
+pseudo-device tun 4 # Network "tunnel" device
+pseudo-device bpfilter 16 # Berkeley Packet Filter
+pseudo-device vnd 4 # disk-like interface to files
+pseudo-device ccd 4 # concatenated and striped disks
+#pseudo-device strip 1 # radio clock
diff --git a/sys/arch/sparc/conf/GENERIC_SCSI3 b/sys/arch/sparc/conf/GENERIC_SCSI3
index dd784f33ec5..a160b2c2a29 100644
--- a/sys/arch/sparc/conf/GENERIC_SCSI3
+++ b/sys/arch/sparc/conf/GENERIC_SCSI3
@@ -1,32 +1,55 @@
-# $NetBSD: GENERIC,v 1.19 1995/10/08 11:45:39 pk Exp $
+# $NetBSD: GENERIC_SCSI3,v 1.28.2.1 1996/07/02 23:55:24 jtc Exp $
+# Machine architecture; required by config(8)
machine sparc
-options SUN4, SUN4C, SUN4M
-#options MMU_3L
-#options DDB,DEBUG,DIAGNOSTIC
-# obsolete timezone spec
-options TIMEZONE=0, DST=0
+# Options for variants of the Sun SPARC architecure.
+# At least one is required.
+options SUN4 # sun4/100, sun4/200, sun4/300
+options SUN4C # sun4c - SS1, 1+, 2, ELC, SLC, IPC, IPX, etc.
+options SUN4M # sun4m - SS10, SS20, Classic, etc.
+
+#options MMU_3L # 3-level MMU on sun4/400
+
+# obsolete timezone spec; optional
+#options TIMEZONE=0, DST=0
# Standard system options
-options SWAPPAGER, VNODEPAGER, DEVPAGER # paging
-#options DEBUG, DIAGNOSTIC # extra kernel debugging
-options KTRACE # system call tracing support
-#options KGDB # support for kernel gdb
-#options KGDBDEV=0xc01, KGDBRATE=38400 # device & baud rate
-options RASTERCONSOLE # fast rasterop console
-options SYSVMSG,SYSVSEM,SYSVSHM
-options COMPAT_09, COMPAT_10
+options SWAPPAGER # swap and anonymous memory; required
+options VNODEPAGER # mapped files; required
+options DEVPAGER # mapped devices; required
+options KTRACE # system call tracing
+options SYSVMSG # System V message queues
+options SYSVSEM # System V semaphores
+options SYSVSHM # System V shared memory
+#options SHMMAXPGS=1024 # 1024 pages is the default
+options LKM # loadable kernel modules
+#options INSECURE # disable kernel security level
+#options UCONSOLE # allow anyone to steal the virtual console
+
+# Debugging options
+#options DDB # kernel dynamic debugger
+#options DEBUG # kernel debugging code
+#options DIAGNOSTIC # extra kernel sanity checking
+#options KGDB # support for kernel gdb
+#options KGDBDEV=0xc01 # kgdb device number (dev_t)
+#options KGDBRATE=38400 # baud rate
+
+# Compatibility options
+options COMPAT_43 # 4.3BSD system interfaces
+options COMPAT_10 # NetBSD 1.0 binary compatibility
+options COMPAT_11 # NetBSD 1.1 binary compatibility
+options COMPAT_SUNOS # SunOS 4.x binary compatibility
+options COMPAT_SVR4 # SunOS 5.x binary compatibility
# Filesystem options
-options FFS
-options NFSSERVER # Sun NFS-compatible filesystem
-options NFSCLIENT # Sun NFS-compatible filesystem
+options FFS # Berkeley Fast Filesystem
+options QUOTA # FFS quotas
+options NFSSERVER # Sun NFS-compatible filesystem server
+options NFSCLIENT # Sun NFS-compatible filesystem client
options KERNFS # kernel data-structure filesystem
-options FIFO # POSIX fifo support (in all filesystems)
-options QUOTA # fast filesystem with user and group quotas
+options NULLFS # NULL layered filesystem
options MFS # memory-based filesystem
-options LOFS # Loop-back filesystem
options FDESC # user file descriptor filesystem
options UMAPFS # uid/gid remapping filesystem
options LFS # Log-based filesystem (still experimental)
@@ -34,132 +57,231 @@ options PORTAL # portal filesystem (still experimental)
options PROCFS # /proc
options CD9660 # ISO 9660 + Rock Ridge file system
options UNION # union file system
+options FIFO # POSIX fifo support (in all filesystems)
# Networking options
-options INET
+options INET # IP stack
options TCP_COMPAT_42 # compatibility with 4.2BSD TCP/IP
#options GATEWAY # IP packet forwarding
-#options ISO # OSI networking
-#options TPIP
-#options EON
-options COMPAT_43
-
-options LKM
+#options ISO,TPIP # OSI networking
+#options EON # OSI tunneling over IP
+#options CCITT,LLC,HDLC # X.25
# Options for SPARCstation hardware
-options COMPAT_SUNOS # compatibility with SunOS binaries
-options COMPAT_SVR4 # compatibility with SVR4 binaries
+options RASTERCONSOLE # fast rasterop console
-config bsd swap generic
+# Generic swap; second partition of root disk or network.
+config netbsd swap generic
+# Main bus and CPU .. all systems.
mainbus0 at root
cpu0 at mainbus0
-sbus0 at mainbus0
-obio0 at mainbus0
-vmes0 at mainbus0
-vmel0 at mainbus0
-
-audio0 at mainbus0
-auxreg0 at mainbus0
-clock0 at mainbus0
-clock0 at obio0 addr 0xf2000000
-oclock0 at obio0 addr 0xf3000000
-oclock0 at obio0 addr 0x03000000 # 4/100
-memreg0 at mainbus0
-memreg0 at obio0 addr 0xf4000000
-memreg0 at obio0 addr 0x04000000 # 4/100
-timer0 at mainbus0
-timer0 at obio0 addr 0xef000000
-eeprom0 at obio0 addr 0xf2000000
-eeprom0 at obio0 addr 0x02000000 # 4/100
-
-zs0 at mainbus0
-zs0 at obio0 addr 0xf1000000 level 12 flags 0x103
-zs0 at obio0 addr 0x01000000 level 12 flags 0x103 # 4/100
-zs1 at mainbus0
-zs1 at obio0 addr 0xf0000000 level 12 flags 0x103
-zs1 at obio0 addr 0x00000000 level 12 flags 0x103 # 4/100
-zs2 at obio0 addr 0xe0000000 level 12 flags 0x103
-
-# FSBE/S or sun4m
-espdma0 at sbus0 slot ? offset ?
-espdma* at sbus0 slot ? offset ?
-esp0 at espdma? slot ? offset ?
-esp* at espdma? slot ? offset ?
-
-# old sun4c SCSI
-dma0 at sbus0 slot ? offset ?
-dma* at sbus0 slot ? offset ?
-dma0 at obio0 addr 0xfa001000 level 4
-esp0 at sbus0 slot ? offset ?
-esp* at sbus0 slot ? offset ?
-esp0 at obio0 addr 0xfa000000 level 4
-
-# old sun4c
-le0 at sbus? slot ? offset ?
+# Bus types found on SPARC systems.
+sbus0 at mainbus0 # sun4c
+obio0 at mainbus0 # sun4 and sun4m
+vmes0 at mainbus0 # sun4
+vmel0 at mainbus0 # sun4
+iommu0 at mainbus0 # sun4m
+sbus0 at iommu0 # sun4m
+
+audio0 at mainbus0 # sun4c
+audio0 at obio0 # sun4m
+
+auxreg0 at mainbus0 # sun4c
+auxreg0 at obio0 # sun4m
+
+# Power status and control register found on Sun4m systems
+power0 at obio0
+
+# Mostek clock found on 4/300, sun4c, and sun4m systems.
+# The Mostek clock NVRAM is the "eeprom" on sun4/300 systems.
+clock0 at mainbus0 # sun4c
+clock0 at obio0 # sun4m
+clock0 at obio0 addr 0xf2000000 # sun4/300
+
+# Intersil clock found on 4/100 and 4/200 systems.
+oclock0 at obio0 addr 0xf3000000 # sun4/200
+oclock0 at obio0 addr 0x03000000 # sun4/100
+
+# Memory error registers.
+memreg0 at mainbus0 # sun4c
+memreg0 at obio0 # sun4m
+memreg0 at obio0 addr 0xf4000000 # sun4/200 and sun4/300
+memreg0 at obio0 addr 0x04000000 # sun4/100
+
+# Timer chip found on 4/300, sun4c, and sun4m systems.
+timer0 at mainbus0 # sun4c
+timer0 at obio0 # sun4m
+timer0 at obio0 addr 0xef000000 # sun4/300
+
+# EEPROM found on 4/100 and 4/200 systems. Note that the 4/300
+# doesn't use this driver; the `EEPROM' is in the NVRAM on the
+# Mostek clock chip on 4/300 systems.
+eeprom0 at obio0 addr 0xf2000000 # sun4/200
+eeprom0 at obio0 addr 0x02000000 # sun4/100
+
+# Zilog 8530 serial chips. Each has two-channels.
+# zs0 is ttya and ttyb. zs1 is the keyboard and mouse.
+zs0 at mainbus0 # sun4c
+zs0 at obio0 # sun4m
+zs0 at obio0 addr 0xf1000000 level 12 flags 0x103 # sun4/200 and sun4/300
+zs0 at obio0 addr 0x01000000 level 12 flags 0x103 # sun4/100
+zs1 at mainbus0 # sun4c
+zs1 at obio0 # sun4m
+zs1 at obio0 addr 0xf0000000 level 12 flags 0x103 # sun4/200 and sun4/300
+zs1 at obio0 addr 0x00000000 level 12 flags 0x103 # sun4/100
+zs2 at obio0 addr 0xe0000000 level 12 flags 0x103 # sun4/300
+
+#
+# Note the flags on the esp entries below, that work around
+# deficiencies in the current driver:
+# bits 0-7: disable disconnect/reselect for the corresponding target
+# bits 8-15: disable synch negotiation for target [bit-8]
+# Note: targets 4-7 have disconnect/reselect enabled on the premise
+# that tape devices normally have one of these targets. Tape
+# devices should be allowed to disconnect for the SCSI bus
+# to operate acceptably.
+#
+
+# sun4/300 SCSI - an NCR53c94 or equivalent behind
+# an LSI Logic DMA controller
+dma0 at obio0 addr 0xfa001000 level 4 # sun4/300
+esp0 at obio0 addr 0xfa000000 level 4 flags 0xff0f #
+
+# sun4c or sun4m SCSI - an NCR53c94 or equivalent behind
+# specialized DMA glue
+dma0 at sbus0 slot ? offset ? # on-board SCSI
+esp0 at sbus0 slot ? offset ? flags 0xff0f # sun4c
+esp0 at dma0 flags 0xff0f # sun4m
+
+# FSBE/S SCSI - an NCR53c94 or equivalent behind
+dma* at sbus? slot ? offset ? # SBus SCSI
+esp* at sbus? slot ? offset ? flags 0xff0f # two flavours
+esp* at dma? flags 0xff0f # depending on model
+
+# sun4m Ethernet - an AMD 7990 LANCE behind
+# specialized DMA glue
+ledma0 at sbus0 slot ? offset ? # sun4m on-board
+le0 at ledma0 #
+
+# Additional SBus LANCE devices - glued on by lebuffer (not yet implemented)
+#lebuffer* at sbus0 slot ? offset ? # sun4m SBus
+#le* at lebuffer? #
+
+# sun4/300 and sun4c Ethernet - an AMD 7990 LANCE
+le0 at sbus0 slot ? offset ? # sun4c on-board
le* at sbus? slot ? offset ?
-le0 at obio0 addr 0xf9000000 level 6
-ie0 at obio0 addr 0xf6000000 level 6
-ie0 at obio0 addr 0x06000000 level 6 # 4/100
+le0 at obio0 addr 0xf9000000 level 6 # sun4/300
+
+# sun4/100 and sun4/200 Ethernet - an Intel 82586 on-board
+# or on a Multibus/VME card.
+ie0 at obio0 addr 0xf6000000 level 6 # sun4/200 on-board
+ie0 at obio0 addr 0x06000000 level 6 # sun4/100 on-board
+
ie1 at vmes0 addr 0xffe88000 level 5 vect 0x75
ie2 at vmes0 addr 0xff31ff02 level 5 vect 0x76
ie3 at vmes0 addr 0xff35ff02 level 5 vect 0x77
ie4 at vmes0 addr 0xff2dff02 level 5 vect 0x7c
+# Xylogics 753 or 7053 VME SMD disk controllers and disks, found
+# on sun4 systems.
xdc0 at vmel0 addr 0xffffee80 level 3 vect 0x44
xdc1 at vmel0 addr 0xffffee90 level 3 vect 0x45
xdc2 at vmel0 addr 0xffffeea0 level 3 vect 0x46
xdc3 at vmel0 addr 0xffffeeb0 level 3 vect 0x47
xd* at xdc? drive ?
+# Xylogics 451 or 451 VME SMD disk controllers and disks, found
+# on sun4 systems.
xyc0 at vmes0 addr 0xffffee40 level 3 vect 0x48
xyc1 at vmes0 addr 0xffffee48 level 3 vect 0x49
xy* at xyc? drive ?
+# NCR5380-based "Sun SCSI 3" VME SCSI controller.
+# This driver has several flags which may be enabled by OR'ing
+# the values and using the "flags" directive.
+# Valid flags are:
+#
+# 0x01 Use DMA (may be polled)
+# 0x02 Use DMA completion interrupts
+# 0x04 Allow disconnect/reselect
+#
+# E.g. the following would enable DMA, interrupts, and reselect:
+# si0 at vmes0 addr 0xff200000 level 3 vect 0x40 flags 0x07
+#
+# By default, DMA is enabled in the driver.
si0 at vmes0 addr 0xff200000 level 3 vect 0x40
-si1 at vmes0 addr 0xff204000 level 3 vect 0x41
+
+# NCR5380-based "SCSI Weird" on-board SCSI interface found
+# on sun4/100 systems. The flags are the same as the "si"
+# controller. Note, while DMA is enabled by default, only
+# polled DMA works at this time, and reselects do not work
+# on this particular controller.
sw0 at obio0 addr 0x0a000000 level 3
-bwtwo0 at sbus? slot ? offset ?
-bwtwo* at sbus? slot ? offset ?
-bwtwo0 at obio0 addr 0xfd000000 level 4 # 4/260
-bwtwo0 at obio0 addr 0xfb300000 level 4 # 4/{300/400} ??
+# Sun "bwtwo" black and white framebuffer, found on sun4, sun4c, and sun4m
+# systems. If your sun4 system has a cgfour installed in the P4 slot,
+# the P4 entries for "bwtwo" will attach to the overlay plane of the
+# "cgfour".
+bwtwo0 at sbus0 slot ? offset ? # sun4c on-board
+bwtwo* at sbus? slot ? offset ? # sun4c and sun4m
+bwtwo0 at obio0 addr 0xfd000000 level 4 # sun4/200
+bwtwo0 at obio0 addr 0xfb300000 level 4 # sun4/300 in P4 slot
+bwtwo0 at obio0 addr 0x0b300000 level 4 # sun4/100 in P4 slot
+
+# Sun "cgtwo" VME color framebuffer
cgtwo0 at vmes0 addr 0xff400000 level 4 vect 0xa8
-#cgnine0 at vmel0 addr 0x08000000 level 4
+
+# Sun "cgthree" Sbus color framebuffer
cgthree0 at sbus? slot ? offset ?
cgthree* at sbus? slot ? offset ?
-cgsix0 at sbus? slot ? offset ?
-cgsix* at sbus? slot ? offset ?
+#cgthree0 at obio? slot ? offset ? # sun4m
+#cgthree* at obio? slot ? offset ? # sun4m
-pfour0 at obio0 addr 0xfb000000
-pfour0 at obio0 addr 0x0b000000 # 4/100
-bwtwo0 at pfour0
-cgfour0 at pfour0
-cgeight0 at pfour0
-cgsix0 at pfour0
+# Sun "cgfour" color framebuffer with overlay plane. See above comment
+# regarding overlay plane.
+cgfour0 at obio0 addr 0xfb300000 level 4 # sun4/300 P4
+cgfour0 at obio0 addr 0x0b300000 level 4 # sun4/100 P4
+# Sun "cgsix" accelerated color framebuffer.
+cgsix0 at sbus? slot ? offset ?
+cgsix* at sbus? slot ? offset ?
+cgsix0 at obio0 addr 0xfb000000 level 4 # sun4/300 P4
+cgsix0 at obio0 addr 0x0b000000 level 4 # sun4/100 P4
+
+# Sun "cgeight" 24-bit framebuffer
+cgeight0 at obio0 addr 0xfb300000 level 4 # sun4/300 P4
+cgeight0 at obio0 addr 0x0b300000 level 4 # sun4/100 P4
+
+# SCSI bus layer. SCSI devices attach to the SCSI bus, which attaches
+# to the underlying hardware controller.
scsibus* at esp?
scsibus* at si?
scsibus* at sw?
-# Map scsi-id 3 to sd0; no questions asked. Map the remaining drives
-# in the default order. This is because SunOS's rom swaps scsi-id 3 to
-# sd0, and scsi-id 0 to sd3.
+# These entries find devices on all SCSI busses and assign
+# unit numers dynamically.
+# Map the SCSI disk at target 3 to unit 0, a'la SunOS.
sd0 at scsibus? target 3 lun ?
-sd* at scsibus? target ? lun ?
-st* at scsibus? target ? lun ?
-cd* at scsibus? target ? lun ?
-
-fdc0 at mainbus0
-fd* at fdc0
-
-pseudo-device loop
-pseudo-device pty 32
-pseudo-device sl 2
-pseudo-device kbd
-pseudo-device ppp 2
-pseudo-device tun 4
-pseudo-device vnd 3
-pseudo-device bpfilter 16
+sd* at scsibus? target ? lun ? # SCSI disks
+st* at scsibus? target ? lun ? # SCSI tapes
+cd* at scsibus? target ? lun ? # SCSI CD-ROMs
+ch* at scsibus? target ? lun ? # SCSI changer devices
+
+# Floppy controller and drive found on SPARCstations.
+fdc0 at mainbus0 # sun4c controller
+fdc0 at obio0 # sun4m controller
+fd* at fdc0 # the drive itself
+
+pseudo-device loop # loopback interface; required
+pseudo-device pty 32 # pseudo-ttys (for network, etc.)
+pseudo-device sl 2 # SLIP interfaces
+pseudo-device kbd # Sun keyboard
+pseudo-device ppp 2 # PPP interfaces
+pseudo-device tun 4 # Network "tunnel" device
+pseudo-device bpfilter 16 # Berkeley Packet Filter
+pseudo-device vnd 4 # disk-like interface to files
+pseudo-device ccd 4 # concatenated and striped disks
+#pseudo-device strip 1 # radio clock
diff --git a/sys/arch/sparc/conf/Makefile.sparc b/sys/arch/sparc/conf/Makefile.sparc
index b812cd783b7..f22d4eef230 100644
--- a/sys/arch/sparc/conf/Makefile.sparc
+++ b/sys/arch/sparc/conf/Makefile.sparc
@@ -1,16 +1,15 @@
-# $NetBSD: Makefile.sparc,v 1.22 1995/09/19 23:26:58 thorpej Exp $
+# $NetBSD: Makefile.sparc,v 1.32.4.1 1996/06/12 20:26:32 pk Exp $
-# @(#)Makefile.sparc 8.1 (Berkeley) 7/19/93
-# Makefile for 4.4 BSD
+# Makefile for NetBSD
#
# This makefile is constructed from a machine description:
# config machineid
# Most changes should be made in the machine description
-# /sys/conf/``machineid''
+# /sys/arch/sparc/conf/``machineid''
# after which you should do
-# config machineid
+# config machineid
# Machine generic makefile changes should be made in
-# /sys/conf/Makefile.``machinetype''
+# /sys/arch/sparc/conf/Makefile.sparc
# after which config should be rerun for all machines of that type.
#
# N.B.: NO DEPENDENCIES ON FOLLOWING FLAGS ARE VISIBLE TO MAKEFILE
@@ -23,20 +22,24 @@
# PROF is set to -pg if profiling.
AS?= as
-
-CC= cc ${DEBUG}
-CPP= cpp
-LD= ld
-TOUCH= touch -f -c
-AWK= awk
+CC?= cc
+CPP?= cpp
+LD?= ld
+STRIP?= strip -d
+TOUCH?= touch -f -c
# source tree is located via $S relative to the compilation directory
S= ../../../..
SPARC= ../..
-INCLUDES= -I. -I$S/arch -I$S -I$S/sys
-COPTS= ${INCLUDES} ${IDENT} -D_KERNEL
-CFLAGS= ${COPTS} -O2 -Werror
+INCLUDES= -I. -I$S/arch -I$S
+CPPFLAGS= ${INCLUDES} ${IDENT} -D_KERNEL
+GCCWFLAGS=-Werror -Wall -Wstrict-prototypes -Wmissing-prototypes
+CFLAGS= ${DEBUG} -O2 ${GCCWFLAGS}
+# add `-mno-fpu' to work around gcc (last noticed in v2.7.2) bug
+CFLAGS+= -mno-fpu
+AFLAGS= -x assembler-with-cpp -traditional-cpp -D_LOCORE
+LINKFLAGS= -N -p -Ttext F8004000 -e start
### find out what to use for libkern
.include "$S/lib/libkern/Makefile.inc"
@@ -59,109 +62,117 @@ LIBCOMPAT= ${COMPATLIB_PROF}
# capitalized (e.g. C for a .c file), and CONFIG_DEP is _C if the file
# is marked as config-dependent.
-# sparc kernel uses volatile, rather than heavy "device-driver"s.
+NORMAL_C= ${CC} ${CFLAGS} ${CPPFLAGS} ${PROF} -c $<
+NORMAL_C_C= ${CC} ${CFLAGS} ${CPPFLAGS} ${PROF} ${PARAM} -c $<
-NORMAL_C= ${CC} -c ${CFLAGS} ${PROF} $<
-NORMAL_C_C= ${CC} -c ${CFLAGS} ${PROF} ${PARAM} $<
+DRIVER_C= ${CC} ${CFLAGS} ${CPPFLAGS} ${PROF} -c $<
+DRIVER_C_C= ${CC} ${CFLAGS} ${CPPFLAGS} ${PROF} ${PARAM} -c $<
-# optimizer breaks sometimes; temporary workaround uses ${NOOPT_C}
-NOOPT_C= ${CC} -c ${COPTS} ${PROF} ${PARAM} $<
-
-NORMAL_S= rm -f $*.c; ln -s $< $*.c; \
- ${CC} ${COPTS} -I${SPARC}/sparc -E $*.c > $*.i; \
- ${AS} -o $@ $*.i; rm -f $*.c $*.i
-NORMAL_S_C= rm -f $*.c; ln -s $< $*.c; \
- ${CC} ${COPTS} ${PARAM} -I${SPARC}/sparc -E $*.c > $*.i; \
- ${AS} -o $@ $*.i; rm -f $*.c $*.i
+NORMAL_S= ${CC} ${AFLAGS} ${CPPFLAGS} -c $<
+NORMAL_S_C= ${CC} ${AFLAGS} ${CPPFLAGS} ${PARAM} -c $<
%OBJS
%CFILES
+%SFILES
+
# load lines for config "xxx" will be emitted as:
# xxx: ${SYSTEM_DEP} swapxxx.o
# ${SYSTEM_LD_HEAD}
# ${SYSTEM_LD} swapxxx.o
# ${SYSTEM_LD_TAIL}
+SYSTEM_OBJ= locore.o \
+ param.o ioconf.o ${OBJS} ${LIBKERN} ${LIBCOMPAT}
+SYSTEM_DEP= Makefile ${SYSTEM_OBJ}
+SYSTEM_LD_HEAD= @rm -f $@
+SYSTEM_LD= @echo ${LD} ${LINKFLAGS} -o $@ '$${SYSTEM_OBJ}' vers.o; \
+ ${LD} ${LINKFLAGS} -o $@ ${SYSTEM_OBJ} vers.o
+SYSTEM_LD_TAIL= @size $@; chmod 755 $@
+
DEBUG?=
.if ${DEBUG} == "-g"
-LDX=-X
+LINKFLAGS+= -X
+SYSTEM_LD_TAIL+=; \
+ echo cp $@ $@.gdb; rm -f $@.gdb; cp $@ $@.gdb; \
+ echo ${STRIP} $@; ${STRIP} $@
.else
-LDX=-x
+LINKFLAGS+= -x
.endif
-SYSTEM_OBJ= locore.o vnode_if.o ${OBJS} param.o ioconf.o \
- ${LIBKERN} ${LIBCOMPAT}
-SYSTEM_DEP= Makefile ${SYSTEM_OBJ}
-SYSTEM_LD_HEAD= @echo loading $@; rm -f $@
-SYSTEM_LD= @${LD} ${LDX} -N -e start -Ttext f8004000 -o $@ \
- ${SYSTEM_OBJ} vers.o
-SYSTEM_LD_TAIL= @echo rearranging symbols; size $@; chmod 755 $@
-#.if ${DEBUG} == "-g"
-#SYSTEM_LD_TAIL+=; echo cp $@ $@.gdb; rm -f $@.gdb; cp $@ $@.gdb; \
-# echo strip -d $@; strip -d $@
-#.endif
%LOAD
-newvers: vers.o
+assym.h: genassym
+ ./genassym >assym.h
-vers.o: ${SYSTEM_DEP} ${SYSTEM_SWAP_DEP}
- sh $S/conf/newvers.sh
- ${CC} ${CFLAGS} -c vers.c
+genassym: genassym.o
+ ${CC} -o $@ genassym.o
-clean::
- rm -f eddep bsd bsd.gdb tags vnode_if.[ch] tags1 *.[io] \
- [a-z]*.s Errs errs linterrs makelinks genassym
+genassym.o: ${SPARC}/sparc/genassym.c
+ ${NORMAL_C_C:S/^-pg$//:S/^-p$//}
-locore.o: ${SPARC}/sparc/locore.s assym.s
- ${NORMAL_S}
+param.c: $S/conf/param.c
+ rm -f param.c
+ cp $S/conf/param.c .
-# depend on maxusers
-assym.s: Makefile
+param.o: param.c Makefile
+ ${NORMAL_C_C}
-./assym.s: assym.s
-assym.s: genassym
- ./genassym >assym.s
+ioconf.o: ioconf.c
+ ${NORMAL_C}
+
+newvers: ${SYSTEM_DEP} ${SYSTEM_SWAP_DEP}
+ sh $S/conf/newvers.sh
+ ${CC} ${CFLAGS} ${CPPFLAGS} ${PROF} -c vers.c
-genassym: genassym.o
- ${CC} -static -o $@ genassym.o
-genassym.o: ${SPARC}/sparc/genassym.c Makefile
- ${CC} -c ${CFLAGS} ${PARAM} $<
+clean::
+ rm -f eddep *netbsd netbsd.gdb tags *.[io] [a-z]*.s \
+ [Ee]rrs linterrs makelinks genassym genassym.o assym.h
+
+lint:
+ @lint -hbxncez -DGENERIC -Dvolatile= ${CPPFLAGS} ${PARAM} -UKGDB \
+ ${SPARC}/sparc/Locore.c ${CFILES} ${SPARC}/sparc/swapgeneric.c \
+ ioconf.c param.c | \
+ grep -v 'static function .* unused'
-SRCS= ${CFILES} ioconf.c param.c vnode_if.c
-depend: .depend
-.depend: ${SRCS} assym.s
- mkdep ${COPTS} ${SRCS}
- mkdep -a -p ${COPTS} ${SPARC}/sparc/genassym.c
+tags:
+ @echo "see $S/kern/Makefile for tags"
links:
egrep '#if' ${CFILES} | sed -f $S/conf/defines | \
sed -e 's/:.*//' -e 's/\.c/.o/' | sort -u > dontlink
echo ${CFILES} | tr -s ' ' '\12' | sed 's/\.c/.o/' | \
sort -u | comm -23 - dontlink | \
- sed 's,../.*/\(.*.o\),rm -f \1;ln -s ../GENERIC/\1 \1,' > makelinks
+ sed 's,../.*/\(.*.o\),rm -f \1; ln -s ../GENERIC/\1 \1,' > makelinks
sh makelinks && rm -f dontlink
-tags: depend
- sh $S/conf/systags.sh
- rm -f tags1
- sed -e 's, ../, ,' tags > tags1
+SRCS= ${SPARC}/sparc/locore.s \
+ param.c ioconf.c ${CFILES} ${SFILES}
+depend:: .depend
+.depend: ${SRCS} assym.h param.c
+ mkdep ${AFLAGS} ${CPPFLAGS} ${SPARC}/sparc/locore.s
+ mkdep -a ${CFLAGS} ${CPPFLAGS} param.c ioconf.c ${CFILES}
+ mkdep -a ${AFLAGS} ${CPPFLAGS} ${SFILES}
+ mkdep -a ${CFLAGS} ${CPPFLAGS} ${PARAM} ${SPARC}/sparc/genassym.c
-ioconf.o: ioconf.c
- ${CC} -c ${CFLAGS} ioconf.c
-param.c: $S/conf/param.c
- rm -f param.c
- cp $S/conf/param.c .
+# depend on root or device configuration
+autoconf.o conf.o: Makefile
+
+# depend on network or filesystem configuration
+uipc_proto.o vfs_conf.o: Makefile
-param.o: param.c Makefile
- ${CC} -c ${CFLAGS} ${PARAM} param.c
+# depend on maxusers
+genassym.o machdep.o: 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
+# depend on CPU configuration
+bwtwo.o cgsix.o cgthree.o cgtwo.o cons.o dma.o esp.o fb.o if_ie.o: Makefile
+ms.c obio.o zs.c autoconf.o clock.o cpu.o disksubr.o locore.o: Makefile
+machdep.o mem.o openprom.o pmap.o vm_machdep.o: Makefile
-%RULES
-# DO NOT DELETE THIS LINE -- make depend uses it
+locore.o: ${SPARC}/sparc/locore.s assym.h
+ ${NORMAL_S}
+%RULES
diff --git a/sys/arch/sparc/conf/files.sparc b/sys/arch/sparc/conf/files.sparc
index 3a6851377c8..43a0adeb80a 100644
--- a/sys/arch/sparc/conf/files.sparc
+++ b/sys/arch/sparc/conf/files.sparc
@@ -1,112 +1,152 @@
-# $NetBSD: files.sparc,v 1.12 1995/10/05 00:21:35 pk Exp $
+# $NetBSD: files.sparc,v 1.23 1996/05/21 19:06:26 pk Exp $
# @(#)files.sparc 8.1 (Berkeley) 7/19/93
# sparc-specific configuration info
-# maxpartitions must be first item in files.${ARCH}
+# maxpartitions must be first item in files.${ARCH}.newconf
maxpartitions 8
maxusers 2 8 64
-device mainbus at root {}
+device mainbus {}
+attach mainbus at root
-device obio at mainbus { [addr = -1], [level = -1] }
-device vmel at mainbus { [addr = -1], [level = -1], [vect = -1] }
-device vmes at mainbus { [addr = -1], [level = -1], [vect = -1] }
-file arch/sparc/dev/obio.c obio vmel vmes
+device obio { [addr = -1], [level = -1] }
+attach obio at mainbus
+device vmel { [addr = -1], [level = -1], [vect = -1] }
+attach vmel at mainbus
+device vmes { [addr = -1], [level = -1], [vect = -1] }
+attach vmes at mainbus
+file arch/sparc/dev/obio.c obio | vmel | vmes
-device pfour at obio {}
-file arch/sparc/dev/pfour.c pfour needs-count
+device audio
+attach audio at mainbus, obio
+file arch/sparc/dev/amd7930.c audio
-device audio at mainbus
-file arch/sparc/dev/amd7930.c audio needs-flag
-file arch/sparc/sparc/amd7930intr.s audio
-
-device auxreg at mainbus
+device auxreg
+attach auxreg at mainbus, obio
file arch/sparc/sparc/auxreg.c
-device clock at mainbus, obio
-device oclock at obio
-device timer at mainbus, obio
-device eeprom at obio
+device clock
+attach clock at mainbus, obio
+
+device oclock
+attach oclock at obio
+
+device timer
+attach timer at mainbus, obio
+
+device eeprom
+attach eeprom at obio
file arch/sparc/sparc/clock.c
-device cpu at mainbus
+device power
+attach power at obio
+file arch/sparc/dev/power.c power
+
+device cpu
+attach cpu at mainbus
file arch/sparc/sparc/cpu.c
-device memreg at mainbus, obio
+device memreg
+attach memreg at mainbus, obio
file arch/sparc/sparc/memreg.c
-device zs at mainbus, obio
-file arch/sparc/dev/zs.c zs needs-count
+device zs
+attach zs at mainbus, obio
+file arch/sparc/dev/zs.c zs needs-count
+
+device fdc {}
+attach fdc at mainbus, obio
+device fd: disk
+attach fd at fdc
+file arch/sparc/dev/fd.c fdc | fd needs-flag
+file arch/sparc/sparc/bsd_fdintr.s fdc
+
+device iommu {}
+attach iommu at mainbus
+file arch/sparc/sparc/iommu.c iommu
-device fdc at mainbus {}
-device fd at fdc: disk
-file arch/sparc/dev/fd.c fd needs-flag
-file arch/sparc/sparc/bsd_fdintr.s fd
+device sbus { slot = -1, offset = -1 }
+attach sbus at mainbus, iommu
+file arch/sparc/dev/sbus.c sbus
-device sbus at mainbus { slot = -1, offset = -1 }
-file arch/sparc/dev/sbus.c sbus
+#
+# Machine-independent SCSI drivers
+#
include "../../../scsi/files.scsi"
major { sd = 7 }
major { vnd = 8 }
major { cd = 18 }
-device dma at sbus, obio
-device espdma at sbus { slot = -1, offset = -1 }
-device ledma at sbus { slot = -1, offset = -1 }
-file arch/sparc/dev/dma.c dma espdma ledma needs-flag
+device dma {}
+attach dma at sbus, obio
+device ledma {}
+attach ledma at sbus
+file arch/sparc/dev/dma.c dma | ledma
-device esp at sbus, espdma, obio: scsi
+device esp: scsi
+attach esp at sbus, dma, obio
file arch/sparc/dev/esp.c esp
-device bwtwo at sbus, obio, vmes, vmel, pfour
+# Brooktree DAC attribute
+define bt_dac
+
+device bwtwo
+attach bwtwo at sbus, obio, vmes, vmel
file arch/sparc/dev/bwtwo.c bwtwo needs-flag
-device cgtwo at vmes
+device cgtwo
+attach cgtwo at sbus, obio, vmes
file arch/sparc/dev/cgtwo.c cgtwo needs-flag
-device cgthree at sbus
+device cgthree: bt_dac
+attach cgthree at sbus
file arch/sparc/dev/cgthree.c cgthree needs-flag
-device cgfour at pfour
+device cgfour: bt_dac
+attach cgfour at obio
file arch/sparc/dev/cgfour.c cgfour needs-flag
-device cgsix at sbus, pfour
+device cgsix: bt_dac
+attach cgsix at sbus, obio
file arch/sparc/dev/cgsix.c cgsix needs-flag
-device cgeight at pfour
+device cgeight: bt_dac
+attach cgeight at obio
file arch/sparc/dev/cgeight.c cgeight needs-flag
-file arch/sparc/dev/bt_subr.c cgsix cgthree cgfour
+file arch/sparc/dev/bt_subr.c bt_dac
-device le at sbus, ledma, obio: ifnet, ether
+# device definition in sys/conf/files
+attach le at sbus, ledma, obio
file arch/sparc/dev/if_le.c le
-device ie at obio, vmes, vmel: ifnet, ether
+device ie: ifnet, ether
+attach ie at obio, vmes, vmel
file arch/sparc/dev/if_ie.c ie
-device xdc at vmel {drive = -1}
-device xd at xdc: disk
+device xdc {drive = -1}
+attach xdc at vmel
+device xd: disk
+attach xd at xdc
file arch/sparc/dev/xd.c xd needs-flag
-major { xd = 10 }
+major {xd = 10}
-device xyc at vmes {drive = -1}
-device xy at xyc: disk
+device xyc {drive = -1}
+attach xyc at vmes
+device xy: disk
+attach xy at xyc
file arch/sparc/dev/xy.c xy needs-flag
-major { xy = 3 }
-
-device si at vmes: scsi, ncr5380sbc
-device sw at obio: scsi, ncr5380sbc
-file arch/sparc/dev/si.c si sw
+major {xy = 3}
-# ENI ATM driver
-device en: atm, ifnet
-attach en at sbus with en_sbus
-file arch/sparc/dev/if_en_sbus.c en
-file dev/ic/midway.c en
+device si: scsi, ncr5380sbc
+attach si at vmes
+device sw: scsi, ncr5380sbc
+attach sw at obio
+file arch/sparc/dev/si.c si | sw
pseudo-device kbd
@@ -126,7 +166,10 @@ file arch/sparc/fpu/fpu_mul.c
file arch/sparc/fpu/fpu_sqrt.c
file arch/sparc/fpu/fpu_subr.c
+# N.B.: optimizer breaks pmap.c and/or cache.c somehow -- have not
+# identified the exact problem yet. NOOPT_C suffices for now.
file arch/sparc/sparc/autoconf.c
+file arch/sparc/sparc/amd7930intr.s audio
file arch/sparc/sparc/cache.c
file arch/sparc/sparc/conf.c
file arch/sparc/sparc/in_cksum.c
@@ -140,21 +183,36 @@ file arch/sparc/sparc/process_machdep.c
file arch/sparc/sparc/mem.c
file arch/sparc/sparc/openprom.c
file arch/sparc/sparc/pmap.c
+# the following overrides the generic "sys_process.c"
+# commented out by deraadt
+#file arch/sparc/sparc/sys_process.c
file arch/sparc/sparc/sys_machdep.c
file arch/sparc/sparc/trap.c
file arch/sparc/sparc/vm_machdep.c
file arch/sparc/sparc/disksubr.c
-file arch/sparc/sparc/db_interface.c ddb
+file kludge_for_in_proto.c hy needs-flag
+
+file arch/sparc/sparc/db_interface.c ddb
file arch/sparc/sparc/db_trace.c ddb
file arch/sparc/sparc/db_disasm.c ddb
+#
+# Raster Console
+#
include "../../../dev/rcons/files.rcons"
+#
+# Compatibility modules
+#
+
# SVR4 Binary Compatibility (COMPAT_SVR4)
include "../../../compat/svr4/files.svr4"
file arch/sparc/sparc/svr4_machdep.c compat_svr4
+file arch/sparc/sparc/sunos_machdep.c compat_sunos
# SunOS Binary Compatibility (COMPAT_SUNOS)
include "../../../compat/sunos/files.sunos"
-file arch/sparc/sparc/sunos_machdep.c compat_sunos
+
+# Miscellaneous
+file netns/ns_cksum.c ns
diff --git a/sys/arch/sparc/dev/amd7930.c b/sys/arch/sparc/dev/amd7930.c
index c7a133e3eaf..d2af316243d 100644
--- a/sys/arch/sparc/dev/amd7930.c
+++ b/sys/arch/sparc/dev/amd7930.c
@@ -1,4 +1,4 @@
-/* $NetBSD: amd7930.c,v 1.2 1995/05/04 19:43:27 pk Exp $ */
+/* $NetBSD: amd7930.c,v 1.10 1996/03/31 22:38:29 pk Exp $ */
/*
* Copyright (c) 1995 Rolf Grossmann
@@ -69,7 +69,7 @@ struct amd7930_softc {
int sc_open; /* single use device */
int sc_locked; /* true when transfering data */
struct mapreg sc_map; /* current contents of map registers */
-
+
u_char sc_rlevel; /* record level */
u_char sc_plevel; /* play level */
u_char sc_mlevel; /* monitor level */
@@ -83,7 +83,7 @@ struct amd7930_softc {
/* sc_au is special in that the hardware interrupt handler uses it */
struct auio sc_au; /* recv and xmit buffers, etc */
-#define sc_intrcnt sc_au.au_intrcnt
+#define sc_intrcnt sc_au.au_intrcnt /* statistics */
};
/* interrupt interfaces */
@@ -92,7 +92,6 @@ int amd7930hwintr __P((void *));
#define AUDIO_SET_SWINTR ienab_bis(IE_L4)
#else
struct auio *auiop;
-extern void amd7930_trap();
#endif
int amd7930swintr __P((void *));
@@ -101,12 +100,15 @@ void audio_setmap __P((volatile struct amd7930 *, struct mapreg *));
static void init_amd __P((volatile struct amd7930 *));
/* autoconfiguration driver */
-void amd9730attach __P((struct device *, struct device *, void *));
-int amd9730match __P((struct device *, void *, void *));
+void amd7930attach __P((struct device *, struct device *, void *));
+int amd7930match __P((struct device *, void *, void *));
+
+struct cfattach audio_ca = {
+ sizeof(struct amd7930_softc), amd7930match, amd7930attach
+};
-struct cfdriver audiocd = {
- NULL, "audio", amd9730match, amd9730attach,
- DV_DULL, sizeof(struct amd7930_softc)
+struct cfdriver audio_cd = {
+ NULL, "audio", DV_DULL
};
struct audio_device amd7930_device = {
@@ -212,8 +214,10 @@ int amd7930_set_in_port __P((void *, int));
int amd7930_get_in_port __P((void *));
int amd7930_commit_settings __P((void *));
u_int amd7930_get_silence __P((int));
-int amd7930_start_output __P((void *, void *, int, void (*)(), void *));
-int amd7930_start_input __P((void *, void *, int, void (*)(), void *));
+int amd7930_start_output __P((void *, void *, int, void (*)(void *),
+ void *));
+int amd7930_start_input __P((void *, void *, int, void (*)(void *),
+ void *));
int amd7930_halt_output __P((void *));
int amd7930_halt_input __P((void *));
int amd7930_cont_output __P((void *));
@@ -268,7 +272,7 @@ struct audio_hw_if sa_hw_if = {
/* autoconfig routines */
int
-amd9730match(parent, vcf, aux)
+amd7930match(parent, vcf, aux)
struct device *parent;
void *vcf, *aux;
{
@@ -276,7 +280,7 @@ amd9730match(parent, vcf, aux)
register struct confargs *ca = aux;
register struct romaux *ra = &ca->ca_ra;
- if (cputyp==CPU_SUN4)
+ if (CPU_ISSUN4)
return (0);
return (strcmp(cf->cf_driver->cd_name, ra->ra_name) == 0);
}
@@ -285,7 +289,7 @@ amd9730match(parent, vcf, aux)
* Audio chip found.
*/
void
-amd9730attach(parent, self, args)
+amd7930attach(parent, self, args)
struct device *parent, *self;
void *args;
{
@@ -301,8 +305,9 @@ amd9730attach(parent, self, args)
}
pri = ra->ra_intr[0].int_pri;
printf(" pri %d, softpri %d\n", pri, PIL_AUSOFT);
- amd = (volatile struct amd7930 *)(ra->ra_vaddr ? ra->ra_vaddr :
- mapiodev(ra->ra_reg, 0, sizeof (*amd), ca->ca_bustype));
+ amd = (volatile struct amd7930 *)(ra->ra_vaddr ?
+ ra->ra_vaddr : mapiodev(ra->ra_reg, 0, sizeof (*amd),
+ ca->ca_bustype));
sc->sc_map.mr_mmr1 = AMD_MMR1_GX | AMD_MMR1_GER |
AMD_MMR1_GR | AMD_MMR1_STG;
@@ -312,7 +317,7 @@ amd9730attach(parent, self, args)
sc->sc_plevel = 128;
sc->sc_mlevel = 0;
sc->sc_out_port = SUNAUDIO_SPEAKER;
-
+
init_amd(amd);
#ifndef AUDIO_C_HANDLER
@@ -360,14 +365,13 @@ amd7930_open(dev, flags)
int flags;
{
register struct amd7930_softc *sc;
- register volatile struct amd7930 *amd;
int unit = AUDIOUNIT(dev);
DPRINTF(("sa_open: unit %d\n",unit));
- if (unit >= audiocd.cd_ndevs)
+ if (unit >= audio_cd.cd_ndevs)
return (ENODEV);
- if ((sc = audiocd.cd_devs[unit]) == NULL)
+ if ((sc = audio_cd.cd_devs[unit]) == NULL)
return (ENXIO);
if (sc->sc_open)
return (EBUSY);
@@ -410,7 +414,7 @@ amd7930_set_in_sr(addr, sr)
{
if (sr != 8000)
return EINVAL;
-
+
return(0); /* no other sampling rates supported by amd chip */
}
@@ -463,7 +467,7 @@ amd7930_set_encoding(addr, enc)
{
if (enc != AUDIO_ENCODING_ULAW)
return(EINVAL);
-
+
return(0); /* no other encoding supported by amd chip */
}
@@ -571,7 +575,7 @@ amd7930_commit_settings(addr)
register struct mapreg *map;
register volatile struct amd7930 *amd;
register int s, level;
-
+
DPRINTF(("sa_commit.\n"));
map = &sc->sc_map;
@@ -579,7 +583,7 @@ amd7930_commit_settings(addr)
map->mr_gx = gx_coeff[sc->sc_rlevel];
map->mr_stgr = gx_coeff[sc->sc_mlevel];
-
+
level = (sc->sc_plevel * (256 + NGER)) >> 8;
if (level >= 256) {
map->mr_ger = ger_coeff[level - 256];
@@ -593,9 +597,9 @@ amd7930_commit_settings(addr)
map->mr_mmr2 |= AMD_MMR2_LS;
else
map->mr_mmr2 &= ~AMD_MMR2_LS;
-
+
s = splaudio();
-
+
amd->cr = AMDR_MAP_MMR1;
amd->dr = map->mr_mmr1;
amd->cr = AMDR_MAP_GX;
@@ -625,7 +629,7 @@ amd7930_start_output(addr, p, cc, intr, arg)
void *addr;
void *p;
int cc;
- void (*intr)();
+ void (*intr) __P((void *));
void *arg;
{
register struct amd7930_softc *sc = addr;
@@ -657,7 +661,7 @@ amd7930_start_input(addr, p, cc, intr, arg)
void *addr;
void *p;
int cc;
- void (*intr)();
+ void (*intr) __P((void *));
void *arg;
{
register struct amd7930_softc *sc = addr;
@@ -760,7 +764,7 @@ amd7930_set_port(addr, cp)
if (cp->type != AUDIO_MIXER_VALUE || cp->un.value.num_channels != 1)
return(EINVAL);
-
+
switch(cp->dev) {
case SUNAUDIO_MIC_PORT:
sc->sc_rlevel = cp->un.value.level[AUDIO_MIXER_LEVEL_MONO];
@@ -790,7 +794,7 @@ amd7930_get_port(addr, cp)
if (cp->type != AUDIO_MIXER_VALUE || cp->un.value.num_channels != 1)
return(EINVAL);
-
+
switch(cp->dev) {
case SUNAUDIO_MIC_PORT:
cp->un.value.level[AUDIO_MIXER_LEVEL_MONO] = sc->sc_rlevel;
@@ -814,8 +818,6 @@ amd7930_query_devinfo(addr, dip)
void *addr;
register mixer_devinfo_t *dip;
{
- register struct amd7930_softc *sc = addr;
-
switch(dip->index) {
case SUNAUDIO_MIC_PORT:
dip->type = AUDIO_MIXER_VALUE;
@@ -867,7 +869,7 @@ amd7930_query_devinfo(addr, dip)
}
DPRINTF(("AUDIO_MIXER_DEVINFO: name=%s\n", dip->label.name));
-
+
return(0);
}
@@ -913,7 +915,6 @@ amd7930hwintr(au0)
}
}
- *(au->au_intrcnt)++;
return (1);
}
#endif /* AUDIO_C_HANDLER */
diff --git a/sys/arch/sparc/dev/amd7930var.h b/sys/arch/sparc/dev/amd7930var.h
index 0faaf75c41e..2d2e5f28257 100644
--- a/sys/arch/sparc/dev/amd7930var.h
+++ b/sys/arch/sparc/dev/amd7930var.h
@@ -1,4 +1,4 @@
-/* $NetBSD: amd7930var.h,v 1.2 1995/05/04 19:43:32 pk Exp $ */
+/* $NetBSD: amd7930var.h,v 1.3 1996/02/01 22:32:25 mycroft Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -44,7 +44,7 @@
* @(#)bsd_audiovar.h 8.1 (Berkeley) 6/11/93
*/
-#ifndef LOCORE
+#ifndef _LOCORE
/* XXX I think these defines should go into some other header file */
#define SUNAUDIO_MIC_PORT 0
@@ -80,4 +80,4 @@ struct mapreg {
u_char mr_mmr2;
};
-#endif /* !LOCORE */
+#endif /* !_LOCORE */
diff --git a/sys/arch/sparc/dev/bt_subr.c b/sys/arch/sparc/dev/bt_subr.c
index c7ba266d197..2a1df0ad71b 100644
--- a/sys/arch/sparc/dev/bt_subr.c
+++ b/sys/arch/sparc/dev/bt_subr.c
@@ -1,4 +1,4 @@
-/* $NetBSD: bt_subr.c,v 1.4 1994/11/20 20:51:54 deraadt Exp $ */
+/* $NetBSD: bt_subr.c,v 1.5 1996/03/14 19:44:32 christos Exp $ */
/*
* Copyright (c) 1993
@@ -45,9 +45,12 @@
*/
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/buf.h>
#include <sys/errno.h>
+#include <vm/vm.h>
+
#include <machine/fbio.h>
#include <sparc/dev/btreg.h>
diff --git a/sys/arch/sparc/dev/btreg.h b/sys/arch/sparc/dev/btreg.h
index a1f4692da60..aa3f5fbc0db 100644
--- a/sys/arch/sparc/dev/btreg.h
+++ b/sys/arch/sparc/dev/btreg.h
@@ -1,4 +1,4 @@
-/* $NetBSD: btreg.h,v 1.2 1994/11/20 20:51:55 deraadt Exp $ */
+/* $NetBSD: btreg.h,v 1.4 1996/02/27 22:09:21 thorpej Exp $ */
/*
* Copyright (c) 1993
@@ -78,3 +78,63 @@ struct bt_regs {
u_int bt_ctrl; /* control register */
u_int bt_omap; /* overlay (cursor) map register */
};
+#define BT_INIT(bt, shift) do { /* whatever this means.. */ \
+ (bt)->bt_addr = 0x06 << (shift); /* command reg */ \
+ (bt)->bt_ctrl = 0x73 << (shift); /* overlay plane */ \
+ (bt)->bt_addr = 0x04 << (shift); /* read mask */ \
+ (bt)->bt_ctrl = 0xff << (shift); /* color planes */ \
+} while(0)
+#define BT_UNBLANK(bt, x, shift) do { \
+ /* restore color 0 (and R of color 1) */ \
+ (bt)->bt_addr = 0 << (shift); \
+ (bt)->bt_cmap = (x); \
+ if ((shift)) { \
+ (bt)->bt_cmap = (x) << 8; \
+ (bt)->bt_cmap = (x) << 16; \
+ /* restore read mask */ \
+ BT_INIT((bt), (shift)); \
+} while(0)
+#define BT_BLANK(bt, shift) do { \
+ (bt)->bt_addr = 0x06 << (shift); /* command reg */ \
+ (bt)->bt_ctrl = 0x70 << (shift); /* overlay plane */ \
+ (bt)->bt_addr = 0x04 << (shift); /* read mask */ \
+ (bt)->bt_ctrl = 0x00 << (shift); /* color planes */ \
+ /* Set color 0 to black -- note that this overwrites R of color 1. */\
+ (bt)->bt_addr = 0 << (shift); \
+ (bt)->bt_cmap = 0 << (shift); \
+ /* restore read mask */ \
+ BT_INIT((bt), (shift)); \
+} while(0)
+
+
+/*
+ * Sbus framebuffer control look like this (usually at offset 0x400000).
+ */
+struct fbcontrol {
+ struct bt_regs fbc_dac;
+ u_char fbc_ctrl;
+ u_char fbc_status;
+ u_char fbc_cursor_start;
+ u_char fbc_cursor_end;
+ u_char fbc_vcontrol[12]; /* 12 bytes of video timing goo */
+};
+/* fbc_ctrl bits: */
+#define FBC_IENAB 0x80 /* Interrupt enable */
+#define FBC_VENAB 0x40 /* Video enable */
+#define FBC_TIMING 0x20 /* Master timing enable */
+#define FBC_CURSOR 0x10 /* Cursor compare enable */
+#define FBC_XTALMSK 0x0c /* Xtal select (0,1,2,test) */
+#define FBC_DIVMSK 0x03 /* Divisor (1,2,3,4) */
+
+/* fbc_status bits: */
+#define FBS_INTR 0x80 /* Interrupt pending */
+#define FBS_MSENSE 0x70 /* Monitor sense mask */
+#define FBS_1024X768 0x10
+#define FBS_1152X900 0x30
+#define FBS_1280X1024 0x40
+#define FBS_1600X1280 0x50
+#define FBS_ID_MASK 0x0f /* ID mask */
+#define FBS_ID_COLOR 0x01
+#define FBS_ID_MONO 0x02
+#define FBS_ID_MONO_ECL 0x03 /* ? */
+
diff --git a/sys/arch/sparc/dev/bwtwo.c b/sys/arch/sparc/dev/bwtwo.c
index b024a82640a..78be5d0b885 100644
--- a/sys/arch/sparc/dev/bwtwo.c
+++ b/sys/arch/sparc/dev/bwtwo.c
@@ -1,35 +1,7 @@
-/* $NetBSD: bwtwo.c,v 1.15 1995/10/09 15:39:34 pk Exp $ */
+/* $NetBSD: bwtwo.c,v 1.26 1996/04/01 17:30:15 christos Exp $ */
/*
- * Copyright (c) 1995 Theo de Raadt
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- * 3. All advertising materials mentioning features or use of this software
- * must display the following acknowledgement:
- * This product includes software developed under OpenBSD by
- * Theo de Raadt.
- * 4. The name of the author may not be used to endorse or promote products
- * derived from this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
- * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
- * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
- * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
- * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
- * SUCH DAMAGE.
- *
+ * Copyright (c) 1996 Jason R. Thorpe. All rights reserved.
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
*
@@ -77,14 +49,19 @@
* black&white display (bwtwo) driver.
*
* Does not handle interrupts, even though they can occur.
+ *
+ * P4 and overlay plane support by Jason R. Thorpe <thorpej@NetBSD.ORG>.
+ * Overlay plane handling hints and ideas provided by Brad Spencer.
*/
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/device.h>
#include <sys/ioctl.h>
#include <sys/malloc.h>
#include <sys/mman.h>
#include <sys/tty.h>
+#include <sys/conf.h>
#include <vm/vm.h>
@@ -92,25 +69,36 @@
#include <machine/autoconf.h>
#include <machine/pmap.h>
#include <machine/fbvar.h>
-#if defined(SUN4)
#include <machine/eeprom.h>
#include <machine/ctlreg.h>
+#include <machine/conf.h>
#include <sparc/sparc/asm.h>
-#endif
+#include <sparc/dev/btreg.h>
#include <sparc/dev/bwtworeg.h>
-#include <sparc/dev/pfourreg.h>
#include <sparc/dev/sbusvar.h>
-#include "pfour.h"
+#if defined(SUN4)
+#include <sparc/dev/pfourreg.h>
+#endif
/* per-display variables */
struct bwtwo_softc {
struct device sc_dev; /* base device */
struct sbusdev sc_sd; /* sbus device */
struct fbdevice sc_fb; /* frame buffer device */
- volatile struct bwtworeg *sc_reg;/* control registers */
- struct rom_reg sc_phys; /* display RAM (phys addr) */
- int sc_bustype;
+ volatile struct fbcontrol *sc_reg;/* control registers */
+ struct rom_reg sc_phys; /* phys address description */
+ int sc_bustype; /* type of bus we live on */
+ int sc_pixeloffset; /* offset to framebuffer */
+#if defined(SUN4)
+ /*
+ * Additional overlay plane goo.
+ */
+ int sc_ovtype; /* what kind of color fb? */
+#define BWO_NONE 0x00
+#define BWO_CGFOUR 0x01
+#define BWO_CGEIGHT 0x02
+#endif
};
/* autoconfiguration driver */
@@ -121,20 +109,24 @@ int bwtwoclose __P((dev_t, int, int, struct proc *));
int bwtwoioctl __P((dev_t, u_long, caddr_t, int, struct proc *));
int bwtwommap __P((dev_t, int, int));
static void bwtwounblank __P((struct device *));
+static void bwtwo_set_video __P((struct bwtwo_softc *, int));
+static int bwtwo_get_video __P((struct bwtwo_softc *));
+
+struct cfattach bwtwo_ca = {
+ sizeof(struct bwtwo_softc), bwtwomatch, bwtwoattach
+};
-struct cfdriver bwtwocd = {
- NULL, "bwtwo", bwtwomatch, bwtwoattach,
- DV_DULL, sizeof(struct bwtwo_softc)
+struct cfdriver bwtwo_cd = {
+ NULL, "bwtwo", DV_DULL
};
+/* XXX we do not handle frame buffer interrupts (do not know how) */
+
/* frame buffer generic driver */
static struct fbdriver bwtwofbdriver = {
bwtwounblank, bwtwoopen, bwtwoclose, bwtwoioctl, bwtwommap
};
-static void bwtwoenable __P((struct bwtwo_softc *, int));
-static int bwtwostatus __P((struct bwtwo_softc *));
-
extern int fbnode;
extern struct tty *fbconstty;
@@ -150,18 +142,48 @@ bwtwomatch(parent, vcf, aux)
struct confargs *ca = aux;
struct romaux *ra = &ca->ca_ra;
+ if (CPU_ISSUN4 && cf->cf_unit != 0)
+ return (0);
+
if (strcmp(cf->cf_driver->cd_name, ra->ra_name))
return (0);
+
+ /*
+ * Mask out invalid flags from the user.
+ */
+ cf->cf_flags &= FB_USERMASK;
+
if (ca->ca_bustype == BUS_SBUS)
return(1);
-#if NPFOUR > 0
- if (ca->ca_bustype == BUS_PFOUR) {
- if (PFOUR_ID(ra->ra_pfour) == PFOUR_ID_BW)
- return (1);
+
+ /*
+ * Make sure there's hardware there.
+ */
+ if (probeget(ra->ra_vaddr, 4) == -1)
return (0);
+
+#if defined(SUN4)
+ if (CPU_ISSUN4 && (ca->ca_bustype == BUS_OBIO)) {
+ /*
+ * Check for a pfour framebuffer.
+ */
+ switch (fb_pfour_id(ra->ra_vaddr)) {
+ case PFOUR_ID_BW:
+ case PFOUR_ID_COLOR8P1: /* bwtwo in ... */
+ case PFOUR_ID_COLOR24: /* ...overlay plane */
+ cf->cf_flags |= FB_PFOUR;
+ /* FALLTHROUGH */
+
+ case PFOUR_NOTPFOUR:
+ return (1);
+
+ default:
+ return (0);
+ }
}
#endif
- return (probeget(ra->ra_vaddr, 4) != -1);
+
+ return (0);
}
/*
@@ -175,107 +197,180 @@ bwtwoattach(parent, self, args)
register struct bwtwo_softc *sc = (struct bwtwo_softc *)self;
register struct confargs *ca = args;
register int node = ca->ca_ra.ra_node, ramsize;
- int isconsole;
- char *nam;
+ struct fbdevice *fb = &sc->sc_fb;
+ int isconsole = 0;
+ int sbus = 1;
+ char *nam = NULL;
- sc->sc_fb.fb_driver = &bwtwofbdriver;
- sc->sc_fb.fb_device = &sc->sc_dev;
- sc->sc_fb.fb_type.fb_type = FBTYPE_SUN2BW;
+ fb->fb_driver = &bwtwofbdriver;
+ fb->fb_device = &sc->sc_dev;
+ fb->fb_type.fb_type = FBTYPE_SUN2BW;
+ fb->fb_flags = sc->sc_dev.dv_cfdata->cf_flags;
+
+ /*
+ * Map the control register.
+ */
+ if (fb->fb_flags & FB_PFOUR) {
+ fb->fb_pfour =
+ (volatile u_int32_t *)mapiodev(ca->ca_ra.ra_reg, 0,
+ sizeof(u_int32_t), ca->ca_bustype);
+ sc->sc_reg = NULL;
+ } else {
+ sc->sc_reg =
+ (volatile struct fbcontrol *)mapiodev(ca->ca_ra.ra_reg,
+ BWREG_REG, sizeof(struct fbcontrol), ca->ca_bustype);
+ fb->fb_pfour = NULL;
+ }
+
+ /* Set up default pixel offset. May be changed below. */
+ sc->sc_pixeloffset = BWREG_MEM;
- sc->sc_bustype = ca->ca_bustype;
switch (ca->ca_bustype) {
-#if defined(SUN4)
-#if NPFOUR > 0
- case BUS_PFOUR:
- node = 0;
- pfour_reset();
- pfour_videosize(ca->ca_ra.ra_pfour,
- &sc->sc_fb.fb_type.fb_width,
- &sc->sc_fb.fb_type.fb_height);
- sc->sc_fb.fb_linebytes = sc->sc_fb.fb_type.fb_width / 8;
- nam = "bwtwo";
- break;
-#endif
-#endif
case BUS_OBIO:
-#if defined(SUN4M)
- if (cputyp == CPU_SUN4M) { /* 4m has framebuffer on obio */
- nam = getpropstring(node, "model");
- break;
- }
-#endif
+ if (CPU_ISSUN4M) /* 4m has framebuffer on obio */
+ goto obp_name;
+
+ sbus = node = 0;
#if defined(SUN4)
- node = 0;
+ if (fb->fb_flags & FB_PFOUR) {
+ nam = "bwtwo/p4";
+ /*
+ * Notice if this is an overlay plane on a color
+ * framebuffer. Note that PFOUR_COLOR_OFF_OVERLAY
+ * is the same as PFOUR_BW_OFF, but we use the
+ * different names anyway.
+ */
+ switch (PFOUR_ID(*fb->fb_pfour)) {
+ case PFOUR_ID_COLOR8P1:
+ sc->sc_ovtype = BWO_CGFOUR;
+ sc->sc_pixeloffset = PFOUR_COLOR_OFF_OVERLAY;
+ break;
+
+ case PFOUR_ID_COLOR24:
+ sc->sc_ovtype = BWO_CGEIGHT;
+ sc->sc_pixeloffset = PFOUR_COLOR_OFF_OVERLAY;
+ break;
+
+ default:
+ sc->sc_ovtype = BWO_NONE;
+ sc->sc_pixeloffset = PFOUR_BW_OFF;
+ break;
+ }
+ } else
+#endif
+ nam = "bwtwo";
+ break;
+
+ case BUS_VME32:
+ case BUS_VME16:
+ sbus = node = 0;
nam = "bwtwo";
break;
-#endif
+
case BUS_SBUS:
+ obp_name:
#if defined(SUN4C) || defined(SUN4M)
nam = getpropstring(node, "model");
- break;
#endif
+ break;
}
+
sc->sc_phys = ca->ca_ra.ra_reg[0];
sc->sc_bustype = ca->ca_bustype;
- sc->sc_fb.fb_type.fb_depth = 1;
- fb_setsize(&sc->sc_fb, sc->sc_fb.fb_type.fb_depth,
- 1152, 900, node, ca->ca_bustype);
+ fb->fb_type.fb_depth = 1;
+ fb_setsize(fb, fb->fb_type.fb_depth, 1152, 900, node, ca->ca_bustype);
- ramsize = sc->sc_fb.fb_type.fb_height * sc->sc_fb.fb_linebytes;
- sc->sc_fb.fb_type.fb_cmsize = 0;
- sc->sc_fb.fb_type.fb_size = ramsize;
+ ramsize = fb->fb_type.fb_height * fb->fb_linebytes;
+ fb->fb_type.fb_cmsize = 0;
+ fb->fb_type.fb_size = ramsize;
printf(": %s, %d x %d", nam,
- sc->sc_fb.fb_type.fb_width, sc->sc_fb.fb_type.fb_height);
+ fb->fb_type.fb_width, fb->fb_type.fb_height);
#if defined(SUN4)
- if (cputyp == CPU_SUN4) {
+ if (CPU_ISSUN4) {
struct eeprom *eep = (struct eeprom *)eeprom_va;
+ int constype = (fb->fb_flags & FB_PFOUR) ? EE_CONS_P4OPT :
+ EE_CONS_BW;
/*
* Assume this is the console if there's no eeprom info
* to be found.
*/
- if (eep == NULL || eep->ee_diag.eed_console == EED_CONS_BW)
+ if (eep == NULL || eep->eeConsole == constype)
isconsole = (fbconstty != NULL);
else
isconsole = 0;
}
#endif
-#if defined(SUN4C) || defined(SUN4M)
- if (cputyp == CPU_SUN4C || cputyp == CPU_SUN4M)
+
+ if (CPU_ISSUN4COR4M)
isconsole = node == fbnode && fbconstty != NULL;
-#endif
+
/*
* When the ROM has mapped in a bwtwo display, the address
- * maps only the video RAM, so in any case we have to map the
+ * maps only the video RAM, hence we always map the control
* registers ourselves. We only need the video RAM if we are
* going to print characters via rconsole.
*/
- if ((sc->sc_fb.fb_pixels = ca->ca_ra.ra_vaddr) == NULL && isconsole) {
+ if ((fb->fb_pixels = ca->ca_ra.ra_vaddr) == NULL && isconsole) {
/* this probably cannot happen (on sun4c), but what the heck */
- sc->sc_fb.fb_pixels = mapiodev(ca->ca_ra.ra_reg, BWREG_MEM,
+ fb->fb_pixels =
+ mapiodev(ca->ca_ra.ra_reg, sc->sc_pixeloffset,
ramsize, ca->ca_bustype);
}
- sc->sc_reg = (volatile struct bwtworeg *)mapiodev(ca->ca_ra.ra_reg,
- BWREG_REG, sizeof(struct bwtworeg), ca->ca_bustype);
/* Insure video is enabled */
- bwtwoenable(sc, 1);
+ bwtwo_set_video(sc, 1);
if (isconsole) {
printf(" (console)\n");
#ifdef RASTERCONSOLE
- fbrcons_init(&sc->sc_fb);
+ fbrcons_init(fb);
#endif
} else
printf("\n");
+
#if defined(SUN4C) || defined(SUN4M)
- if (ca->ca_bustype == BUS_SBUS)
+ if (sbus)
sbus_establish(&sc->sc_sd, &sc->sc_dev);
#endif
- if ((node == fbnode && cputyp != CPU_SUN4) ||
- (isconsole && cputyp == CPU_SUN4))
- fb_attach(&sc->sc_fb);
+
+#if defined(SUN4)
+ if ((fb->fb_flags & FB_PFOUR) && (sc->sc_ovtype != BWO_NONE)) {
+ char *ovnam;
+
+ switch (sc->sc_ovtype) {
+ case BWO_CGFOUR:
+ ovnam = "cgfour";
+ break;
+
+ case BWO_CGEIGHT:
+ ovnam = "cgeight";
+ break;
+
+ default:
+ ovnam = "unknown";
+ break;
+ }
+ printf("%s: %s overlay plane\n", sc->sc_dev.dv_xname, ovnam);
+ }
+#endif
+
+ if (CPU_ISSUN4 || node == fbnode) {
+#if defined(SUN4)
+ /*
+ * If we're on an overlay plane of a color framebuffer,
+ * then we don't force the issue in fb_attach() because
+ * we'd like the color framebuffer to actually be the
+ * "console framebuffer". We're only around to speed
+ * up rconsole.
+ */
+ if ((fb->fb_flags & FB_PFOUR) && (sc->sc_ovtype != BWO_NONE ))
+ fb_attach(fb, 0);
+ else
+#endif
+ fb_attach(fb, isconsole);
+ }
}
int
@@ -286,8 +381,9 @@ bwtwoopen(dev, flags, mode, p)
{
int unit = minor(dev);
- if (unit >= bwtwocd.cd_ndevs || bwtwocd.cd_devs[unit] == NULL)
+ if (unit >= bwtwo_cd.cd_ndevs || bwtwo_cd.cd_devs[unit] == NULL)
return (ENXIO);
+
return (0);
}
@@ -309,7 +405,7 @@ bwtwoioctl(dev, cmd, data, flags, p)
int flags;
struct proc *p;
{
- struct bwtwo_softc *sc = bwtwocd.cd_devs[minor(dev)];
+ struct bwtwo_softc *sc = bwtwo_cd.cd_devs[minor(dev)];
switch (cmd) {
@@ -318,11 +414,11 @@ bwtwoioctl(dev, cmd, data, flags, p)
break;
case FBIOGVIDEO:
- bwtwostatus(sc);
+ *(int *)data = bwtwo_get_video(sc);
break;
case FBIOSVIDEO:
- bwtwoenable(sc, *(int *)data);
+ bwtwo_set_video(sc, (*(int *)data));
break;
default:
@@ -337,7 +433,7 @@ bwtwounblank(dev)
{
struct bwtwo_softc *sc = (struct bwtwo_softc *)dev;
- bwtwoenable(sc, 1);
+ bwtwo_set_video(sc, 1);
}
/*
@@ -349,7 +445,7 @@ bwtwommap(dev, off, prot)
dev_t dev;
int off, prot;
{
- register struct bwtwo_softc *sc = bwtwocd.cd_devs[minor(dev)];
+ register struct bwtwo_softc *sc = bwtwo_cd.cd_devs[minor(dev)];
if (off & PGOFSET)
panic("bwtwommap");
@@ -359,55 +455,59 @@ bwtwommap(dev, off, prot)
* I turned on PMAP_NC here to disable the cache as I was
* getting horribly broken behaviour with it on.
*/
- return (REG2PHYS(&sc->sc_phys, BWREG_MEM+off, sc->sc_bustype) | PMAP_NC);
+ return (REG2PHYS(&sc->sc_phys, sc->sc_pixeloffset + off,
+ sc->sc_bustype) | PMAP_NC);
}
-
-int
-bwtwostatus(sc)
+static int
+bwtwo_get_video(sc)
struct bwtwo_softc *sc;
{
-#ifdef SUN4
-#if NPFOUR > 0
- if (sc->sc_bustype == BUS_PFOUR)
- return pfourstatus();
-#endif
- if (sc->sc_bustype == BUS_OBIO)
- return (lduba(AC_SYSENABLE, ASI_CONTROL) & SYSEN_VIDEO);
-#endif
-#if defined(SUN4C) || defined(SUN4M)
- return (sc->sc_reg->bw_ctl & CTL_VE);
+
+#if defined(SUN4)
+ if (CPU_ISSUN4 && (sc->sc_bustype == BUS_OBIO)) {
+ if (sc->sc_fb.fb_flags & FB_PFOUR) {
+ /*
+ * This handles the overlay plane case, too.
+ */
+ return (fb_pfour_get_video(&sc->sc_fb));
+ } else
+ return ((lduba(AC_SYSENABLE,
+ ASI_CONTROL) & SYSEN_VIDEO) != 0);
+ }
#endif
+
+ return ((sc->sc_reg->fbc_ctrl & FBC_VENAB) != 0);
}
-void
-bwtwoenable(sc, on)
+static void
+bwtwo_set_video(sc, enable)
struct bwtwo_softc *sc;
- int on;
+ int enable;
{
-#if NPFOUR > 0
- if (sc->sc_bustype == BUS_PFOUR) {
- pfourenable(on);
+
+#if defined(SUN4)
+ if (CPU_ISSUN4 && (sc->sc_bustype == BUS_OBIO)) {
+ if (sc->sc_fb.fb_flags & FB_PFOUR) {
+ /*
+ * This handles the overlay plane case, too.
+ */
+ fb_pfour_set_video(&sc->sc_fb, enable);
return;
}
-#endif
- if (on) {
-#ifdef SUN4
- if (sc->sc_bustype == BUS_OBIO) {
+ if (enable)
stba(AC_SYSENABLE, ASI_CONTROL,
lduba(AC_SYSENABLE, ASI_CONTROL) | SYSEN_VIDEO);
- return;
- }
-#endif
- sc->sc_reg->bw_ctl |= CTL_VE;
- } else {
-#ifdef SUN4
- if (sc->sc_bustype == BUS_OBIO) {
+ else
stba(AC_SYSENABLE, ASI_CONTROL,
lduba(AC_SYSENABLE, ASI_CONTROL) & ~SYSEN_VIDEO);
- return;
- }
-#endif
- sc->sc_reg->bw_ctl &= ~CTL_VE;
+
+ return;
}
+#endif
+
+ if (enable)
+ sc->sc_reg->fbc_ctrl |= FBC_VENAB;
+ else
+ sc->sc_reg->fbc_ctrl &= ~FBC_VENAB;
}
diff --git a/sys/arch/sparc/dev/bwtworeg.h b/sys/arch/sparc/dev/bwtworeg.h
index cae902991a8..236f4a44929 100644
--- a/sys/arch/sparc/dev/bwtworeg.h
+++ b/sys/arch/sparc/dev/bwtworeg.h
@@ -1,4 +1,4 @@
-/* $NetBSD: bwtworeg.h,v 1.2 1994/11/20 20:51:58 deraadt Exp $ */
+/* $NetBSD: bwtworeg.h,v 1.3 1996/02/27 00:32:39 pk Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -53,20 +53,9 @@
* the sparc addressing modes work well.
*/
struct bwtworeg {
- /*
- * The xxx0 range is all 0xff on my IPC but causes a screen glitch
- * on my SS1+, so it must do *some*thing... the xxx1 range is full
- * of values but I do not know what they are. bw_ctl changes for
- * a blanked screen.
- */
- char bw_xxx0[16];
- u_char bw_ctl; /* contains video enable */
- char bw_xxx1[15];
+ struct fbcontrol bw_fbc;
};
-/* bits in bw_ctl */
-#define CTL_VE 0x40 /* video enable */
-
/* offsets */
#define BWREG_ID 0
#define BWREG_REG 0x400000
@@ -76,7 +65,7 @@ struct bwtworeg {
struct bwtwo_all {
long ba_id; /* ID = 0xfe010104 on my IPC */
char ba_xxx0[0x400000-4];
- struct bwtworeg ba_reg; /* control registers */
+ struct fbcontrol ba_reg; /* control registers */
char ba_xxx1[0x400000-32];
char ba_ram[4096]; /* actually larger */
};
diff --git a/sys/arch/sparc/dev/cgeight.c b/sys/arch/sparc/dev/cgeight.c
index 546a6992797..aea11563608 100644
--- a/sys/arch/sparc/dev/cgeight.c
+++ b/sys/arch/sparc/dev/cgeight.c
@@ -1,6 +1,7 @@
-/* $NetBSD: cgeight.c,v 1.12 1994/11/23 07:02:07 deraadt Exp $ */
+/* $NetBSD: cgeight.c,v 1.7 1996/04/01 17:29:57 christos Exp $ */
/*
+ * Copyright (c) 1996 Jason R. Thorpe. All rights reserved.
* Copyright (c) 1995 Theo de Raadt
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
@@ -54,12 +55,14 @@
*/
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/buf.h>
#include <sys/device.h>
#include <sys/ioctl.h>
#include <sys/malloc.h>
#include <sys/mman.h>
#include <sys/tty.h>
+#include <sys/conf.h>
#include <vm/vm.h>
@@ -67,20 +70,20 @@
#include <machine/autoconf.h>
#include <machine/pmap.h>
#include <machine/fbvar.h>
+#include <machine/eeprom.h>
+#include <machine/conf.h>
#include <sparc/dev/btreg.h>
#include <sparc/dev/btvar.h>
-#include <sparc/dev/cgeightreg.h>
#include <sparc/dev/pfourreg.h>
/* per-display variables */
struct cgeight_softc {
struct device sc_dev; /* base device */
struct fbdevice sc_fb; /* frame buffer device */
- volatile struct bt_regs *sc_bt; /* Brooktree registers */
struct rom_reg sc_phys; /* display RAM (phys addr) */
+ volatile struct fbcontrol *sc_fbc; /* Brooktree registers */
int sc_bustype; /* type of bus we live on */
- int sc_blanked; /* true if blanked */
union bt_cmap sc_cmap; /* Brooktree color map */
};
@@ -91,13 +94,19 @@ int cgeightopen __P((dev_t, int, int, struct proc *));
int cgeightclose __P((dev_t, int, int, struct proc *));
int cgeightioctl __P((dev_t, u_long, caddr_t, int, struct proc *));
int cgeightmmap __P((dev_t, int, int));
+#if defined(SUN4)
static void cgeightunblank __P((struct device *));
+#endif
+
+struct cfattach cgeight_ca = {
+ sizeof(struct cgeight_softc), cgeightmatch, cgeightattach
+};
-struct cfdriver cgeightcd = {
- NULL, "cgeight", cgeightmatch, cgeightattach,
- DV_DULL, sizeof(struct cgeight_softc)
+struct cfdriver cgeight_cd = {
+ NULL, "cgeight", DV_DULL
};
+#if defined(SUN4)
/* frame buffer generic driver */
static struct fbdriver cgeightfbdriver = {
cgeightunblank, cgeightopen, cgeightclose, cgeightioctl, cgeightmmap
@@ -107,6 +116,9 @@ extern int fbnode;
extern struct tty *fbconstty;
static void cgeightloadcmap __P((struct cgeight_softc *, int, int));
+static int cgeight_get_video __P((struct cgeight_softc *));
+static void cgeight_set_video __P((struct cgeight_softc *, int));
+#endif
/*
* Match a cgeight.
@@ -122,8 +134,40 @@ cgeightmatch(parent, vcf, aux)
if (strcmp(cf->cf_driver->cd_name, ra->ra_name))
return (0);
- if (PFOUR_ID(ra->ra_pfour) == PFOUR_ID_COLOR24)
+
+ /*
+ * Mask out invalid flags from the user.
+ */
+ cf->cf_flags &= FB_USERMASK;
+
+ /*
+ * Only exists on a sun4.
+ */
+ if (!CPU_ISSUN4)
+ return (0);
+
+ /*
+ * Only exists on obio.
+ */
+ if (ca->ca_bustype != BUS_OBIO)
+ return (0);
+
+ /*
+ * Make sure there's hardware there.
+ */
+ if (probeget(ra->ra_vaddr, 4) == -1)
+ return (0);
+
+#if defined(SUN4)
+ /*
+ * Check the pfour register.
+ */
+ if (fb_pfour_id(ra->ra_vaddr) == PFOUR_ID_COLOR24) {
+ cf->cf_flags |= FB_PFOUR;
return (1);
+ }
+#endif
+
return (0);
}
@@ -135,99 +179,115 @@ cgeightattach(parent, self, args)
struct device *parent, *self;
void *args;
{
+#if defined(SUN4)
register struct cgeight_softc *sc = (struct cgeight_softc *)self;
register struct confargs *ca = args;
register int node = 0, ramsize, i;
register volatile struct bt_regs *bt;
- register struct cgeight_all *p;
+ struct fbdevice *fb = &sc->sc_fb;
int isconsole;
-#ifdef RASTERCONSOLE
- struct fbdevice fbd;
-#endif
- sc->sc_fb.fb_driver = &cgeightfbdriver;
- sc->sc_fb.fb_device = &sc->sc_dev;
- sc->sc_fb.fb_type.fb_type = FBTYPE_MEMCOLOR;
+ fb->fb_driver = &cgeightfbdriver;
+ fb->fb_device = &sc->sc_dev;
+ fb->fb_type.fb_type = FBTYPE_MEMCOLOR;
+ fb->fb_flags = sc->sc_dev.dv_cfdata->cf_flags;
+
+ /*
+ * Only pfour cgfours, thank you...
+ */
+ if ((ca->ca_bustype != BUS_OBIO) ||
+ ((fb->fb_flags & FB_PFOUR) == 0)) {
+ printf("%s: ignoring; not a pfour\n", sc->sc_dev.dv_xname);
+ return;
+ }
- pfour_reset();
- pfour_videosize(ca->ca_ra.ra_pfour, &sc->sc_fb.fb_type.fb_width,
- &sc->sc_fb.fb_type.fb_height);
+ /* Map the pfour register. */
+ fb->fb_pfour = (volatile u_int32_t *)mapiodev(ca->ca_ra.ra_reg, 0,
+ sizeof(u_int32_t), ca->ca_bustype);
- sc->sc_fb.fb_linebytes = sc->sc_fb.fb_type.fb_width * 4;
+ ramsize = PFOUR_COLOR_OFF_END - PFOUR_COLOR_OFF_OVERLAY;
+
+ fb->fb_type.fb_depth = 24;
+ fb_setsize(fb, fb->fb_type.fb_depth, 1152, 900, node, ca->ca_bustype);
- ramsize = CG8REG_END - CG8REG_OVERLAY;
- sc->sc_fb.fb_type.fb_depth = 24;
sc->sc_fb.fb_type.fb_cmsize = 256;
sc->sc_fb.fb_type.fb_size = ramsize;
- printf(": %d x %d", sc->sc_fb.fb_type.fb_width,
- sc->sc_fb.fb_type.fb_height);
+ printf(": cgeight/p4, %d x %d", fb->fb_type.fb_width,
+ fb->fb_type.fb_height);
+
+ isconsole = 0;
+
+ if (cputyp == CPU_SUN4) {
+ struct eeprom *eep = (struct eeprom *)eeprom_va;
+
+ /*
+ * Assume this is the console if there's no eeprom info
+ * to be found.
+ */
+ if (eep == NULL || eep->eeConsole == EE_CONS_P4OPT)
+ isconsole = (fbconstty != NULL);
+ }
+
+#if 0
+ /*
+ * We don't do any of the console handling here. Instead,
+ * we let the bwtwo driver pick up the overlay plane and
+ * use it instead. Rconsole should have better performance
+ * with the 1-bit depth.
+ * -- Jason R. Thorpe <thorpej@NetBSD.ORG>
+ */
/*
- * When the ROM has mapped in a cgeight display, the address
+ * When the ROM has mapped in a cgfour display, the address
* maps only the video RAM, so in any case we have to map the
* registers ourselves. We only need the video RAM if we are
* going to print characters via rconsole.
- *
- * XXX: it is insane to map the full 0x800000 space, when
- * the mmap code down below doesn't want it that way.
- * Ridiculous!
*/
- isconsole = node == fbnode && fbconstty != NULL;
- if (ca->ca_ra.ra_vaddr == NULL) {
- /* this probably cannot happen, but what the heck */
- ca->ca_ra.ra_vaddr = mapiodev(ca->ca_ra.ra_reg, 0,
- ramsize, ca->ca_bustype);
+
+ if (isconsole) {
+ /* XXX this is kind of a waste */
+ fb->fb_pixels = mapiodev(ca->ca_ra.ra_reg,
+ PFOUR_COLOR_OFF_OVERLAY, ramsize, ca->ca_bustype);
}
- sc->sc_fb.fb_pixels = (char *)((int)ca->ca_ra.ra_vaddr +
- CG8REG_COLOUR - CG8REG_OVERLAY);
+#endif
+
+ /* Map the Brooktree. */
+ sc->sc_fbc = (volatile struct fbcontrol *)mapiodev(ca->ca_ra.ra_reg,
+ PFOUR_COLOR_OFF_CMAP, sizeof(struct fbcontrol), ca->ca_bustype);
-#define O(memb) ((u_int)(&((struct cgeight_all *)0)->memb))
- sc->sc_bt = bt = (volatile struct bt_regs *)mapiodev(ca->ca_ra.ra_reg,
- O(ba_btreg), sizeof(struct bt_regs), ca->ca_bustype);
sc->sc_phys = ca->ca_ra.ra_reg[0];
sc->sc_bustype = ca->ca_bustype;
+#if 0 /* XXX thorpej ??? */
/* tell the enable plane to look at the mono image */
memset(ca->ca_ra.ra_vaddr, 0xff,
sc->sc_fb.fb_type.fb_width * sc->sc_fb.fb_type.fb_height / 8);
-#if 0
- memset((caddr_t)((int)ca->ca_ra.ra_vaddr +
- CG8REG_ENABLE - CG8REG_OVERLAY), 0x00,
- sc->sc_fb.fb_type.fb_width * sc->sc_fb.fb_type.fb_height / 8);
#endif
/* grab initial (current) color map */
+ bt = &sc->sc_fbc->fbc_dac;
bt->bt_addr = 0;
for (i = 0; i < 256 * 3 / 4; i++)
sc->sc_cmap.cm_chip[i] = bt->bt_cmap;
- /* make sure we are not blanked (see cgeightunblank) */
- bt->bt_addr = 0x06; /* command reg */
- bt->bt_ctrl = 0x73; /* overlay plane */
- bt->bt_addr = 0x04; /* read mask */
- bt->bt_ctrl = 0xff; /* color planes */
+ BT_INIT(bt, 0);
+
+#if 0 /* see above */
if (isconsole) {
printf(" (console)\n");
-#ifdef RASTERCONSOLE
- /*
- * Like SunOS and the bootrom, we want to do full-screen
- * text on the overlay plane. But rcons_init() requires
- * our fbdevice pointer to remain the same; so we hack
- * the fbdevice, pass it in, and then restore it's
- * values. Ugly -- should change rcons_init()'s interface.
- */
- bcopy(&sc->sc_fb, &fbd, sizeof fbd);
- sc->sc_fb.fb_type.fb_depth = 1;
- sc->sc_fb.fb_linebytes = sc->sc_fb.fb_type.fb_width / 8;
- sc->sc_fb.fb_pixels = ca->ca_ra.ra_vaddr;
- rcons_init(&fbd);
- bcopy(&fbd, &sc->sc_fb, sizeof fbd);
+#if defined(RASTERCONSOLE) && 0 /* XXX been told it doesn't work well. */
+ fbrcons_init(fb);
#endif
} else
+#endif /* 0 */
printf("\n");
- if ((node == fbnode && cputyp != CPU_SUN4) ||
- (isconsole && cputyp == CPU_SUN4))
- fb_attach(&sc->sc_fb);
+
+ /*
+ * Even though we're not using rconsole, we'd still like
+ * to notice if we're the console framebuffer.
+ */
+ fb_attach(&sc->sc_fb, isconsole);
+#endif
}
int
@@ -238,7 +298,7 @@ cgeightopen(dev, flags, mode, p)
{
int unit = minor(dev);
- if (unit >= cgeightcd.cd_ndevs || cgeightcd.cd_devs[unit] == NULL)
+ if (unit >= cgeight_cd.cd_ndevs || cgeight_cd.cd_devs[unit] == NULL)
return (ENXIO);
return (0);
}
@@ -261,7 +321,8 @@ cgeightioctl(dev, cmd, data, flags, p)
int flags;
struct proc *p;
{
- register struct cgeight_softc *sc = cgeightcd.cd_devs[minor(dev)];
+#if defined(SUN4)
+ register struct cgeight_softc *sc = cgeight_cd.cd_devs[minor(dev)];
register struct fbgattr *fba;
int error;
@@ -299,83 +360,21 @@ cgeightioctl(dev, cmd, data, flags, p)
break;
case FBIOGVIDEO:
- *(int *)data = sc->sc_blanked;
+ *(int *)data = cgeight_get_video(sc);
break;
case FBIOSVIDEO:
- if (*(int *)data)
- cgeightunblank(&sc->sc_dev);
- else if (!sc->sc_blanked) {
- register volatile struct bt_regs *bt;
-
- bt = sc->sc_bt;
- bt->bt_addr = 0x06; /* command reg */
- bt->bt_ctrl = 0x70; /* overlay plane */
- bt->bt_addr = 0x04; /* read mask */
- bt->bt_ctrl = 0x00; /* color planes */
- /*
- * Set color 0 to black -- note that this overwrites
- * R of color 1.
- */
- bt->bt_addr = 0;
- bt->bt_cmap = 0;
-
- sc->sc_blanked = 1;
- }
+ cgeight_set_video(sc, *(int *)data);
break;
default:
return (ENOTTY);
}
+#endif
return (0);
}
/*
- * Undo the effect of an FBIOSVIDEO that turns the video off.
- */
-static void
-cgeightunblank(dev)
- struct device *dev;
-{
- struct cgeight_softc *sc = (struct cgeight_softc *)dev;
- register volatile struct bt_regs *bt;
-
- if (sc->sc_blanked) {
- sc->sc_blanked = 0;
- bt = sc->sc_bt;
- /* restore color 0 (and R of color 1) */
- bt->bt_addr = 0;
- bt->bt_cmap = sc->sc_cmap.cm_chip[0];
-
- /* restore read mask */
- bt->bt_addr = 0x06; /* command reg */
- bt->bt_ctrl = 0x73; /* overlay plane */
- bt->bt_addr = 0x04; /* read mask */
- bt->bt_ctrl = 0xff; /* color planes */
- }
-}
-
-/*
- * Load a subset of the current (new) colormap into the Brooktree DAC.
- */
-static void
-cgeightloadcmap(sc, start, ncolors)
- register struct cgeight_softc *sc;
- register int start, ncolors;
-{
- register volatile struct bt_regs *bt;
- register u_int *ip;
- register int count;
-
- ip = &sc->sc_cmap.cm_chip[BT_D4M3(start)]; /* start/4 * 3 */
- count = BT_D4M3(start + ncolors - 1) - BT_D4M3(start) + 3;
- bt = sc->sc_bt;
- bt->bt_addr = BT_D4M4(start);
- while (--count >= 0)
- bt->bt_cmap = *ip++;
-}
-
-/*
* Return the address that would map the given device at the given
* offset, allowing for the given protection, or return -1 for error.
*
@@ -389,14 +388,14 @@ cgeightmmap(dev, off, prot)
dev_t dev;
int off, prot;
{
- register struct cgeight_softc *sc = cgeightcd.cd_devs[minor(dev)];
+ register struct cgeight_softc *sc = cgeight_cd.cd_devs[minor(dev)];
int poff;
#define START_ENABLE (128*1024)
-#define START_COLOUR (128*1024 + 128*1024)
-#define COLOUR_SIZE (sc->sc_fb.fb_type.fb_width * \
- sc->sc_fb.fb_type.fb_height * 4)
-#define END_COLOUR (START_COLOUR + COLOUR_SIZE)
+#define START_COLOR ((128*1024) + (128*1024))
+#define COLOR_SIZE (sc->sc_fb.fb_type.fb_width * \
+ sc->sc_fb.fb_type.fb_height * 3)
+#define END_COLOR (START_COLOR + COLOR_SIZE)
#define START_SPECIAL 0x800000
#define PROMSIZE 0x40000
#define NOOVERLAY (0x04000000)
@@ -405,36 +404,108 @@ cgeightmmap(dev, off, prot)
panic("cgeightmap");
if ((u_int)off >= NOOVERLAY) {
- off = off - NOOVERLAY;
+ off =- NOOVERLAY;
/*
* X11 maps a huge chunk of the frame buffer; far more than
* there really is. We compensate by double-mapping the
* first page for as many other pages as it wants
*/
- while (off >= COLOUR_SIZE)
- off = 0;
- poff = off + (CG8REG_COLOUR - CG8REG_OVERLAY);
- } else if ((u_int)off < START_ENABLE) /* in overlay plane */
- poff = off;
- else if ((u_int)off < START_COLOUR) /* in enable plane */
- poff = off + (CG8REG_ENABLE - CG8REG_OVERLAY) - START_ENABLE;
- else if ((u_int)off < END_COLOUR) /* in colour plane */
- poff = off + (CG8REG_COLOUR - CG8REG_OVERLAY) - START_COLOUR;
- else if ((u_int)off < START_SPECIAL) /* hole */
+ while (off >= COLOR_SIZE)
+ off -= COLOR_SIZE; /* XXX thorpej ??? */
+
+ poff = off + PFOUR_COLOR_OFF_COLOR;
+ } else if ((u_int)off < START_ENABLE) {
+ /*
+ * in overlay plane
+ */
+ poff = PFOUR_COLOR_OFF_OVERLAY + off;
+ } else if ((u_int)off < START_COLOR) {
+ /*
+ * in enable plane
+ */
+ poff = (off - START_ENABLE) + PFOUR_COLOR_OFF_ENABLE;
+ } else if ((u_int)off < END_COLOR) {
+ /*
+ * in colour plane
+ */
+ poff = (off - START_COLOR) + PFOUR_COLOR_OFF_COLOR;
+ } else if ((u_int)off < START_SPECIAL) {
+ /*
+ * hole
+ */
+ poff = 0; /* XXX */
+ } else if ((u_int)off == START_SPECIAL) {
+ /*
+ * colour map (Brooktree)
+ */
+ poff = PFOUR_COLOR_OFF_CMAP;
+ } else if ((u_int)off == START_SPECIAL + NBPG) {
+ /*
+ * p4 register
+ */
poff = 0;
- else if ((u_int)off == START_SPECIAL) /* colour map */
- poff = CG8REG_CMAP;
- else if ((u_int)off == START_SPECIAL + NBPG) /* p4 register */
- poff = PFOUR_REG;
- else if ((u_int)off > START_SPECIAL + NBPG*2 &&
- (u_int) off < START_SPECIAL + NBPG*2 + PROMSIZE) /* rom */
- poff = CG8REG_P4REG + 0x8000 + off - START_SPECIAL + NBPG*2;
- else
+ } else if ((u_int)off > (START_SPECIAL + (NBPG * 2)) &&
+ (u_int) off < (START_SPECIAL + (NBPG * 2) + PROMSIZE)) {
+ /*
+ * rom
+ */
+ poff = 0x8000 + (off - (START_SPECIAL + (NBPG * 2)));
+ } else
return (-1);
/*
* I turned on PMAP_NC here to disable the cache as I was
* getting horribly broken behaviour with it on.
*/
- return (REG2PHYS(&sc->sc_phys, off, sc->sc_bustype) | PMAP_NC);
+ return (REG2PHYS(&sc->sc_phys, poff, sc->sc_bustype) | PMAP_NC);
+}
+
+#if defined(SUN4)
+/*
+ * Undo the effect of an FBIOSVIDEO that turns the video off.
+ */
+static void
+cgeightunblank(dev)
+ struct device *dev;
+{
+
+ cgeight_set_video((struct cgeight_softc *)dev, 1);
+}
+
+static int
+cgeight_get_video(sc)
+ struct cgeight_softc *sc;
+{
+
+ return (fb_pfour_get_video(&sc->sc_fb));
}
+
+static void
+cgeight_set_video(sc, enable)
+ struct cgeight_softc *sc;
+ int enable;
+{
+
+ fb_pfour_set_video(&sc->sc_fb, enable);
+}
+
+/*
+ * Load a subset of the current (new) colormap into the Brooktree DAC.
+ */
+static void
+cgeightloadcmap(sc, start, ncolors)
+ register struct cgeight_softc *sc;
+ register int start, ncolors;
+{
+ register volatile struct bt_regs *bt;
+ register u_int *ip;
+ register int count;
+
+ ip = &sc->sc_cmap.cm_chip[BT_D4M3(start)]; /* start/4 * 3 */
+ count = BT_D4M3(start + ncolors - 1) - BT_D4M3(start) + 3;
+ bt = &sc->sc_fbc->fbc_dac;
+ bt->bt_addr = BT_D4M4(start);
+ while (--count >= 0)
+ bt->bt_cmap = *ip++;
+}
+#endif
diff --git a/sys/arch/sparc/dev/cgfour.c b/sys/arch/sparc/dev/cgfour.c
index ebd6b1fecd7..d5dde467a9c 100644
--- a/sys/arch/sparc/dev/cgfour.c
+++ b/sys/arch/sparc/dev/cgfour.c
@@ -1,6 +1,7 @@
-/* $NetBSD: cgfour.c,v 1.12 1994/11/23 07:02:07 deraadt Exp $ */
+/* $NetBSD: cgfour.c,v 1.7 1996/04/01 17:29:58 christos Exp $ */
/*
+ * Copyright (c) 1996 Jason R. Thorpe. All rights reserved.
* Copyright (c) 1995 Theo de Raadt. All rights reserved.
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
@@ -58,12 +59,14 @@
*/
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/buf.h>
#include <sys/device.h>
#include <sys/ioctl.h>
#include <sys/malloc.h>
#include <sys/mman.h>
#include <sys/tty.h>
+#include <sys/conf.h>
#include <vm/vm.h>
@@ -71,20 +74,20 @@
#include <machine/autoconf.h>
#include <machine/pmap.h>
#include <machine/fbvar.h>
+#include <machine/eeprom.h>
+#include <machine/conf.h>
#include <sparc/dev/btreg.h>
#include <sparc/dev/btvar.h>
-#include <sparc/dev/cgfourreg.h>
#include <sparc/dev/pfourreg.h>
/* per-display variables */
struct cgfour_softc {
struct device sc_dev; /* base device */
struct fbdevice sc_fb; /* frame buffer device */
- volatile struct bt_regs *sc_bt; /* Brooktree registers */
struct rom_reg sc_phys; /* display RAM (phys addr) */
+ volatile struct fbcontrol *sc_fbc; /* Brooktree registers */
int sc_bustype; /* type of bus we live on */
- int sc_blanked; /* true if blanked */
union bt_cmap sc_cmap; /* Brooktree color map */
};
@@ -95,13 +98,19 @@ int cgfouropen __P((dev_t, int, int, struct proc *));
int cgfourclose __P((dev_t, int, int, struct proc *));
int cgfourioctl __P((dev_t, u_long, caddr_t, int, struct proc *));
int cgfourmmap __P((dev_t, int, int));
+#if defined(SUN4)
static void cgfourunblank __P((struct device *));
+#endif
+
+struct cfattach cgfour_ca = {
+ sizeof(struct cgfour_softc), cgfourmatch, cgfourattach
+};
-struct cfdriver cgfourcd = {
- NULL, "cgfour", cgfourmatch, cgfourattach,
- DV_DULL, sizeof(struct cgfour_softc)
+struct cfdriver cgfour_cd = {
+ NULL, "cgfour", DV_DULL
};
+#if defined(SUN4)
/* frame buffer generic driver */
static struct fbdriver cgfourfbdriver = {
cgfourunblank, cgfouropen, cgfourclose, cgfourioctl, cgfourmmap
@@ -111,6 +120,9 @@ extern int fbnode;
extern struct tty *fbconstty;
static void cgfourloadcmap __P((struct cgfour_softc *, int, int));
+static int cgfour_get_video __P((struct cgfour_softc *));
+static void cgfour_set_video __P((struct cgfour_softc *, int));
+#endif
/*
* Match a cgfour.
@@ -126,8 +138,40 @@ cgfourmatch(parent, vcf, aux)
if (strcmp(cf->cf_driver->cd_name, ra->ra_name))
return (0);
- if (PFOUR_ID(ra->ra_pfour) == PFOUR_ID_COLOR8P1)
+
+ /*
+ * Mask out invalid flags from the user.
+ */
+ cf->cf_flags &= FB_USERMASK;
+
+ /*
+ * Only exists on a sun4.
+ */
+ if (!CPU_ISSUN4)
+ return (0);
+
+ /*
+ * Only exists on obio.
+ */
+ if (ca->ca_bustype != BUS_OBIO)
+ return (0);
+
+ /*
+ * Make sure there's hardware there.
+ */
+ if (probeget(ra->ra_vaddr, 4) == -1)
+ return (0);
+
+#if defined(SUN4)
+ /*
+ * Check the pfour register.
+ */
+ if (fb_pfour_id(ra->ra_vaddr) == PFOUR_ID_COLOR8P1) {
+ cf->cf_flags |= FB_PFOUR;
return (1);
+ }
+#endif
+
return (0);
}
@@ -139,74 +183,109 @@ cgfourattach(parent, self, args)
struct device *parent, *self;
void *args;
{
+#if defined(SUN4)
register struct cgfour_softc *sc = (struct cgfour_softc *)self;
register struct confargs *ca = args;
register int node = 0, ramsize, i;
register volatile struct bt_regs *bt;
+ struct fbdevice *fb = &sc->sc_fb;
int isconsole;
- sc->sc_fb.fb_driver = &cgfourfbdriver;
- sc->sc_fb.fb_device = &sc->sc_dev;
- sc->sc_fb.fb_type.fb_type = FBTYPE_SUN4COLOR;
+ fb->fb_driver = &cgfourfbdriver;
+ fb->fb_device = &sc->sc_dev;
+ fb->fb_type.fb_type = FBTYPE_SUN4COLOR;
+ fb->fb_flags = sc->sc_dev.dv_cfdata->cf_flags;
+
+ /*
+ * Only pfour cgfours, thank you...
+ */
+ if ((ca->ca_bustype != BUS_OBIO) ||
+ ((fb->fb_flags & FB_PFOUR) == 0)) {
+ printf("%s: ignoring; not a pfour\n", sc->sc_dev.dv_xname);
+ return;
+ }
+
+ /* Map the pfour register. */
+ fb->fb_pfour = (volatile u_int32_t *)mapiodev(ca->ca_ra.ra_reg, 0,
+ sizeof(u_int32_t), ca->ca_bustype);
+
+ ramsize = PFOUR_COLOR_OFF_END - PFOUR_COLOR_OFF_OVERLAY;
+
+ fb->fb_type.fb_depth = 8;
+ fb_setsize(fb, fb->fb_type.fb_depth, 1152, 900, node, ca->ca_bustype);
+
+ fb->fb_type.fb_cmsize = 256;
+ fb->fb_type.fb_size = ramsize;
+ printf(": cgfour/p4, %d x %d", fb->fb_type.fb_width,
+ fb->fb_type.fb_height);
+
+ isconsole = 0;
- pfour_reset();
- pfour_videosize(ca->ca_ra.ra_pfour, &sc->sc_fb.fb_type.fb_width,
- &sc->sc_fb.fb_type.fb_height);
+ if (CPU_ISSUN4) {
+ struct eeprom *eep = (struct eeprom *)eeprom_va;
- sc->sc_fb.fb_linebytes = sc->sc_fb.fb_type.fb_width;
+ /*
+ * Assume this is the console if there's no eeprom info
+ * to be found.
+ */
+ if (eep == NULL || eep->eeConsole == EE_CONS_P4OPT)
+ isconsole = (fbconstty != NULL);
+ }
- ramsize = CG4REG_END - CG4REG_OVERLAY;
- sc->sc_fb.fb_type.fb_depth = 8;
- sc->sc_fb.fb_type.fb_cmsize = 256;
- sc->sc_fb.fb_type.fb_size = ramsize;
- printf(": %d x %d", sc->sc_fb.fb_type.fb_width,
- sc->sc_fb.fb_type.fb_height);
+#if 0
+ /*
+ * We don't do any of the console handling here. Instead,
+ * we let the bwtwo driver pick up the overlay plane and
+ * use it instead. Rconsole should have better performance
+ * with the 1-bit depth.
+ * -- Jason R. Thorpe <thorpej@NetBSD.ORG>
+ */
/*
* When the ROM has mapped in a cgfour display, the address
* maps only the video RAM, so in any case we have to map the
* registers ourselves. We only need the video RAM if we are
* going to print characters via rconsole.
- *
- * XXX: it is insane to map the full 0x800000 space, when
- * the mmap code down below doesn't use it all. Ridiculous!
*/
- isconsole = node == fbnode && fbconstty != NULL;
- if (ca->ca_ra.ra_vaddr == NULL) {
- /* this probably cannot happen, but what the heck */
- ca->ca_ra.ra_vaddr = mapiodev(ca->ca_ra.ra_reg, 0,
- ramsize, ca->ca_bustype);
+
+ if (isconsole) {
+ /* XXX this is kind of a waste */
+ fb->fb_pixels = mapiodev(ca->ca_ra.ra_reg,
+ PFOUR_COLOR_OFF_OVERLAY, ramsize, ca->ca_bustype);
}
- sc->sc_fb.fb_pixels = (char *)((int)ca->ca_ra.ra_vaddr +
- CG4REG_COLOUR - CG4REG_OVERLAY);
+#endif
+
+ /* Map the Brooktree. */
+ sc->sc_fbc = (volatile struct fbcontrol *)mapiodev(ca->ca_ra.ra_reg,
+ PFOUR_COLOR_OFF_CMAP, sizeof(struct fbcontrol), ca->ca_bustype);
-#define O(memb) ((u_int)(&((struct cgfour_all *)0)->memb))
- sc->sc_bt = bt = (volatile struct bt_regs *)mapiodev(ca->ca_ra.ra_reg,
- O(ba_btreg), sizeof(struct bt_regs), ca->ca_bustype);
sc->sc_phys = ca->ca_ra.ra_reg[0];
sc->sc_bustype = ca->ca_bustype;
/* grab initial (current) color map */
+ bt = &sc->sc_fbc->fbc_dac;
bt->bt_addr = 0;
for (i = 0; i < 256 * 3 / 4; i++)
((char *)&sc->sc_cmap)[i] = bt->bt_cmap >> 24;
- /* make sure we are not blanked (see cgfourunblank) */
- bt->bt_addr = 0x06 << 24; /* command reg */
- bt->bt_ctrl = 0x73 << 24; /* overlay plane */
- bt->bt_addr = 0x04 << 24; /* read mask */
- bt->bt_ctrl = 0xff << 24; /* color planes */
+ BT_INIT(bt, 24);
+#if 0 /* See above. */
if (isconsole) {
printf(" (console)\n");
-#ifdef RASTERCONSOLE
- fbrcons_init(&sc->sc_fb);
+#if defined(RASTERCONSOLE) && 0 /* XXX been told it doesn't work well. */
+ fbrcons_init(fb);
#endif
} else
+#endif /* 0 */
printf("\n");
- if ((node == fbnode && cputyp != CPU_SUN4) ||
- (isconsole && cputyp == CPU_SUN4))
- fb_attach(&sc->sc_fb);
+
+ /*
+ * Even though we're not using rconsole, we'd still like
+ * to notice if we're the console framebuffer.
+ */
+ fb_attach(fb, isconsole);
+#endif
}
int
@@ -217,7 +296,7 @@ cgfouropen(dev, flags, mode, p)
{
int unit = minor(dev);
- if (unit >= cgfourcd.cd_ndevs || cgfourcd.cd_devs[unit] == NULL)
+ if (unit >= cgfour_cd.cd_ndevs || cgfour_cd.cd_devs[unit] == NULL)
return (ENXIO);
return (0);
}
@@ -240,7 +319,8 @@ cgfourioctl(dev, cmd, data, flags, p)
int flags;
struct proc *p;
{
- register struct cgfour_softc *sc = cgfourcd.cd_devs[minor(dev)];
+#if defined(SUN4)
+ register struct cgfour_softc *sc = cgfour_cd.cd_devs[minor(dev)];
register struct fbgattr *fba;
int error;
@@ -278,62 +358,109 @@ cgfourioctl(dev, cmd, data, flags, p)
break;
case FBIOGVIDEO:
- *(int *)data = sc->sc_blanked;
+ *(int *)data = cgfour_get_video(sc);
break;
case FBIOSVIDEO:
- if (*(int *)data)
- cgfourunblank(&sc->sc_dev);
- else if (!sc->sc_blanked) {
- register volatile struct bt_regs *bt;
-
- bt = sc->sc_bt;
- bt->bt_addr = 0x06 << 24; /* command reg */
- bt->bt_ctrl = 0x70 << 24; /* overlay plane */
- bt->bt_addr = 0x04 << 24; /* read mask */
- bt->bt_ctrl = 0x00 << 24; /* color planes */
- /*
- * Set color 0 to black -- note that this overwrites
- * R of color 1.
- */
- bt->bt_addr = 0 << 24;
- bt->bt_cmap = 0 << 24;
-
- sc->sc_blanked = 1;
- }
+ cgfour_set_video(sc, *(int *)data);
break;
default:
return (ENOTTY);
}
+#endif
return (0);
}
/*
+ * Return the address that would map the given device at the given
+ * offset, allowing for the given protection, or return -1 for error.
+ *
+ * the cg4 maps it's overlay plane for 128K, followed by the enable
+ * plane for 128K, followed by the colour plane (for as much colour
+ * as their is.)
+ *
+ * As well, mapping at an offset of 0x04000000 causes the cg4 to map
+ * only it's colour plane, at 0.
+ */
+int
+cgfourmmap(dev, off, prot)
+ dev_t dev;
+ int off, prot;
+{
+ register struct cgfour_softc *sc = cgfour_cd.cd_devs[minor(dev)];
+ int poff;
+
+#define START_ENABLE (128*1024)
+#define START_COLOR ((128*1024) + (128*1024))
+#define COLOR_SIZE (sc->sc_fb.fb_type.fb_width * \
+ sc->sc_fb.fb_type.fb_height)
+#define END_COLOR (START_COLOR + COLOR_SIZE)
+#define NOOVERLAY (0x04000000)
+
+ if (off & PGOFSET)
+ panic("cgfourmap");
+
+ if ((u_int)off >= NOOVERLAY) {
+ off =- NOOVERLAY;
+
+ /*
+ * X11 maps a huge chunk of the frame buffer; far more than
+ * there really is. We compensate by double-mapping the
+ * first page for as many other pages as it wants
+ */
+ while (off >= COLOR_SIZE)
+ off -= COLOR_SIZE; /* XXX thorpej ??? */
+
+ poff = off + PFOUR_COLOR_OFF_COLOR;
+ } else if ((u_int)off < START_ENABLE) {
+ /*
+ * in overlay plane
+ */
+ poff = PFOUR_COLOR_OFF_OVERLAY + off;
+ } else if ((u_int)off < START_COLOR) {
+ /*
+ * in enable plane
+ */
+ poff = (off - START_ENABLE) + PFOUR_COLOR_OFF_ENABLE;
+ } else if ((u_int)off < END_COLOR) {
+ /*
+ * in colour plane
+ */
+ poff = (off - START_COLOR) + PFOUR_COLOR_OFF_COLOR;
+ } else
+ return (-1);
+
+ return (REG2PHYS(&sc->sc_phys, poff, sc->sc_bustype) | PMAP_NC);
+}
+
+#if defined(SUN4)
+/*
* Undo the effect of an FBIOSVIDEO that turns the video off.
*/
static void
cgfourunblank(dev)
struct device *dev;
{
- struct cgfour_softc *sc = (struct cgfour_softc *)dev;
- register volatile struct bt_regs *bt;
- if (sc->sc_blanked) {
- sc->sc_blanked = 0;
- bt = sc->sc_bt;
- /* restore color 0 (and R of color 1) */
- bt->bt_addr = 0 << 24;
- bt->bt_cmap = sc->sc_cmap.cm_chip[0];
- bt->bt_cmap = sc->sc_cmap.cm_chip[0] << 8;
- bt->bt_cmap = sc->sc_cmap.cm_chip[0] << 16;
-
- /* restore read mask */
- bt->bt_addr = 0x06 << 24; /* command reg */
- bt->bt_ctrl = 0x73 << 24; /* overlay plane */
- bt->bt_addr = 0x04 << 24; /* read mask */
- bt->bt_ctrl = 0xff << 24; /* color planes */
- }
+ cgfour_set_video((struct cgfour_softc *)dev, 1);
+}
+
+static int
+cgfour_get_video(sc)
+ struct cgfour_softc *sc;
+{
+
+ return (fb_pfour_get_video(&sc->sc_fb));
+}
+
+static void
+cgfour_set_video(sc, enable)
+ struct cgfour_softc *sc;
+ int enable;
+{
+
+ fb_pfour_set_video(&sc->sc_fb, enable);
}
/*
@@ -350,7 +477,7 @@ cgfourloadcmap(sc, start, ncolors)
ip = &sc->sc_cmap.cm_chip[BT_D4M3(start)]; /* start/4 * 3 */
count = BT_D4M3(start + ncolors - 1) - BT_D4M3(start) + 3;
- bt = sc->sc_bt;
+ bt = &sc->sc_fbc->fbc_dac;
bt->bt_addr = BT_D4M4(start) << 24;
while (--count >= 0) {
i = *ip++;
@@ -361,53 +488,4 @@ cgfourloadcmap(sc, start, ncolors)
bt->bt_cmap = i << 24;
}
}
-
-/*
- * Return the address that would map the given device at the given
- * offset, allowing for the given protection, or return -1 for error.
- *
- * the cg4 maps it's overlay plane for 128K, followed by the enable
- * plane for 128K, followed by the colour plane (for as much colour
- * as their is.)
- *
- * As well, mapping at an offset of 0x04000000 causes the cg4 to map
- * only it's colour plane, at 0.
- */
-int
-cgfourmmap(dev, off, prot)
- dev_t dev;
- int off, prot;
-{
- register struct cgfour_softc *sc = cgfourcd.cd_devs[minor(dev)];
- int poff;
-
-#define START_ENABLE (128*1024)
-#define START_COLOUR (128*1024 + 128*1024)
-#define COLOUR_SIZE (sc->sc_fb.fb_type.fb_width * sc->sc_fb.fb_type.fb_height)
-#define END_COLOUR (START_COLOUR + COLOUR_SIZE)
-#define NOOVERLAY (0x04000000)
-
- if (off & PGOFSET)
- panic("cgfourmap");
-
- if ((u_int)off >= NOOVERLAY) {
- off = off - NOOVERLAY;
-
- /*
- * X11 maps a huge chunk of the frame buffer; far more than
- * there really is. We compensate by double-mapping the
- * first page for as many other pages as it wants
- */
- while (off >= COLOUR_SIZE)
- off = 0;
- poff = off + (CG4REG_COLOUR - CG4REG_OVERLAY);
- } else if ((u_int)off < START_ENABLE) /* in overlay plane */
- poff = off;
- else if ((u_int)off < START_COLOUR) /* in enable plane */
- poff = off + (CG4REG_ENABLE - CG4REG_OVERLAY) - START_ENABLE;
- else if ((u_int)off < (CG4REG_END - CG4REG_OVERLAY)) /* in colour plane */
- poff = off + (CG4REG_COLOUR - CG4REG_OVERLAY) - START_COLOUR;
- else
- return (-1);
- return (REG2PHYS(&sc->sc_phys, off, sc->sc_bustype) | PMAP_NC);
-}
+#endif
diff --git a/sys/arch/sparc/dev/cgsix.c b/sys/arch/sparc/dev/cgsix.c
index 9dab9af06b0..259024c7da6 100644
--- a/sys/arch/sparc/dev/cgsix.c
+++ b/sys/arch/sparc/dev/cgsix.c
@@ -1,4 +1,4 @@
-/* $NetBSD: cgsix.c,v 1.16 1995/10/08 01:39:16 pk Exp $ */
+/* $NetBSD: cgsix.c,v 1.25 1996/04/01 17:30:00 christos Exp $ */
/*
* Copyright (c) 1993
@@ -53,6 +53,7 @@
*/
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/buf.h>
#include <sys/device.h>
#include <machine/fbio.h>
@@ -60,6 +61,7 @@
#include <sys/malloc.h>
#include <sys/mman.h>
#include <sys/tty.h>
+#include <sys/conf.h>
#ifdef DEBUG
#include <sys/proc.h>
@@ -71,12 +73,19 @@
#include <machine/autoconf.h>
#include <machine/pmap.h>
#include <machine/fbvar.h>
+#include <machine/cpu.h>
+#if defined(SUN4)
+#include <machine/eeprom.h>
+#endif
+#include <machine/conf.h>
#include <sparc/dev/btreg.h>
#include <sparc/dev/btvar.h>
#include <sparc/dev/cgsixreg.h>
#include <sparc/dev/sbusvar.h>
+#if defined(SUN4)
#include <sparc/dev/pfourreg.h>
+#endif
union cursor_cmap { /* colormap, like bt_cmap, but tiny */
u_char cm_map[2][3]; /* 2 R/G/B entries */
@@ -118,9 +127,12 @@ int cgsixioctl __P((dev_t, u_long, caddr_t, int, struct proc *));
int cgsixmmap __P((dev_t, int, int));
static void cg6_unblank __P((struct device *));
-struct cfdriver cgsixcd = {
- NULL, "cgsix", cgsixmatch, cgsixattach,
- DV_DULL, sizeof(struct cgsix_softc)
+struct cfattach cgsix_ca = {
+ sizeof(struct cgsix_softc), cgsixmatch, cgsixattach
+};
+
+struct cfdriver cgsix_cd = {
+ NULL, "cgsix", DV_DULL
};
/* frame buffer generic driver */
@@ -155,19 +167,35 @@ cgsixmatch(parent, vcf, aux)
if (strcmp(cf->cf_driver->cd_name, ra->ra_name))
return (0);
+
+ /*
+ * Mask out invalid flags from the user.
+ */
+ cf->cf_flags &= FB_USERMASK;
+
if (ca->ca_bustype == BUS_SBUS)
return (1);
-#ifdef SUN4
- if (ca->ca_bustype == BUS_PFOUR &&
- PFOUR_ID(ra->ra_pfour) == PFOUR_ID_FASTCOLOR)
- return (1);
- if (ca->ca_bustype == BUS_OBIO) {
- struct cg6_layout *p = (struct cg6_layout *)ra->ra_paddr;
- void *tmp = bus_tmp(&p->cg6_fhc_un.un_fhc, ca->ca_bustype);
- return (probeget(tmp, 4) != 0);
- }
+#if defined(SUN4)
+ if (CPU_ISSUN4 && (ca->ca_bustype == BUS_OBIO)) {
+ void *tmp;
+
+ /*
+ * Check for a pfour framebuffer. This is done somewhat
+ * differently on the cgsix than other pfour framebuffers.
+ */
+ bus_untmp();
+ tmp = bus_tmp(ra->ra_paddr + CGSIX_FHC_OFFSET, ca->ca_bustype);
+ if (probeget(tmp, 4) == -1)
+ return (0);
+
+ if (fb_pfour_id(tmp) == PFOUR_ID_FASTCOLOR) {
+ cf->cf_flags |= FB_PFOUR;
+ return (1);
+ }
+ }
#endif
+
return (0);
}
@@ -181,79 +209,105 @@ cgsixattach(parent, self, args)
{
register struct cgsix_softc *sc = (struct cgsix_softc *)self;
register struct confargs *ca = args;
- register int node = ca->ca_ra.ra_node, ramsize, i, isconsole;
+ register int node = 0, ramsize, i;
register volatile struct bt_regs *bt;
- register volatile struct cg6_layout *p;
- char *nam;
+ struct fbdevice *fb = &sc->sc_fb;
+ int isconsole = 0, sbus = 1;
+ char *nam = NULL;
extern struct tty *fbconstty;
- sc->sc_fb.fb_driver = &cg6_fbdriver;
- sc->sc_fb.fb_device = &sc->sc_dev;
- sc->sc_fb.fb_type.fb_type = FBTYPE_SUNFAST_COLOR;
+ fb->fb_driver = &cg6_fbdriver;
+ fb->fb_device = &sc->sc_dev;
+ fb->fb_type.fb_type = FBTYPE_SUNFAST_COLOR;
+ fb->fb_flags = sc->sc_dev.dv_cfdata->cf_flags;
+
+ /*
+ * Dunno what the PROM has mapped, though obviously it must have
+ * the video RAM mapped. Just map what we care about for ourselves
+ * (the FHC, THC, and Brooktree registers).
+ */
+#define O(memb) ((u_int)(&((struct cg6_layout *)0)->memb))
+ sc->sc_physadr = ca->ca_ra.ra_reg[0];
+ sc->sc_bustype = ca->ca_bustype;
+ sc->sc_bt = bt = (volatile struct bt_regs *)
+ mapiodev(ca->ca_ra.ra_reg, O(cg6_bt_un.un_btregs),
+ sizeof *sc->sc_bt, ca->ca_bustype);
+ sc->sc_fhc = (volatile int *)
+ mapiodev(ca->ca_ra.ra_reg, O(cg6_fhc_un.un_fhc),
+ sizeof *sc->sc_fhc, ca->ca_bustype);
+ sc->sc_thc = (volatile struct cg6_thc *)
+ mapiodev(ca->ca_ra.ra_reg, O(cg6_thc_un.un_thc),
+ sizeof *sc->sc_thc, ca->ca_bustype);
+ sc->sc_tec = (volatile struct cg6_tec_xxx *)
+ mapiodev(ca->ca_ra.ra_reg, O(cg6_tec_un.un_tec),
+ sizeof *sc->sc_tec, ca->ca_bustype);
switch (ca->ca_bustype) {
-#if defined(SUN4)
- case BUS_PFOUR:
- node = 0;
-#if 0
- /*
- * XXX cg6 reset routine is not good enough to
- * rebuild state correctly!
- */
- pfour_reset();
-#endif
- /*
- * XXX pfour register is confused?
- */
- sc->sc_fb.fb_type.fb_width = 1152;
- sc->sc_fb.fb_type.fb_height = 900;
- nam = "cgsix";
- break;
- case BUS_VME32:
- node = 0;
- nam = "cgsix";
- break;
-#endif /* SUN4 */
case BUS_OBIO:
+ sbus = node = 0;
+ if (fb->fb_flags & FB_PFOUR)
+ nam = "cgsix/p4";
+ else
+ nam = "cgsix";
+
#if defined(SUN4M)
- if (cputyp == CPU_SUN4M) { /* 4m has framebuffer on obio */
+ if (CPU_ISSUN4M) { /* 4m has framebuffer on obio */
+ node = ca->ca_ra.ra_node;
nam = getpropstring(node, "model");
break;
}
#endif
break;
+
+ case BUS_VME32:
+ case BUS_VME16:
+ sbus = node = 0;
+ nam = "cgsix";
+ break;
+
case BUS_SBUS:
+ node = ca->ca_ra.ra_node;
nam = getpropstring(node, "model");
break;
+
+ case BUS_MAIN:
+ printf("cgsix on mainbus?\n");
+ return;
}
- sc->sc_fb.fb_type.fb_depth = 8;
- fb_setsize(&sc->sc_fb, sc->sc_fb.fb_type.fb_depth,
- 1152, 900, node, ca->ca_bustype);
+ /* Don't have to map the pfour register on the cgsix. */
+ fb->fb_pfour = NULL;
- ramsize = sc->sc_fb.fb_type.fb_height * sc->sc_fb.fb_linebytes;
- sc->sc_fb.fb_type.fb_cmsize = 256;
- sc->sc_fb.fb_type.fb_size = ramsize;
- printf(": %s, %d x %d", nam,
- sc->sc_fb.fb_type.fb_width, sc->sc_fb.fb_type.fb_height);
- isconsole = node == fbnode && fbconstty != NULL;
+ fb->fb_type.fb_depth = 8;
+ fb_setsize(fb, fb->fb_type.fb_depth, 1152, 900,
+ node, ca->ca_bustype);
- /*
- * Dunno what the PROM has mapped, though obviously it must have
- * the video RAM mapped. Just map what we care about for ourselves
- * (the FHC, THC, and Brooktree registers).
- */
-#define O(memb) ((u_int)(&((struct cg6_layout *)0)->memb))
- sc->sc_physadr = ca->ca_ra.ra_reg[0];
- sc->sc_bustype = ca->ca_bustype;
- sc->sc_bt = bt = (volatile struct bt_regs *)mapiodev(ca->ca_ra.ra_reg,
- O(cg6_bt_un.un_btregs), sizeof *sc->sc_bt, ca->ca_bustype);
- sc->sc_fhc = (volatile int *)mapiodev(ca->ca_ra.ra_reg,
- O(cg6_fhc_un.un_fhc), sizeof *sc->sc_fhc, ca->ca_bustype);
- sc->sc_thc = (volatile struct cg6_thc *)mapiodev(ca->ca_ra.ra_reg,
- O(cg6_thc_un.un_thc), sizeof *sc->sc_thc, ca->ca_bustype);
- sc->sc_tec = (volatile struct cg6_tec_xxx *)mapiodev(ca->ca_ra.ra_reg,
- O(cg6_tec_un.un_tec), sizeof *sc->sc_tec, ca->ca_bustype);
+ ramsize = fb->fb_type.fb_height * fb->fb_linebytes;
+ fb->fb_type.fb_cmsize = 256;
+ fb->fb_type.fb_size = ramsize;
+ printf(": %s, %d x %d", nam, fb->fb_type.fb_width,
+ fb->fb_type.fb_height);
+
+#if defined(SUN4)
+ if (CPU_ISSUN4) {
+ struct eeprom *eep = (struct eeprom *)eeprom_va;
+ int constype = (fb->fb_flags & FB_PFOUR) ? EE_CONS_P4OPT :
+ EE_CONS_COLOR;
+ /*
+ * Assume this is the console if there's no eeprom info
+ * to be found.
+ */
+ if (eep == NULL || eep->eeConsole == constype)
+ isconsole = (fbconstty != NULL);
+ else
+ isconsole = 0;
+ }
+#endif
+
+#if defined(SUN4C) || defined(SUN4M)
+ if (CPU_ISSUN4COR4M)
+ isconsole = node == fbnode && fbconstty != NULL;
+#endif
sc->sc_fhcrev = (*sc->sc_fhc >> FHC_REV_SHIFT) &
(FHC_REV_MASK >> FHC_REV_SHIFT);
@@ -273,19 +327,19 @@ cgsixattach(parent, self, args)
if (isconsole) {
printf(" (console)\n");
#ifdef RASTERCONSOLE
- sc->sc_fb.fb_pixels = (caddr_t)mapiodev(ca->ca_ra.ra_reg,
- O(cg6_ram[0]), ramsize, ca->ca_bustype);
+ sc->sc_fb.fb_pixels = (caddr_t)
+ mapiodev(ca->ca_ra.ra_reg, O(cg6_ram[0]),
+ ramsize, ca->ca_bustype);
fbrcons_init(&sc->sc_fb);
#endif
} else
printf("\n");
#if defined(SUN4C) || defined(SUN4M)
- if (ca->ca_bustype == BUS_SBUS)
+ if (sbus)
sbus_establish(&sc->sc_sd, &sc->sc_dev);
-#endif /* SUN4C || SUN4M */
- if ((node == fbnode && cputyp != CPU_SUN4) ||
- (isconsole && cputyp == CPU_SUN4))
- fb_attach(&sc->sc_fb);
+#endif
+ if (CPU_ISSUN4 || (node == fbnode))
+ fb_attach(&sc->sc_fb, isconsole);
}
int
@@ -296,7 +350,7 @@ cgsixopen(dev, flags, mode, p)
{
int unit = minor(dev);
- if (unit >= cgsixcd.cd_ndevs || cgsixcd.cd_devs[unit] == NULL)
+ if (unit >= cgsix_cd.cd_ndevs || cgsix_cd.cd_devs[unit] == NULL)
return (ENXIO);
return (0);
}
@@ -307,7 +361,7 @@ cgsixclose(dev, flags, mode, p)
int flags, mode;
struct proc *p;
{
- struct cgsix_softc *sc = cgsixcd.cd_devs[minor(dev)];
+ struct cgsix_softc *sc = cgsix_cd.cd_devs[minor(dev)];
cg6_reset(sc);
return (0);
@@ -321,9 +375,9 @@ cgsixioctl(dev, cmd, data, flags, p)
int flags;
struct proc *p;
{
- register struct cgsix_softc *sc = cgsixcd.cd_devs[minor(dev)];
+ register struct cgsix_softc *sc = cgsix_cd.cd_devs[minor(dev)];
u_int count;
- int i, v, error;
+ int v, error;
union cursor_cmap tcm;
switch (cmd) {
@@ -480,7 +534,7 @@ cgsixioctl(dev, cmd, data, flags, p)
default:
#ifdef DEBUG
- log(LOG_NOTICE, "cgsixioctl(%x) (%s[%d])\n", cmd,
+ log(LOG_NOTICE, "cgsixioctl(%lx) (%s[%d])\n", cmd,
p->p_comm, p->p_pid);
#endif
return (ENOTTY);
@@ -663,7 +717,7 @@ cgsixmmap(dev, off, prot)
dev_t dev;
int off, prot;
{
- register struct cgsix_softc *sc = cgsixcd.cd_devs[minor(dev)];
+ register struct cgsix_softc *sc = cgsix_cd.cd_devs[minor(dev)];
register struct mmo *mo;
register u_int u, sz;
#define O(memb) ((u_int)(&((struct cg6_layout *)0)->memb))
@@ -698,7 +752,7 @@ cgsixmmap(dev, off, prot)
sz = mo->mo_size ? mo->mo_size : sc->sc_fb.fb_type.fb_size;
if (u < sz)
return (REG2PHYS(&sc->sc_physadr, u + mo->mo_physoff,
- sc->sc_bustype) | PMAP_NC);
+ sc->sc_bustype) | PMAP_NC);
}
#ifdef DEBUG
{
diff --git a/sys/arch/sparc/dev/cgsixreg.h b/sys/arch/sparc/dev/cgsixreg.h
index 15a7d603f9e..d9b90977c85 100644
--- a/sys/arch/sparc/dev/cgsixreg.h
+++ b/sys/arch/sparc/dev/cgsixreg.h
@@ -1,4 +1,4 @@
-/* $NetBSD: cgsixreg.h,v 1.3 1994/11/20 20:52:00 deraadt Exp $ */
+/* $NetBSD: cgsixreg.h,v 1.4 1996/02/27 22:09:31 thorpej Exp $ */
/*
* Copyright (c) 1993
@@ -71,6 +71,9 @@
* we have only seen rev 1 & 2.
*/
+/* offsets */
+#define CGSIX_FHC_OFFSET 0x300000
+
/* bits in FHC register */
#define FHC_FBID_MASK 0xff000000 /* bits 24..31 are frame buffer ID */
#define FHC_FBID_SHIFT 24
diff --git a/sys/arch/sparc/dev/cgthree.c b/sys/arch/sparc/dev/cgthree.c
index 06c62b15e91..1599048c61d 100644
--- a/sys/arch/sparc/dev/cgthree.c
+++ b/sys/arch/sparc/dev/cgthree.c
@@ -1,4 +1,4 @@
-/* $NetBSD: cgthree.c,v 1.16 1995/10/08 01:39:18 pk Exp $ */
+/* $NetBSD: cgthree.c,v 1.27 1996/04/01 17:30:03 christos Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -53,12 +53,14 @@
*/
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/buf.h>
#include <sys/device.h>
#include <sys/ioctl.h>
#include <sys/malloc.h>
#include <sys/mman.h>
#include <sys/tty.h>
+#include <sys/conf.h>
#include <vm/vm.h>
@@ -66,6 +68,8 @@
#include <machine/autoconf.h>
#include <machine/pmap.h>
#include <machine/fbvar.h>
+#include <machine/cpu.h>
+#include <machine/conf.h>
#include <sparc/dev/btreg.h>
#include <sparc/dev/btvar.h>
@@ -77,10 +81,9 @@ struct cgthree_softc {
struct device sc_dev; /* base device */
struct sbusdev sc_sd; /* sbus device */
struct fbdevice sc_fb; /* frame buffer device */
- volatile struct bt_regs *sc_bt; /* Brooktree registers */
- struct rom_reg sc_phys; /* display RAM (phys addr) */
+ struct rom_reg sc_phys; /* phys address description */
+ volatile struct fbcontrol *sc_fbc; /* Brooktree registers */
int sc_bustype; /* type of bus we live on */
- int sc_blanked; /* true if blanked */
union bt_cmap sc_cmap; /* Brooktree color map */
};
@@ -93,9 +96,12 @@ int cgthreeioctl __P((dev_t, u_long, caddr_t, int, struct proc *));
int cgthreemmap __P((dev_t, int, int));
static void cgthreeunblank(struct device *);
-struct cfdriver cgthreecd = {
- NULL, "cgthree", cgthreematch, cgthreeattach,
- DV_DULL, sizeof(struct cgthree_softc)
+struct cfattach cgthree_ca = {
+ sizeof(struct cgthree_softc), cgthreematch, cgthreeattach
+};
+
+struct cfdriver cgthree_cd = {
+ NULL, "cgthree", DV_DULL
};
/* frame buffer generic driver */
@@ -107,6 +113,8 @@ extern int fbnode;
extern struct tty *fbconstty;
static void cgthreeloadcmap __P((struct cgthree_softc *, int, int));
+static void cgthree_set_video __P((struct cgthree_softc *, int));
+static int cgthree_get_video __P((struct cgthree_softc *));
/*
* Match a cgthree.
@@ -120,6 +128,11 @@ cgthreematch(parent, vcf, aux)
struct confargs *ca = aux;
struct romaux *ra = &ca->ca_ra;
+ /*
+ * Mask out invalid flags from the user.
+ */
+ cf->cf_flags &= FB_USERMASK;
+
if (strcmp(cf->cf_driver->cd_name, ra->ra_name))
return (0);
if (ca->ca_bustype == BUS_SBUS)
@@ -138,32 +151,36 @@ cgthreeattach(parent, self, args)
{
register struct cgthree_softc *sc = (struct cgthree_softc *)self;
register struct confargs *ca = args;
- register int node = ca->ca_ra.ra_node, ramsize, i;
+ register int node = 0, ramsize, i;
register volatile struct bt_regs *bt;
- register struct cgthree_all *p;
int isconsole;
- char *nam;
+ int sbus = 1;
+ char *nam = NULL;
sc->sc_fb.fb_driver = &cgthreefbdriver;
sc->sc_fb.fb_device = &sc->sc_dev;
+ sc->sc_fb.fb_flags = sc->sc_dev.dv_cfdata->cf_flags;
/*
* The defaults below match my screen, but are not guaranteed
* to be correct as defaults go...
*/
sc->sc_fb.fb_type.fb_type = FBTYPE_SUN3COLOR;
switch (ca->ca_bustype) {
-#if defined(SUN4M)
case BUS_OBIO:
- if (cputyp == CPU_SUN4M) /* 4m has framebuffer on obio */
+ if (CPU_ISSUN4M) { /* 4m has framebuffer on obio */
+ sbus = 0;
+ node = ca->ca_ra.ra_node;
nam = getpropstring(node, "model");
- break;
-#endif
-
+ break;
+ }
case BUS_VME32:
- node = 0;
+ case BUS_VME16:
+ sbus = node = 0;
nam = "cgthree";
break;
+
case BUS_SBUS:
+ node = ca->ca_ra.ra_node;
nam = getpropstring(node, "model");
break;
}
@@ -173,7 +190,7 @@ cgthreeattach(parent, self, args)
1152, 900, node, ca->ca_bustype);
ramsize = roundup(sc->sc_fb.fb_type.fb_height * sc->sc_fb.fb_linebytes,
- NBPG);
+ NBPG);
sc->sc_fb.fb_type.fb_cmsize = 256;
sc->sc_fb.fb_type.fb_size = ramsize;
printf(": %s, %d x %d", nam,
@@ -186,27 +203,27 @@ cgthreeattach(parent, self, args)
* going to print characters via rconsole.
*/
isconsole = node == fbnode && fbconstty != NULL;
- p = (struct cgthree_all *)ca->ca_ra.ra_paddr;
if ((sc->sc_fb.fb_pixels = ca->ca_ra.ra_vaddr) == NULL && isconsole) {
/* this probably cannot happen, but what the heck */
sc->sc_fb.fb_pixels = mapiodev(ca->ca_ra.ra_reg, CG3REG_MEM,
- ramsize, ca->ca_bustype);
+ ramsize, ca->ca_bustype);
}
- sc->sc_bt = bt = (volatile struct bt_regs *)mapiodev(ca->ca_ra.ra_reg,
- CG3REG_REG, sizeof(struct bt_regs), ca->ca_bustype);
+ sc->sc_fbc = (volatile struct fbcontrol *)
+ mapiodev(ca->ca_ra.ra_reg, CG3REG_REG,
+ sizeof(struct fbcontrol), ca->ca_bustype);
sc->sc_phys = ca->ca_ra.ra_reg[0];
sc->sc_bustype = ca->ca_bustype;
/* grab initial (current) color map */
+ bt = &sc->sc_fbc->fbc_dac;
bt->bt_addr = 0;
for (i = 0; i < 256 * 3 / 4; i++)
sc->sc_cmap.cm_chip[i] = bt->bt_cmap;
- /* make sure we are not blanked (see cgthreeunblank) */
- bt->bt_addr = 0x06; /* command reg */
- bt->bt_ctrl = 0x73; /* overlay plane */
- bt->bt_addr = 0x04; /* read mask */
- bt->bt_ctrl = 0xff; /* color planes */
+
+ /* make sure we are not blanked */
+ cgthree_set_video(sc, 1);
+ BT_INIT(bt, 0);
if (isconsole) {
printf(" (console)\n");
@@ -215,12 +232,10 @@ cgthreeattach(parent, self, args)
#endif
} else
printf("\n");
-#if defined(SUN4C) || defined(SUN4M)
- if (ca->ca_bustype == BUS_SBUS)
+ if (sbus)
sbus_establish(&sc->sc_sd, &sc->sc_dev);
-#endif /* SUN4C || SUN4M */
if (node == fbnode)
- fb_attach(&sc->sc_fb);
+ fb_attach(&sc->sc_fb, isconsole);
}
int
@@ -231,7 +246,7 @@ cgthreeopen(dev, flags, mode, p)
{
int unit = minor(dev);
- if (unit >= cgthreecd.cd_ndevs || cgthreecd.cd_devs[unit] == NULL)
+ if (unit >= cgthree_cd.cd_ndevs || cgthree_cd.cd_devs[unit] == NULL)
return (ENXIO);
return (0);
}
@@ -254,7 +269,7 @@ cgthreeioctl(dev, cmd, data, flags, p)
int flags;
struct proc *p;
{
- register struct cgthree_softc *sc = cgthreecd.cd_devs[minor(dev)];
+ register struct cgthree_softc *sc = cgthree_cd.cd_devs[minor(dev)];
register struct fbgattr *fba;
int error;
@@ -292,29 +307,11 @@ cgthreeioctl(dev, cmd, data, flags, p)
break;
case FBIOGVIDEO:
- *(int *)data = sc->sc_blanked;
+ *(int *)data = cgthree_get_video(sc);
break;
case FBIOSVIDEO:
- if (*(int *)data)
- cgthreeunblank(&sc->sc_dev);
- else if (!sc->sc_blanked) {
- register volatile struct bt_regs *bt;
-
- bt = sc->sc_bt;
- bt->bt_addr = 0x06; /* command reg */
- bt->bt_ctrl = 0x70; /* overlay plane */
- bt->bt_addr = 0x04; /* read mask */
- bt->bt_ctrl = 0x00; /* color planes */
- /*
- * Set color 0 to black -- note that this overwrites
- * R of color 1.
- */
- bt->bt_addr = 0;
- bt->bt_cmap = 0;
-
- sc->sc_blanked = 1;
- }
+ cgthree_set_video(sc, *(int *)data);
break;
default:
@@ -330,22 +327,28 @@ static void
cgthreeunblank(dev)
struct device *dev;
{
- struct cgthree_softc *sc = (struct cgthree_softc *)dev;
- register volatile struct bt_regs *bt;
- if (sc->sc_blanked) {
- sc->sc_blanked = 0;
- bt = sc->sc_bt;
- /* restore color 0 (and R of color 1) */
- bt->bt_addr = 0;
- bt->bt_cmap = sc->sc_cmap.cm_chip[0];
-
- /* restore read mask */
- bt->bt_addr = 0x06; /* command reg */
- bt->bt_ctrl = 0x73; /* overlay plane */
- bt->bt_addr = 0x04; /* read mask */
- bt->bt_ctrl = 0xff; /* color planes */
- }
+ cgthree_set_video((struct cgthree_softc *)dev, 1);
+}
+
+static void
+cgthree_set_video(sc, enable)
+ struct cgthree_softc *sc;
+ int enable;
+{
+
+ if (enable)
+ sc->sc_fbc->fbc_ctrl |= FBC_VENAB;
+ else
+ sc->sc_fbc->fbc_ctrl &= ~FBC_VENAB;
+}
+
+static int
+cgthree_get_video(sc)
+ struct cgthree_softc *sc;
+{
+
+ return ((sc->sc_fbc->fbc_ctrl & FBC_VENAB) != 0);
}
/*
@@ -362,7 +365,7 @@ cgthreeloadcmap(sc, start, ncolors)
ip = &sc->sc_cmap.cm_chip[BT_D4M3(start)]; /* start/4 * 3 */
count = BT_D4M3(start + ncolors - 1) - BT_D4M3(start) + 3;
- bt = sc->sc_bt;
+ bt = &sc->sc_fbc->fbc_dac;
bt->bt_addr = BT_D4M4(start);
while (--count >= 0)
bt->bt_cmap = *ip++;
@@ -378,7 +381,7 @@ cgthreeloadcmap(sc, start, ncolors)
* map the whole thing, so we repeatedly map the first 256K to the
* first page of the color screen. If someone tries to use the overlay
* and enable regions, they will get a surprise....
- *
+ *
* As well, mapping at an offset of 0x04000000 causes the cg3 to be
* mapped in flat mode without the cg4 emulation.
*/
@@ -387,7 +390,7 @@ cgthreemmap(dev, off, prot)
dev_t dev;
int off, prot;
{
- register struct cgthree_softc *sc = cgthreecd.cd_devs[minor(dev)];
+ register struct cgthree_softc *sc = cgthree_cd.cd_devs[minor(dev)];
#define START (128*1024 + 128*1024)
#define NOOVERLAY (0x04000000)
diff --git a/sys/arch/sparc/dev/cgthreereg.h b/sys/arch/sparc/dev/cgthreereg.h
index 0b8356e4126..073cc52e66c 100644
--- a/sys/arch/sparc/dev/cgthreereg.h
+++ b/sys/arch/sparc/dev/cgthreereg.h
@@ -1,4 +1,4 @@
-/* $NetBSD: cgthreereg.h,v 1.4 1994/11/20 20:52:03 deraadt Exp $ */
+/* $NetBSD: cgthreereg.h,v 1.5 1996/02/27 00:14:17 pk Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -58,7 +58,7 @@
struct cgthree_all {
long ba_id; /* ID = 0xfe010104 on my IPC */
char ba_xxx0[0x400000-4];
- struct bt_regs ba_btreg; /* Brooktree registers */
- char ba_xxx1[0x400000-sizeof(struct bt_regs)];
+ struct fbcontrol ba_fbc; /* Brooktree regs (+ misc control) */
+ char ba_xxx1[0x400000-sizeof(struct fbcontrol)];
char ba_ram[4096]; /* actually larger */
};
diff --git a/sys/arch/sparc/dev/cgtwo.c b/sys/arch/sparc/dev/cgtwo.c
index 17be2ccbc12..243c7cde111 100644
--- a/sys/arch/sparc/dev/cgtwo.c
+++ b/sys/arch/sparc/dev/cgtwo.c
@@ -1,4 +1,4 @@
-/* $NetBSD: cgtwo.c,v 1.5 1995/10/08 01:39:15 pk Exp $ */
+/* $NetBSD: cgtwo.c,v 1.16 1996/05/18 12:19:14 mrg Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -53,12 +53,14 @@
*/
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/buf.h>
#include <sys/device.h>
#include <sys/ioctl.h>
#include <sys/malloc.h>
#include <sys/mman.h>
#include <sys/tty.h>
+#include <sys/conf.h>
#include <vm/vm.h>
@@ -66,10 +68,13 @@
#include <machine/autoconf.h>
#include <machine/pmap.h>
#include <machine/fbvar.h>
+#if defined(SUN4)
#include <machine/eeprom.h>
-
+#endif
+#include <machine/conf.h>
#include <machine/cgtworeg.h>
+
/* per-display variables */
struct cgtwo_softc {
struct device sc_dev; /* base device */
@@ -91,10 +96,15 @@ int cgtwoclose __P((dev_t, int, int, struct proc *));
int cgtwoioctl __P((dev_t, u_long, caddr_t, int, struct proc *));
int cgtwommap __P((dev_t, int, int));
static void cgtwounblank __P((struct device *));
+int cgtwogetcmap __P((struct cgtwo_softc *, struct fbcmap *));
+int cgtwoputcmap __P((struct cgtwo_softc *, struct fbcmap *));
-struct cfdriver cgtwocd = {
- NULL, "cgtwo", cgtwomatch, cgtwoattach,
- DV_DULL, sizeof(struct cgtwo_softc)
+struct cfattach cgtwo_ca = {
+ sizeof(struct cgtwo_softc), cgtwomatch, cgtwoattach
+};
+
+struct cfdriver cgtwo_cd = {
+ NULL, "cgtwo", DV_DULL
};
/* frame buffer generic driver */
@@ -105,8 +115,6 @@ static struct fbdriver cgtwofbdriver = {
extern int fbnode;
extern struct tty *fbconstty;
-static void cgtwoloadcmap __P((struct cgtwo_softc *, int, int));
-
/*
* Match a cgtwo.
*/
@@ -118,18 +126,32 @@ cgtwomatch(parent, vcf, aux)
struct cfdata *cf = vcf;
struct confargs *ca = aux;
struct romaux *ra = &ca->ca_ra;
- int probe;
+#if defined(SUN4)
caddr_t tmp;
+#endif
+
+ /*
+ * Mask out invalid flags from the user.
+ */
+ cf->cf_flags &= FB_USERMASK;
+
+ if (ca->ca_bustype != BUS_VME16)
+ return (0);
if (strcmp(cf->cf_driver->cd_name, ra->ra_name))
return (0);
+#if defined(SUN4)
+ if (!CPU_ISSUN4 || cf->cf_unit != 0)
+ return (0);
+
/* XXX - Must do our own mapping at CG2_CTLREG_OFF */
bus_untmp();
tmp = (caddr_t)bus_tmp(ra->ra_paddr + CG2_CTLREG_OFF, ca->ca_bustype);
if (probeget(tmp, 2) != -1)
return 1;
- return (0);
+#endif
+ return 0;
}
/*
@@ -142,15 +164,25 @@ cgtwoattach(parent, self, args)
{
register struct cgtwo_softc *sc = (struct cgtwo_softc *)self;
register struct confargs *ca = args;
- register int node = 0, i;
- register struct cgtwo_all *p;
- struct eeprom *eep = (struct eeprom *)eeprom_va;
- int isconsole;
- char *nam;
+ register int node = 0;
+ int isconsole = 0;
+ char *nam = NULL;
sc->sc_fb.fb_driver = &cgtwofbdriver;
sc->sc_fb.fb_device = &sc->sc_dev;
sc->sc_fb.fb_type.fb_type = FBTYPE_SUN2COLOR;
+ sc->sc_fb.fb_flags = sc->sc_dev.dv_cfdata->cf_flags;
+
+ switch (ca->ca_bustype) {
+ case BUS_VME16:
+ node = 0;
+ nam = "cgtwo";
+ break;
+
+ default:
+ panic("cgtwoattach: impossible bustype");
+ /* NOTREACHED */
+ }
sc->sc_fb.fb_type.fb_depth = 8;
fb_setsize(&sc->sc_fb, sc->sc_fb.fb_type.fb_depth,
@@ -158,8 +190,8 @@ cgtwoattach(parent, self, args)
sc->sc_fb.fb_type.fb_cmsize = 256;
sc->sc_fb.fb_type.fb_size = roundup(CG2_MAPPED_SIZE, NBPG);
- printf(": cgtwo, %d x %d", sc->sc_fb.fb_type.fb_width,
- sc->sc_fb.fb_type.fb_height);
+ printf(": %s, %d x %d", nam,
+ sc->sc_fb.fb_type.fb_width, sc->sc_fb.fb_type.fb_height);
/*
* When the ROM has mapped in a cgtwo display, the address
@@ -167,30 +199,41 @@ cgtwoattach(parent, self, args)
* registers ourselves. We only need the video RAM if we are
* going to print characters via rconsole.
*/
- if (eep == NULL || eep->ee_diag.eed_console == EED_CONS_COLOR)
- isconsole = (fbconstty != NULL);
- else
- isconsole = 0;
-
+#if defined(SUN4)
+ if (CPU_ISSUN4) {
+ struct eeprom *eep = (struct eeprom *)eeprom_va;
+ /*
+ * Assume this is the console if there's no eeprom info
+ * to be found.
+ */
+ if (eep == NULL || eep->eeConsole == EE_CONS_COLOR)
+ isconsole = (fbconstty != NULL);
+ else
+ isconsole = 0;
+ }
+#endif
sc->sc_phys = ca->ca_ra.ra_reg[0];
sc->sc_bustype = ca->ca_bustype;
if ((sc->sc_fb.fb_pixels = ca->ca_ra.ra_vaddr) == NULL && isconsole) {
/* this probably cannot happen, but what the heck */
sc->sc_fb.fb_pixels = mapiodev(ca->ca_ra.ra_reg, CG2_PIXMAP_OFF,
- CG2_PIXMAP_SIZE, ca->ca_bustype);
+ CG2_PIXMAP_SIZE,
+ PMAP_VME32/*ca->ca_bustype*/);
}
#ifndef offsetof
#define offsetof(type, member) ((size_t)(&((type *)0)->member))
#endif
- sc->sc_reg = (volatile struct cg2statusreg *)mapiodev(ca->ca_ra.ra_reg,
- CG2_ROPMEM_OFF + offsetof(struct cg2fb, status.reg),
- sizeof(struct cg2statusreg), ca->ca_bustype);
+ sc->sc_reg = (volatile struct cg2statusreg *)
+ mapiodev(ca->ca_ra.ra_reg,
+ CG2_ROPMEM_OFF + offsetof(struct cg2fb, status.reg),
+ sizeof(struct cg2statusreg), ca->ca_bustype);
- sc->sc_cmap = (volatile u_short *)mapiodev(ca->ca_ra.ra_reg,
- CG2_ROPMEM_OFF + offsetof(struct cg2fb, redmap[0]),
- 3 * CG2_CMSIZE, ca->ca_bustype);
+ sc->sc_cmap = (volatile u_short *)
+ mapiodev(ca->ca_ra.ra_reg,
+ CG2_ROPMEM_OFF + offsetof(struct cg2fb, redmap[0]),
+ 3 * CG2_CMSIZE, ca->ca_bustype);
if (isconsole) {
printf(" (console)\n");
@@ -199,8 +242,9 @@ cgtwoattach(parent, self, args)
#endif
} else
printf("\n");
- if (isconsole)
- fb_attach(&sc->sc_fb);
+
+ if (node == fbnode || CPU_ISSUN4)
+ fb_attach(&sc->sc_fb, isconsole);
}
int
@@ -211,7 +255,7 @@ cgtwoopen(dev, flags, mode, p)
{
int unit = minor(dev);
- if (unit >= cgtwocd.cd_ndevs || cgtwocd.cd_devs[unit] == NULL)
+ if (unit >= cgtwo_cd.cd_ndevs || cgtwo_cd.cd_devs[unit] == NULL)
return (ENXIO);
return (0);
}
@@ -234,9 +278,8 @@ cgtwoioctl(dev, cmd, data, flags, p)
int flags;
struct proc *p;
{
- register struct cgtwo_softc *sc = cgtwocd.cd_devs[minor(dev)];
+ register struct cgtwo_softc *sc = cgtwo_cd.cd_devs[minor(dev)];
register struct fbgattr *fba;
- int error;
switch (cmd) {
@@ -257,10 +300,10 @@ cgtwoioctl(dev, cmd, data, flags, p)
break;
case FBIOGETCMAP:
- return cgtwogetcmap(sc, data);
+ return cgtwogetcmap(sc, (struct fbcmap *) data);
case FBIOPUTCMAP:
- return cgtwoputcmap(sc, data);
+ return cgtwoputcmap(sc, (struct fbcmap *) data);
case FBIOGVIDEO:
*(int *)data = sc->sc_reg->video_enab;
@@ -284,7 +327,6 @@ cgtwounblank(dev)
struct device *dev;
{
struct cgtwo_softc *sc = (struct cgtwo_softc *)dev;
-
sc->sc_reg->video_enab = 1;
}
@@ -381,7 +423,7 @@ cgtwommap(dev, off, prot)
dev_t dev;
int off, prot;
{
- register struct cgtwo_softc *sc = cgtwocd.cd_devs[minor(dev)];
+ register struct cgtwo_softc *sc = cgtwo_cd.cd_devs[minor(dev)];
if (off & PGOFSET)
panic("cgtwommap");
@@ -389,5 +431,5 @@ cgtwommap(dev, off, prot)
if ((unsigned)off >= sc->sc_fb.fb_type.fb_size)
return (-1);
- return (REG2PHYS(&sc->sc_phys, off, PMAP_VME32) | PMAP_NC);
+ return (REG2PHYS(&sc->sc_phys, off, PMAP_VME32/*sc->sc_bustype*/) | PMAP_NC);
}
diff --git a/sys/arch/sparc/dev/cons.c b/sys/arch/sparc/dev/cons.c
index d4e096bf88d..08df311233b 100644
--- a/sys/arch/sparc/dev/cons.c
+++ b/sys/arch/sparc/dev/cons.c
@@ -1,4 +1,4 @@
-/* $NetBSD: cons.c,v 1.15 1995/07/13 12:02:19 pk Exp $ */
+/* $NetBSD: cons.c,v 1.23.4.1 1996/06/02 09:07:53 mrg Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -49,7 +49,6 @@
*/
#include <sys/param.h>
-#include <sys/types.h>
#include <sys/proc.h>
#include <sys/systm.h>
#include <sys/ioctl.h>
@@ -57,11 +56,18 @@
#include <sys/file.h>
#include <sys/conf.h>
+#include <dev/cons.h>
+
#include <machine/bsd_openprom.h>
+#include <machine/eeprom.h>
#include <machine/psl.h>
-#ifdef SUN4
+#include <machine/cpu.h>
+#include <machine/kbd.h>
+#if defined(SUN4)
#include <machine/oldmon.h>
#endif
+#include <machine/autoconf.h>
+#include <machine/conf.h>
#include "zs.h"
@@ -71,25 +77,25 @@ int rom_console_input; /* when set, hardclock calls cnrom() */
int cons_ocount; /* output byte count */
-extern struct promvec *promvec;
-
/*
* The output driver may munge the minor number in cons.t_dev.
*/
struct tty cons; /* rom console tty device */
-static void (*fcnstop) __P((struct tty *, int));
+static int (*fcnstop) __P((struct tty *, int));
static void cnstart __P((struct tty *));
-void cnstop __P((struct tty *, int));
+int cnstop __P((struct tty *, int));
static void cnfbstart __P((struct tty *));
-static void cnfbstop __P((struct tty *, int));
+static int cnfbstop __P((struct tty *, int));
static void cnfbdma __P((void *));
+static struct tty *xxcntty __P((dev_t));
extern char char_type[];
-/*XXX*/static struct tty *
-cntty()
+/*XXX*/
+static struct tty *
+xxcntty(dev_t dev)
{
return &cons;
}
@@ -99,16 +105,121 @@ consinit()
{
register struct tty *tp = &cons;
register int in, out;
- void zsconsole();
-/*XXX*/ cdevsw[0].d_tty = cntty;
+/*XXX*/ cdevsw[0].d_tty = xxcntty;
tp->t_dev = makedev(0, 0); /* /dev/console */
tp->t_ispeed = tp->t_ospeed = TTYDEF_SPEED;
tp->t_param = (int (*)(struct tty *, struct termios *))nullop;
- in = *promvec->pv_stdin;
- out = *promvec->pv_stdout;
- switch (in) {
+ if (promvec->pv_romvec_vers > 2) {
+ /* We need to probe the PROM device tree */
+ register int node,fd;
+ char buffer[128];
+ register struct nodeops *no;
+ register struct v2devops *op;
+ register char *cp;
+ extern int fbnode;
+
+ in = out = -1;
+ no = promvec->pv_nodeops;
+ op = &promvec->pv_v2devops;
+
+ node = findroot();
+ if (no->no_proplen(node, "stdin-path") >= sizeof(buffer)) {
+ printf("consinit: increase buffer size and recompile\n");
+ goto setup_output;
+ }
+ /* XXX: fix above */
+
+ no->no_getprop(node, "stdin-path",buffer);
+
+ /*
+ * Open an "instance" of this device.
+ * You'd think it would be appropriate to call v2_close()
+ * on the handle when we're done with it. But that seems
+ * to cause the device to shut down somehow; for the moment,
+ * we simply leave it open...
+ */
+ if ((fd = op->v2_open(buffer)) == 0 ||
+ (node = op->v2_fd_phandle(fd)) == 0) {
+ printf("consinit: bogus stdin path %s.\n",buffer);
+ goto setup_output;
+ }
+ if (no->no_proplen(node,"keyboard") >= 0) {
+ in = PROMDEV_KBD;
+ goto setup_output;
+ }
+ if (strcmp(getpropstring(node,"device_type"),"serial") != 0) {
+ /* not a serial, not keyboard. what is it?!? */
+ in = -1;
+ goto setup_output;
+ }
+ /*
+ * At this point we assume the device path is in the form
+ * ....device@x,y:a for ttya and ...device@x,y:b for ttyb.
+ * If it isn't, we defer to the ROM
+ */
+ cp = buffer;
+ while (*cp)
+ cp++;
+ cp -= 2;
+#ifdef DEBUG
+ if (cp < buffer)
+ panic("consinit: bad stdin path %s",buffer);
+#endif
+ /* XXX: only allows tty's a->z, assumes PROMDEV_TTYx contig */
+ if (cp[0]==':' && cp[1] >= 'a' && cp[1] <= 'z')
+ in = PROMDEV_TTYA + (cp[1] - 'a');
+ /* else use rom */
+setup_output:
+ node = findroot();
+ if (no->no_proplen(node, "stdout-path") >= sizeof(buffer)) {
+ printf("consinit: increase buffer size and recompile\n");
+ goto setup_console;
+ }
+ /* XXX: fix above */
+
+ no->no_getprop(node, "stdout-path", buffer);
+
+ if ((fd = op->v2_open(buffer)) == 0 ||
+ (node = op->v2_fd_phandle(fd)) == 0) {
+ printf("consinit: bogus stdout path %s.\n",buffer);
+ goto setup_output;
+ }
+ if (strcmp(getpropstring(node,"device_type"),"display") == 0) {
+ /* frame buffer output */
+ out = PROMDEV_SCREEN;
+ fbnode = node;
+ } else if (strcmp(getpropstring(node,"device_type"), "serial")
+ != 0) {
+ /* not screen, not serial. Whatzit? */
+ out = -1;
+ } else { /* serial console. which? */
+ /*
+ * At this point we assume the device path is in the
+ * form:
+ * ....device@x,y:a for ttya, etc.
+ * If it isn't, we defer to the ROM
+ */
+ cp = buffer;
+ while (*cp)
+ cp++;
+ cp -= 2;
+#ifdef DEBUG
+ if (cp < buffer)
+ panic("consinit: bad stdout path %s",buffer);
+#endif
+ /* XXX: only allows tty's a->z, assumes PROMDEV_TTYx contig */
+ if (cp[0]==':' && cp[1] >= 'a' && cp[1] <= 'z')
+ out = PROMDEV_TTYA + (cp[1] - 'a');
+ else out = -1;
+ }
+ } else {
+ in = *promvec->pv_stdin;
+ out = *promvec->pv_stdout;
+ }
+setup_console:
+ switch (in) {
#if NZS > 0
case PROMDEV_TTYA:
zsconsole(tp, 0, 0, NULL);
@@ -152,12 +263,13 @@ consinit()
default:
printf("unknown console output sink %d; using rom\n", out);
tp->t_oproc = cnstart;
- fcnstop = (void (*)(struct tty *, int))nullop;
+ fcnstop = (int (*)(struct tty *, int))nullop;
break;
}
}
/* ARGSUSED */
+int
cnopen(dev, flag, mode, p)
dev_t dev;
int flag, mode;
@@ -165,12 +277,42 @@ cnopen(dev, flag, mode, p)
{
register struct tty *tp = &cons;
static int firstopen = 1;
+ static int rows = 0, cols = 0;
- if(firstopen) {
+ if (firstopen) {
clalloc(&tp->t_rawq, 1024, 1);
clalloc(&tp->t_canq, 1024, 1);
/* output queue doesn't need quoting */
clalloc(&tp->t_outq, 1024, 0);
+ tty_attach(tp);
+ /*
+ * get the console struct winsize.
+ */
+ if (CPU_ISSUN4COR4M) {
+ int i;
+ char *prop;
+
+ if ((prop = getpropstring(optionsnode, "screen-#rows"))) {
+ i = 0;
+ while (*prop != '\0')
+ i = i * 10 + *prop++ - '0';
+ rows = (unsigned short)i;
+ }
+ if ((prop = getpropstring(optionsnode, "screen-#columns"))) {
+ i = 0;
+ while (*prop != '\0')
+ i = i * 10 + *prop++ - '0';
+ cols = (unsigned short)i;
+ }
+ }
+ if (CPU_ISSUN4) {
+ struct eeprom *ep = (struct eeprom *)eeprom_va;
+
+ if (ep) {
+ rows = (u_short)ep->eeTtyRows;
+ cols = (u_short)ep->eeTtyCols;
+ }
+ }
firstopen = 0;
}
@@ -186,12 +328,15 @@ cnopen(dev, flag, mode, p)
tp->t_state = TS_ISOPEN | TS_CARR_ON;
(void)(*tp->t_param)(tp, &tp->t_termios);
ttsetwater(tp);
+ tp->t_winsize.ws_row = rows;
+ tp->t_winsize.ws_col = cols;
} else if (tp->t_state & TS_XCLUDE && p->p_ucred->cr_uid != 0)
return (EBUSY);
return ((*linesw[tp->t_line].l_open)(dev, tp));
}
/* ARGSUSED */
+int
cnclose(dev, flag, mode, p)
dev_t dev;
int flag, mode;
@@ -205,6 +350,7 @@ cnclose(dev, flag, mode, p)
}
/* ARGSUSED */
+int
cnread(dev, uio, flag)
dev_t dev;
struct uio *uio;
@@ -216,19 +362,21 @@ cnread(dev, uio, flag)
}
/* ARGSUSED */
+int
cnwrite(dev, uio, flag)
dev_t dev;
struct uio *uio;
int flag;
{
register struct tty *tp;
-
+
if ((tp = constty) == NULL ||
(tp->t_state & (TS_CARR_ON|TS_ISOPEN)) != (TS_CARR_ON|TS_ISOPEN))
tp = &cons;
return ((*linesw[tp->t_line].l_write)(tp, uio, flag));
}
+int
cnioctl(dev, cmd, data, flag, p)
dev_t dev;
u_long cmd;
@@ -258,6 +406,7 @@ cnioctl(dev, cmd, data, flag, p)
return (ENOTTY);
}
+int
cnselect(dev, which, p)
dev_t dev;
int which;
@@ -281,9 +430,9 @@ cnstart(tp)
register int c, s;
register union {
void (*v1)__P((int));
- void (*v3)__P((int, u_char *, int));
+ int (*v3)__P((int, void *, int));
} putc;
- register int fd, v;
+ register int fd = 0, v;
s = spltty();
if (tp->t_state & (TS_TIMEOUT | TS_TTSTOP)) {
@@ -291,7 +440,7 @@ cnstart(tp)
return;
}
if ((v = promvec->pv_romvec_vers) > 2) {
- putc.v3 = (void (*))promvec->pv_v2devops.v2_write;
+ putc.v3 = promvec->pv_v2devops.v2_write;
fd = *promvec->pv_v2bootargs.v2_fd1;
} else
putc.v1 = promvec->pv_putchar;
@@ -304,7 +453,7 @@ cnstart(tp)
*/
(void) splhigh();
if (v > 2) {
- u_char c0 = c & 0177;
+ unsigned char c0 = c & 0177;
(*putc.v3)(fd, &c0, 1);
} else
(*putc.v1)(c & 0177);
@@ -318,12 +467,13 @@ cnstart(tp)
splx(s);
}
-void
+int
cnstop(tp, flag)
register struct tty *tp;
int flag;
{
- (*fcnstop)(tp, flag);
+ (void)(*fcnstop)(tp, flag);
+ return 0;
}
/*
@@ -367,7 +517,7 @@ cnfbstart(tp)
/*
* Stop frame buffer output: just assert TS_FLUSH if necessary.
*/
-static void
+static int
cnfbstop(tp, flag)
register struct tty *tp;
int flag;
@@ -377,6 +527,7 @@ cnfbstop(tp, flag)
if ((tp->t_state & (TS_BUSY | TS_TTSTOP)) == TS_BUSY)
tp->t_state |= TS_FLUSH;
splx(s);
+ return 0;
}
/*
@@ -387,7 +538,7 @@ cnfbdma(tpaddr)
void *tpaddr;
{
register struct tty *tp = tpaddr;
- register char *p, *q;
+ register unsigned char *p, *q;
register int n, c, s;
s = spltty(); /* paranoid */
@@ -406,7 +557,7 @@ cnfbdma(tpaddr)
(*promvec->pv_v2devops.v2_write)
(*promvec->pv_v2bootargs.v2_fd1, p, n);
} else
- (*promvec->pv_putstr)(p, n);
+ (*promvec->pv_putstr)((char *)p, n);
ndflush(&tp->t_outq, n);
}
if (tp->t_line)
@@ -423,6 +574,7 @@ cnfbdma(tpaddr)
volatile int cn_rxc = -1; /* XXX receive `silo' */
/* called from hardclock, which is above spltty, so no tty calls! */
+int
cnrom()
{
register int c;
@@ -442,6 +594,7 @@ cnrom()
}
/* pseudo console software interrupt scheduled when cnrom() returns 1 */
+void
cnrint()
{
register struct tty *tp;
@@ -466,6 +619,7 @@ cnrint()
(*linesw[tp->t_line].l_rint)(c, tp);
}
+int
cngetc()
{
register int s, c;
@@ -473,30 +627,30 @@ cngetc()
if (promvec->pv_romvec_vers > 2) {
register int n = 0;
unsigned char c0;
+ s = splhigh();
while (n <= 0) {
- s = splhigh();
n = (*promvec->pv_v2devops.v2_read)
(*promvec->pv_v2bootargs.v2_fd0, &c0, 1);
- splx(s);
}
+ splx(s);
c = c0;
} else {
-#ifdef SUN4
+#if defined(SUN4)
/* SUN4 PROM: must turn off echo to avoid double char echo */
extern struct om_vector *oldpvec;
- int saveecho;
+ int saveecho = 0;
#endif
s = splhigh();
-#ifdef SUN4
- if (cputyp == CPU_SUN4) {
+#if defined(SUN4)
+ if (CPU_ISSUN4) {
saveecho = *(oldpvec->echo);
*(oldpvec->echo) = 0;
}
#endif
c = (*promvec->pv_getchar)();
-#ifdef SUN4
- if (cputyp == CPU_SUN4)
+#if defined(SUN4)
+ if (CPU_ISSUN4)
*(oldpvec->echo) = saveecho;
#endif
splx(s);
@@ -506,6 +660,7 @@ cngetc()
return (c);
}
+void
cnputc(c)
register int c;
{
@@ -523,6 +678,7 @@ cnputc(c)
splx(s);
}
+void
cnpollc(on)
int on;
{
diff --git a/sys/arch/sparc/dev/dma.c b/sys/arch/sparc/dev/dma.c
index 54be0c4fdf8..ee40392c0fe 100644
--- a/sys/arch/sparc/dev/dma.c
+++ b/sys/arch/sparc/dev/dma.c
@@ -1,35 +1,7 @@
-/* $NetBSD: dma.c,v 1.8 1995/02/01 12:37:21 pk Exp $ */
+/* $NetBSD: dma.c,v 1.28.2.2 1996/07/02 23:46:29 jtc Exp $ */
/*
- * Copyright (c) 1995 Theo de Raadt
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- * 3. All advertising materials mentioning features or use of this software
- * must display the following acknowledgement:
- * This product includes software developed under OpenBSD by
- * Theo de Raadt.
- * 4. The name of the author may not be used to endorse or promote products
- * derived from this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
- * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
- * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
- * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
- * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
- * SUCH DAMAGE.
- *
+ * Copyright (c) 1994 Paul Kranenburg. All rights reserved.
* Copyright (c) 1994 Peter Galbavy. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
@@ -77,46 +49,63 @@
#include <scsi/scsiconf.h>
#include <sparc/dev/sbusvar.h>
-#include <sparc/dev/sbusreg.h>
#include <sparc/dev/dmareg.h>
#include <sparc/dev/dmavar.h>
#include <sparc/dev/espreg.h>
#include <sparc/dev/espvar.h>
-#include <sparc/sparc/cache.h>
-
-/*#define DMA_TEST*/
-
-extern int sbus_print __P((void *, char *));
-
+int dmaprint __P((void *, char *));
void dmaattach __P((struct device *, struct device *, void *));
int dmamatch __P((struct device *, void *, void *));
+void dma_reset __P((struct dma_softc *));
+void dma_enintr __P((struct dma_softc *));
+int dma_isintr __P((struct dma_softc *));
+int espdmaintr __P((struct dma_softc *));
+int ledmaintr __P((struct dma_softc *));
+int dma_setup __P((struct dma_softc *, caddr_t *, size_t *,
+ int, size_t *));
+void dma_go __P((struct dma_softc *));
+
+struct cfattach dma_ca = {
+ sizeof(struct dma_softc), dmamatch, dmaattach
+};
-struct cfdriver dmacd = {
- NULL, "dma", dmamatch, dmaattach,
- DV_DULL, sizeof(struct dma_softc)
+struct cfdriver dma_cd = {
+ NULL, "dma", DV_DULL
};
-struct cfdriver ledmacd = {
- NULL, "ledma", dmamatch, dmaattach,
- DV_DULL, sizeof(struct dma_softc)
+struct cfattach ledma_ca = {
+ sizeof(struct dma_softc), matchbyname, dmaattach
};
-struct cfdriver espdmacd = {
- NULL, "espdma", dmamatch, dmaattach,
- DV_DULL, sizeof(struct dma_softc)
+struct cfdriver ledma_cd = {
+ NULL, "ledma", DV_DULL
};
int
+dmaprint(aux, name)
+ void *aux;
+ char *name;
+{
+ register struct confargs *ca = aux;
+
+ if (name)
+ printf("[%s at %s]", ca->ca_ra.ra_name, name);
+ printf(" slot 0x%x offset 0x%x", ca->ca_slot, ca->ca_offset);
+ return (UNCONF);
+}
+
+int
dmamatch(parent, vcf, aux)
- struct device *parent;
- void *vcf, *aux;
+ struct device *parent;
+ void *vcf, *aux;
{
- struct cfdata *cf = vcf;
+ struct cfdata *cf = vcf;
register struct confargs *ca = aux;
register struct romaux *ra = &ca->ca_ra;
- if (strcmp(cf->cf_driver->cd_name, ra->ra_name))
+ if (strcmp(cf->cf_driver->cd_name, ra->ra_name) &&
+ strcmp("espdma", ra->ra_name))
return (0);
if (ca->ca_bustype == BUS_SBUS)
return (1);
@@ -129,353 +118,408 @@ dmamatch(parent, vcf, aux)
*/
void
dmaattach(parent, self, aux)
- struct device *parent, *self;
- void *aux;
+ struct device *parent, *self;
+ void *aux;
{
register struct confargs *ca = aux;
- struct confargs oca;
- struct dma_softc *sc = (void *)self;
- int node, base, slot;
- char *name;
-
- /* XXX modifying ra_vaddr is bad! */
- if (ca->ca_ra.ra_vaddr == NULL)
- ca->ca_ra.ra_vaddr = mapiodev(ca->ca_ra.ra_reg, 0,
- ca->ca_ra.ra_len, ca->ca_bustype);
- if ((u_long)ca->ca_ra.ra_paddr & PGOFSET)
- (u_long)ca->ca_ra.ra_vaddr |= ((u_long)ca->ca_ra.ra_paddr & PGOFSET);
+ struct dma_softc *sc = (void *)self;
+#if defined(SUN4C) || defined(SUN4M)
+ int node;
+ struct confargs oca;
+ char *name;
+#endif
+
+ if (ca->ca_ra.ra_vaddr == NULL || ca->ca_ra.ra_nvaddrs == 0)
+ ca->ca_ra.ra_vaddr =
+ mapiodev(ca->ca_ra.ra_reg, 0, ca->ca_ra.ra_len,
+ ca->ca_bustype);
+
sc->sc_regs = (struct dma_regs *) ca->ca_ra.ra_vaddr;
/*
- * What happens here is that if the esp driver has not been
- * configured, then this returns a NULL pointer. Then when the
- * esp actually gets configured, it does the opposing test, and
- * if the sc->sc_dma field in it's softc is NULL, then tries to
- * find the matching dma driver.
+ * If we're a ledma, check to see what cable type is currently
+ * active and set the appropriate bit in the ledma csr so that
+ * it gets used. If we didn't netboot, the PROM won't have the
+ * "cable-selection" property; default to TP and then the user
+ * can change it via a "link0" option to ifconfig.
+ */
+ if (strcmp(ca->ca_ra.ra_name, "ledma") == 0) {
+ char *cabletype = getpropstring(ca->ca_ra.ra_node,
+ "cable-selection");
+ if (strcmp(cabletype, "tpe") == 0) {
+ sc->sc_regs->csr |= DE_AUI_TP;
+ } else if (strcmp(cabletype, "aui") == 0) {
+ sc->sc_regs->csr &= ~DE_AUI_TP;
+ } else {
+ /* assume TP if nothing there */
+ sc->sc_regs->csr |= DE_AUI_TP;
+ }
+ delay(20000); /* manual says we need 20ms delay */
+ }
+
+ /*
+ * Get transfer burst size from PROM and plug it into the controller
+ * registers. This is needed on the Sun4m; do others need it too?
+ * XXX
*/
- sc->sc_esp = ((struct esp_softc *)
- getdevunit("esp", sc->sc_dev.dv_unit));
- if (sc->sc_esp)
- sc->sc_esp->sc_dma = sc;
+ if (CPU_ISSUN4M) {
+ sc->sc_burst = getpropint(ca->ca_ra.ra_node,"burst-sizes", -1);
+ if (sc->sc_burst == -1) {
+ /* check parent SBus for burst sizes */
+ if (((struct sbus_softc *)parent)->sc_burst == 0)
+ sc->sc_burst = SBUS_BURST_32 - 1; /* 1->16 */
+ else
+ sc->sc_burst =
+ ((struct sbus_softc *)parent)->sc_burst;
+ }
+ sc->sc_regs->csr &= ~D_BURST_SIZE; /* must clear first */
+ if (sc->sc_burst & SBUS_BURST_32) {
+ sc->sc_regs->csr |= D_BURST_32;
+ } else if (sc->sc_burst & SBUS_BURST_16) {
+ sc->sc_regs->csr |= D_BURST_16;
+ } else if (strcmp(ca->ca_ra.ra_name,"espdma") == 0) {
+ /* only espdma supports non-burst */
+ sc->sc_regs->csr |= D_BURST_0;
+#ifdef DIAGNOSTIC
+ } else {
+ printf(" <unknown burst size 0x%x>", sc->sc_burst);
+#endif
+ }
+ }
printf(": rev ");
sc->sc_rev = sc->sc_regs->csr & D_DEV_ID;
switch (sc->sc_rev) {
- case DMAREV_4300:
- printf("4/300\n");
+ case DMAREV_0:
+ printf("0");
break;
case DMAREV_1:
- printf("1\n");
- break;
- case DMAREV_ESC1:
- printf("ESC1\n");
+ printf("1");
break;
case DMAREV_PLUS:
- printf("1+\n");
+ printf("1+");
break;
case DMAREV_2:
- printf("2\n");
+ printf("2");
break;
default:
- printf("unknown\n");
+ printf("unknown");
+ }
+ printf("\n");
+
+ /* indirect functions */
+ if (sc->sc_dev.dv_cfdata->cf_attach == &dma_ca) {
+ sc->intr = espdmaintr;
+ } else {
+ sc->intr = ledmaintr;
}
+ sc->enintr = dma_enintr;
+ sc->isintr = dma_isintr;
+ sc->reset = dma_reset;
+ sc->setup = dma_setup;
+ sc->go = dma_go;
+
+ sc->sc_node = ca->ca_ra.ra_node;
+ if (CPU_ISSUN4)
+ goto espsearch;
#if defined(SUN4C) || defined(SUN4M)
- if (ca->ca_bustype == BUS_SBUS) {
- sc->sc_node = ca->ca_ra.ra_node;
+ if (ca->ca_bustype == BUS_SBUS)
sbus_establish(&sc->sc_sd, &sc->sc_dev);
+ /* Propagate bootpath */
+ if (ca->ca_ra.ra_bp != NULL &&
+ (strcmp(ca->ca_ra.ra_bp->name, "espdma") == 0 ||
+ strcmp(ca->ca_ra.ra_bp->name, "dma") == 0 ||
+ strcmp(ca->ca_ra.ra_bp->name, "ledma") == 0))
+ oca.ca_ra.ra_bp = ca->ca_ra.ra_bp + 1;
+ else
+ oca.ca_ra.ra_bp = NULL;
+
+ /* search through children */
+ node = firstchild(sc->sc_node);
+ if (node != 0) do {
+ name = getpropstring(node, "name");
+ if (!romprop(&oca.ca_ra, name, node))
+ continue;
+
+ sbus_translate(parent, &oca);
+ oca.ca_bustype = BUS_SBUS;
+ (void) config_found(&sc->sc_dev, (void *)&oca, dmaprint);
+ } while ((node = nextsibling(node)) != 0); else
+#endif /* SUN4C || SUN4M */
+
+ if (strcmp(ca->ca_ra.ra_name, "dma") == 0) {
+espsearch:
/*
- * If the device is in an SBUS slave slot, report
- * it (but we don't care, because the corresponding
- * ESP will also realize the same thing.)
+ * find the ESP by poking around the esp device structures
+ *
+ * What happens here is that if the esp driver has not been
+ * configured, then this returns a NULL pointer. Then when the
+ * esp actually gets configured, it does the opposing test, and
+ * if the sc->sc_dma field in it's softc is NULL, then tries to
+ * find the matching dma driver.
+ *
*/
- (void) sbus_slavecheck(self, ca);
+ sc->sc_esp = (struct esp_softc *)
+ getdevunit("esp", sc->sc_dev.dv_unit);
/*
- * if our name is not "dma", we may have subdevices
- * below us in the device tree (like an esp)
- * XXX: TDR: should we do this even if it is "dma"?
+ * and a back pointer to us, for DMA
*/
- if (strcmp(ca->ca_ra.ra_name, "dma") == 0)
- return;
-
- /* search through children */
- for (node = firstchild(sc->sc_node); node;
- node = nextsibling(node)) {
- name = getpropstring(node, "name");
- if (!romprop(&oca.ca_ra, name, node))
- continue;
-
- /*
- * advance bootpath if it currently points to us
- * XXX There appears to be strangeness in the unit
- * number on at least one espdma system (SS5 says
- * espdma5, but nothing is available to compare
- * against that digit 5...
- */
- if (ca->ca_ra.ra_bp &&
- !strcmp(ca->ca_ra.ra_bp->name, ca->ca_ra.ra_name) &&
- ca->ca_ra.ra_bp->val[1] == (int)ca->ca_ra.ra_paddr)
- oca.ca_ra.ra_bp = ca->ca_ra.ra_bp + 1;
- else
- oca.ca_ra.ra_bp = NULL;
-
- base = (int)oca.ca_ra.ra_paddr;
- if (SBUS_ABS(base)) {
- oca.ca_slot = SBUS_ABS_TO_SLOT(base);
- oca.ca_offset = SBUS_ABS_TO_OFFSET(base);
- } else {
- oca.ca_slot = slot = ca->ca_ra.ra_iospace;
- oca.ca_offset = base;
- oca.ca_ra.ra_paddr = (void *)SBUS_ADDR(slot, base);
- }
- oca.ca_bustype = BUS_SBUS;
- (void) config_found(&sc->sc_dev, (void *)&oca, sbus_print);
- }
-
+ if (sc->sc_esp)
+ sc->sc_esp->sc_dma = sc;
}
-#endif /* SUN4C || SUN4M */
}
void
-dmareset(sc)
- struct dma_softc *sc;
+dma_reset(sc)
+ struct dma_softc *sc;
{
- DMAWAIT_PEND(sc);
+ DMAWAIT(sc);
+ DMA_DRAIN(sc); /* Drain (DMA rev 1) */
+ DMACSR(sc) &= ~D_EN_DMA; /* Stop DMA */
+ DMAWAIT1(sc); /* let things drain */
DMACSR(sc) |= D_RESET; /* reset DMA */
DELAY(200); /* what should this be ? */
+ DMAWAIT1(sc);
DMACSR(sc) &= ~D_RESET; /* de-assert reset line */
+ DMACSR(sc) |= D_INT_EN; /* enable interrupts */
+ if (sc->sc_rev > DMAREV_1)
+ DMACSR(sc) |= D_FASTER;
+ sc->sc_active = 0; /* and of course we aren't */
+}
- switch (sc->sc_rev) {
- case DMAREV_1:
- case DMAREV_4300:
- case DMAREV_ESC1:
- break;
- case DMAREV_PLUS:
- case DMAREV_2:
- if (sc->sc_esp->sc_rev >= ESP100A)
- DMACSR(sc) |= D_FASTER;
- break;
- }
- sc->sc_active = 0; /* and of course we aren't */
+void
+dma_enintr(sc)
+ struct dma_softc *sc;
+{
+ sc->sc_regs->csr |= D_INT_EN;
}
int
-dmapending(sc)
+dma_isintr(sc)
struct dma_softc *sc;
{
return (sc->sc_regs->csr & (D_INT_PEND|D_ERR_PEND));
}
-/* bytes between loc and the end of this 16M region of memory */
-#define DMAMAX(loc) (0x01000000 - ((loc) & 0x00ffffff))
+#define DMAMAX(a) (0x01000000 - ((a) & 0x00ffffff))
-#define ESPMAX ((sc->sc_esp->sc_rev > ESP100A) ? \
- (16 * 1024 * 1024) : (64 * 1024))
/*
- * Start a dma transfer or keep it going.
- * We do the loading of the transfer counter.
- * XXX: what about write-back caches?
+ * setup a dma transfer
*/
-void
-dmastart(sc, addr, len, datain, poll)
- struct dma_softc *sc;
- void *addr;
- size_t *len;
- int datain, poll;
+int
+dma_setup(sc, addr, len, datain, dmasize)
+ struct dma_softc *sc;
+ caddr_t *addr;
+ size_t *len;
+ int datain;
+ size_t *dmasize; /* IN-OUT */
{
- struct espregs *espr = sc->sc_esp->sc_regs;
- int size;
+ u_long csr;
- if (sc->sc_active)
- panic("dma: dmastart() called while active\n");
+ /* clear errors and D_TC flag */
+ DMAWAIT(sc);
+ DMA_DRAIN(sc); /* ? */
+ DMAWAIT1(sc);
+ DMACSR(sc) |= D_INVALIDATE;
+ DMAWAIT1(sc);
+#if 0
+ DMACSR(sc) &= ~D_INT_EN;
+#endif
sc->sc_dmaaddr = addr;
sc->sc_dmalen = len;
- sc->sc_dmapolling = poll;
- sc->sc_dmadev2mem = datain;
+
+ ESP_DMA(("%s: start %d@%p,%d\n", sc->sc_dev.dv_xname,
+ *sc->sc_dmalen, *sc->sc_dmaaddr, datain ? 1 : 0));
/*
* the rules say we cannot transfer more than the limit
* of this DMA chip (64k for old and 16Mb for new),
* and we cannot cross a 16Mb boundary.
*/
- size = min(*sc->sc_dmalen, ESPMAX);
- size = min(size, DMAMAX((size_t) *sc->sc_dmaaddr));
- sc->sc_segsize = size;
-
-#ifdef DMA_TEST
- printf("%s: start %d@0x%08x [%s scsi] [chunk=%d] %d\n",
- sc->sc_dev.dv_xname,
- *sc->sc_dmalen, *sc->sc_dmaaddr,
- datain ? "read from" : "write to",
- sc->sc_segsize, poll);
-#endif
+ *dmasize = sc->sc_dmasize =
+ min(*dmasize, DMAMAX((size_t) *sc->sc_dmaaddr));
- espr->espr_tcl = size;
- espr->espr_tcm = size >> 8;
- if (sc->sc_esp->sc_rev > ESP100A)
- espr->espr_tch = size >> 16;
- espr->espr_cmd = ESPCMD_DMA|ESPCMD_NOP; /* load the count in */
+ ESP_DMA(("dma_setup: dmasize = %d\n", sc->sc_dmasize));
- DMADDR(sc) = *sc->sc_dmaaddr;
- DMACSR(sc) = (DMACSR(sc) & ~(D_WRITE|D_INT_EN)) | D_EN_DMA |
- (datain ? D_WRITE : 0) | (poll ? 0 : D_INT_EN);
+ /* Program the DMA address */
+ if (CPU_ISSUN4M && sc->sc_dmasize) {
+ /*
+ * Use dvma mapin routines to map the buffer into DVMA space.
+ */
+ sc->sc_dvmaaddr = *sc->sc_dmaaddr;
+ sc->sc_dvmakaddr = kdvma_mapin(sc->sc_dvmaaddr,
+ sc->sc_dmasize, 0);
+ if (sc->sc_dvmakaddr == NULL)
+ panic("dma: cannot allocate DVMA address");
+ DMADDR(sc) = sc->sc_dvmakaddr;
+ } else
+ DMADDR(sc) = *sc->sc_dmaaddr;
+
+ /* Setup DMA control register */
+ csr = DMACSR(sc);
+ if (datain)
+ csr |= D_WRITE;
+ else
+ csr &= ~D_WRITE;
+ csr |= D_INT_EN;
+ DMACSR(sc) = csr;
+
+ return 0;
+}
- /* and kick the SCSI */
- espr->espr_cmd = ESPCMD_DMA|ESPCMD_TRANS;
+void
+dma_go(sc)
+ struct dma_softc *sc;
+{
+ /* Start DMA */
+ DMACSR(sc) |= D_EN_DMA;
sc->sc_active = 1;
}
/*
* Pseudo (chained) interrupt from the esp driver to kick the
- * current running DMA transfer. espintr() cleans up errors.
+ * current running DMA transfer. I am replying on espintr() to
+ * pickup and clean errors for now
*
- * return 1 if a dma operation is being continued (for when all
- * the data could not be transferred in one dma operation).
+ * return 1 if it was a DMA continue.
*/
int
-dmaintr(sc, restart)
- struct dma_softc *sc;
- int restart;
+espdmaintr(sc)
+ struct dma_softc *sc;
{
- struct espregs *espr = sc->sc_esp->sc_regs;
- int trans = 0, resid;
+ int trans, resid;
+ u_long csr;
+ csr = DMACSR(sc);
-#ifdef DMA_TEST
- printf("%s: intr\n", sc->sc_dev.dv_xname);
-#endif
+ ESP_DMA(("%s: intr: addr %p, csr %b\n", sc->sc_dev.dv_xname,
+ DMADDR(sc), csr, DMACSRBITS));
- if (DMACSR(sc) & D_ERR_PEND) {
- printf("%s: error", sc->sc_dev.dv_xname);
- if (sc->sc_rev == DMAREV_4300)
- DMAWAIT_PEND(sc);
+ if (csr & D_ERR_PEND) {
+ DMACSR(sc) &= ~D_EN_DMA; /* Stop DMA */
DMACSR(sc) |= D_INVALIDATE;
- return (0);
+ printf("%s: error: csr=%b\n", sc->sc_dev.dv_xname,
+ csr, DMACSRBITS);
+ return 0;
}
+ /* This is an "assertion" :) */
if (sc->sc_active == 0)
panic("dmaintr: DMA wasn't active");
- /* DMA has stopped, but flush it first */
- dmadrain(sc);
- if (DMACSR(sc) & D_DRAINING)
- printf("drain failed %d left\n", DMACSR(sc) & D_DRAINING);
+ /* clear errors and D_TC flag */
+ DMAWAIT(sc);
+ DMA_DRAIN(sc); /* ? */
+ DMAWAIT1(sc);
+ DMACSR(sc) |= D_INVALIDATE;
+ DMAWAIT1(sc);
+
+ /* DMA has stopped */
DMACSR(sc) &= ~D_EN_DMA;
sc->sc_active = 0;
- /*
- * XXX: The subtracting of resid and throwing away up to 31
- * bytes cannot be the best/right way to do this. There's got
- * to be a better way, such as letting the DMA take control
- * again and putting it into memory, or pulling it out and
- * putting it in memory by ourselves.
- * XXX in the meantime, just do this job silently
- */
+ if (sc->sc_dmasize == 0) {
+ /* A "Transfer Pad" operation completed */
+ ESP_DMA(("dmaintr: discarded %d bytes (tcl=%d, tcm=%d)\n",
+ ESP_READ_REG(sc->sc_esp, ESP_TCL) |
+ (ESP_READ_REG(sc->sc_esp, ESP_TCM) << 8),
+ ESP_READ_REG(sc->sc_esp, ESP_TCL),
+ ESP_READ_REG(sc->sc_esp, ESP_TCM)));
+ return 0;
+ }
+
resid = 0;
- if (!sc->sc_dmadev2mem &&
- (resid = (espr->espr_fflag & ESPFIFO_FF)) != 0) {
-#if 0
- printf("empty FIFO of %d ", resid);
-#endif
- espr->espr_cmd = ESPCMD_FLUSH;
- DELAY(1);
+ /*
+ * If a transfer onto the SCSI bus gets interrupted by the device
+ * (e.g. for a SAVEPOINTER message), the data in the FIFO counts
+ * as residual since the ESP counter registers get decremented as
+ * bytes are clocked into the FIFO.
+ */
+ if (!(csr & D_WRITE) &&
+ (resid = (ESP_READ_REG(sc->sc_esp, ESP_FFLAG) & ESPFIFO_FF)) != 0) {
+ ESP_DMA(("dmaintr: empty esp FIFO of %d ", resid));
+ ESPCMD(sc->sc_esp, ESPCMD_FLUSH);
+ }
+
+ if ((sc->sc_esp->sc_espstat & ESPSTAT_TC) == 0) {
+ /*
+ * `Terminal count' is off, so read the residue
+ * out of the ESP counter registers.
+ */
+ resid += ( ESP_READ_REG(sc->sc_esp, ESP_TCL) |
+ (ESP_READ_REG(sc->sc_esp, ESP_TCM) << 8) |
+ ((sc->sc_esp->sc_cfg2 & ESPCFG2_FE)
+ ? (ESP_READ_REG(sc->sc_esp, ESP_TCH) << 16)
+ : 0));
+
+ if (resid == 0 && sc->sc_dmasize == 65536 &&
+ (sc->sc_esp->sc_cfg2 & ESPCFG2_FE) == 0)
+ /* A transfer of 64K is encoded as `TCL=TCM=0' */
+ resid = 65536;
}
- resid += espr->espr_tcl | (espr->espr_tcm << 8) |
- (sc->sc_esp->sc_rev > ESP100A ? (espr->espr_tch << 16) : 0);
- trans = sc->sc_segsize - resid;
+
+ trans = sc->sc_dmasize - resid;
if (trans < 0) { /* transferred < 0 ? */
printf("%s: xfer (%d) > req (%d)\n",
- sc->sc_dev.dv_xname, trans, sc->sc_segsize);
- trans = sc->sc_segsize;
+ sc->sc_dev.dv_xname, trans, sc->sc_dmasize);
+ trans = sc->sc_dmasize;
}
-#ifdef DMA_TEST
- printf("dmaintr: resid=%d, trans=%d\n", resid, trans);
-#endif
+ ESP_DMA(("dmaintr: tcl=%d, tcm=%d, tch=%d; trans=%d, resid=%d\n",
+ ESP_READ_REG(sc->sc_esp, ESP_TCL),
+ ESP_READ_REG(sc->sc_esp, ESP_TCM),
+ (sc->sc_esp->sc_cfg2 & ESPCFG2_FE)
+ ? ESP_READ_REG(sc->sc_esp, ESP_TCH) : 0,
+ trans, resid));
- if (sc->sc_dmadev2mem && vactype != VAC_NONE)
+ if (csr & D_WRITE)
cache_flush(*sc->sc_dmaaddr, trans);
- sc->sc_segsize -= trans;
+ if (CPU_ISSUN4M && sc->sc_dvmakaddr)
+ dvma_mapout((vm_offset_t)sc->sc_dvmakaddr,
+ (vm_offset_t)sc->sc_dvmaaddr, sc->sc_dmasize);
+
*sc->sc_dmalen -= trans;
*sc->sc_dmaaddr += trans;
-#ifdef DMA_TEST
- printf("%s: %d/%d bytes left\n", sc->sc_dev.dv_xname,
- *sc->sc_dmalen, sc->sc_segsize);
-#endif
-
- /* completely finished with the DMA transaction */
- if (sc->sc_segsize == 0 && *sc->sc_dmalen == 0)
- return (0);
+#if 0 /* this is not normal operation just yet */
+ if (*sc->sc_dmalen == 0 ||
+ sc->sc_esp->sc_phase != sc->sc_esp->sc_prevphase)
+ return 0;
- if (restart == 0)
- return (1);
/* and again */
- dmastart(sc, sc->sc_dmaaddr, sc->sc_dmalen, sc->sc_dmadev2mem,
- sc->sc_dmapolling);
- return (1);
-}
-
-/*
- * We have to ask rev 1 and 4/300 dma controllers to drain their fifo.
- * Apparently the other chips have drained by the time we get an
- * interrupt, but we check anyways.
- */
-void
-dmadrain(sc)
- struct dma_softc *sc;
-{
- switch (sc->sc_rev) {
- case DMAREV_1:
- case DMAREV_4300:
- case DMAREV_PLUS:
- case DMAREV_2:
- if (DMACSR(sc) & D_DRAINING)
- DMAWAIT_DRAIN(sc);
- break;
- case DMAREV_ESC1:
- DMAWAIT_PEND(sc)
- DMAWAIT_DRAIN(sc); /* XXX: needed? */
- break;
- }
- DMACSR(sc) |= D_INVALIDATE;
+ dma_start(sc, sc->sc_dmaaddr, sc->sc_dmalen, DMACSR(sc) & D_WRITE);
+ return 1;
+#endif
+ return 0;
}
/*
- * XXX
- * During autoconfig we are in polled mode and we execute some commands.
- * eventually we execute the last polled command. esp interrupts go through
- * the dma chip's D_INT_EN gate. thus, because we have our data, we're happy
- * and return. the esp chip has not, however, become unbusy. as soon as we
- * execute our first non-polled command, we find the esp state machine is
- * non-idle. it has not finished getting off the scsi bus, because it didn't
- * get interrupts (and the polled code has long since gone it's merry way.)
- *
- * therefore, whenever we finish with a polled mode command, we enable
- * interrupts so that we can get our data. it is probably safe to do so,
- * since the scsi transfer has happened without error. the interrupts that
- * will happen have no bearing on the higher level scsi subsystem, since it
- * just functions to let the esp chip "clean up" it's state.
+ * Pseudo (chained) interrupt from the le driver to handle DMA
+ * errors.
+ *
+ * XXX: untested
*/
-void
-dmaenintr(sc)
- struct dma_softc *sc;
-{
- DMACSR(sc) |= D_INT_EN;
-}
-
int
-dmadisintr(sc)
+ledmaintr(sc)
struct dma_softc *sc;
{
- int x = DMACSR(sc) & D_INT_EN;
+ u_long csr;
- DMACSR(sc) &= ~D_INT_EN;
- return x;
+ csr = DMACSR(sc);
+
+ if (csr & D_ERR_PEND) {
+ printf("Lance DMA error, see your doctor!\n");
+ DMACSR(sc) &= ~D_EN_DMA; /* Stop DMA */
+ DMACSR(sc) |= D_INVALIDATE;
+ printf("%s: error: csr=%b\n", sc->sc_dev.dv_xname,
+ (u_int)csr, DMACSRBITS);
+ }
+ return 1;
}
diff --git a/sys/arch/sparc/dev/dmareg.h b/sys/arch/sparc/dev/dmareg.h
index d19c5c2960e..3998581845a 100644
--- a/sys/arch/sparc/dev/dmareg.h
+++ b/sys/arch/sparc/dev/dmareg.h
@@ -1,9 +1,7 @@
-/* $NetBSD: dmareg.h,v 1.5 1994/11/20 20:52:06 deraadt Exp $ */
+/* $NetBSD: dmareg.h,v 1.8 1996/04/22 02:34:58 abrown Exp $ */
/*
* Copyright (c) 1994 Peter Galbavy. All rights reserved.
- * Copyright (c) 1995 Theo de Raadt. All rights reserved.
- *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
@@ -14,8 +12,7 @@
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
- * This product includes software developed by Peter Galbavy and
- * Theo de Raadt.
+ * This product includes software developed by Peter Galbavy.
* 4. The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission.
*
@@ -31,51 +28,64 @@
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
+#define DMACSRBITS "\020\01INT\02ERR\03DR1\04DR2\05IEN\011WRITE\016ENCNT\017TC\032DMAON"
+
struct dma_regs {
- volatile u_long csr; /* DMA CSR */
-#define D_INT_PEND 0x00000001 /* interrupt pending */
-#define D_ERR_PEND 0x00000002 /* error pending */
-#define D_DRAINING 0x0000000c /* fifo draining */
-#define D_INT_EN 0x00000010 /* interrupt enable */
-#define D_INVALIDATE 0x00000020 /* invalidate fifo */
-#define D_SLAVE_ERR 0x00000040 /* slave access size error */
-#define D_DRAIN 0x00000040 /* drain fifo if DMAREV_1 */
-#define D_RESET 0x00000080 /* reset scsi */
-#define D_WRITE 0x00000100 /* 1 = dev -> mem */
-#define D_EN_DMA 0x00000200 /* enable DMA requests */
-#define D_R_PEND 0x00000400 /* no reset/flush allowed! */
-#define D_BYTEADDR 0x00001800 /* next byte to be accessed by esp */
-#define D_EN_CNT 0x00002000 /* enable byte counter */
-#define D_TC 0x00004000 /* terminal count */
-#define D_ILLAC 0x00008000 /* enable lance ethernet */
-#define D_DSBL_CSR_DRN 0x00010000 /* disable fifo drain on csr */
-#define D_DSBL_SCSI_DRN 0x00020000 /* disable fifo drain on reg */
-#define D_BURST_SIZE 0x000c0000 /* sbus read/write burst size */
-#define D_DIAG 0x00100000 /* disable fifo drain on addr */
-#define D_TWO_CYCLE 0x00200000 /* 2 clocks per transfer */
-#define D_FASTER 0x00400000 /* 3 clocks per transfer */
-#define D_TCI_DIS 0x00800000 /* disable intr on D_TC */
-#define D_EN_NEXT 0x01000000 /* enable auto next address */
-#define D_DMA_ON 0x02000000 /* enable dma from scsi */
-#define D_A_LOADED 0x04000000 /* address loaded */
-#define D_NA_LOADED 0x08000000 /* next address loaded */
-#define D_DEV_ID 0xf0000000 /* device ID */
-#define DMAREV_4300 0x00000000 /* Sunray DMA */
-#define DMAREV_ESC1 0x40000000 /* ESC gate array */
-#define DMAREV_1 0x80000000 /* 'DMA' */
-#define DMAREV_PLUS 0x90000000 /* 'DMA+' */
-#define DMAREV_2 0xa0000000 /* 'DMA2' */
+ volatile u_long csr; /* DMA CSR */
+#define D_INT_PEND 0x00000001 /* interrupt pending */
+#define D_ERR_PEND 0x00000002 /* error pending */
+#define D_DRAINING 0x0000000c /* fifo draining */
+#define D_INT_EN 0x00000010 /* interrupt enable */
+#define D_INVALIDATE 0x00000020 /* invalidate fifo */
+#define D_SLAVE_ERR 0x00000040 /* slave access size error */
+#define D_DRAIN 0x00000040 /* drain fifo if DMAREV_1 */
+#define D_RESET 0x00000080 /* reset scsi */
+#define D_WRITE 0x00000100 /* 1 = dev -> mem */
+#define D_EN_DMA 0x00000200 /* enable DMA requests */
+#define D_R_PEND 0x00000400 /* something only on ver < 2 */
+#define D_EN_CNT 0x00002000 /* enable byte counter */
+#define D_TC 0x00004000 /* terminal count */
+#define D_DSBL_CSR_DRN 0x00010000 /* disable fifo drain on csr */
+#define D_DSBL_SCSI_DRN 0x00020000 /* disable fifo drain on reg */
+#define D_BURST_SIZE 0x000c0000 /* sbus read/write burst size */
+#define D_BURST_0 0x00080000 /* no bursts (SCSI-only) */
+#define D_BURST_16 0x00040000 /* 16-byte bursts */
+#define D_BURST_32 0x00000000 /* 32-byte bursts */
+#define D_DIAG 0x00100000 /* disable fifo drain on addr */
+#define D_TWO_CYCLE 0x00200000 /* 2 clocks per transfer */
+#define D_FASTER 0x00400000 /* 3 clocks per transfer */
+#define DE_AUI_TP 0x00400000 /* 1 for TP, 0 for AUI */
+#define D_TCI_DIS 0x00800000 /* disable intr on D_TC */
+#define D_EN_NEXT 0x01000000 /* enable auto next address */
+#define D_DMA_ON 0x02000000 /* enable dma from scsi */
+#define D_A_LOADED 0x04000000 /* address loaded */
+#define D_NA_LOADED 0x08000000 /* next address loaded */
+#define D_DEV_ID 0xf0000000 /* device ID */
+#define DMAREV_0 0x00000000 /* Sunray DMA */
+#define DMAREV_1 0x80000000 /* 'DMA' */
+#define DMAREV_PLUS 0x90000000 /* 'DMA+' */
+#define DMAREV_2 0xa0000000 /* 'DMA2' */
- volatile caddr_t addr;
-#define DMA_D_ADDR 0x01 /* DMA ADDR */
+ volatile caddr_t addr;
+#define DMA_D_ADDR 0x01 /* DMA ADDR (in u_longs) */
- /*
- * some versions of dma controller lack the following
- * two registers -- do not use them!
- */
+ volatile u_long bcnt; /* DMA COUNT (in u_longs) */
+#define D_BCNT_MASK 0x00ffffff /* only 24 bits */
- volatile u_long bcnt; /* DMA COUNT */
-#define D_BCNT_MASK 0x00ffffff /* only 24 bits */
+ volatile u_long test; /* DMA TEST (in u_longs) */
+#define en_testcsr addr /* enet registers overlap */
+#define en_cachev bcnt
+#define en_bar test
- volatile u_long test; /* DMA TEST (in u_longs) */
};
+
+/*
+ * PROM-reported DMA burst sizes for the SBus
+ */
+#define SBUS_BURST_1 0x1
+#define SBUS_BURST_2 0x2
+#define SBUS_BURST_4 0x4
+#define SBUS_BURST_8 0x8
+#define SBUS_BURST_16 0x10
+#define SBUS_BURST_32 0x20
+#define SBUS_BURST_64 0x40
diff --git a/sys/arch/sparc/dev/dmavar.h b/sys/arch/sparc/dev/dmavar.h
index c5ebb58357c..6bc42333a98 100644
--- a/sys/arch/sparc/dev/dmavar.h
+++ b/sys/arch/sparc/dev/dmavar.h
@@ -1,10 +1,7 @@
-/* $NetBSD: dmavar.h,v 1.4 1994/11/27 00:08:34 deraadt Exp $ */
+/* $NetBSD: dmavar.h,v 1.8 1996/04/22 02:35:00 abrown Exp $ */
/*
- * Copyright (c) 1994 Peter Galbavy
- * Copyright (c) 1995 Theo de Raadt
- * All rights reserved.
- *
+ * Copyright (c) 1994 Peter Galbavy. All rights reserved.
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
@@ -15,8 +12,7 @@
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
- * This product includes software developed by Peter Galbavy and
- * Theo de Raadt.
+ * This product includes software developed by Peter Galbavy.
* 4. The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission.
*
@@ -33,49 +29,62 @@
*/
struct dma_softc {
- struct device sc_dev; /* us as a device */
- struct sbusdev sc_sd; /* sbus device */
- struct esp_softc *sc_esp; /* my scsi */
- struct dma_regs *sc_regs; /* the registers */
+ struct device sc_dev; /* us as a device */
+ struct sbusdev sc_sd; /* sbus device */
+ struct esp_softc *sc_esp; /* my scsi */
+ struct le_softc *sc_le; /* my ethernet */
+ struct dma_regs *sc_regs; /* the registers */
int sc_active; /* DMA active ? */
int sc_rev; /* revision */
int sc_node; /* PROM node ID */
-
- size_t sc_segsize; /* current operation */
- void **sc_dmaaddr;
+ int sc_burst; /* DVMA burst size in effect */
+ caddr_t sc_dvmakaddr; /* DVMA cookies */
+ caddr_t sc_dvmaaddr; /* */
+ size_t sc_dmasize;
+ caddr_t *sc_dmaaddr;
size_t *sc_dmalen;
- char sc_dmapolling; /* ... is polled */
- char sc_dmadev2mem; /* transfer direction */
+ void (*reset)(struct dma_softc *); /* reset routine */
+ void (*enintr)(struct dma_softc *); /* enable interrupts */
+ int (*isintr)(struct dma_softc *); /* interrupt ? */
+ int (*intr)(struct dma_softc *); /* interrupt ! */
+ int (*setup)(struct dma_softc *, caddr_t *, size_t *, int, size_t *);
+ void (*go)(struct dma_softc *);
};
-void dmareset __P((struct dma_softc *sc));
-void dmastart __P((struct dma_softc *sc, void *addr,
- size_t *len, int datain, int poll));
-int dmaintr __P((struct dma_softc *sc, int restart));
-int dmapending __P((struct dma_softc *sc));
-void dmadrain __P((struct dma_softc *sc));
-void dmaenintr __P((struct dma_softc *sc));
-int dmadisintr __P((struct dma_softc *sc));
-
#define DMACSR(sc) (sc->sc_regs->csr)
#define DMADDR(sc) (sc->sc_regs->addr)
-#define DMABCNT(sc) (sc->sc_regs->bcnt)
-#define TIME_WAIT(cond, msg, sc) { \
- int count = 500000; \
- while (--count > 0 && (cond)) \
- DELAY(1); \
- if (count == 0) { \
- printf("CSR = %x\n", (sc)->sc_regs->csr); \
- panic(msg); \
- } \
-}
+/*
+ * We are not allowed to touch the DMA "flush" and "drain" bits
+ * while it is still thinking about a request (DMA_RP).
+ */
+
+/*
+ * TIME WAIT (to debug hanging machine problem)
+ */
+
+#define TIME_WAIT(COND, MSG, SC) { int count = 500000; \
+ while (--count > 0 && (COND)) DELAY(1); \
+ if (count == 0) { \
+ printf("CSR = %lx\n",\
+ SC->sc_regs->csr);\
+ panic(MSG); } \
+ }
+
+#define DMAWAIT(sc) TIME_WAIT((sc->sc_regs->csr & D_R_PEND), "DMAWAIT", sc)
+#define DMAWAIT1(sc) TIME_WAIT((sc->sc_regs->csr & D_DRAINING), "DMAWAIT1", sc)
+#define DMAREADY(sc) TIME_WAIT((!(sc->sc_regs->csr & D_DMA_ON)), "DMAREADY", sc)
-#define DMAWAIT_PEND(sc) \
- TIME_WAIT((DMACSR(sc) & D_R_PEND), \
- "DMAWAIT_PEND", sc)
+#define DMA_DRAIN(sc) if (sc->sc_rev < DMAREV_2) { \
+ DMACSR(sc) |= D_DRAIN; \
+ DMAWAIT1(sc); \
+ }
-/* keep punching the chip until it's flushed */
-#define DMAWAIT_DRAIN(sc) \
- TIME_WAIT((DMACSR(sc) |= D_DRAIN, DMACSR(sc) & D_DRAINING), \
- "DMAWAIT_DRAIN", sc)
+/* DMA engine functions */
+#define DMA_ENINTR(r) (((r)->enintr)(r))
+#define DMA_ISINTR(r) (((r)->isintr)(r))
+#define DMA_RESET(r) (((r)->reset)(r))
+#define DMA_INTR(r) (((r)->intr)(r))
+#define DMA_ISACTIVE(r) ((r)->sc_active)
+#define DMA_SETUP(a, b, c, d, e) (((a)->setup)(a, b, c, d, e))
+#define DMA_GO(r) (((r)->go)(r))
diff --git a/sys/arch/sparc/dev/esp.c b/sys/arch/sparc/dev/esp.c
index 7941b614917..840ca8100d0 100644
--- a/sys/arch/sparc/dev/esp.c
+++ b/sys/arch/sparc/dev/esp.c
@@ -1,6 +1,10 @@
+/* $NetBSD: esp.c,v 1.47.2.1 1996/06/12 20:46:52 pk Exp $ */
+
/*
- * Copyright (c) 1995 Theo de Raadt
- *
+ * Copyright (c) 1994 Peter Galbavy
+ * Copyright (c) 1995 Paul Kranenburg
+ * All rights reserved.
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
@@ -11,33 +15,29 @@
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
- * This product includes software developed under OpenBSD by
- * Theo de Raadt.
+ * This product includes software developed by Peter Galbavy
* 4. The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission.
*
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
- * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
- * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
- * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
- * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
- * SUCH DAMAGE.
- */
-
-/*
- * Based IN PART on aic6360 by Jarle Greipsland, an older esp driver
- * by Peter Galbavy, and work by Charles Hannum on a few other drivers.
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
+ * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
+ * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
+ * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
+ * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
+ * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
*/
/*
- * todo:
- * fix & enable sync
- * confirm parity and bus failures do not lock up driver
+ * Based on aic6360 by Jarle Greipsland
+ *
+ * Acknowledgements: Many of the algorithms used in this driver are
+ * inspired by the work of Julian Elischer (julian@tfs.com) and
+ * Charles Hannum (mycroft@duality.gnu.ai.mit.edu). Thanks a million!
*/
#include <sys/types.h>
@@ -59,62 +59,68 @@
#include <machine/cpu.h>
#include <machine/autoconf.h>
#include <sparc/dev/sbusvar.h>
-#include <sparc/dev/dmavar.h>
#include <sparc/dev/dmareg.h>
+#include <sparc/dev/dmavar.h>
#include <sparc/dev/espreg.h>
#include <sparc/dev/espvar.h>
-int esp_debug = ESP_SHOWPHASE|ESP_SHOWMISC|ESP_SHOWTRAC|ESP_SHOWCMDS; /**/
-
-#if 1
-#define ESPD(x)
-#else
-#define ESPD(x) printf x
-#endif
+int esp_debug = 0; /*ESP_SHOWPHASE|ESP_SHOWMISC|ESP_SHOWTRAC|ESP_SHOWCMDS;*/
+
+/*static*/ void espattach __P((struct device *, struct device *, void *));
+/*static*/ int espmatch __P((struct device *, void *, void *));
+/*static*/ int espprint __P((void *, char *));
+/*static*/ u_int esp_adapter_info __P((struct esp_softc *));
+/*static*/ void espreadregs __P((struct esp_softc *));
+/*static*/ void espselect __P((struct esp_softc *,
+ u_char, u_char, u_char *, u_char));
+/*static*/ void esp_scsi_reset __P((struct esp_softc *));
+/*static*/ void esp_reset __P((struct esp_softc *));
+/*static*/ void esp_init __P((struct esp_softc *, int));
+/*static*/ int esp_scsi_cmd __P((struct scsi_xfer *));
+/*static*/ int esp_poll __P((struct esp_softc *, struct ecb *));
+/*static*/ void esp_sched __P((struct esp_softc *));
+/*static*/ void esp_done __P((struct ecb *));
+/*static*/ void esp_msgin __P((struct esp_softc *));
+/*static*/ void esp_msgout __P((struct esp_softc *));
+/*static*/ int espintr __P((struct esp_softc *));
+/*static*/ void esp_timeout __P((void *arg));
+/*static*/ void esp_abort __P((struct esp_softc *, struct ecb *));
+int esp_stp2cpb __P((struct esp_softc *, int));
+int esp_cpb2stp __P((struct esp_softc *, int));
+
+/* Linkup to the rest of the kernel */
+struct cfattach esp_ca = {
+ sizeof(struct esp_softc), espmatch, espattach
+};
-void espattach __P((struct device *, struct device *, void *));
-int espmatch __P((struct device *, void *, void *));
-int espprint __P((void *, char *));
-void espreadregs __P((struct esp_softc *));
-int espgetbyte __P((struct esp_softc *, int));
-void espselect __P((struct esp_softc *));
-void esp_reset __P((struct esp_softc *));
-void esp_init __P((struct esp_softc *, int));
-int esp_scsi_cmd __P((struct scsi_xfer *));
-int esp_poll __P((struct esp_softc *, struct ecb *));
-int espphase __P((struct esp_softc *));
-void esp_sched __P((struct esp_softc *));
-void esp_done __P((struct ecb *));
-void esp_msgin __P((struct esp_softc *));
-void esp_makemsg __P((struct esp_softc *));
-void esp_msgout __P((struct esp_softc *));
-int espintr __P((struct esp_softc *));
-void esp_timeout __P((void *arg));
-
-struct cfdriver espcd = {
- NULL, "esp", espmatch, espattach, DV_DULL, sizeof(struct esp_softc)
+struct cfdriver esp_cd = {
+ NULL, "esp", DV_DULL
};
struct scsi_adapter esp_switch = {
- esp_scsi_cmd, minphys, NULL, NULL
+ esp_scsi_cmd,
+ minphys, /* no max at this level; handled by DMA code */
+ NULL,
+ NULL,
};
struct scsi_device esp_dev = {
- NULL, NULL, NULL, NULL
+ NULL, /* Use default error handler */
+ NULL, /* have a queue, served by this */
+ NULL, /* have no async handler */
+ NULL, /* Use default 'done' routine */
};
-/*
- * Does anyone actually use this, and what for ?
- */
int
espprint(aux, name)
void *aux;
char *name;
{
- return (UNCONF);
+ if (name != NULL)
+ printf("%s: scsibus ", name);
+ return UNCONF;
}
-
int
espmatch(parent, vcf, aux)
struct device *parent;
@@ -132,7 +138,6 @@ espmatch(parent, vcf, aux)
return (probeget(ra->ra_vaddr, 1) != -1);
}
-
/*
* Attach this instance, and then all the sub-devices
*/
@@ -143,9 +148,8 @@ espattach(parent, self, aux)
{
register struct confargs *ca = aux;
struct esp_softc *sc = (void *)self;
- struct espregs *espr;
struct bootpath *bp;
- int freq;
+ int dmachild = strncmp(parent->dv_xname, "dma", 3) == 0;
/*
* Make sure things are sane. I don't know if this is ever
@@ -164,13 +168,11 @@ espattach(parent, self, aux)
* address space.
*/
if (ca->ca_ra.ra_vaddr)
- sc->sc_regs = (struct espregs *) ca->ca_ra.ra_vaddr;
+ sc->sc_reg = (volatile u_char *) ca->ca_ra.ra_vaddr;
else {
- sc->sc_regs = (struct espregs *)
- mapiodev(ca->ca_ra.ra_reg, 0, ca->ca_ra.ra_len,
- ca->ca_bustype);
+ sc->sc_reg = (volatile u_char *)
+ mapiodev(ca->ca_ra.ra_reg, 0, ca->ca_ra.ra_len, ca->ca_bustype);
}
- espr = sc->sc_regs;
/* Other settings */
sc->sc_node = ca->ca_ra.ra_node;
@@ -185,92 +187,138 @@ espattach(parent, self, aux)
sc->sc_freq = ((struct sbus_softc *)
sc->sc_dev.dv_parent)->sc_clockfreq;
- freq = sc->sc_freq / 1000000; /* gimme Mhz */
-
- /*
- * This is the value used to start sync negotiations. For a
- * 25Mhz clock, this gives us 40, or 160nS, or 6.25Mb/s. It
- * is constant for each adapter. In turn, notice that the ESP
- * register "SYNCTP" is = (250 / the negotiated period).
- */
- sc->sc_minsync = 1000 / freq;
-
- /* 0 is actually 8, even though the register only has 3 bits */
- sc->sc_ccf = FREQTOCCF(freq) & 0x07;
+ /* gimme Mhz */
+ sc->sc_freq /= 1000000;
- /* The value must not be 1 -- make it 2 */
- if (sc->sc_ccf == 1)
- sc->sc_ccf = 2;
+ if (dmachild) {
+ sc->sc_dma = (struct dma_softc *)parent;
+ sc->sc_dma->sc_esp = sc;
+ } else {
+ /*
+ * find the DMA by poking around the dma device structures
+ *
+ * What happens here is that if the dma driver has not been
+ * configured, then this returns a NULL pointer. Then when the
+ * dma actually gets configured, it does the opposing test, and
+ * if the sc->sc_esp field in it's softc is NULL, then tries to
+ * find the matching esp driver.
+ *
+ */
+ sc->sc_dma = (struct dma_softc *)
+ getdevunit("dma", sc->sc_dev.dv_unit);
- /*
- * The recommended timeout is 250ms (1/4 seconds). This
- * register is loaded with a value calculated as follows:
- *
- * (timout period) x (CLK frequency)
- * reg = -------------------------------------
- * 8192 x (Clock Conversion Factor)
- *
- * We have the CCF from above. For a 25MHz clock this gives
- * a constant of 153 (we round up).
- */
- sc->sc_timeout = (sc->sc_freq / 4 / 8192 / sc->sc_ccf) + 1;
+ /*
+ * and a back pointer to us, for DMA
+ */
+ if (sc->sc_dma)
+ sc->sc_dma->sc_esp = sc;
+ else
+ panic("espattach: no dma found");
+ }
/*
- * find the corresponding DMA controller.
+ * It is necessary to try to load the 2nd config register here,
+ * to find out what rev the esp chip is, else the esp_reset
+ * will not set up the defaults correctly.
*/
- sc->sc_dma = ((struct dma_softc *)getdevunit("dma",
- sc->sc_dev.dv_unit));
- if (sc->sc_dma)
- sc->sc_dma->sc_esp = sc;
-
sc->sc_cfg1 = sc->sc_id | ESPCFG1_PARENB;
- sc->sc_cfg2 = 0;
- sc->sc_cfg3 = 0;
+ sc->sc_cfg2 = ESPCFG2_SCSI2 | ESPCFG2_RPE;
+ sc->sc_cfg3 = ESPCFG3_CDB;
+ ESP_WRITE_REG(sc, ESP_CFG2, sc->sc_cfg2);
- /*
- * The ESP100 only has a cfg1 register. The ESP100A has it, but
- * lacks the cfg3 register. Thus, we can tell which chip we have.
- * XXX: what about the FAS100A?
- */
- espr->espr_cfg2 = ESPCFG2_SCSI2 | ESPCFG2_RPE;
- if ((espr->espr_cfg2 & ~ESPCFG2_RSVD) != (ESPCFG2_SCSI2 | ESPCFG2_RPE)) {
+ if ((ESP_READ_REG(sc, ESP_CFG2) & ~ESPCFG2_RSVD) != (ESPCFG2_SCSI2 | ESPCFG2_RPE)) {
printf(": ESP100");
sc->sc_rev = ESP100;
} else {
- espr->espr_cfg2 = 0;
- espr->espr_cfg3 = 0;
- espr->espr_cfg3 = ESPCFG3_CDB | ESPCFG3_FCLK;
- if (espr->espr_cfg3 != (ESPCFG3_CDB | ESPCFG3_FCLK)) {
+ sc->sc_cfg2 = ESPCFG2_SCSI2;
+ ESP_WRITE_REG(sc, ESP_CFG2, sc->sc_cfg2);
+ sc->sc_cfg3 = 0;
+ ESP_WRITE_REG(sc, ESP_CFG3, sc->sc_cfg3);
+ sc->sc_cfg3 = (ESPCFG3_CDB | ESPCFG3_FCLK);
+ ESP_WRITE_REG(sc, ESP_CFG3, sc->sc_cfg3);
+ if (ESP_READ_REG(sc, ESP_CFG3) != (ESPCFG3_CDB | ESPCFG3_FCLK)) {
printf(": ESP100A");
- /* XXX sc->sc_cfg2 = ESPCFG2_SCSI2 | ESPCFG2_FE; */
sc->sc_rev = ESP100A;
} else {
- espr->espr_cfg3 = 0;
+ /* ESPCFG2_FE enables > 64K transfers */
+ sc->sc_cfg2 |= ESPCFG2_FE;
+ sc->sc_cfg3 = 0;
+ ESP_WRITE_REG(sc, ESP_CFG3, sc->sc_cfg3);
printf(": ESP200");
sc->sc_rev = ESP200;
}
}
+ /*
+ * This is the value used to start sync negotiations
+ * Note that the ESP register "SYNCTP" is programmed
+ * in "clocks per byte", and has a minimum value of 4.
+ * The SCSI period used in negotiation is one-fourth
+ * of the time (in nanoseconds) needed to transfer one byte.
+ * Since the chip's clock is given in MHz, we have the following
+ * formula: 4 * period = (1000 / freq) * 4
+ */
+ sc->sc_minsync = 1000 / sc->sc_freq;
+
+ /*
+ * Alas, we must now modify the value a bit, because it's
+ * only valid when can switch on FASTCLK and FASTSCSI bits
+ * in config register 3...
+ */
+ switch (sc->sc_rev) {
+ case ESP100:
+ sc->sc_maxxfer = 64 * 1024;
+ sc->sc_minsync = 0; /* No synch on old chip? */
+ break;
+ case ESP100A:
+ sc->sc_maxxfer = 64 * 1024;
+ sc->sc_minsync = esp_cpb2stp(sc, 5); /* Min clocks/byte is 5 */
+ break;
+ case ESP200:
+ sc->sc_maxxfer = 16 * 1024 * 1024;
+ /* XXX - do actually set FAST* bits */
+ }
+
+ sc->sc_ccf = FREQTOCCF(sc->sc_freq);
+
+ /* The value *must not* be == 1. Make it 2 */
+ if (sc->sc_ccf == 1)
+ sc->sc_ccf = 2;
+
+ /*
+ * The recommended timeout is 250ms. This register is loaded
+ * with a value calculated as follows, from the docs:
+ *
+ * (timout period) x (CLK frequency)
+ * reg = -------------------------------------
+ * 8192 x (Clock Conversion Factor)
+ *
+ * Since CCF has a linear relation to CLK, this generally computes
+ * to the constant of 153.
+ */
+ sc->sc_timeout = ((250 * 1000) * sc->sc_freq) / (8192 * sc->sc_ccf);
+
+ /* CCF register only has 3 bits; 0 is actually 8 */
+ sc->sc_ccf &= 7;
+
+ /* Reset state & bus */
sc->sc_state = 0;
esp_init(sc, 1);
- printf(" %dMhz, target %d\n", freq, sc->sc_id);
+ printf(" %dMhz, target %d\n", sc->sc_freq, sc->sc_id);
+ /* add me to the sbus structures */
+ sc->sc_sd.sd_reset = (void *) esp_reset;
#if defined(SUN4C) || defined(SUN4M)
if (ca->ca_bustype == BUS_SBUS) {
- /* add to the sbus structures */
- sc->sc_sd.sd_reset = (void *) esp_reset;
- sbus_establish(&sc->sc_sd, &sc->sc_dev);
-
- /*
- * If the device is in an SBUS slave slot, bail now.
- */
- if (sbus_slavecheck(self, ca))
- return;
+ if (dmachild)
+ sbus_establish(&sc->sc_sd, sc->sc_dev.dv_parent);
+ else
+ sbus_establish(&sc->sc_sd, &sc->sc_dev);
}
#endif /* SUN4C || SUN4M */
- /* and the interupts */
+ /* and the interuppts */
sc->sc_ih.ih_fun = (void *) espintr;
sc->sc_ih.ih_arg = sc;
intr_establish(sc->sc_pri, &sc->sc_ih);
@@ -286,23 +334,20 @@ espattach(parent, self, aux)
sc->sc_link.openings = 2;
/*
- * If the boot path is "esp", and the numbers point to
- * this controller, then advance the bootpath a step.
- * XXX boot /sbus/esp/sd@3,0 passes in esp@0,0 -- which
- * is almost always wrong, but match in that case anyways.
+ * If the boot path is "esp" at the moment and it's me, then
+ * walk our pointer to the sub-device, ready for the config
+ * below.
*/
bp = ca->ca_ra.ra_bp;
switch (ca->ca_bustype) {
case BUS_SBUS:
if (bp != NULL && strcmp(bp->name, "esp") == 0 &&
- ((bp->val[0]==ca->ca_slot &&
- (bp->val[1]==ca->ca_offset || bp->val[1]==0)) ||
- (bp->val[0]==-1 && bp->val[1]==sc->sc_dev.dv_unit)))
+ SAME_ESP(sc, bp, ca))
bootpath_store(1, bp + 1);
break;
default:
if (bp != NULL && strcmp(bp->name, "esp") == 0 &&
- bp->val[0] == -1 && bp->val[1] == sc->sc_dev.dv_unit)
+ bp->val[0] == -1 && bp->val[1] == sc->sc_dev.dv_unit)
bootpath_store(1, bp + 1);
break;
}
@@ -315,266 +360,317 @@ espattach(parent, self, aux)
bootpath_store(1, NULL);
}
-
/*
- * Warning: This inserts two commands into the ESP command fifo. Because
- * there are only 2 entries in that fifo, you may need to put a DELAY(1)
- * after calling this, if you are going to issue another command soon.
+ * This is the generic esp reset function. It does not reset the SCSI bus,
+ * only this controllers, but kills any on-going commands, and also stops
+ * and resets the DMA.
+ *
+ * After reset, registers are loaded with the defaults from the attach
+ * routine above.
*/
void
-espflush(sc)
+esp_reset(sc)
struct esp_softc *sc;
{
- struct espregs *espr = sc->sc_regs;
- if (espr->espr_fflag & ESPFIFO_FF) {
- espr->espr_cmd = ESPCMD_FLUSH;
- espr->espr_cmd = ESPCMD_NOP;
+ /* reset DMA first */
+ DMA_RESET(sc->sc_dma);
+
+ /* reset SCSI chip */
+ ESPCMD(sc, ESPCMD_RSTCHIP);
+ ESPCMD(sc, ESPCMD_NOP);
+ DELAY(500);
+
+ /* do these backwards, and fall through */
+ switch (sc->sc_rev) {
+ case ESP200:
+ ESP_WRITE_REG(sc, ESP_CFG3, sc->sc_cfg3);
+ case ESP100A:
+ ESP_WRITE_REG(sc, ESP_CFG2, sc->sc_cfg2);
+ case ESP100:
+ ESP_WRITE_REG(sc, ESP_CFG1, sc->sc_cfg1);
+ ESP_WRITE_REG(sc, ESP_CCF, sc->sc_ccf);
+ ESP_WRITE_REG(sc, ESP_SYNCOFF, 0);
+ ESP_WRITE_REG(sc, ESP_TIMEOUT, sc->sc_timeout);
+ break;
+ default:
+ printf("%s: unknown revision code, assuming ESP100\n",
+ sc->sc_dev.dv_xname);
+ ESP_WRITE_REG(sc, ESP_CFG1, sc->sc_cfg1);
+ ESP_WRITE_REG(sc, ESP_CCF, sc->sc_ccf);
+ ESP_WRITE_REG(sc, ESP_SYNCOFF, 0);
+ ESP_WRITE_REG(sc, ESP_TIMEOUT, sc->sc_timeout);
}
}
/*
- * returns -1 if no byte is available because fifo is empty.
+ * Reset the SCSI bus, but not the chip
*/
-int
-espgetbyte(sc, dotrans)
+void
+esp_scsi_reset(sc)
struct esp_softc *sc;
- int dotrans;
{
- struct espregs *espr = sc->sc_regs;
+ /* stop DMA first, as the chip will return to Bus Free phase */
+ DMACSR(sc->sc_dma) &= ~D_EN_DMA;
- if (espr->espr_fflag & ESPFIFO_FF)
- return espr->espr_fifo;
- return -1;
+ printf("esp: resetting SCSI bus\n");
+ ESPCMD(sc, ESPCMD_RSTSCSI);
}
/*
- * Send a command to a target, as well as any required messages.
- * There are three cases. The first two cases are fairly simple..
- * 1) command alone
- * load command into fifo, and use ESPCMD_SELNATN
- * 2) MSG_IDENTIFY + command
- * load message and command into fifo, and use ESPCMD_SELATN
- * 3) a bunch of messages + command
- * load first message byte into fifo. Use ESPCMD_SELATNS. When the
- * next interrupt occurs, load the remainer of the message into
- * the fifo and use ESPCMD_TRANS. When the device is ready to
- * receive the command, it will switch into COMMAND_PHASE, and
- * at that point we will feed it the command.
+ * Initialize esp state machine
*/
void
-espselect(sc)
+esp_init(sc, doreset)
struct esp_softc *sc;
+ int doreset;
{
- struct espregs *espr = sc->sc_regs;
- register struct ecb *ecb = sc->sc_nexus;
- register struct scsi_link *sc_link = ecb->xs->sc_link;
- struct esp_tinfo *ti = &sc->sc_tinfo[sc_link->target];
- u_char *cmd = (u_char *)&ecb->cmd;
- int loadcmd = 1;
- int outcmd, i;
-
- espr->espr_selid = sc_link->target;
- espr->espr_syncoff = ti->offset;
- espr->espr_synctp = ti->synctp;
-
- sc->sc_state = ESPS_SELECTING;
-
- if (ecb->xs->flags & SCSI_RESET)
- sc->sc_msgpriq = SEND_DEV_RESET;
- else if (ti->flags & DO_NEGOTIATE)
- sc->sc_msgpriq = SEND_SDTR;
- else
- sc->sc_msgpriq = SEND_IDENTIFY;
-
- if (sc->sc_msgpriq) {
- esp_makemsg(sc);
+ struct ecb *ecb;
+ int r;
- ESPD(("OM["));
- for (i = 0; i < sc->sc_omlen; i++)
- ESPD(("%02x%c", sc->sc_omp[i],
- (i == sc->sc_omlen-1) ? ']' : ' '));
- ESPD((" "));
+ ESP_TRACE(("[ESP_INIT(%d)] ", doreset));
- espr->espr_fifo = sc->sc_omp[0]; /* 1st msg byte only */
- if (sc->sc_omlen == 1) {
- outcmd = ESPCMD_SELATN;
- } else {
- outcmd = ESPCMD_SELATNS;
- /* and this will will load the rest of the msg bytes */
- sc->sc_state = ESPS_SELECTSTOP;
- loadcmd = 0;
+ if (sc->sc_state == 0) { /* First time through */
+ TAILQ_INIT(&sc->ready_list);
+ TAILQ_INIT(&sc->nexus_list);
+ TAILQ_INIT(&sc->free_list);
+ sc->sc_nexus = NULL;
+ ecb = sc->sc_ecb;
+ bzero(ecb, sizeof(sc->sc_ecb));
+ for (r = 0; r < sizeof(sc->sc_ecb) / sizeof(*ecb); r++) {
+ TAILQ_INSERT_TAIL(&sc->free_list, ecb, chain);
+ ECB_SETQ(ecb, ECB_QFREE);
+ ecb++;
+ }
+ bzero(sc->sc_tinfo, sizeof(sc->sc_tinfo));
+ } else {
+ sc->sc_flags |= ESP_ABORTING;
+ sc->sc_state = ESP_IDLE;
+ ecb = sc->sc_nexus;
+ if (ecb != NULL) {
+ ecb->xs->error = XS_TIMEOUT;
+ esp_done(ecb);
+ sc->sc_nexus = NULL;
+ }
+ while ((ecb = sc->nexus_list.tqh_first) != NULL) {
+ ecb->xs->error = XS_TIMEOUT;
+ esp_done(ecb);
}
- } else
- outcmd = ESPCMD_SELNATN;
+ }
- ESPD(("P%d/%02x/%d ", (ecb->xs->flags & SCSI_POLL) ? 1 : 0, outcmd, loadcmd));
+ /*
+ * reset the chip to a known state
+ */
+ esp_reset(sc);
- if (loadcmd) {
- ESPD(("CMD["));
- for (i = 0; i < ecb->clen; i++)
- ESPD(("%02x%c", cmd[i],
- (i == ecb->clen-1) ? ']' : ' '));
- ESPD((" "));
+ sc->sc_phase = sc->sc_prevphase = INVALID_PHASE;
+ for (r = 0; r < 8; r++) {
+ struct esp_tinfo *tp = &sc->sc_tinfo[r];
+/* XXX - config flags per target: low bits: no reselect; high bits: no synch */
+ int fl = sc->sc_dev.dv_cfdata->cf_flags;
+
+ tp->flags = ((sc->sc_minsync && !(fl & (1<<(r+8))))
+ ? T_NEGOTIATE : 0) |
+ ((fl & (1<<r)) ? T_RSELECTOFF : 0) |
+ T_NEED_TO_RESET;
+ tp->period = sc->sc_minsync;
+ tp->offset = 0;
+ }
+ sc->sc_flags &= ~ESP_ABORTING;
- /* load the command into the FIFO */
- for (i = 0; i < ecb->clen; i++)
- espr->espr_fifo = cmd[i];
+ if (doreset) {
+ sc->sc_state = ESP_SBR;
+ ESPCMD(sc, ESPCMD_RSTSCSI);
+ return;
}
- espr->espr_cmd = outcmd;
- if (!(ecb->xs->flags & SCSI_POLL))
- sc->sc_flags |= ESPF_MAYDISCON;
+ sc->sc_state = ESP_IDLE;
+ esp_sched(sc);
+ return;
}
-
/*
- * Reset the ESP and DMA chips. Stops any transactions dead in the water;
- * does not cause an interrupt.
+ * Read the ESP registers, and save their contents for later use.
+ * ESP_STAT, ESP_STEP & ESP_INTR are mostly zeroed out when reading
+ * ESP_INTR - so make sure it is the last read.
+ *
+ * I think that (from reading the docs) most bits in these registers
+ * only make sense when he DMA CSR has an interrupt showing. Call only
+ * if an interrupt is pending.
*/
-void
-esp_reset(sc)
+void
+espreadregs(sc)
struct esp_softc *sc;
{
- struct espregs *espr = sc->sc_regs;
- dmareset(sc->sc_dma); /* reset DMA first */
+ sc->sc_espstat = ESP_READ_REG(sc, ESP_STAT);
+ /* Only the stepo bits are of interest */
+ sc->sc_espstep = ESP_READ_REG(sc, ESP_STEP) & ESPSTEP_MASK;
+ sc->sc_espintr = ESP_READ_REG(sc, ESP_INTR);
- espr->espr_cmd = ESPCMD_RSTCHIP;
- DELAY(5);
- espr->espr_cmd = ESPCMD_NOP;
- DELAY(5);
+ /*
+ * Determine the SCSI bus phase, return either a real SCSI bus phase
+ * or some pseudo phase we use to detect certain exceptions.
+ */
- /* do these backwards, and fall through */
- switch (sc->sc_rev) {
- case ESP200:
- espr->espr_cfg3 = sc->sc_cfg3;
- case ESP100A:
- espr->espr_cfg2 = sc->sc_cfg2;
- case ESP100:
- espr->espr_cfg1 = sc->sc_cfg1;
- espr->espr_ccf = sc->sc_ccf;
- espr->espr_syncoff = 0;
- espr->espr_timeout = sc->sc_timeout;
- break;
- }
+ sc->sc_phase = (sc->sc_espintr & ESPINTR_DIS)
+ ? /* Disconnected */ BUSFREE_PHASE
+ : sc->sc_espstat & ESPSTAT_PHASE;
+
+ ESP_MISC(("regs[intr=%02x,stat=%02x,step=%02x] ",
+ sc->sc_espintr, sc->sc_espstat, sc->sc_espstep));
}
/*
- * Initialize esp state machine
+ * Convert Synchronous Transfer Period to chip register Clock Per Byte value.
+ */
+int
+esp_stp2cpb(sc, period)
+ struct esp_softc *sc;
+ int period;
+{
+ int v;
+ v = (sc->sc_freq * period) / 250;
+ if (esp_cpb2stp(sc, v) < period)
+ /* Correct round-down error */
+ v++;
+ return v;
+}
+
+/*
+ * Convert chip register Clock Per Byte value to Synchronous Transfer Period.
+ */
+int
+esp_cpb2stp(sc, cpb)
+ struct esp_softc *sc;
+ int cpb;
+{
+ return ((250 * cpb) / sc->sc_freq);
+}
+
+/*
+ * Send a command to a target, set the driver state to ESP_SELECTING
+ * and let the caller take care of the rest.
+ *
+ * Keeping this as a function allows me to say that this may be done
+ * by DMA instead of programmed I/O soon.
*/
void
-esp_init(sc, doreset)
+espselect(sc, target, lun, cmd, clen)
struct esp_softc *sc;
- int doreset;
+ u_char target, lun;
+ u_char *cmd;
+ u_char clen;
{
- struct espregs *espr = sc->sc_regs;
- struct ecb *ecb;
+ struct esp_tinfo *ti = &sc->sc_tinfo[target];
int i;
-
- /*
- * reset the chip to a known state
- */
- esp_reset(sc);
- if (doreset) {
- espr->espr_cmd = ESPCMD_RSTSCSI;
- DELAY(500);
- /* cheat: we don't want the state machine to reset again.. */
- esp_reset(sc);
- }
+ ESP_TRACE(("[espselect(t%d,l%d,cmd:%x)] ", target, lun, *(u_char *)cmd));
- if (sc->sc_state == 0) { /* First time through */
- TAILQ_INIT(&sc->ready_list);
- TAILQ_INIT(&sc->nexus_list);
- TAILQ_INIT(&sc->free_list);
- sc->sc_nexus = 0;
- ecb = sc->sc_ecb;
- bzero(ecb, sizeof(sc->sc_ecb));
- for (i = 0; i < sizeof(sc->sc_ecb) / sizeof(*ecb); i++) {
- TAILQ_INSERT_TAIL(&sc->free_list, ecb, chain);
- ecb++;
- }
- bzero(sc->sc_tinfo, sizeof(sc->sc_tinfo));
+ /* new state ESP_SELECTING */
+ sc->sc_state = ESP_SELECTING;
+
+ ESPCMD(sc, ESPCMD_FLUSH);
+
+ /*
+ * The docs say the target register is never reset, and I
+ * can't think of a better place to set it
+ */
+ ESP_WRITE_REG(sc, ESP_SELID, target);
+ if (ti->flags & T_SYNCMODE) {
+ ESP_WRITE_REG(sc, ESP_SYNCOFF, ti->offset);
+ ESP_WRITE_REG(sc, ESP_SYNCTP, esp_stp2cpb(sc, ti->period));
} else {
- sc->sc_state = ESPS_IDLE;
- if (sc->sc_nexus != NULL) {
- sc->sc_nexus->xs->error = XS_DRIVER_STUFFUP;
- untimeout(esp_timeout, sc->sc_nexus);
- esp_done(sc->sc_nexus);
- }
- sc->sc_nexus = NULL;
- while (ecb = sc->nexus_list.tqh_first) {
- ecb->xs->error = XS_DRIVER_STUFFUP;
- untimeout(esp_timeout, ecb);
- esp_done(ecb);
- }
+ ESP_WRITE_REG(sc, ESP_SYNCOFF, 0);
+ ESP_WRITE_REG(sc, ESP_SYNCTP, 0);
}
-
- for (i = 0; i < 8; i++) {
- struct esp_tinfo *ti = &sc->sc_tinfo[i];
-
- ti->flags = DO_NEGOTIATE;
- ti->period = sc->sc_minsync;
- ti->synctp = 250 / ti->period;
- ti->offset = ESP_SYNC_REQOFF;
+
+ /*
+ * Who am I. This is where we tell the target that we are
+ * happy for it to disconnect etc.
+ */
+ ESP_WRITE_REG(sc, ESP_FIFO,
+ MSG_IDENTIFY(lun, (ti->flags & T_RSELECTOFF)?0:1));
+
+ if (ti->flags & T_NEGOTIATE) {
+ /* Arbitrate, select and stop after IDENTIFY message */
+ ESPCMD(sc, ESPCMD_SELATNS);
+ return;
}
- sc->sc_state = ESPS_IDLE;
+
+ /* Now the command into the FIFO */
+ for (i = 0; i < clen; i++)
+ ESP_WRITE_REG(sc, ESP_FIFO, *cmd++);
+
+ /* And get the targets attention */
+ ESPCMD(sc, ESPCMD_SELATN);
+
}
/*
- * Start a SCSI-command: This function is called by the higher level
- * SCSI-driver to queue/run SCSI-commands.
+ * DRIVER FUNCTIONS CALLABLE FROM HIGHER LEVEL DRIVERS
+ */
+
+/*
+ * Start a SCSI-command
+ * This function is called by the higher level SCSI-driver to queue/run
+ * SCSI-commands.
*/
-int
+int
esp_scsi_cmd(xs)
struct scsi_xfer *xs;
{
struct scsi_link *sc_link = xs->sc_link;
struct esp_softc *sc = sc_link->adapter_softc;
- struct ecb *ecb;
- int s;
-
- /*ESPD(("NS%08x/%08x/%d ", xs, xs->data, xs->datalen));*/
+ struct ecb *ecb;
+ int s, flags;
+
+ ESP_TRACE(("[esp_scsi_cmd] "));
+ ESP_CMDS(("[0x%x, %d]->%d ", (int)xs->cmd->opcode, xs->cmdlen,
+ sc_link->target));
- /* XXX: set lun */
- xs->cmd->bytes[0] |= (sc_link->lun << SCSI_CMD_LUN_SHIFT);
+ flags = xs->flags;
/* Get a esp command block */
s = splbio();
ecb = sc->free_list.tqh_first;
if (ecb) {
TAILQ_REMOVE(&sc->free_list, ecb, chain);
+ ECB_SETQ(ecb, ECB_QNONE);
}
splx(s);
-
+
if (ecb == NULL) {
- xs->error = XS_DRIVER_STUFFUP;
+ ESP_MISC(("TRY_AGAIN_LATER"));
return TRY_AGAIN_LATER;
}
/* Initialize ecb */
- ecb->flags = ECB_ACTIVE;
ecb->xs = xs;
bcopy(xs->cmd, &ecb->cmd, xs->cmdlen);
ecb->clen = xs->cmdlen;
ecb->daddr = xs->data;
ecb->dleft = xs->datalen;
ecb->stat = 0;
-
+
s = splbio();
TAILQ_INSERT_TAIL(&sc->ready_list, ecb, chain);
- if ((xs->flags & SCSI_POLL) == 0)
- timeout(esp_timeout, ecb, (xs->timeout * hz) / 1000);
+ ECB_SETQ(ecb, ECB_QREADY);
+ timeout(esp_timeout, ecb, (xs->timeout*hz)/1000);
- if (sc->sc_state == ESPS_IDLE)
+ if (sc->sc_state == ESP_IDLE)
esp_sched(sc);
+
splx(s);
- /* Not allowed to use interrupts, use polling instead */
- if (xs->flags & SCSI_POLL)
+ if (flags & SCSI_POLL) {
+ /* Not allowed to use interrupts, use polling instead */
return esp_poll(sc, ecb);
+ }
+
+ ESP_MISC(("SUCCESSFULLY_QUEUED"));
return SUCCESSFULLY_QUEUED;
+
}
/*
@@ -586,57 +682,45 @@ esp_poll(sc, ecb)
struct ecb *ecb;
{
struct scsi_xfer *xs = ecb->xs;
- int count = xs->timeout * 1000;
+ int count = xs->timeout * 100;
- ESP_TRACE(("esp_poll\n"));
+ ESP_TRACE(("[esp_poll] "));
while (count) {
- if (dmapending(sc->sc_dma)) {
- /*
- * We decrement the interrupt event counter to
- * repair it... because this isn't a real interrupt.
- */
- if (espintr(sc))
- --sc->sc_intrcnt.ev_count;
- continue;
+ if (DMA_ISINTR(sc->sc_dma)) {
+ espintr(sc);
}
+#if alternatively
+ if (ESP_READ_REG(sc, ESP_STAT) & ESPSTAT_INT)
+ espintr(sc);
+#endif
if (xs->flags & ITSDONE)
break;
- DELAY(1);
+ DELAY(10);
+ if (sc->sc_state == ESP_IDLE) {
+ ESP_TRACE(("[esp_poll: rescheduling] "));
+ esp_sched(sc);
+ }
count--;
}
if (count == 0) {
- ecb->xs->error = XS_TIMEOUT;
- sc_print_addr(ecb->xs->sc_link);
- printf("timed out\n");
- esp_reset(sc);
+ ESP_MISC(("esp_poll: timeout"));
+ esp_timeout((caddr_t)ecb);
}
- dmaenintr(sc->sc_dma);
+
return COMPLETE;
}
/*
- * Notice that we do not read the live register on an ESP100. On the
- * ESP100A and above the FE (Feature Enable) bit in config 2 latches
- * the phase in the register so it is safe to read.
+ * LOW LEVEL SCSI UTILITIES
*/
-int
-espphase(sc)
- struct esp_softc *sc;
-{
-
- if (sc->sc_rev > ESP100)
- return (sc->sc_regs->espr_stat & ESPSTAT_PHASE);
- return (sc->sc_espstat & ESPSTAT_PHASE);
-}
-
/*
* Schedule a scsi operation. This has now been pulled out of the interrupt
* handler so that we may call it from esp_scsi_cmd and esp_done. This may
* save us an unecessary interrupt just to get things going. Should only be
- * called when state == ESPS_IDLE and at bio pl.
+ * called when state == ESP_IDLE and at bio pl.
*/
void
esp_sched(sc)
@@ -645,29 +729,40 @@ esp_sched(sc)
struct scsi_link *sc_link;
struct ecb *ecb;
int t;
-
- ESP_TRACE(("esp_sched\n"));
+
+ ESP_TRACE(("[esp_sched] "));
+ if (sc->sc_state != ESP_IDLE)
+ panic("esp_sched: not IDLE (state=%d)", sc->sc_state);
+
+ if (sc->sc_flags & ESP_ABORTING)
+ return;
/*
* Find first ecb in ready queue that is for a target/lunit
* combinations that is not busy.
*/
for (ecb = sc->ready_list.tqh_first; ecb; ecb = ecb->chain.tqe_next) {
- caddr_t cmd = (caddr_t) &ecb->cmd;
sc_link = ecb->xs->sc_link;
t = sc_link->target;
if (!(sc->sc_tinfo[t].lubusy & (1 << sc_link->lun))) {
- struct esp_tinfo *ti = &sc->sc_tinfo[ecb->xs->sc_link->target];
+ struct esp_tinfo *ti = &sc->sc_tinfo[t];
+ if ((ecb->flags & ECB_QBITS) != ECB_QREADY)
+ panic("esp: busy entry on ready list");
TAILQ_REMOVE(&sc->ready_list, ecb, chain);
+ ECB_SETQ(ecb, ECB_QNONE);
sc->sc_nexus = ecb;
sc->sc_flags = 0;
- sc_link = ecb->xs->sc_link;
- espselect(sc);
- ti = &sc->sc_tinfo[sc_link->target];
+ sc->sc_prevphase = INVALID_PHASE;
sc->sc_dp = ecb->daddr;
sc->sc_dleft = ecb->dleft;
ti->lubusy |= (1<<sc_link->lun);
+/*XXX*/if (sc->sc_msgpriq) {
+ printf("esp: message queue not empty: %x!\n", sc->sc_msgpriq);
+}
+/*XXX*/sc->sc_msgpriq = sc->sc_msgout = 0;
+ espselect(sc, t, sc_link->lun,
+ (u_char *)&ecb->cmd, ecb->clen);
break;
} else
ESP_MISC(("%d:%d busy\n", t, sc_link->lun));
@@ -684,24 +779,27 @@ esp_done(ecb)
struct scsi_xfer *xs = ecb->xs;
struct scsi_link *sc_link = xs->sc_link;
struct esp_softc *sc = sc_link->adapter_softc;
+ struct esp_tinfo *ti = &sc->sc_tinfo[sc_link->target];
- ESP_TRACE(("esp_done "));
+ ESP_TRACE(("[esp_done(error:%x)] ", xs->error));
+
+ untimeout(esp_timeout, ecb);
/*
- * Now, if we've come here with no error code, i.e. we've kept the
+ * Now, if we've come here with no error code, i.e. we've kept the
* initial XS_NOERROR, and the status code signals that we should
- * check sense, we'll need to set up a request sense cmd block and
- * push the command back into the ready queue *before* any other
+ * check sense, we'll need to set up a request sense cmd block and
+ * push the command back into the ready queue *before* any other
* commands for this target/lunit, else we lose the sense info.
* We don't support chk sense conditions for the request sense cmd.
*/
if (xs->error == XS_NOERROR) {
- if (ecb->flags & ECB_CHKSENSE)
+ if ((ecb->flags & ECB_ABORTED) != 0) {
+ xs->error = XS_TIMEOUT;
+ } else if ((ecb->flags & ECB_CHKSENSE) != 0) {
xs->error = XS_SENSE;
- else if ((ecb->stat & ST_MASK) == SCSI_CHECK) {
+ } else if ((ecb->stat & ST_MASK) == SCSI_CHECK) {
struct scsi_sense *ss = (void *)&ecb->cmd;
-
- ESPD(("RS "));
ESP_MISC(("requesting sense "));
/* First, save the return values */
xs->resid = ecb->dleft;
@@ -709,35 +807,43 @@ esp_done(ecb)
/* Next, setup a request sense command block */
bzero(ss, sizeof(*ss));
ss->opcode = REQUEST_SENSE;
- ss->byte2 = sc_link->lun << SCSI_CMD_LUN_SHIFT;
+ /*ss->byte2 = sc_link->lun << 5;*/
ss->length = sizeof(struct scsi_sense_data);
ecb->clen = sizeof(*ss);
ecb->daddr = (char *)&xs->sense;
ecb->dleft = sizeof(struct scsi_sense_data);
- ecb->flags = ECB_ACTIVE|ECB_CHKSENSE;
+ ecb->flags |= ECB_CHKSENSE;
+/*XXX - must take off queue here */
+ if (ecb != sc->sc_nexus) {
+ panic("%s: esp_sched: floating ecb %p",
+ sc->sc_dev.dv_xname, ecb);
+ }
TAILQ_INSERT_HEAD(&sc->ready_list, ecb, chain);
- sc->sc_tinfo[sc_link->target].lubusy &=
- ~(1<<sc_link->lun);
- sc->sc_tinfo[sc_link->target].senses++;
- /* found it */
- if (sc->sc_nexus != ecb)
- TAILQ_INSERT_HEAD(&sc->ready_list, ecb, chain);
- sc->sc_state = ESPS_IDLE;
- esp_sched(sc);
+ ECB_SETQ(ecb, ECB_QREADY);
+ ti->lubusy &= ~(1<<sc_link->lun);
+ ti->senses++;
+ timeout(esp_timeout, ecb, (xs->timeout*hz)/1000);
+ if (sc->sc_nexus == ecb) {
+ sc->sc_nexus = NULL;
+ sc->sc_state = ESP_IDLE;
+ esp_sched(sc);
+ }
return;
- } else
+ } else {
xs->resid = ecb->dleft;
+ }
}
xs->flags |= ITSDONE;
-#if ESP_DEBUG > 1
+#ifdef ESP_DEBUG
if (esp_debug & ESP_SHOWMISC) {
- printf("err=0x%02x ", xs->error);
- if (xs->error == XS_SENSE)
- printf("sense=%02x\n", xs->sense.error_code);
+ printf("err=0x%02x ",xs->error);
+ if (xs->error == XS_SENSE) {
+ printf("sense=%2x; ", xs->sense.error_code);
+ }
}
- if ((xs->resid || xs->error > XS_SENSE) && (esp_debug & ESP_SHOWMISC)) {
+ if ((xs->resid || xs->error > XS_SENSE) && esp_debug & ESP_SHOWMISC) {
if (xs->resid)
printf("esp_done: resid=%d\n", xs->resid);
if (xs->error)
@@ -746,45 +852,41 @@ esp_done(ecb)
#endif
/*
- * Remove the ECB from whatever queue it's on. We have to do a bit of
- * a hack to figure out which queue it's on. Note that it is *not*
- * necessary to cdr down the ready queue, but we must cdr down the
- * nexus queue and see if it's there, so we can mark the unit as no
- * longer busy. This code is sickening, but it works.
+ * Remove the ECB from whatever queue it's on.
*/
- if (ecb == sc->sc_nexus) {
- sc->sc_tinfo[sc_link->target].lubusy &= ~(1<<sc_link->lun);
+ switch (ecb->flags & ECB_QBITS) {
+ case ECB_QNONE:
+ if (ecb != sc->sc_nexus) {
+ panic("%s: floating ecb", sc->sc_dev.dv_xname);
+ }
sc->sc_nexus = NULL;
- sc->sc_state = ESPS_IDLE;
+ sc->sc_state = ESP_IDLE;
+ ti->lubusy &= ~(1<<sc_link->lun);
esp_sched(sc);
- } else if (sc->ready_list.tqh_last == &ecb->chain.tqe_next) {
+ break;
+ case ECB_QREADY:
TAILQ_REMOVE(&sc->ready_list, ecb, chain);
- } else {
- register struct ecb *ecb2;
-
- for (ecb2 = sc->nexus_list.tqh_first; ecb2;
- ecb2 = ecb2->chain.tqe_next)
- if (ecb2 == ecb)
- break;
- if (ecb2) {
- TAILQ_REMOVE(&sc->nexus_list, ecb, chain);
- sc->sc_tinfo[sc_link->target].lubusy &=
- ~(1<<sc_link->lun);
- } else if (ecb->chain.tqe_next) {
- TAILQ_REMOVE(&sc->ready_list, ecb, chain);
- } else {
- printf("%s: can't find matching ecb\n",
- sc->sc_dev.dv_xname);
- Debugger();
- }
+ break;
+ case ECB_QNEXUS:
+ TAILQ_REMOVE(&sc->nexus_list, ecb, chain);
+ ti->lubusy &= ~(1<<sc_link->lun);
+ break;
+ case ECB_QFREE:
+ panic("%s: dequeue: busy ecb on free list",
+ sc->sc_dev.dv_xname);
+ break;
+ default:
+ panic("%s: dequeue: unknown queue %d",
+ sc->sc_dev.dv_xname, ecb->flags & ECB_QBITS);
}
- /* Put it on the free list. */
- ecb->flags = ECB_FREE;
+
+ /* Put it on the free list, and clear flags. */
TAILQ_INSERT_HEAD(&sc->free_list, ecb, chain);
+ ecb->flags = ECB_QFREE;
- sc->sc_tinfo[sc_link->target].cmds++;
- ESPD(("DONE%d ", xs->resid));
+ ti->cmds++;
scsi_done(xs);
+ return;
}
/*
@@ -797,13 +899,13 @@ esp_done(ecb)
* else there will be an illegal command interrupt.
*/
#define esp_sched_msgout(m) \
- do { \
- sc->sc_regs->espr_cmd = ESPCMD_SETATN; \
- sc->sc_msgpriq |= (m); \
- ESPD(("Mq%02x ", sc->sc_msgpriq)); \
+ do { \
+ ESP_MISC(("esp_sched_msgout %d ", m)); \
+ ESPCMD(sc, ESPCMD_SETATN); \
+ sc->sc_msgpriq |= (m); \
} while (0)
-#define IS1BYTEMSG(m) (((m) != 1 && ((unsigned)(m)) < 0x20) || (m) & 0x80)
+#define IS1BYTEMSG(m) (((m) != 1 && (m) < 0x20) || (m) & 0x80)
#define IS2BYTEMSG(m) (((m) & 0xf0) == 0x20)
#define ISEXTMSG(m) ((m) == 1)
@@ -817,22 +919,16 @@ void
esp_msgin(sc)
register struct esp_softc *sc;
{
- struct espregs *espr = sc->sc_regs;
- struct ecb *ecb;
- int extlen;
- int x, i;
-
- ESP_TRACE(("esp_msgin "));
-
- /* is something wrong ? */
- if (sc->sc_phase != MESSAGE_IN_PHASE) {
- printf("%s: not MESSAGE_IN_PHASE\n", sc->sc_dev.dv_xname);
+ register int v;
+
+ ESP_TRACE(("[esp_msgin(curmsglen:%d)] ", sc->sc_imlen));
+
+ if ((ESP_READ_REG(sc, ESP_FFLAG) & ESPFIFO_FF) == 0) {
+ printf("%s: msgin: no msg byte available\n",
+ sc->sc_dev.dv_xname);
return;
}
- ESPD(("MSGIN:%d ", espr->espr_fflag & ESPFIFO_FF));
-
-#ifdef fixme
/*
* Prepare for a new message. A message should (according
* to the SCSI standard) be transmitted in one single
@@ -840,187 +936,125 @@ esp_msgin(sc)
* then this is a new message.
*/
if (sc->sc_prevphase != MESSAGE_IN_PHASE) {
- sc->sc_flags &= ~ESPF_DROP_MSGI;
- if (sc->sc_imlen > 0) {
- printf("%s: message type %02x",
- sc->sc_dev.dv_xname, sc->sc_imess[0]);
- if (!IS1BYTEMSG(sc->sc_imess[0]))
- printf(" %02x", sc->sc_imess[1]);
- printf(" was dropped\n");
- }
+ sc->sc_flags &= ~ESP_DROP_MSGI;
sc->sc_imlen = 0;
}
-#endif
- /*
- * Which target is reselecting us? (The ID bit really)
- * On a reselection, there should be the reselecting target's
- * id and an identify message in the fifo.
- */
- if (sc->sc_state == ESPS_RESELECTED && sc->sc_imlen == 0) {
- x = espgetbyte(sc, 0);
- if (x == -1) {
- printf("%s: msgin reselection found fifo empty\n",
- sc->sc_dev.dv_xname);
- return;
- }
- ESPD(("ID=%02x ", x));
- sc->sc_selid = (u_char)x & ~(1<<sc->sc_id);
- ESP_MISC(("selid=0x%02x ", sc->sc_selid));
- }
+ v = ESP_READ_REG(sc, ESP_FIFO);
+ ESP_MISC(("<msgbyte:0x%02x>", v));
- /*
- * If parity errors just dump everything on the floor
- */
- if (sc->sc_espstat & ESPSTAT_PE) {
- printf("%s: SCSI bus parity error\n",
- sc->sc_dev.dv_xname);
- esp_sched_msgout(SEND_PARITY_ERROR);
- DELAY(1);
- sc->sc_flags |= ESPF_DROP_MSGI;
+#if 0
+ if (sc->sc_state == ESP_RESELECTED && sc->sc_imlen == 0) {
+ /*
+ * Which target is reselecting us? (The ID bit really)
+ */
+ sc->sc_selid = v;
+ sc->sc_selid &= ~(1<<sc->sc_id);
+ ESP_MISC(("selid=0x%2x ", sc->sc_selid));
+ return;
}
+#endif
+
+ sc->sc_imess[sc->sc_imlen] = v;
/*
* If we're going to reject the message, don't bother storing
* the incoming bytes. But still, we need to ACK them.
- * XXX: the above comment might be true -- this is not being
- * done though!
*/
- if ((sc->sc_flags & ESPF_DROP_MSGI) == 0) {
- x = espgetbyte(sc, 1);
- if (x == -1) {
- printf("%s: msgin fifo empty at %d\n",
- sc->sc_dev.dv_xname, sc->sc_imlen);
- if (sc->sc_espintr & ESPINTR_BS) {
- espr->espr_cmd = ESPCMD_TRANS;
- ESPD(("MSI:y "));
- } else
- ESPD(("MSI:n "));
- return;
- }
- ESPD(("{%02x} ", (u_char)x));
- sc->sc_imess[sc->sc_imlen] = (u_char)x;
- if (sc->sc_imlen >= ESP_MSGLEN_MAX) {
- esp_sched_msgout(SEND_REJECT);
- DELAY(1);
- sc->sc_flags |= ESPF_DROP_MSGI;
- } else {
- sc->sc_imlen++;
- /*
- * This testing is suboptimal, but most
- * messages will be of the one byte variety, so
- * it should not effect performance
- * significantly.
- */
- if (sc->sc_imlen == 1 && IS1BYTEMSG(sc->sc_imess[0]))
- goto gotit;
- if (sc->sc_imlen == 2 && IS2BYTEMSG(sc->sc_imess[0]))
- goto gotit;
- if (sc->sc_imlen >= 3 && ISEXTMSG(sc->sc_imess[0]) &&
- sc->sc_imlen == sc->sc_imess[1] + 2)
- goto gotit;
- }
- }
- goto done;
-gotit:
- ESPD(("MP["));
- for (i=0; i<sc->sc_imlen; i++)
- ESPD(("%02x%c", sc->sc_imess[i],
- (i==sc->sc_imlen-1) ? ']' : ' '));
- ESPD((" "));
+ if ((sc->sc_flags & ESP_DROP_MSGI)) {
+ ESPCMD(sc, ESPCMD_SETATN);
+ ESPCMD(sc, ESPCMD_MSGOK);
+ printf("<dropping msg byte %x>",
+ sc->sc_imess[sc->sc_imlen]);
+ return;
+ }
- sc->sc_imlen = 0; /* message is fully received */
+ if (sc->sc_imlen >= ESP_MAX_MSG_LEN) {
+ esp_sched_msgout(SEND_REJECT);
+ sc->sc_flags |= ESP_DROP_MSGI;
+ } else {
+ sc->sc_imlen++;
+ /*
+ * This testing is suboptimal, but most
+ * messages will be of the one byte variety, so
+ * it should not effect performance
+ * significantly.
+ */
+ if (sc->sc_imlen == 1 && IS1BYTEMSG(sc->sc_imess[0]))
+ goto gotit;
+ if (sc->sc_imlen == 2 && IS2BYTEMSG(sc->sc_imess[0]))
+ goto gotit;
+ if (sc->sc_imlen >= 3 && ISEXTMSG(sc->sc_imess[0]) &&
+ sc->sc_imlen == sc->sc_imess[1] + 2)
+ goto gotit;
+ }
+ /* Ack what we have so far */
+ ESPCMD(sc, ESPCMD_MSGOK);
+ return;
- ESP_MISC(("gotmsg "));
+gotit:
+ ESP_MSGS(("gotmsg(%x)", sc->sc_imess[0]));
/*
* Now we should have a complete message (1 byte, 2 byte
* and moderately long extended messages). We only handle
* extended messages which total length is shorter than
- * ESP_MSGLEN_MAX. Longer messages will be amputated.
+ * ESP_MAX_MSG_LEN. Longer messages will be amputated.
*/
- if (sc->sc_state == ESPS_NEXUS ||
- sc->sc_state == ESPS_DOINGMSGIN ||
- sc->sc_state == ESPS_DOINGSTATUS) {
- struct esp_tinfo *ti;
-
- ecb = sc->sc_nexus;
- ti = &sc->sc_tinfo[ecb->xs->sc_link->target];
-
- sc->sc_state = ESPS_NEXUS; /* where we go after this */
+ if (sc->sc_state == ESP_HASNEXUS) {
+ struct ecb *ecb = sc->sc_nexus;
+ struct esp_tinfo *ti = &sc->sc_tinfo[ecb->xs->sc_link->target];
switch (sc->sc_imess[0]) {
case MSG_CMDCOMPLETE:
- ESP_MISC(("cmdcomplete "));
- if (!ecb) {
- esp_sched_msgout(SEND_ABORT);
- DELAY(1);
- printf("%s: CMDCOMPLETE but no command?\n",
- sc->sc_dev.dv_xname);
- break;
- }
+ ESP_MSGS(("cmdcomplete "));
if (sc->sc_dleft < 0) {
- sc_print_addr(ecb->xs->sc_link);
- printf("%d extra bytes\n", -sc->sc_dleft);
- ecb->dleft = 0;
+ struct scsi_link *sc_link = ecb->xs->sc_link;
+ printf("esp: %d extra bytes from %d:%d\n",
+ -sc->sc_dleft,
+ sc_link->target, sc_link->lun);
+ sc->sc_dleft = 0;
}
ecb->xs->resid = ecb->dleft = sc->sc_dleft;
- ecb->flags |= ECB_DONE;
+ sc->sc_flags |= ESP_BUSFREE_OK;
break;
+
case MSG_MESSAGE_REJECT:
- if (esp_debug & ESP_SHOWMISC)
- printf("%s targ %d: our msg rejected by target\n",
- sc->sc_dev.dv_xname,
- ecb->xs->sc_link->target);
- if (sc->sc_flags & ESPF_SYNCHNEGO) {
- /*
- * Device doesn't even know what sync is
- * (right?)
- */
- ti->offset = 0;
- ti->period = sc->sc_minsync;
- ti->synctp = 250 / ti->period;
- sc->sc_flags &= ~ESPF_SYNCHNEGO;
- ti->flags &= ~DO_NEGOTIATE;
+ if (esp_debug & ESP_SHOWMSGS)
+ printf("%s: our msg rejected by target\n",
+ sc->sc_dev.dv_xname);
+#if 1 /* XXX - must remember last message */
+sc_print_addr(ecb->xs->sc_link); printf("MSG_MESSAGE_REJECT>>");
+#endif
+ if (sc->sc_flags & ESP_SYNCHNEGO) {
+ ti->period = ti->offset = 0;
+ sc->sc_flags &= ~ESP_SYNCHNEGO;
+ ti->flags &= ~T_NEGOTIATE;
}
/* Not all targets understand INITIATOR_DETECTED_ERR */
- if (sc->sc_msgout == SEND_INIT_DET_ERR) {
+ if (sc->sc_msgout == SEND_INIT_DET_ERR)
esp_sched_msgout(SEND_ABORT);
- DELAY(1);
- }
break;
case MSG_NOOP:
+ ESP_MSGS(("noop "));
break;
case MSG_DISCONNECT:
- if (!ecb) {
- esp_sched_msgout(SEND_ABORT);
- DELAY(1);
- printf("%s: nothing to DISCONNECT\n",
- sc->sc_dev.dv_xname);
- break;
- }
+ ESP_MSGS(("disconnect "));
ti->dconns++;
- TAILQ_INSERT_HEAD(&sc->nexus_list, ecb, chain);
- ecb = sc->sc_nexus = NULL;
- sc->sc_state = ESPS_EXPECTDISC;
- sc->sc_flags |= ESPF_MAYDISCON;
- break;
- case MSG_SAVEDATAPOINTER:
- if (!ecb) {
- esp_sched_msgout(SEND_ABORT);
- DELAY(1);
- printf("%s: no DATAPOINTERs to save\n",
- sc->sc_dev.dv_xname);
+ sc->sc_flags |= ESP_DISCON;
+ sc->sc_flags |= ESP_BUSFREE_OK;
+ if ((ecb->xs->sc_link->quirks & SDEV_AUTOSAVE) == 0)
break;
- }
- ESPD(("DL%d/%d ", sc->sc_dleft, ecb->dleft));
+ /*FALLTHROUGH*/
+ case MSG_SAVEDATAPOINTER:
+ ESP_MSGS(("save datapointer "));
ecb->dleft = sc->sc_dleft;
ecb->daddr = sc->sc_dp;
break;
case MSG_RESTOREPOINTERS:
+ ESP_MSGS(("restore datapointer "));
if (!ecb) {
esp_sched_msgout(SEND_ABORT);
- DELAY(1);
printf("%s: no DATAPOINTERs to restore\n",
sc->sc_dev.dv_xname);
break;
@@ -1028,43 +1062,95 @@ gotit:
sc->sc_dp = ecb->daddr;
sc->sc_dleft = ecb->dleft;
break;
+ case MSG_PARITY_ERROR:
+ printf("%s:target%d: MSG_PARITY_ERROR\n",
+ sc->sc_dev.dv_xname,
+ ecb->xs->sc_link->target);
+ break;
case MSG_EXTENDED:
+ ESP_MSGS(("extended(%x) ", sc->sc_imess[2]));
switch (sc->sc_imess[2]) {
case MSG_EXT_SDTR:
+ ESP_MSGS(("SDTR period %d, offset %d ",
+ sc->sc_imess[3], sc->sc_imess[4]));
ti->period = sc->sc_imess[3];
ti->offset = sc->sc_imess[4];
- sc->sc_flags &= ~ESPF_SYNCHNEGO;
- ti->flags &= ~DO_NEGOTIATE;
- if (ti->offset == 0 ||
- ti->offset > ESP_SYNC_MAXOFF ||
- ti->period < sc->sc_minsync) {
+ if (sc->sc_minsync == 0) {
+ /* We won't do synch */
ti->offset = 0;
- ti->period = sc->sc_minsync;
- ti->synctp = 250 / ti->period;
- break;
+ esp_sched_msgout(SEND_SDTR);
+ } else if (ti->offset == 0) {
+ printf("%s:%d: async\n", "esp",
+ ecb->xs->sc_link->target);
+ ti->offset = 0;
+ sc->sc_flags &= ~ESP_SYNCHNEGO;
+ } else if (ti->period > 124) {
+ printf("%s:%d: async\n", "esp",
+ ecb->xs->sc_link->target);
+ ti->offset = 0;
+ esp_sched_msgout(SEND_SDTR);
+ } else {
+ int r = 250/ti->period;
+ int s = (100*250)/ti->period - 100*r;
+ int p;
+ p = esp_stp2cpb(sc, ti->period);
+ ti->period = esp_cpb2stp(sc, p);
+#ifdef ESP_DEBUG
+ sc_print_addr(ecb->xs->sc_link);
+#endif
+ if ((sc->sc_flags&ESP_SYNCHNEGO) == 0) {
+ /* Target initiated negotiation */
+ if (ti->flags & T_SYNCMODE) {
+ ti->flags &= ~T_SYNCMODE;
+#ifdef ESP_DEBUG
+ printf("renegotiated ");
+#endif
+ }
+ ESP_WRITE_REG(sc, ESP_SYNCOFF,
+ 0);
+ ESP_WRITE_REG(sc, ESP_SYNCTP,
+ 0);
+ /* Clamp to our maxima */
+ if (ti->period < sc->sc_minsync)
+ ti->period = sc->sc_minsync;
+ if (ti->offset > 15)
+ ti->offset = 15;
+ esp_sched_msgout(SEND_SDTR);
+ } else {
+ /* we are sync */
+ sc->sc_flags &= ~ESP_SYNCHNEGO;
+ ESP_WRITE_REG(sc, ESP_SYNCOFF,
+ ti->offset);
+ ESP_WRITE_REG(sc, ESP_SYNCTP,
+ p);
+ ti->flags |= T_SYNCMODE;
+ }
+#ifdef ESP_DEBUG
+ printf("max sync rate %d.%02dMb/s\n",
+ r, s);
+#endif
}
- printf("%s targ %d: sync, offset %d, period %dnsec\n",
- sc->sc_dev.dv_xname, ecb->xs->sc_link->target,
- ti->offset, ti->period * 4);
- ti->synctp = 250 / ti->period;
+ ti->flags &= ~T_NEGOTIATE;
break;
- default:
- /* Extended messages we don't handle */
- esp_sched_msgout(SEND_REJECT);
- DELAY(1);
+ default: /* Extended messages we don't handle */
+ ESPCMD(sc, ESPCMD_SETATN);
break;
}
break;
default:
- /* thanks for that ident... XXX: we should check it? */
+ ESP_MSGS(("ident "));
+ /* thanks for that ident... */
if (!MSG_ISIDENTIFY(sc->sc_imess[0])) {
- esp_sched_msgout(SEND_REJECT);
- DELAY(1);
+ ESP_MISC(("unknown "));
+printf("%s: unimplemented message: %d\n", sc->sc_dev.dv_xname, sc->sc_imess[0]);
+ ESPCMD(sc, ESPCMD_SETATN);
}
break;
}
- } else if (sc->sc_state == ESPS_RESELECTED) {
- struct scsi_link *sc_link;
+ } else if (sc->sc_state == ESP_RESELECTED) {
+ struct scsi_link *sc_link = NULL;
+ struct ecb *ecb;
+ struct esp_tinfo *ti;
u_char lunit;
if (MSG_ISIDENTIFY(sc->sc_imess[0])) { /* Identify? */
@@ -1073,115 +1159,65 @@ gotit:
* Search wait queue for disconnected cmd
* The list should be short, so I haven't bothered with
* any more sophisticated structures than a simple
- * singly linked list.
+ * singly linked list.
*/
lunit = sc->sc_imess[0] & 0x07;
for (ecb = sc->nexus_list.tqh_first; ecb;
- ecb = ecb->chain.tqe_next) {
+ ecb = ecb->chain.tqe_next) {
sc_link = ecb->xs->sc_link;
if (sc_link->lun == lunit &&
sc->sc_selid == (1<<sc_link->target)) {
TAILQ_REMOVE(&sc->nexus_list, ecb,
chain);
+ ECB_SETQ(ecb, ECB_QNONE);
break;
}
}
- if (!ecb) {
- /* Invalid reselection! */
+ if (!ecb) { /* Invalid reselection! */
esp_sched_msgout(SEND_ABORT);
- DELAY(1);
printf("esp: invalid reselect (idbit=0x%2x)\n",
sc->sc_selid);
- } else {
+ } else { /* Reestablish nexus */
/*
- * Reestablish nexus:
* Setup driver data structures and
* do an implicit RESTORE POINTERS
*/
+ ti = &sc->sc_tinfo[sc_link->target];
sc->sc_nexus = ecb;
sc->sc_dp = ecb->daddr;
sc->sc_dleft = ecb->dleft;
- sc->sc_tinfo[sc_link->target].lubusy |=
- (1<<sc_link->lun);
- espr->espr_syncoff =
- sc->sc_tinfo[sc_link->target].offset;
- espr->espr_synctp = 250 /
- sc->sc_tinfo[sc_link->target].period;
+ sc->sc_tinfo[sc_link->target].lubusy
+ |= (1<<sc_link->lun);
+ if (ti->flags & T_SYNCMODE) {
+ ESP_WRITE_REG(sc, ESP_SYNCOFF,
+ ti->offset);
+ ESP_WRITE_REG(sc, ESP_SYNCTP,
+ esp_stp2cpb(sc, ti->period));
+ } else {
+ ESP_WRITE_REG(sc, ESP_SYNCOFF, 0);
+ ESP_WRITE_REG(sc, ESP_SYNCTP, 0);
+ }
ESP_MISC(("... found ecb"));
- sc->sc_state = ESPS_NEXUS;
+ sc->sc_state = ESP_HASNEXUS;
}
} else {
printf("%s: bogus reselect (no IDENTIFY) %0x2x\n",
sc->sc_dev.dv_xname, sc->sc_selid);
esp_sched_msgout(SEND_DEV_RESET);
- DELAY(1);
}
} else { /* Neither ESP_HASNEXUS nor ESP_RESELECTED! */
printf("%s: unexpected message in; will send DEV_RESET\n",
sc->sc_dev.dv_xname);
esp_sched_msgout(SEND_DEV_RESET);
- DELAY(1);
}
-done:
- espr->espr_cmd = ESPCMD_MSGOK;
- espr->espr_cmd = ESPCMD_NOP;
-}
-
-void
-esp_makemsg(sc)
- register struct esp_softc *sc;
-{
- struct esp_tinfo *ti;
- struct ecb *ecb = sc->sc_nexus;
- int i;
+ /* Ack last message byte */
+ ESPCMD(sc, ESPCMD_MSGOK);
- sc->sc_msgout = sc->sc_msgpriq & -sc->sc_msgpriq;
- ESPD(("MQ%02x/%02x ", sc->sc_msgpriq, sc->sc_msgout));
-
- sc->sc_omlen = 1; /* "Default" message len */
- switch (sc->sc_msgout) {
- case SEND_SDTR: /* implies an IDENTIFY message */
- sc->sc_flags |= ESPF_SYNCHNEGO;
- ti = &sc->sc_tinfo[ecb->xs->sc_link->target];
- sc->sc_omess[0] = MSG_IDENTIFY(ecb->xs->sc_link->lun,
- !(ecb->xs->flags & SCSI_POLL));
- sc->sc_omess[1] = MSG_EXTENDED;
- sc->sc_omess[2] = 3;
- sc->sc_omess[3] = MSG_EXT_SDTR;
- sc->sc_omess[4] = ti->period;
- sc->sc_omess[5] = ti->offset;
- sc->sc_omlen = 6;
- /* Fallthrough! */
- case SEND_IDENTIFY:
- sc->sc_omess[0] = MSG_IDENTIFY(ecb->xs->sc_link->lun,
- !(ecb->xs->flags & SCSI_POLL));
- break;
- case SEND_DEV_RESET:
- sc->sc_omess[0] = MSG_BUS_DEV_RESET;
- if (sc->sc_nexus)
- sc->sc_nexus->flags |= ECB_DONE;
- break;
- case SEND_PARITY_ERROR:
- sc->sc_omess[0] = MSG_PARITY_ERROR;
- break;
- case SEND_ABORT:
- sc->sc_omess[0] = MSG_ABORT;
- if (sc->sc_nexus)
- sc->sc_nexus->flags |= ECB_DONE;
- break;
- case SEND_INIT_DET_ERR:
- sc->sc_omess[0] = MSG_INITIATOR_DET_ERR;
- break;
- case SEND_REJECT:
- sc->sc_omess[0] = MSG_MESSAGE_REJECT;
- break;
- default:
- sc->sc_omess[0] = MSG_NOOP;
- break;
- }
- sc->sc_omp = sc->sc_omess;
+ /* Done, reset message pointer. */
+ sc->sc_flags &= ~ESP_DROP_MSGI;
+ sc->sc_imlen = 0;
}
@@ -1192,650 +1228,746 @@ void
esp_msgout(sc)
register struct esp_softc *sc;
{
- struct espregs *espr = sc->sc_regs;
struct esp_tinfo *ti;
struct ecb *ecb;
- int i;
-
- /* Pick up highest priority message */
- sc->sc_msgout = sc->sc_msgpriq & -sc->sc_msgpriq;
- esp_makemsg(sc);
-
- for (i = 0; i < sc->sc_omlen; i++)
- espr->espr_fifo = sc->sc_omp[i];
- espr->espr_cmd = ESPCMD_TRANS;
-}
-
+ size_t size;
-void
-esp_timeout(arg)
- void *arg;
-{
- struct ecb *ecb = (struct ecb *)arg;
- struct esp_softc *sc;
+ ESP_TRACE(("[esp_msgout(priq:%x, prevphase:%x)]", sc->sc_msgpriq, sc->sc_prevphase));
- sc = ecb->xs->sc_link->adapter_softc;
- sc_print_addr(ecb->xs->sc_link);
- printf("timed out at %d %d\n", sc->sc_state, sc->sc_phase);
+ if (sc->sc_prevphase != MESSAGE_OUT_PHASE) {
+ /* Pick up highest priority message */
+ sc->sc_msgout = sc->sc_msgpriq & -sc->sc_msgpriq;
+ sc->sc_omlen = 1; /* "Default" message len */
+ switch (sc->sc_msgout) {
+ case SEND_SDTR:
+ ecb = sc->sc_nexus;
+ ti = &sc->sc_tinfo[ecb->xs->sc_link->target];
+ sc->sc_omess[0] = MSG_EXTENDED;
+ sc->sc_omess[1] = 3;
+ sc->sc_omess[2] = MSG_EXT_SDTR;
+ sc->sc_omess[3] = ti->period;
+ sc->sc_omess[4] = ti->offset;
+ sc->sc_omlen = 5;
+ break;
+ case SEND_IDENTIFY:
+ if (sc->sc_state != ESP_HASNEXUS) {
+ printf("esp at line %d: no nexus", __LINE__);
+ }
+ ecb = sc->sc_nexus;
+ sc->sc_omess[0] = MSG_IDENTIFY(ecb->xs->sc_link->lun,0);
+ break;
+ case SEND_DEV_RESET:
+ sc->sc_omess[0] = MSG_BUS_DEV_RESET;
+ sc->sc_flags |= ESP_BUSFREE_OK;
+ ecb = sc->sc_nexus;
+ ti = &sc->sc_tinfo[ecb->xs->sc_link->target];
+ ti->flags &= ~T_SYNCMODE;
+ ti->flags |= T_NEGOTIATE;
+ break;
+ case SEND_PARITY_ERROR:
+ sc->sc_omess[0] = MSG_PARITY_ERROR;
+ break;
+ case SEND_ABORT:
+ sc->sc_omess[0] = MSG_ABORT;
+ sc->sc_flags |= ESP_BUSFREE_OK;
+ break;
+ case SEND_INIT_DET_ERR:
+ sc->sc_omess[0] = MSG_INITIATOR_DET_ERR;
+ break;
+ case SEND_REJECT:
+ sc->sc_omess[0] = MSG_MESSAGE_REJECT;
+ break;
+ default:
+ sc->sc_omess[0] = MSG_NOOP;
+ break;
+ }
+ sc->sc_omp = sc->sc_omess;
+ }
- ecb->xs->error = XS_TIMEOUT;
- esp_reset(sc);
- esp_done(ecb);
- printf("new ecb %08x\n", sc->sc_nexus);
+#if 1
+ /* (re)send the message */
+ size = min(sc->sc_omlen, sc->sc_maxxfer);
+ DMA_SETUP(sc->sc_dma, &sc->sc_omp, &sc->sc_omlen, 0, &size);
+ /* Program the SCSI counter */
+ ESP_WRITE_REG(sc, ESP_TCL, size);
+ ESP_WRITE_REG(sc, ESP_TCM, size >> 8);
+ if (sc->sc_cfg2 & ESPCFG2_FE) {
+ ESP_WRITE_REG(sc, ESP_TCH, size >> 16);
+ }
+ /* load the count in */
+ ESPCMD(sc, ESPCMD_NOP|ESPCMD_DMA);
+ ESPCMD(sc, (size==0?ESPCMD_TRPAD:ESPCMD_TRANS)|ESPCMD_DMA);
+ DMA_GO(sc->sc_dma);
+#else
+ { int i;
+ ESPCMD(sc, ESPCMD_FLUSH);
+ for (i = 0; i < sc->sc_omlen; i++)
+ ESP_WRITE_REG(sc, FIFO, sc->sc_omess[i]);
+ ESPCMD(sc, ESPCMD_TRANS);
+#if test_stuck_on_msgout
+printf("<<XXXmsgoutdoneXXX>>");
+#endif
+ }
+#endif
}
/*
- * Read the ESP registers, and save their contents for later use.
- * ESP_STAT, ESP_STEP & ESP_INTR are mostly zeroed out when reading
- * ESP_INTR - so make sure it is the last read.
+ * This is the most critical part of the driver, and has to know
+ * how to deal with *all* error conditions and phases from the SCSI
+ * bus. If there are no errors and the DMA was active, then call the
+ * DMA pseudo-interrupt handler. If this returns 1, then that was it
+ * and we can return from here without further processing.
*
- * XXX: TDR: this logic seems unsound
- * I think that (from reading the docs) most bits in these registers
- * only make sense when the DMA CSR has an interrupt showing. So I have
- * coded this to not do anything if there is no interrupt or error
- * pending.
+ * Most of this needs verifying.
*/
-void
-espreadregs(sc)
- struct esp_softc *sc;
-{
- struct espregs *espr = sc->sc_regs;
-
- /* they mean nothing if the is no pending interrupt ??? */
- if (!(dmapending(sc->sc_dma)))
- return;
-
- /* Only the stepo bits are of interest */
- sc->sc_espstep = espr->espr_step & ESPSTEP_MASK;
- sc->sc_espstat = espr->espr_stat;
- sc->sc_espintr = espr->espr_intr;
-
- ESP_MISC(("regs[intr=%02x,stat=%02x,step=%02x] ", sc->sc_espintr,
- sc->sc_espstat, sc->sc_espstep));
-}
-
-/*
- * Whatever we do, we must generate an interrupt if we expect to go
- * to the next state.
- * Note: this increments the events even if called from esp_poll()
- */
-int
+int
espintr(sc)
register struct esp_softc *sc;
{
- struct espregs *espr = sc->sc_regs;
- register struct ecb *ecb = sc->sc_nexus;
+ register struct ecb *ecb;
register struct scsi_link *sc_link;
- u_char *cmd;
struct esp_tinfo *ti;
- int dmaintrwas, i;
+ int loop;
+ size_t size;
- /*
- * Revision 1 DMA's must have their interrupts disabled,
- * otherwise we can get bus timeouts while reading ESP
- * registers.
- */
- if (sc->sc_dma->sc_rev == DMAREV_1)
- dmaintrwas = dmadisintr(sc->sc_dma);
-
- espreadregs(sc);
+ ESP_TRACE(("[espintr]"));
/*
- * If either of these two things is true, we caused an interrupt.
- * If we didn't, let's get out of here.
+ * I have made some (maybe seriously flawed) assumptions here,
+ * but basic testing (uncomment the printf() below), show that
+ * certainly something happens when this loop is here.
+ *
+ * The idea is that many of the SCSI operations take very little
+ * time, and going away and getting interrupted is too high an
+ * overhead to pay. For example, selecting, sending a message
+ * and command and then doing some work can be done in one "pass".
+ *
+ * The DELAY is not variable because I do not understand that the
+ * DELAY loop should be fixed-time regardless of CPU speed, but
+ * I am *assuming* that the faster SCSI processors get things done
+ * quicker (sending a command byte etc), and so there is no
+ * need to be too slow.
+ *
+ * This is a heuristic. It is 2 when at 20Mhz, 2 at 25Mhz and 1
+ * at 40Mhz. This needs testing.
*/
- if ((sc->sc_espintr | dmapending(sc->sc_dma)) == 0) {
- if (sc->sc_dma->sc_rev == DMAREV_1 && dmaintrwas)
- dmaenintr(sc->sc_dma);
- return (0);
- }
+ for (loop = 0; 1;loop++, DELAY(50/sc->sc_freq)) {
+ /* a feeling of deja-vu */
+ if (!DMA_ISINTR(sc->sc_dma))
+ return (loop != 0);
+#if 0
+ if (loop)
+ printf("*");
+#endif
- sc->sc_phase = espphase(sc);
+ /* and what do the registers say... */
+ espreadregs(sc);
- ESPD(("I%02x/%02x/%02x ", sc->sc_espintr, sc->sc_state, sc->sc_phase));
+ sc->sc_intrcnt.ev_count++;
- if (sc->sc_espintr & ESPINTR_SBR) {
- espflush(sc);
- printf("%s: scsi bus reset\n", sc->sc_dev.dv_xname);
- esp_init(sc, 0); /* Restart everything */
/*
- * No interrupt expected, since there are now no
- * ecb's in progress.
+ * At the moment, only a SCSI Bus Reset or Illegal
+ * Command are classed as errors. A disconnect is a
+ * valid condition, and we let the code check is the
+ * "ESP_BUSFREE_OK" flag was set before declaring it
+ * and error.
+ *
+ * Also, the status register tells us about "Gross
+ * Errors" and "Parity errors". Only the Gross Error
+ * is really bad, and the parity errors are dealt
+ * with later
+ *
+ * TODO
+ * If there are too many parity error, go to slow
+ * cable mode ?
*/
- goto done;
- }
- if (sc->sc_espintr & ESPINTR_ILL) {
- printf("%s: illegal command %02x\n",
- sc->sc_dev.dv_xname, espr->espr_cmd);
- if (sc->sc_state == ESPS_NEXUS) {
- ecb->xs->error = XS_DRIVER_STUFFUP;
- untimeout(esp_timeout, ecb);
- esp_done(ecb);
+ /* SCSI Reset */
+ if (sc->sc_espintr & ESPINTR_SBR) {
+ if (ESP_READ_REG(sc, ESP_FFLAG) & ESPFIFO_FF) {
+ ESPCMD(sc, ESPCMD_FLUSH);
+ DELAY(1);
+ }
+ if (sc->sc_state != ESP_SBR) {
+ printf("%s: SCSI bus reset\n",
+ sc->sc_dev.dv_xname);
+ esp_init(sc, 0); /* Restart everything */
+ return 1;
+ }
+#if 0
+ /*XXX*/ printf("<expected bus reset: "
+ "[intr %x, stat %x, step %d]>\n",
+ sc->sc_espintr, sc->sc_espstat,
+ sc->sc_espstep);
+#endif
+ if (sc->sc_nexus)
+ panic("%s: nexus in reset state",
+ sc->sc_dev.dv_xname);
+ sc->sc_state = ESP_IDLE;
+ esp_sched(sc);
+ return 1;
}
- esp_reset(sc);
- esp_sched(sc); /* start next command */
- goto done;
- }
- if (sc->sc_espstat & ESPSTAT_GE) {
- /* no target ? */
- espflush(sc);
- printf("%s: gross error\n", sc->sc_dev.dv_xname);
- if (sc->sc_state == ESPS_NEXUS) {
- ecb->xs->error = XS_DRIVER_STUFFUP;
- untimeout(esp_timeout, ecb);
- esp_done(ecb);
- }
- esp_reset(sc);
- esp_sched(sc);
- /* started next command, which will interrupt us */
- goto done;
- }
+ ecb = sc->sc_nexus;
- /*
- * The `bad interrupts' have already been checked.
- */
-states:
- switch (sc->sc_state) {
- case ESPS_IDLE:
- ESPD(("I "));
+#define ESPINTR_ERR (ESPINTR_SBR|ESPINTR_ILL)
+ if (sc->sc_espintr & ESPINTR_ERR ||
+ sc->sc_espstat & ESPSTAT_GE) {
- sc->sc_nexus = NULL;
+ if (sc->sc_espstat & ESPSTAT_GE) {
+ /* no target ? */
+ if (ESP_READ_REG(sc, ESP_FFLAG) & ESPFIFO_FF) {
+ ESPCMD(sc, ESPCMD_FLUSH);
+ DELAY(1);
+ }
+ if (sc->sc_state == ESP_HASNEXUS ||
+ sc->sc_state == ESP_SELECTING) {
+ ecb->xs->error = XS_DRIVER_STUFFUP;
+ esp_done(ecb);
+ }
+ return 1;
+ }
- if (sc->sc_espintr & ESPINTR_RESEL) {
- sc->sc_state = ESPS_RESELECTED;
- goto states;
+ if (sc->sc_espintr & ESPINTR_ILL) {
+ /* illegal command, out of sync ? */
+ printf("%s: illegal command: 0x%x (state %d, phase %x, prevphase %x)\n",
+ sc->sc_dev.dv_xname, sc->sc_lastcmd,
+ sc->sc_state, sc->sc_phase,
+ sc->sc_prevphase);
+ if (ESP_READ_REG(sc, ESP_FFLAG) & ESPFIFO_FF) {
+ ESPCMD(sc, ESPCMD_FLUSH);
+ DELAY(1);
+ }
+ esp_init(sc, 0); /* Restart everything */
+ return 1;
+ }
}
- esp_sched(sc);
/*
- * We will get an interrupt if esp_sched() queues
- * anything to the scsi bus.
+ * Call if DMA is active.
+ *
+ * If DMA_INTR returns true, then maybe go 'round the loop
+ * again in case there is no more DMA queued, but a phase
+ * change is expected.
*/
- goto done;
- case ESPS_SELECTING:
+ if (DMA_ISACTIVE(sc->sc_dma)) {
+ DMA_INTR(sc->sc_dma);
+ /* If DMA active here, then go back to work... */
+ if (DMA_ISACTIVE(sc->sc_dma))
+ return 1;
+
+ if (sc->sc_dleft == 0 &&
+ (sc->sc_espstat & ESPSTAT_TC) == 0)
+ printf("%s: !TC [intr %x, stat %x, step %d]"
+ " prevphase %x, resid %x\n",
+ sc->sc_dev.dv_xname,
+ sc->sc_espintr,
+ sc->sc_espstat,
+ sc->sc_espstep,
+ sc->sc_prevphase,
+ ecb?ecb->dleft:-1);
+ }
+
+#if 0 /* Unreliable on some ESP revisions? */
+ if ((sc->sc_espstat & ESPSTAT_INT) == 0) {
+ printf("%s: spurious interrupt\n", sc->sc_dev.dv_xname);
+ return 1;
+ }
+#endif
+
/*
- * For a simple select, we sent either
- * ESPCMD_SELATN or ESPCMD_SELNATN
+ * check for less serious errors
*/
- ESPD(("S "));
-
- if (sc->sc_espintr & ESPINTR_DIS) {
- sc->sc_state = ESPS_NEXUS; /* will cleanup */
- goto states;
+ if (sc->sc_espstat & ESPSTAT_PE) {
+ printf("%s: SCSI bus parity error\n",
+ sc->sc_dev.dv_xname);
+ if (sc->sc_prevphase == MESSAGE_IN_PHASE)
+ esp_sched_msgout(SEND_PARITY_ERROR);
+ else
+ esp_sched_msgout(SEND_INIT_DET_ERR);
}
- if (sc->sc_espintr & ESPINTR_RESEL) {
+ if (sc->sc_espintr & ESPINTR_DIS) {
+ ESP_MISC(("<DISC [intr %x, stat %x, step %d]>",
+ sc->sc_espintr,sc->sc_espstat,sc->sc_espstep));
+ if (ESP_READ_REG(sc, ESP_FFLAG) & ESPFIFO_FF) {
+ ESPCMD(sc, ESPCMD_FLUSH);
+ DELAY(1);
+ }
/*
- * If we were trying to select a target,
- * push our command back into the ready list.
+ * This command must (apparently) be issued within
+ * 250mS of a disconnect. So here you are...
*/
- if (sc->sc_state == ESPS_SELECTING) {
- TAILQ_INSERT_HEAD(&sc->ready_list,
- sc->sc_nexus, chain);
- ecb = sc->sc_nexus = NULL;
- }
- sc->sc_state = ESPS_RESELECTED;
- goto states;
- }
-
- if (sc->sc_espintr == (ESPINTR_FC|ESPINTR_BS)) {
- ESPD(("STEP%d ", sc->sc_espstep));
- if (sc->sc_espstep == 0) {
- /* ATN: cannot happen */
- /* NATN: target did not assert message phase */
- sc->sc_state = ESPS_IDLE;
- goto states;
- } else if (sc->sc_espstep == 2) {
- /* BOTH: target did not assert command phase */
- sc->sc_state = ESPS_IDLE;
- goto states;
- } else if (sc->sc_espstep == 3) {
- /* BOTH: changed phase during cmd transfer */
- ESPD(("FF%d ", espr->espr_fflag & ESPFIFO_FF));
- if (espr->espr_fflag & ESPFIFO_FF) {
- sc->sc_state = ESPS_IDLE;
- goto states;
+ ESPCMD(sc, ESPCMD_ENSEL);
+ if (sc->sc_state != ESP_IDLE) {
+ if ((sc->sc_flags & ESP_SYNCHNEGO)) {
+#ifdef ESP_DEBUG
+ if (ecb)
+ sc_print_addr(ecb->xs->sc_link);
+ printf("sync nego not completed!\n");
+#endif
+ sc->sc_flags &= ~ESP_SYNCHNEGO;
+ sc->sc_tinfo[ecb->xs->sc_link->target].offset = 0;
+ sc->sc_tinfo[ecb->xs->sc_link->target].flags &= ~T_NEGOTIATE;
+ }
+ /*XXX*/sc->sc_msgpriq = sc->sc_msgout = 0;
+
+ /* it may be OK to disconnect */
+ if (!(sc->sc_flags & ESP_BUSFREE_OK)) {
+ if (sc->sc_state == ESP_HASNEXUS) {
+ sc_print_addr(ecb->xs->sc_link);
+ printf("disconnect without"
+ "warning\n");
+ }
+ ecb->xs->error = XS_TIMEOUT;
+ } else if (sc->sc_flags & ESP_DISCON) {
+ TAILQ_INSERT_HEAD(&sc->nexus_list, ecb, chain);
+ ECB_SETQ(ecb, ECB_QNEXUS);
+ sc->sc_nexus = NULL;
+ sc->sc_flags &= ~ESP_DISCON;
+ sc->sc_state = ESP_IDLE;
+#if ESP_DEBUG
+if ((esp_debug & 0x10000) && ecb->dleft == 0) {
+ printf("%s: silly disconnect (ecb %p [stat %x])\n",
+ sc->sc_dev.dv_xname, ecb, ecb->stat);
+}
+#endif
+ esp_sched(sc);
+ return 1;
}
- }
- espr->espr_cmd = ESPCMD_NOP; /* unlatch fifo counter */
- if (sc->sc_phase != DATA_IN_PHASE &&
- sc->sc_tinfo[ecb->xs->sc_link->target].offset == 0) {
- DELAY(1);
- espflush(sc);
+ esp_done(ecb);
+ return 1;
}
+ printf("%s: DISCONNECT in IDLE state!\n",
+ sc->sc_dev.dv_xname);
+ }
- ecb = sc->sc_nexus;
- if (!ecb)
- panic("esp: not nexus at sc->sc_nexus");
- sc_link = ecb->xs->sc_link;
- ti = &sc->sc_tinfo[sc_link->target];
- sc->sc_flags = 0;
-
- /* Clearly we succeeded at sending the message */
+ /* did a message go out OK ? This must be broken */
+ if (sc->sc_prevphase == MESSAGE_OUT_PHASE &&
+ sc->sc_phase != MESSAGE_OUT_PHASE) {
+ /* we have sent it */
+ if (sc->sc_msgout == SEND_SDTR &&
+ (sc->sc_flags & ESP_SYNCHNEGO) == 0) {
+ /* We've just accepted new sync parameters */
+ sc->sc_tinfo[ecb->xs->sc_link->target].flags |=
+ T_SYNCMODE;
+if (ecb) sc_print_addr(ecb->xs->sc_link);else printf("NO nexus: ");
+printf("target put in SYNC mode\n");
+ }
sc->sc_msgpriq &= ~sc->sc_msgout;
sc->sc_msgout = 0;
- if (sc->sc_msgpriq)
- espr->espr_cmd = ESPCMD_SETATN;
-
- sc->sc_dp = ecb->daddr; /* implicit RESTOREDATAPOINTERS */
- sc->sc_dleft = ecb->dleft;
- ti->lubusy |= (1<<sc_link->lun);
-
- sc->sc_state = ESPS_NEXUS;
}
- break; /* handle the phase */
- case ESPS_SELECTSTOP:
- /*
- * We wanted to select with multiple message bytes.
- * As a result, we needed to use SELATNS. One
- * message byte has been sent at this point. We need
- * to send the other bytes, then the target will either
- * switch to command phase to fetch the command or
- * send us a message.
- */
- ESPD(("SS%d ", sc->sc_espstep));
- if (sc->sc_espintr & ESPINTR_DIS) {
- sc->sc_state = ESPS_NEXUS;
- goto states;
- }
+ switch (sc->sc_state) {
+
+ case ESP_SBR:
+ printf("%s: waiting for SCSI Bus Reset to happen\n",
+ sc->sc_dev.dv_xname);
+ return 1;
- if (sc->sc_espintr & ESPINTR_RESEL) {
+ case ESP_RESELECTED:
/*
- * If we were trying to select a target,
- * push our command back into the ready list.
+ * we must be continuing a message ?
*/
- if (sc->sc_state == ESPS_SELECTING) {
- TAILQ_INSERT_HEAD(&sc->ready_list,
- sc->sc_nexus, chain);
- ecb = sc->sc_nexus = NULL;
+ if (sc->sc_phase != MESSAGE_IN_PHASE) {
+ printf("%s: target didn't identify\n",
+ sc->sc_dev.dv_xname);
+ esp_init(sc, 1);
+ return 1;
}
- sc->sc_state = ESPS_RESELECTED;
- goto states;
- }
+printf("<<RESELECT CONT'd>>");
+#if XXXX
+ esp_msgin(sc);
+ if (sc->sc_state != ESP_HASNEXUS) {
+ /* IDENTIFY fail?! */
+ printf("%s: identify failed\n",
+ sc->sc_dev.dv_xname);
+ esp_init(sc, 1);
+ return 1;
+ }
+#endif
+ break;
- if (sc->sc_espintr & (ESPINTR_BS|ESPINTR_FC)) {
- int i;
+ case ESP_IDLE:
+if (sc->sc_flags & ESP_ICCS) printf("[[esp: BUMMER]]");
+ case ESP_SELECTING:
- /*
- * Shove the remainder of the message bytes
- * into the fifo. After we send them, command
- * phase will request the command...
- */
- espflush(sc);
- for (i = 1; i < sc->sc_omlen; i++)
- espr->espr_fifo = sc->sc_omp[i];
- espr->espr_cmd = ESPCMD_TRANS;
- sc->sc_state = ESPS_MULTMSGEND;
- goto done;
- }
- goto done;
- case ESPS_MULTMSGEND:
- ESPD(("MME "));
- if (sc->sc_espintr & ESPINTR_DIS) {
- sc->sc_state = ESPS_NEXUS;
- goto states;
- }
+ if (sc->sc_espintr & ESPINTR_RESEL) {
+ /*
+ * If we're trying to select a
+ * target ourselves, push our command
+ * back into the ready list.
+ */
+ if (sc->sc_state == ESP_SELECTING) {
+ ESP_MISC(("backoff selector "));
+ sc_link = sc->sc_nexus->xs->sc_link;
+ ti = &sc->sc_tinfo[sc_link->target];
+ TAILQ_INSERT_HEAD(&sc->ready_list,
+ sc->sc_nexus, chain);
+ ECB_SETQ(sc->sc_nexus, ECB_QREADY);
+ ti->lubusy &= ~(1<<sc_link->lun);
+ ecb = sc->sc_nexus = NULL;
+ }
+ sc->sc_state = ESP_RESELECTED;
+ if (sc->sc_phase != MESSAGE_IN_PHASE) {
+ /*
+ * Things are seriously fucked up.
+ * Pull the brakes, i.e. reset
+ */
+ printf("%s: target didn't identify\n",
+ sc->sc_dev.dv_xname);
+ esp_init(sc, 1);
+ return 1;
+ }
+ if ((ESP_READ_REG(sc, ESP_FFLAG) & ESPFIFO_FF) != 2) {
+ printf("%s: RESELECT: %d bytes in FIFO!\n",
+ sc->sc_dev.dv_xname,
+ ESP_READ_REG(sc, ESP_FFLAG) &
+ ESPFIFO_FF);
+ esp_init(sc, 1);
+ return 1;
+ }
+ sc->sc_selid = ESP_READ_REG(sc, ESP_FIFO);
+ sc->sc_selid &= ~(1<<sc->sc_id);
+ ESP_MISC(("selid=0x%2x ", sc->sc_selid));
+ esp_msgin(sc); /* Handle identify message */
+ if (sc->sc_state != ESP_HASNEXUS) {
+ /* IDENTIFY fail?! */
+ printf("%s: identify failed\n",
+ sc->sc_dev.dv_xname);
+ esp_init(sc, 1);
+ return 1;
+ }
+ continue; /* ie. next phase expected soon */
+ }
- sc->sc_msgpriq &= ~sc->sc_msgout;
- sc->sc_msgout = 0;
+#define ESPINTR_DONE (ESPINTR_FC|ESPINTR_BS)
+ if ((sc->sc_espintr & ESPINTR_DONE) == ESPINTR_DONE) {
+ ecb = sc->sc_nexus;
+ if (!ecb)
+ panic("esp: not nexus at sc->sc_nexus");
- ESPD(("F%d ", espr->espr_fflag & ESPFIFO_FF));
- if (espr->espr_fflag & ESPFIFO_FF) {
- espflush(sc);
- DELAY(1);
- }
+ sc_link = ecb->xs->sc_link;
+ ti = &sc->sc_tinfo[sc_link->target];
+
+ switch (sc->sc_espstep) {
+ case 0:
+ printf("%s: select timeout/no disconnect\n",
+ sc->sc_dev.dv_xname);
+ esp_abort(sc, ecb);
+ return 1;
+ case 1:
+ if ((ti->flags & T_NEGOTIATE) == 0) {
+ printf("%s: step 1 & !NEG\n",
+ sc->sc_dev.dv_xname);
+ esp_abort(sc, ecb);
+ return 1;
+ }
+ if (sc->sc_phase != MESSAGE_OUT_PHASE) {
+ printf("%s: !MSGOUT\n",
+ sc->sc_dev.dv_xname);
+ esp_abort(sc, ecb);
+ return 1;
+ }
+ /* Start negotiating */
+ ti->period = sc->sc_minsync;
+ ti->offset = 15;
+ sc->sc_msgpriq = SEND_SDTR;
+ sc->sc_flags |= ESP_SYNCHNEGO;
+ break;
+ case 3:
+ /*
+ * Grr, this is supposed to mean
+ * "target left command phase
+ * prematurely". It seems to happen
+ * regularly when sync mode is on.
+ * Look at FIFO to see if command
+ * went out.
+ * (Timing problems?)
+ */
+ if ((ESP_READ_REG(sc, ESP_FFLAG)&ESPFIFO_FF) == 0) {
+ /* Hope for the best.. */
+ break;
+ }
+ printf("(%s:%d:%d): selection failed;"
+ " %d left in FIFO "
+ "[intr %x, stat %x, step %d]\n",
+ sc->sc_dev.dv_xname,
+ sc_link->target,
+ sc_link->lun,
+ ESP_READ_REG(sc, ESP_FFLAG) & ESPFIFO_FF,
+ sc->sc_espintr, sc->sc_espstat,
+ sc->sc_espstep);
+ ESPCMD(sc, ESPCMD_FLUSH);
+ sc->sc_flags |= ESP_ABORTING;
+ esp_sched_msgout(SEND_ABORT);
+ return 1;
+ case 2:
+ /* Select stuck at Command Phase */
+ ESPCMD(sc, ESPCMD_FLUSH);
+ case 4:
+ /* So far, everything went fine */
+ sc->sc_msgpriq = 0;
+ break;
+ }
+#if 0
+/* Why set msgpriq? (and not raise ATN) */
+ if (ecb->xs->flags & SCSI_RESET)
+ sc->sc_msgpriq = SEND_DEV_RESET;
+ else if (ti->flags & T_NEGOTIATE)
+ sc->sc_msgpriq =
+ SEND_IDENTIFY | SEND_SDTR;
+ else
+ sc->sc_msgpriq = SEND_IDENTIFY;
+#endif
+ sc->sc_state = ESP_HASNEXUS;
+ /*???sc->sc_flags = 0; */
+ sc->sc_prevphase = INVALID_PHASE; /* ?? */
+ sc->sc_dp = ecb->daddr;
+ sc->sc_dleft = ecb->dleft;
+ ti->lubusy |= (1<<sc_link->lun);
+ break;
+ } else {
+ printf("%s: unexpected status after select"
+ ": [intr %x, stat %x, step %x]\n",
+ sc->sc_dev.dv_xname,
+ sc->sc_espintr, sc->sc_espstat,
+ sc->sc_espstep);
+ ESPCMD(sc, ESPCMD_FLUSH);
+ DELAY(1);
+ esp_abort(sc, ecb);
+ }
+ if (sc->sc_state == ESP_IDLE) {
+ printf("%s: stray interrupt\n", sc->sc_dev.dv_xname);
+ return 0;
+ }
+ break;
- /*
- * If there are more messages, re-assert
- * ATN so that we will enter MSGOUT phase
- * again. Also, pause till that interrupt is
- * done to see if the phase changes.
- */
- if (sc->sc_msgpriq)
- espr->espr_cmd = ESPCMD_SETATN;
- sc->sc_state = ESPS_NEXUS;
- goto states;
- case ESPS_RESELECTED:
- ESPD(("RS "));
-
- if (sc->sc_phase != MESSAGE_IN_PHASE) {
- printf("%s: target didn't identify\n",
- sc->sc_dev.dv_xname);
- esp_init(sc, 1);
- goto done;
- }
- esp_msgin(sc);
- if (sc->sc_state != ESPS_NEXUS) {
- /* IDENTIFY failed?! */
- printf("%s: identify failed\n", sc->sc_dev.dv_xname);
- esp_init(sc, 1);
- goto done;
- }
+ case ESP_HASNEXUS:
+ if (sc->sc_flags & ESP_ICCS) {
+ unsigned char msg;
- /*
- * We will get an interrupt because of the ESPCMD_MSGOK
- * interrupt inside esp_msgin()
- */
- goto done;
- case ESPS_DOINGDMA:
- /*
- * Call if DMA is still active. If dmaintr() has finished,
- * delay a while to let things settle, and then re-read the
- * phase.
- *
- * XXX: should check ESPINTR_DIS
- */
- if (sc->sc_espstat & ESPSTAT_PE) {
- printf("%s: SCSI bus parity error\n",
- sc->sc_dev.dv_xname);
- esp_sched_msgout(SEND_INIT_DET_ERR);
- /* XXX: anything else to do? */
- }
+ sc->sc_flags &= ~ESP_ICCS;
- if (sc->sc_espintr & ESPINTR_BS) {
- /*
- * ESP says we have changed phase, or have
- * finished transferring all the bytes. It
- * doesn't matter which case, either way we
- * will let the phase determine what happens
- * next.
- */
- ESPD(("DOINGDMA:BS:%02x/%02x ", sc->sc_espstat,
- espr->espr_tcl));
- if (sc->sc_dma->sc_active) {
- dmaintr(sc->sc_dma, 0);
- if (sc->sc_dma->sc_active)
- goto done;
- }
- } else {
- ESPD(("DOINGDMA:cont "));
- if (sc->sc_dma->sc_active) {
- dmaintr(sc->sc_dma, 1);
- if (sc->sc_dma->sc_active)
- goto done;
+ if (!(sc->sc_espintr & ESPINTR_DONE)) {
+ printf("%s: ICCS: "
+ ": [intr %x, stat %x, step %x]\n",
+ sc->sc_dev.dv_xname,
+ sc->sc_espintr, sc->sc_espstat,
+ sc->sc_espstep);
+ }
+ if ((ESP_READ_REG(sc, ESP_FFLAG) & ESPFIFO_FF) != 2) {
+ printf("%s: ICCS: expected 2, got %d "
+ ": [intr %x, stat %x, step %x]\n",
+ sc->sc_dev.dv_xname,
+ ESP_READ_REG(sc, ESP_FFLAG) & ESPFIFO_FF,
+ sc->sc_espintr, sc->sc_espstat,
+ sc->sc_espstep);
+ ESPCMD(sc, ESPCMD_FLUSH);
+ esp_abort(sc, ecb);
+ return 1;
+ }
+ ecb->stat = ESP_READ_REG(sc, ESP_FIFO);
+ msg = ESP_READ_REG(sc, ESP_FIFO);
+ ESP_PHASE(("<stat:(%x,%x)>", ecb->stat, msg));
+ if (msg == MSG_CMDCOMPLETE) {
+ sc->sc_flags |= ESP_BUSFREE_OK;
+ ecb->xs->resid = ecb->dleft = sc->sc_dleft;
+ } else
+ printf("%s: STATUS_PHASE: msg %d\n",
+ sc->sc_dev.dv_xname, msg);
+ ESPCMD(sc, ESPCMD_MSGOK);
+ continue; /* ie. wait for disconnect */
}
+ break;
+ default:
+ panic("%s: invalid state: %d",
+ sc->sc_dev.dv_xname,
+ sc->sc_state);
}
- /* finished a DMA transaction */
- sc->sc_flags = 0;
-
- ecb->daddr = sc->sc_dp; /* implicit SAVEDATAPOINTERS */
- ecb->dleft = sc->sc_dleft;
-
/*
- * ESP100 strangeness -- need to fetch a new phase --
- * why?
+ * Driver is now in state ESP_HASNEXUS, i.e. we
+ * have a current command working the SCSI bus.
*/
- if (sc->sc_rev==ESP100 &&
- sc->sc_tinfo[ecb->xs->sc_link->target].offset) {
- sc->sc_espstat = espr->espr_stat & ESPSTAT_PHASE;
- sc->sc_phase = espphase(sc);
- /* XXX should check if sync under/overran */
- }
- sc->sc_state = ESPS_NEXUS;
- break; /* handle the phase */
- case ESPS_DOINGMSGOUT:
- ESPD(("DMO "));
- if (sc->sc_espintr & ESPINTR_DIS) {
- sc->sc_state = ESPS_NEXUS; /* will cleanup */
- goto states;
+ if (sc->sc_state != ESP_HASNEXUS || ecb == NULL) {
+ panic("esp no nexus");
}
- if (sc->sc_espintr & ESPINTR_BS) {
- sc->sc_msgpriq &= ~sc->sc_msgout;
- sc->sc_msgout = 0;
+ switch (sc->sc_phase) {
+ case MESSAGE_OUT_PHASE:
+ ESP_PHASE(("MESSAGE_OUT_PHASE "));
+ esp_msgout(sc);
+ sc->sc_prevphase = MESSAGE_OUT_PHASE;
+ break;
+ case MESSAGE_IN_PHASE:
+ ESP_PHASE(("MESSAGE_IN_PHASE "));
+ if (sc->sc_espintr & ESPINTR_BS) {
+ ESPCMD(sc, ESPCMD_FLUSH);
+ sc->sc_flags |= ESP_WAITI;
+ ESPCMD(sc, ESPCMD_TRANS);
+ } else if (sc->sc_espintr & ESPINTR_FC) {
+ if ((sc->sc_flags & ESP_WAITI) == 0) {
+ printf("%s: MSGIN: unexpected FC bit: "
+ "[intr %x, stat %x, step %x]\n",
+ sc->sc_dev.dv_xname,
+ sc->sc_espintr, sc->sc_espstat,
+ sc->sc_espstep);
+ }
+ sc->sc_flags &= ~ESP_WAITI;
+ esp_msgin(sc);
+ } else {
+ printf("%s: MSGIN: weird bits: "
+ "[intr %x, stat %x, step %x]\n",
+ sc->sc_dev.dv_xname,
+ sc->sc_espintr, sc->sc_espstat,
+ sc->sc_espstep);
+ }
+ sc->sc_prevphase = MESSAGE_IN_PHASE;
+ break;
+ case COMMAND_PHASE: {
+ /* well, this means send the command again */
+ u_char *cmd = (u_char *)&ecb->cmd;
+ int i;
- ESPD(("F%d ", espr->espr_fflag & ESPFIFO_FF));
- if (espr->espr_fflag & ESPFIFO_FF) {
- espflush(sc);
+ ESP_PHASE(("COMMAND_PHASE 0x%02x (%d) ",
+ ecb->cmd.opcode, ecb->clen));
+ if (ESP_READ_REG(sc, ESP_FFLAG) & ESPFIFO_FF) {
+ ESPCMD(sc, ESPCMD_FLUSH);
DELAY(1);
}
-
- /*
- * If there are more messages, re-assert
- * ATN so that we will enter MSGOUT phase
- * again. Also, pause till that interrupt is
- * done to see if the phase changes.
- */
- if (sc->sc_msgpriq)
- espr->espr_cmd = ESPCMD_SETATN;
+ /* Now the command into the FIFO */
+ for (i = 0; i < ecb->clen; i++)
+ ESP_WRITE_REG(sc, ESP_FIFO, *cmd++);
+ ESPCMD(sc, ESPCMD_TRANS);
+ sc->sc_prevphase = COMMAND_PHASE;
+ }
break;
- }
-
-#if 0
- if (sc->sc_espintr & ESPINTR_FC) {
- /*
- * XXX: the book claims that an ESPINTR_BS is
- * generated, not ESPINTR_FC. Check if that is
- * true.
- */
- }
-#endif
- break; /* handle the phase */
- case ESPS_DOINGMSGIN:
- /*
- * We receive two interrupts per message byte here: one for
- * the MSGOK command in esp_msgin(), and one for the TRANS
- * command below.
- */
- ESPD(("DMI "));
-
- if (sc->sc_espintr & ESPINTR_DIS) {
- sc->sc_state = ESPS_NEXUS; /* will cleanup */
- goto states;
- }
-
- if (sc->sc_espstat & ESPSTAT_PE) {
- printf("%s: SCSI bus parity error\n",
- sc->sc_dev.dv_xname);
- esp_sched_msgout(SEND_PARITY_ERROR);
- espflush(sc);
- sc->sc_state = ESPS_NEXUS;
- /* XXX: are we missing some fifo handling? */
- goto done; /* will be interrupted for MSGOUT */
- }
+ case DATA_OUT_PHASE:
+ ESP_PHASE(("DATA_OUT_PHASE [%d] ", sc->sc_dleft));
+ ESPCMD(sc, ESPCMD_FLUSH);
+ size = min(sc->sc_dleft, sc->sc_maxxfer);
+ DMA_SETUP(sc->sc_dma, &sc->sc_dp, &sc->sc_dleft,
+ 0, &size);
+ sc->sc_prevphase = DATA_OUT_PHASE;
+ goto setup_xfer;
+ case DATA_IN_PHASE:
+ ESP_PHASE(("DATA_IN_PHASE "));
+ if (sc->sc_rev == ESP100)
+ ESPCMD(sc, ESPCMD_FLUSH);
+ size = min(sc->sc_dleft, sc->sc_maxxfer);
+ DMA_SETUP(sc->sc_dma, &sc->sc_dp, &sc->sc_dleft,
+ 1, &size);
+ sc->sc_prevphase = DATA_IN_PHASE;
+ setup_xfer:
+ /* Program the SCSI counter */
+ ESP_WRITE_REG(sc, ESP_TCL, size);
+ ESP_WRITE_REG(sc, ESP_TCM, size >> 8);
+ if (sc->sc_cfg2 & ESPCFG2_FE) {
+ ESP_WRITE_REG(sc, ESP_TCH, size >> 16);
+ }
+ /* load the count in */
+ ESPCMD(sc, ESPCMD_NOP|ESPCMD_DMA);
- /*
- * The two interrupts we receive per byte will alternate
- * between entering the following two if() statements.
- * XXX: just incase, are these two if()'s in the correct
- * order?
- */
- if (sc->sc_espintr & ESPINTR_BS) {
/*
- * interrupted by ESPCMD_MSGOK in esp_msgin()
+ * Note that if `size' is 0, we've already transceived
+ * all the bytes we want but we're still in DATA PHASE.
+ * Apparently, the device needs padding. Also, a
+ * transfer size of 0 means "maximum" to the chip
+ * DMA logic.
*/
- if (sc->sc_phase == MESSAGE_IN_PHASE)
- espr->espr_cmd = ESPCMD_TRANS;
- else {
- ESPD(("PC "));
- /*
- * XXX: The phase has been changed on us.
- * This should lead us to discard the
- * current (incomplete) message, perhaps
- * assert an error(?), and go handle whatever
- * new phase we are now in.
- */
- break;
+ ESPCMD(sc,
+ (size==0?ESPCMD_TRPAD:ESPCMD_TRANS)|ESPCMD_DMA);
+ DMA_GO(sc->sc_dma);
+ return 1;
+ case STATUS_PHASE:
+ ESP_PHASE(("STATUS_PHASE "));
+ sc->sc_flags |= ESP_ICCS;
+ ESPCMD(sc, ESPCMD_ICCS);
+ sc->sc_prevphase = STATUS_PHASE;
+ break;
+ case INVALID_PHASE:
+ break;
+ case BUSFREE_PHASE:
+ if (sc->sc_flags & ESP_BUSFREE_OK) {
+ /*It's fun the 1st time.. */
+ sc->sc_flags &= ~ESP_BUSFREE_OK;
}
- goto done;
- }
- if (sc->sc_espintr & ESPINTR_FC) {
- /*
- * interrupted by ESPCMD_TRANS a few lines above.
- */
- esp_msgin(sc);
- goto done;
+ break;
+ default:
+ panic("esp: bogus bus phase\n");
}
- goto done;
- case ESPS_DOINGSTATUS:
- if (sc->sc_espintr & ESPINTR_DIS) {
- sc->sc_state = ESPS_NEXUS; /* will cleanup */
- goto states;
+ }
+ panic("esp: should not get here..");
+}
+
+void
+esp_abort(sc, ecb)
+ struct esp_softc *sc;
+ struct ecb *ecb;
+{
+ if (ecb == sc->sc_nexus) {
+ if (sc->sc_state == ESP_HASNEXUS) {
+ sc->sc_flags |= ESP_ABORTING;
+ esp_sched_msgout(SEND_ABORT);
}
+ } else {
+ if (sc->sc_state == ESP_IDLE)
+ esp_sched(sc);
+ }
+}
- if (sc->sc_espintr & (ESPINTR_FC|ESPINTR_BS)) {
- i = espgetbyte(sc, 0);
- if (i == -1) {
- printf("%s: ESPS_DOINGSTATUS fifo empty\n",
- sc->sc_dev.dv_xname);
- /* XXX: how do we cleanup from this error? */
- goto done;
- }
- if (sc->sc_espstat & ESPSTAT_PE) {
- printf("%s: SCSI bus parity error\n",
- sc->sc_dev.dv_xname);
- /*
- * Can't tell what the real status should
- * be, so force a later check.
- */
- i = SCSI_CHECK;
- }
- ecb->stat = (u_char)i;
- ESPD(("status=%02x ", (u_char)i));
+void
+esp_timeout(arg)
+ void *arg;
+{
+ int s = splbio();
+ struct ecb *ecb = (struct ecb *)arg;
+ struct esp_softc *sc;
+ struct scsi_xfer *xs = ecb->xs;
- if (sc->sc_espintr & ESPINTR_FC) {
- /*
- * XXX assumes a single byte message --
- * and what about parity errors and other
- * possible botches inside esp_msgin?
- */
- esp_msgin(sc);
- sc->sc_state = ESPS_NEXUS;
- } else
- sc->sc_state = ESPS_DOINGMSGIN;
- goto done;
- }
- goto done;
- case ESPS_EXPECTDISC:
- /*
- * We were told to expect a disconnection interrupt.
- * When we get it, we can go back to other work.
- */
- ESPD(("ED "));
- if (sc->sc_espintr & ESPINTR_DIS) {
- espflush(sc);
- DELAY(1);
- espr->espr_cmd = ESPCMD_ENSEL; /* within 250ms of disconnect */
- sc->sc_state = ESPS_IDLE;
- }
- goto states;
- case ESPS_NEXUS:
- ESPD(("NX "));
+ sc = xs->sc_link->adapter_softc;
+ sc_print_addr(xs->sc_link);
+again:
+ printf("%s: timed out [ecb %p (flags 0x%x, dleft %x, stat %x)], "
+ "<state %d, nexus %p, phase(c %x, p %x), resid %x, msg(q %x,o %x) %s>",
+ sc->sc_dev.dv_xname,
+ ecb, ecb->flags, ecb->dleft, ecb->stat,
+ sc->sc_state, sc->sc_nexus, sc->sc_phase, sc->sc_prevphase,
+ sc->sc_dleft, sc->sc_msgpriq, sc->sc_msgout,
+ DMA_ISACTIVE(sc->sc_dma) ? "DMA active" : "");
+#if ESP_DEBUG > 0
+ printf("TRACE: %s.", ecb->trace);
+#endif
- if (sc->sc_espintr & ESPINTR_DIS) {
- espflush(sc);
- DELAY(1);
- espr->espr_cmd = ESPCMD_ENSEL; /* within 250ms of disconnect */
- sc->sc_msgpriq = sc->sc_msgout = 0;
-
- ESPD(("DD:"));
- if (ecb->flags & ECB_DONE) {
- ESPD(("D "));
- /* successfully finished a transaction */
- untimeout(esp_timeout, ecb);
- esp_done(ecb);
- } else if (sc->sc_flags & ESPF_MAYDISCON) {
- ESPD(("M "));
- /* legal discon/recon is happening */
- TAILQ_INSERT_HEAD(&sc->nexus_list,
- ecb, chain);
- sc->sc_nexus = NULL;
- sc->sc_state = ESPS_IDLE;
- sc->sc_flags &= ~ESPF_MAYDISCON;
- goto states;
- } else {
- ESPD(("U "));
- /* unexpected disconnection! fail. */
- ecb->xs->error = XS_TIMEOUT;
- untimeout(esp_timeout, ecb);
- esp_done(ecb);
+ if (ecb->flags & ECB_ABORTED) {
+ /* abort timed out */
+ printf(" AGAIN\n");
+ esp_init(sc, 1);
+ } else {
+ /* abort the operation that has timed out */
+ printf("\n");
+ xs->error = XS_TIMEOUT;
+ ecb->flags |= ECB_ABORTED;
+ esp_abort(sc, ecb);
+ /* 2 secs for the abort */
+ if ((xs->flags & SCSI_POLL) == 0)
+ timeout(esp_timeout, ecb, 2 * hz);
+ else {
+ int count = 200000;
+ while (count) {
+ if (DMA_ISINTR(sc->sc_dma)) {
+ espintr(sc);
+ }
+ if (xs->flags & ITSDONE)
+ break;
+ DELAY(10);
+ --count;
}
- goto done;
+ if (count == 0)
+ goto again;
}
- break; /* handle the phase */
- default:
- printf("%s: unknown state %d\n",
- sc->sc_dev.dv_xname, sc->sc_state);
- panic("cannot continue");
}
-
- ecb = sc->sc_nexus;
-
- switch (sc->sc_phase) {
- case MESSAGE_OUT_PHASE:
- /*
- * ATN was asserted, and the bus changed into MSGOUT phase.
- * Send a message.
- */
- esp_msgout(sc); /* cause intr */
- sc->sc_state = ESPS_DOINGMSGOUT;
- break;
- case COMMAND_PHASE:
- /*
- * Apparently we need to repeat the previous command.
- */
- espflush(sc);
- DELAY(1);
- cmd = (u_char *)&ecb->cmd;
-
- ESPD(("CMD["));
- for (i = 0; i < ecb->clen; i++)
- ESPD(("%02x%c", cmd[i],
- (i == ecb->clen-1) ? ']' : ' '));
- ESPD((" "));
-
- /* Now the command into the FIFO */
- for (i = 0; i < ecb->clen; i++)
- espr->espr_fifo = cmd[i];
-
- espr->espr_cmd = ESPCMD_TRANS; /* cause intr */
- sc->sc_state = ESPS_NEXUS;
- break;
- case DATA_OUT_PHASE:
- /*
- * We can feed the device the data now.
- */
- espflush(sc);
- DELAY(1);
- dmastart(sc->sc_dma, &sc->sc_dp, &sc->sc_dleft, 0,
- ecb->xs->flags & SCSI_POLL); /* cause intr */
- sc->sc_state = ESPS_DOINGDMA;
- break;
- case DATA_IN_PHASE:
- /*
- * The device is ready to give us the data.
- */
- dmadrain(sc->sc_dma);
- dmastart(sc->sc_dma, &sc->sc_dp, &sc->sc_dleft, 1,
- ecb->xs->flags & SCSI_POLL); /* cause intr */
- sc->sc_state = ESPS_DOINGDMA;
- break;
- case MESSAGE_IN_PHASE:
- espflush(sc);
- DELAY(1);
- espr->espr_cmd = ESPCMD_TRANS; /* cause intr */
- sc->sc_state = ESPS_DOINGMSGIN;
- break;
- case STATUS_PHASE:
- espflush(sc);
- DELAY(1);
- espr->espr_cmd = ESPCMD_ICCS; /* cause intr */
- sc->sc_state = ESPS_DOINGSTATUS;
- break;
- default:
- printf("%s: bogus bus phase %d\n",
- sc->sc_dev.dv_xname);
- break;
- }
-
-done:
- if (sc->sc_dma->sc_rev == DMAREV_1 && dmaintrwas)
- dmaenintr(sc->sc_dma);
- sc->sc_intrcnt.ev_count++;
- return (1);
+ splx(s);
}
diff --git a/sys/arch/sparc/dev/espreg.h b/sys/arch/sparc/dev/espreg.h
index 31db483c1bc..e08eebcad7c 100644
--- a/sys/arch/sparc/dev/espreg.h
+++ b/sys/arch/sparc/dev/espreg.h
@@ -1,9 +1,7 @@
-/* $NetBSD: espreg.h,v 1.5 1995/01/07 05:17:15 mycroft Exp $ */
+/* $NetBSD: espreg.h,v 1.7 1995/12/18 23:58:39 pk Exp $ */
/*
* Copyright (c) 1994 Peter Galbavy. All rights reserved.
- * Copyright (c) 1995 Theo de Raadt. All rights reserved.
- *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
@@ -14,8 +12,7 @@
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
- * This product includes software developed by Peter Galbavy and
- * Theo de Raadt
+ * This product includes software developed by Peter Galbavy.
* 4. The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission.
*
@@ -35,137 +32,117 @@
* Register addresses, relative to some base address
*/
-struct espregs {
- volatile u_char espr_tcl;
- volatile u_char x01;
- volatile u_char x02;
- volatile u_char x03;
- volatile u_char espr_tcm;
- volatile u_char x05;
- volatile u_char x06;
- volatile u_char x07;
- volatile u_char espr_fifo;
- volatile u_char x09;
- volatile u_char x0a;
- volatile u_char x0b;
- volatile u_char espr_cmd;
- volatile u_char x0d;
- volatile u_char x0e;
- volatile u_char x0f;
- volatile u_char espr_stat;
-#define espr_selid espr_stat
- volatile u_char x11;
- volatile u_char x12;
- volatile u_char x13;
- volatile u_char espr_intr;
-#define espr_timeout espr_intr
- volatile u_char x15;
- volatile u_char x16;
- volatile u_char x17;
- volatile u_char espr_step;
-#define espr_synctp espr_step
- volatile u_char x19;
- volatile u_char x1a;
- volatile u_char x1b;
- volatile u_char espr_fflag;
-#define espr_syncoff espr_fflag
- volatile u_char x1d;
- volatile u_char x1e;
- volatile u_char x1f;
- volatile u_char espr_cfg1;
- volatile u_char x21;
- volatile u_char x22;
- volatile u_char x23;
- volatile u_char espr_ccf;
- volatile u_char x25;
- volatile u_char x26;
- volatile u_char x27;
- volatile u_char espr_test;
- volatile u_char x29;
- volatile u_char x2a;
- volatile u_char x2b;
- volatile u_char espr_cfg2;
- volatile u_char x2d;
- volatile u_char x2e;
- volatile u_char x2f;
- volatile u_char espr_cfg3; /* ESP200 only */
- volatile u_char x31;
- volatile u_char x32;
- volatile u_char x33;
- volatile u_char espr_tch; /* ESP200 only */
-};
-
-#define ESPCMD_DMA 0x80 /* DMA Bit */
-#define ESPCMD_NOP 0x00 /* No Operation */
-#define ESPCMD_FLUSH 0x01 /* Flush FIFO */
-#define ESPCMD_RSTCHIP 0x02 /* Reset Chip */
-#define ESPCMD_RSTSCSI 0x03 /* Reset SCSI Bus */
-#define ESPCMD_RESEL 0x40 /* Reselect Sequence */
-#define ESPCMD_SELNATN 0x41 /* Select without ATN */
-#define ESPCMD_SELATN 0x42 /* Select with ATN */
-#define ESPCMD_SELATNS 0x43 /* Select with ATN & Stop */
-#define ESPCMD_ENSEL 0x44 /* Enable (Re)Selection */
-#define ESPCMD_DISSEL 0x45 /* Disable (Re)Selection */
-#define ESPCMD_SELATN3 0x46 /* Select with ATN3 */
-#define ESPCMD_RESEL3 0x47 /* Reselect3 Sequence */
-#define ESPCMD_SNDMSG 0x20 /* Send Message */
-#define ESPCMD_SNDSTAT 0x21 /* Send Status */
-#define ESPCMD_SNDDATA 0x22 /* Send Data */
-#define ESPCMD_DISCSEQ 0x23 /* Disconnect Sequence */
-#define ESPCMD_TERMSEQ 0x24 /* Terminate Sequence */
-#define ESPCMD_TCCS 0x25 /* Target Command Comp Seq */
-#define ESPCMD_DISC 0x27 /* Disconnect */
-#define ESPCMD_RECMSG 0x28 /* Receive Message */
-#define ESPCMD_RECCMD 0x29 /* Receive Command */
-#define ESPCMD_RECDATA 0x2a /* Receive Data */
-#define ESPCMD_RECCSEQ 0x2b /* Receive Command Sequence*/
-#define ESPCMD_ABORT 0x04 /* Target Abort DMA */
-#define ESPCMD_TRANS 0x10 /* Transfer Information */
-#define ESPCMD_ICCS 0x11 /* Initiator Cmd Comp Seq */
-#define ESPCMD_MSGOK 0x12 /* Message Accepted */
-#define ESPCMD_TRPAD 0x18 /* Transfer Pad */
-#define ESPCMD_SETATN 0x1a /* Set ATN */
-#define ESPCMD_RSTATN 0x1b /* Reset ATN */
-
-#define ESPSTAT_INT 0x80 /* Interrupt */
-#define ESPSTAT_GE 0x40 /* Gross Error */
-#define ESPSTAT_PE 0x20 /* Parity Error */
-#define ESPSTAT_TC 0x10 /* Terminal Count */
-#define ESPSTAT_VGC 0x08 /* Valid Group Code */
-#define ESPSTAT_PHASE 0x07 /* Phase bits */
-
-#define ESPINTR_SBR 0x80 /* SCSI Bus Reset */
-#define ESPINTR_ILL 0x40 /* Illegal Command */
-#define ESPINTR_DIS 0x20 /* Disconnect */
-#define ESPINTR_BS 0x10 /* Bus Service */
-#define ESPINTR_FC 0x08 /* Function Complete */
-#define ESPINTR_RESEL 0x04 /* Reselected */
-#define ESPINTR_SELATN 0x02 /* Select with ATN */
-#define ESPINTR_SEL 0x01 /* Selected */
-
-#define ESPSTEP_MASK 0x07 /* the last 3 bits */
-#define ESPSTEP_DONE 0x04 /* command went out */
-
-#define ESPFIFO_SS 0xe0 /* Sequence Step (Dup) */
-#define ESPFIFO_FF 0x1f /* Bytes in FIFO */
-
-#define ESPCFG1_SLOW 0x80 /* Slow Cable Mode */
-#define ESPCFG1_SRR 0x40 /* SCSI Reset Rep Int Dis */
-#define ESPCFG1_PTEST 0x20 /* Parity Test Mod */
-#define ESPCFG1_PARENB 0x10 /* Enable Parity Check */
-#define ESPCFG1_CTEST 0x08 /* Enable Chip Test */
-#define ESPCFG1_BUSID 0x07 /* Bus ID */
-
-#define ESPCFG2_RSVD 0xe0 /* reserved */
-#define ESPCFG2_FE 0x40 /* Features Enable */
-#define ESPCFG2_DREQ 0x10 /* DREQ High Impedance */
-#define ESPCFG2_SCSI2 0x08 /* SCSI-2 Enable */
-#define ESPCFG2_BPA 0x04 /* Target Bad Parity Abort */
-#define ESPCFG2_RPE 0x02 /* Register Parity Error */
-#define ESPCFG2_DPE 0x01 /* DMA Parity Error */
-
-#define ESPCFG3_IDM 0x10 /* ID Message Res Check */
-#define ESPCFG3_QTE 0x08 /* Queue Tag Enable */
-#define ESPCFG3_CDB 0x04 /* CDB 10-bytes OK */
-#define ESPCFG3_FSCSI 0x02 /* Fast SCSI */
-#define ESPCFG3_FCLK 0x01 /* Fast Clock (>25Mhz) */
+#define ESP_TCL 0x00 /* RW - Transfer Count Low */
+#define ESP_TCM 0x01 /* RW - Transfer Count Mid */
+#define ESP_TCH 0x0e /* RW - Transfer Count High */
+ /* NOT on 53C90 */
+
+#define ESP_FIFO 0x02 /* RW - FIFO data */
+
+#define ESP_CMD 0x03 /* RW - Command (2 deep) */
+#define ESPCMD_DMA 0x80 /* DMA Bit */
+#define ESPCMD_NOP 0x00 /* No Operation */
+#define ESPCMD_FLUSH 0x01 /* Flush FIFO */
+#define ESPCMD_RSTCHIP 0x02 /* Reset Chip */
+#define ESPCMD_RSTSCSI 0x03 /* Reset SCSI Bus */
+#define ESPCMD_RESEL 0x40 /* Reselect Sequence */
+#define ESPCMD_SELNATN 0x41 /* Select without ATN */
+#define ESPCMD_SELATN 0x42 /* Select with ATN */
+#define ESPCMD_SELATNS 0x43 /* Select with ATN & Stop */
+#define ESPCMD_ENSEL 0x44 /* Enable (Re)Selection */
+#define ESPCMD_DISSEL 0x45 /* Disable (Re)Selection */
+#define ESPCMD_SELATN3 0x46 /* Select with ATN3 */
+#define ESPCMD_RESEL3 0x47 /* Reselect3 Sequence */
+#define ESPCMD_SNDMSG 0x20 /* Send Message */
+#define ESPCMD_SNDSTAT 0x21 /* Send Status */
+#define ESPCMD_SNDDATA 0x22 /* Send Data */
+#define ESPCMD_DISCSEQ 0x23 /* Disconnect Sequence */
+#define ESPCMD_TERMSEQ 0x24 /* Terminate Sequence */
+#define ESPCMD_TCCS 0x25 /* Target Command Comp Seq */
+#define ESPCMD_DISC 0x27 /* Disconnect */
+#define ESPCMD_RECMSG 0x28 /* Receive Message */
+#define ESPCMD_RECCMD 0x29 /* Receive Command */
+#define ESPCMD_RECDATA 0x2a /* Receive Data */
+#define ESPCMD_RECCSEQ 0x2b /* Receive Command Sequence*/
+#define ESPCMD_ABORT 0x04 /* Target Abort DMA */
+#define ESPCMD_TRANS 0x10 /* Transfer Information */
+#define ESPCMD_ICCS 0x11 /* Initiator Cmd Comp Seq */
+#define ESPCMD_MSGOK 0x12 /* Message Accepted */
+#define ESPCMD_TRPAD 0x18 /* Transfer Pad */
+#define ESPCMD_SETATN 0x1a /* Set ATN */
+#define ESPCMD_RSTATN 0x1b /* Reset ATN */
+
+#define ESP_STAT 0x04 /* RO - Status */
+#define ESPSTAT_INT 0x80 /* Interrupt */
+#define ESPSTAT_GE 0x40 /* Gross Error */
+#define ESPSTAT_PE 0x20 /* Parity Error */
+#define ESPSTAT_TC 0x10 /* Terminal Count */
+#define ESPSTAT_VGC 0x08 /* Valid Group Code */
+#define ESPSTAT_PHASE 0x07 /* Phase bits */
+
+#define ESP_SELID 0x04 /* WO - Select/Reselect Bus ID */
+
+#define ESP_INTR 0x05 /* RO - Interrupt */
+#define ESPINTR_SBR 0x80 /* SCSI Bus Reset */
+#define ESPINTR_ILL 0x40 /* Illegal Command */
+#define ESPINTR_DIS 0x20 /* Disconnect */
+#define ESPINTR_BS 0x10 /* Bus Service */
+#define ESPINTR_FC 0x08 /* Function Complete */
+#define ESPINTR_RESEL 0x04 /* Reselected */
+#define ESPINTR_SELATN 0x02 /* Select with ATN */
+#define ESPINTR_SEL 0x01 /* Selected */
+
+#define ESP_TIMEOUT 0x05 /* WO - Select/Reselect Timeout */
+
+#define ESP_STEP 0x06 /* RO - Sequence Step */
+#define ESPSTEP_MASK 0x07 /* the last 3 bits */
+#define ESPSTEP_DONE 0x04 /* command went out */
+
+#define ESP_SYNCTP 0x06 /* WO - Synch Transfer Period */
+ /* Default 5 (53C9X) */
+
+#define ESP_FFLAG 0x07 /* RO - FIFO Flags */
+#define ESPFIFO_SS 0xe0 /* Sequence Step (Dup) */
+#define ESPFIFO_FF 0x1f /* Bytes in FIFO */
+
+#define ESP_SYNCOFF 0x07 /* WO - Synch Offset */
+ /* 0 = ASYNC */
+ /* 1 - 15 = SYNC bytes */
+
+#define ESP_CFG1 0x08 /* RW - Configuration #1 */
+#define ESPCFG1_SLOW 0x80 /* Slow Cable Mode */
+#define ESPCFG1_SRR 0x40 /* SCSI Reset Rep Int Dis */
+#define ESPCFG1_PTEST 0x20 /* Parity Test Mod */
+#define ESPCFG1_PARENB 0x10 /* Enable Parity Check */
+#define ESPCFG1_CTEST 0x08 /* Enable Chip Test */
+#define ESPCFG1_BUSID 0x07 /* Bus ID */
+
+#define ESP_CCF 0x09 /* WO - Clock Conversion Factor */
+ /* 0 = 35.01 - 40Mhz */
+ /* NEVER SET TO 1 */
+ /* 2 = 10Mhz */
+ /* 3 = 10.01 - 15Mhz */
+ /* 4 = 15.01 - 20Mhz */
+ /* 5 = 20.01 - 25Mhz */
+ /* 6 = 25.01 - 30Mhz */
+ /* 7 = 30.01 - 35Mhz */
+
+#define ESP_TEST 0x0a /* WO - Test (Chip Test Only) */
+
+#define ESP_CFG2 0x0b /* RW - Configuration #2 */
+#define ESPCFG2_RSVD 0xa0 /* reserved */
+#define ESPCFG2_FE 0x40 /* Features Enable */
+#define ESPCFG2_DREQ 0x10 /* DREQ High Impedance */
+#define ESPCFG2_SCSI2 0x08 /* SCSI-2 Enable */
+#define ESPCFG2_BPA 0x04 /* Target Bad Parity Abort */
+#define ESPCFG2_RPE 0x02 /* Register Parity Error */
+#define ESPCFG2_DPE 0x01 /* DMA Parity Error */
+
+/* Config #3 only on 53C9X */
+#define ESP_CFG3 0x0c /* RW - Configuration #3 */
+#define ESPCFG3_RSVD 0xe0 /* reserved */
+#define ESPCFG3_IDM 0x10 /* ID Message Res Check */
+#define ESPCFG3_QTE 0x08 /* Queue Tag Enable */
+#define ESPCFG3_CDB 0x04 /* CDB 10-bytes OK */
+#define ESPCFG3_FSCSI 0x02 /* Fast SCSI */
+#define ESPCFG3_FCLK 0x01 /* Fast Clock (>25Mhz) */
diff --git a/sys/arch/sparc/dev/espvar.h b/sys/arch/sparc/dev/espvar.h
index c8592af5b29..8b0b414a08a 100644
--- a/sys/arch/sparc/dev/espvar.h
+++ b/sys/arch/sparc/dev/espvar.h
@@ -1,37 +1,7 @@
-/* $NetBSD: espvar.h,v 1.3 1994/11/20 20:52:12 deraadt Exp $ */
+/* $NetBSD: espvar.h,v 1.13 1996/05/16 20:31:30 pk Exp $ */
/*
- * Copyright (c) 1995 Theo de Raadt
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- * 3. All advertising materials mentioning features or use of this software
- * must display the following acknowledgement:
- * This product includes software developed under OpenBSD by
- * Theo de Raadt.
- * 4. The name of the author may not be used to endorse or promote products
- * derived from this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
- * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
- * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
- * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
- * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
- * SUCH DAMAGE.
- *
* Copyright (c) 1994 Peter Galbavy. All rights reserved.
- *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
@@ -58,82 +28,103 @@
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
-/* esp chip revisions */
-#define ESP100 0
-#define ESP100A 1
-#define ESP200 2
-
-/* Grabbed from Julians SCSI aha-drivers */
-#ifdef DDB
-int Debugger();
-#else DDB
-#define Debugger() panic("should call debugger here (esp.c)")
-#endif DDB
-
-#define ESP_MSGLEN_MAX 8 /* maximum msglen we handle */
-#define ESP_SYNC_REQOFF 0 /* offset we request */
-#define ESP_SYNC_MAXOFF 15 /* maximum supportable sync offset */
-#if ESP_SYNC_REQOFF > ESP_SYNC_MAXOFF
-#warning requested sync offset is higher than maximum possible sync offset.
-#endif
+#define ESP_DEBUG 0
#define FREQTOCCF(freq) (((freq + 4) / 5))
-/*
- * ECB. Holds additional information for each SCSI command
- * Comments: We need a separate scsi command block because we may need
- * to overwrite it with a request sense command. Basically, we refrain
- * from fiddling with the scsi_xfer struct (except do the expected
- * updating of return values). We'll generally update:
- * xs->{flags,resid,error,sense,status} and occasionally xs->retries.
+/* esp revisions */
+#define ESP100 0x01
+#define ESP100A 0x02
+#define ESP200 0x03
+
+#if 0
+typedef caddr_t physaddr;
+
+struct esp_dma_seg {
+ physaddr addr;
+ long len;
+};
+#endif
+
+/*
+ * ECB. Holds additional information for each SCSI command Comments: We
+ * need a separate scsi command block because we may need to overwrite it
+ * with a request sense command. Basicly, we refrain from fiddling with
+ * the scsi_xfer struct (except do the expected updating of return values).
+ * We'll generally update: xs->{flags,resid,error,sense,status} and
+ * occasionally xs->retries.
*/
struct ecb {
TAILQ_ENTRY(ecb) chain;
- struct scsi_xfer *xs; /* SCSI xfer ctrl block from above */
- int flags; /* Status */
-#define ECB_FREE 0x00
-#define ECB_ACTIVE 0x01
-#define ECB_DONE 0x04
+ struct scsi_xfer *xs; /* SCSI xfer ctrl block from above */
+ int flags; /* Status */
+#define ECB_QNONE 0
+#define ECB_QFREE 1
+#define ECB_QREADY 2
+#define ECB_QNEXUS 3
+#define ECB_QBITS 0x07
#define ECB_CHKSENSE 0x08
- struct scsi_generic cmd; /* SCSI command block */
- int clen;
- caddr_t daddr; /* Saved data pointer */
- int dleft; /* Residue */
- int stat; /* SCSI status byte */
+#define ECB_ABORTED 0x10
+#define ECB_SETQ(e, q) do (e)->flags = ((e)->flags&~ECB_QBITS)|(q); while(0)
+ struct scsi_generic cmd; /* SCSI command block */
+ int clen;
+ char *daddr; /* Saved data pointer */
+ int dleft; /* Residue */
+ u_char stat; /* SCSI status byte */
+#if ESP_DEBUG > 0
+ char trace[1000];
+#endif
};
+#if ESP_DEBUG > 0
+#define ECB_TRACE(ecb, msg, a, b) do { \
+ const char *f = "[" msg "]"; \
+ int n = strlen((ecb)->trace); \
+ if (n < (sizeof((ecb)->trace)-100)) \
+ sprintf((ecb)->trace + n, f, a, b); \
+} while(0)
+#else
+#define ECB_TRACE(ecb, msg, a, b)
+#endif
-/*
- * Some info about each (possible) target on the SCSI bus. This should
- * probably have been a "per target+lunit" structure, but we'll leave it at
+/*
+ * Some info about each (possible) target on the SCSI bus. This should
+ * probably have been a "per target+lunit" structure, but we'll leave it at
* this for now. Is there a way to reliably hook it up to sc->fordriver??
*/
struct esp_tinfo {
- int cmds; /* #commands processed */
- int dconns; /* #disconnects */
- int touts; /* #timeouts */
- int perrs; /* #parity errors */
- int senses; /* #request sense commands sent */
- u_char lubusy; /* What luns are busy? */
- u_char flags;
-#define DO_NEGOTIATE 0x01 /* (re)negotiate synchronous options */
-#define TARGET_BUSY 0x02 /* target is busy, i.e. cmd in progress */
- u_char period; /* sync period */
- u_char synctp; /* sync period translated for SYNCTP */
- u_char offset; /* sync offset */
-};
+ int cmds; /* #commands processed */
+ int dconns; /* #disconnects */
+ int touts; /* #timeouts */
+ int perrs; /* #parity errors */
+ int senses; /* #request sense commands sent */
+ ushort lubusy; /* What local units/subr. are busy? */
+ u_char flags;
+#define T_NEED_TO_RESET 0x01 /* Should send a BUS_DEV_RESET */
+#define T_NEGOTIATE 0x02 /* (Re)Negotiate synchronous options */
+#define T_BUSY 0x04 /* Target is busy, i.e. cmd in progress */
+#define T_SYNCMODE 0x08 /* sync mode has been negotiated */
+#define T_SYNCHOFF 0x10 /* .. */
+#define T_RSELECTOFF 0x20 /* .. */
+ u_char period; /* Period suggestion */
+ u_char offset; /* Offset suggestion */
+} tinfo_t;
/* Register a linenumber (for debugging) */
#define LOGLINE(p)
-#define ESP_SHOWECBS 0x01
-#define ESP_SHOWINTS 0x02
-#define ESP_SHOWCMDS 0x04
-#define ESP_SHOWMISC 0x08
-#define ESP_SHOWTRAC 0x10
-#define ESP_SHOWSTART 0x20
-#define ESP_SHOWPHASE 0x40
+#define ESP_SHOWECBS 0x01
+#define ESP_SHOWINTS 0x02
+#define ESP_SHOWCMDS 0x04
+#define ESP_SHOWMISC 0x08
+#define ESP_SHOWTRAC 0x10
+#define ESP_SHOWSTART 0x20
+#define ESP_SHOWPHASE 0x40
+#define ESP_SHOWDMA 0x80
+#define ESP_SHOWCCMDS 0x100
+#define ESP_SHOWMSGS 0x200
-#if ESP_DEBUG
+#ifdef ESP_DEBUG
+extern int esp_debug;
#define ESP_ECBS(str) do {if (esp_debug & ESP_SHOWECBS) printf str;} while (0)
#define ESP_MISC(str) do {if (esp_debug & ESP_SHOWMISC) printf str;} while (0)
#define ESP_INTS(str) do {if (esp_debug & ESP_SHOWINTS) printf str;} while (0)
@@ -141,6 +132,8 @@ struct esp_tinfo {
#define ESP_CMDS(str) do {if (esp_debug & ESP_SHOWCMDS) printf str;} while (0)
#define ESP_START(str) do {if (esp_debug & ESP_SHOWSTART) printf str;}while (0)
#define ESP_PHASE(str) do {if (esp_debug & ESP_SHOWPHASE) printf str;}while (0)
+#define ESP_DMA(str) do {if (esp_debug & ESP_SHOWDMA) printf str;}while (0)
+#define ESP_MSGS(str) do {if (esp_debug & ESP_SHOWMSGS) printf str;}while (0)
#else
#define ESP_ECBS(str)
#define ESP_MISC(str)
@@ -149,85 +142,104 @@ struct esp_tinfo {
#define ESP_CMDS(str)
#define ESP_START(str)
#define ESP_PHASE(str)
+#define ESP_DMA(str)
#endif
+#define ESP_MAX_MSG_LEN 8
+
struct esp_softc {
- struct device sc_dev;
- struct sbusdev sc_sd; /* sbus device */
- struct intrhand sc_ih; /* intr handler */
- struct evcnt sc_intrcnt; /* intr count */
- struct scsi_link sc_link; /* scsi lint struct */
- struct espregs *sc_regs; /* the registers */
- struct dma_softc *sc_dma; /* corresponding DMA ctrlr */
+ struct device sc_dev; /* us as a device */
+ struct sbusdev sc_sd; /* sbus device */
+ struct intrhand sc_ih; /* intr handler */
+ struct evcnt sc_intrcnt; /* intr count */
+ struct scsi_link sc_link; /* scsi lint struct */
+ volatile u_char *sc_reg; /* the registers */
+ struct dma_softc *sc_dma; /* pointer to my dma */
/* register defaults */
- u_char sc_cfg1; /* config 1 */
- u_char sc_cfg2; /* config 2, only ESP100A/200 */
- u_char sc_cfg3; /* config 3, only ESP200 */
- u_char sc_ccf; /* clock conversion */
+ u_char sc_cfg1; /* Config 1 */
+ u_char sc_cfg2; /* Config 2, not ESP100 */
+ u_char sc_cfg3; /* Config 3, only ESP200 */
+ u_char sc_ccf; /* Clock Conversion */
u_char sc_timeout;
- u_char sc_espintr; /* copies of registers */
+ /* register copies, see espreadregs() */
+ u_char sc_espintr;
u_char sc_espstat;
u_char sc_espstep;
u_char sc_espfflags;
- struct bootpath *sc_bp; /* current boot path component */
-
/* Lists of command blocks */
- TAILQ_HEAD(ecb_list, ecb) free_list, ready_list, nexus_list;
+ TAILQ_HEAD(ecb_list, ecb) free_list,
+ ready_list,
+ nexus_list;
- struct ecb *sc_nexus; /* current command */
- struct ecb sc_ecb[8]; /* one per target */
- struct esp_tinfo sc_tinfo[8];
+ struct ecb *sc_nexus; /* current command */
+ struct ecb sc_ecb[8]; /* one per target */
+ struct esp_tinfo sc_tinfo[8];
- /* data about the current nexus (updated for every cmd switch) */
- caddr_t sc_dp; /* current data pointer */
- size_t sc_dleft; /* data left to transfer */
+ /* Data about the current nexus (updated for every cmd switch) */
+ caddr_t sc_dp; /* Current data pointer */
+ ssize_t sc_dleft; /* Data left to transfer */
- /* adapter state */
- int sc_phase; /* bus phase (based on sc_espstat) */
- u_char sc_state; /* state applicable to the adapter */
+ /* Adapter state */
+ int sc_phase; /* Copy of what bus phase we are in */
+ int sc_prevphase; /* Copy of what bus phase we were in */
+ u_char sc_state; /* State applicable to the adapter */
u_char sc_flags;
u_char sc_selid;
+ u_char sc_lastcmd;
- /* message stuff */
- u_char sc_msgpriq; /* messages to send (see SEND_* below) */
- u_char sc_msgout; /* message that is on it's way out */
- u_char sc_omess[ESP_MSGLEN_MAX];
- u_char *sc_omp; /* message pointer (for multibyte messages) */
+ /* Message stuff */
+ u_char sc_msgpriq; /* One or more messages to send (encoded) */
+ u_char sc_msgout; /* What message is on its way out? */
+ u_char sc_omess[ESP_MAX_MSG_LEN];
+ caddr_t sc_omp; /* Message pointer (for multibyte messages) */
size_t sc_omlen;
- u_char sc_imess[ESP_MSGLEN_MAX + 1];
- u_char *sc_imp; /* message pointer (for multibyte messages) */
+ u_char sc_imess[ESP_MAX_MSG_LEN + 1];
+ caddr_t sc_imp; /* Message pointer (for multibyte messages) */
size_t sc_imlen;
/* hardware/openprom stuff */
- int sc_node; /* PROM node ID */
- int sc_freq; /* Freq in HZ */
- int sc_pri; /* SBUS priority */
- int sc_id; /* our scsi id */
- int sc_rev; /* esp revision */
- int sc_minsync; /* minimum sync period / 4 */
+ int sc_node; /* PROM node ID */
+ int sc_freq; /* Freq in HZ */
+ int sc_pri; /* SBUS priority */
+ int sc_id; /* our scsi id */
+ int sc_rev; /* esp revision */
+ int sc_minsync; /* minimum sync period / 4 */
+ int sc_maxxfer; /* maximum transfer size */
};
+/*
+ * Macros to read and write the chip's registers.
+ */
+#define ESP_READ_REG(sc, reg) \
+ ((sc)->sc_reg[(reg) * 4])
+#define ESP_WRITE_REG(sc, reg, val) \
+ do { \
+ u_char v = (val); \
+ (sc)->sc_reg[(reg) * 4] = v; \
+ } while (0)
+
/* values for sc_state */
-#define ESPS_IDLE 0x01 /* waiting for something to do */
-#define ESPS_EXPECTDISC 0x02 /* a disconnect is going to happen */
-#define ESPS_SELECTING 0x03 /* SCSI command is arbiting */
-#define ESPS_RESELECTED 0x04 /* has been reselected */
-#define ESPS_NEXUS 0x05 /* actively using the SCSI bus */
-#define ESPS_DOINGDMA 0x06 /* doing a DMA transaction */
-#define ESPS_DOINGMSGOUT 0x07 /* message-out in progress */
-#define ESPS_DOINGMSGIN 0x08 /* message-in in progress */
-#define ESPS_DOINGSTATUS 0x09 /* status-in in progress */
-#define ESPS_SELECTSTOP 0x0a /* sent first byte of message.. */
-#define ESPS_MULTMSGEND 0x0b /* done sending multibyte msg */
+#define ESP_IDLE 0x01 /* waiting for something to do */
+#define ESP_TMP_UNAVAIL 0x02 /* Don't accept SCSI commands */
+#define ESP_SELECTING 0x03 /* SCSI command is arbiting */
+#define ESP_RESELECTED 0x04 /* Has been reselected */
+#define ESP_HASNEXUS 0x05 /* Actively using the SCSI bus */
+#define ESP_CLEANING 0x06
+#define ESP_SBR 0x07 /* Expect a SCSI RST because we commanded it */
/* values for sc_flags */
-#define ESPF_DROP_MSGI 0x01 /* Discard all msgs (parity err detected) */
-#define ESPF_MAYDISCON 0x02 /* disconnection allowed */
-#define ESPF_DONE 0x04 /* finished transaction */
-#define ESPF_SYNCHNEGO 0x08 /* Synch negotiation in progress. */
+#define ESP_DROP_MSGI 0x01 /* Discard all msgs (parity err detected) */
+#define ESP_DOINGDMA 0x02 /* The FIFO data path is active! */
+#define ESP_BUSFREE_OK 0x04 /* Bus free phase is OK. */
+#define ESP_SYNCHNEGO 0x08 /* Synch negotiation in progress. */
+/*#define ESP_BLOCKED 0x10 * Don't schedule new scsi bus operations */
+#define ESP_DISCON 0x10 /* Target sent DISCONNECT msg */
+#define ESP_ABORTING 0x20 /* Bailing out */
+#define ESP_ICCS 0x40 /* Expect status phase results */
+#define ESP_WAITI 0x80 /* Waiting for non-DMA data to arrive */
/* values for sc_msgout */
#define SEND_DEV_RESET 0x01
@@ -238,14 +250,25 @@ struct esp_softc {
#define SEND_IDENTIFY 0x20
#define SEND_SDTR 0x40
-#define ST_MASK 0x3e /* bits 0, 6, and 7 are reserved */
+/* SCSI Status codes */
+#define ST_GOOD 0x00
+#define ST_CHKCOND 0x02
+#define ST_CONDMET 0x04
+#define ST_BUSY 0x08
+#define ST_INTERMED 0x10
+#define ST_INTERMED_CONDMET 0x14
+#define ST_RESERVATION_CONFLICT 0x18
+#define ST_CMD_TERM 0x22
+#define ST_QUEUE_FULL 0x28
-/* physical phase bits */
+#define ST_MASK 0x3e /* bit 0,6,7 is reserved */
+
+/* phase bits */
#define IOI 0x01
#define CDI 0x02
#define MSGI 0x04
-/* values for sc_phase (where information transfer happens) */
+/* Information transfer phases */
#define DATA_OUT_PHASE (0)
#define DATA_IN_PHASE (IOI)
#define COMMAND_PHASE (CDI)
@@ -255,5 +278,23 @@ struct esp_softc {
#define PHASE_MASK (MSGI|CDI|IOI)
-#define TARGETNAME(ecb) \
- ((struct device *)(ecb)->xs->sc_link->adapter_softc)->dv_xname
+/* Some pseudo phases for getphase()*/
+#define BUSFREE_PHASE 0x100 /* Re/Selection no longer valid */
+#define INVALID_PHASE 0x101 /* Re/Selection valid, but no REQ yet */
+#define PSEUDO_PHASE 0x100 /* "pseudo" bit */
+
+#ifdef ESP_DEBUG
+#define ESPCMD(sc, cmd) do { \
+ if (esp_debug & ESP_SHOWCCMDS) \
+ printf("<cmd:0x%x>", (unsigned)cmd); \
+ sc->sc_lastcmd = cmd; \
+ ESP_WRITE_REG(sc, ESP_CMD, cmd); \
+} while (0)
+#else
+#define ESPCMD(sc, cmd) ESP_WRITE_REG(sc, ESP_CMD, cmd)
+#endif
+
+#define SAME_ESP(sc, bp, ca) \
+ ((bp->val[0] == ca->ca_slot && bp->val[1] == ca->ca_offset) || \
+ (bp->val[0] == -1 && bp->val[1] == sc->sc_dev.dv_unit))
+
diff --git a/sys/arch/sparc/dev/fb.c b/sys/arch/sparc/dev/fb.c
index ade9db9dbe6..836bc7dbdad 100644
--- a/sys/arch/sparc/dev/fb.c
+++ b/sys/arch/sparc/dev/fb.c
@@ -1,4 +1,4 @@
-/* $NetBSD: fb.c,v 1.11 1995/10/08 01:39:19 pk Exp $ */
+/* $NetBSD: fb.c,v 1.18 1996/04/01 17:29:54 christos Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -50,19 +50,24 @@
*/
#include <sys/param.h>
-#include <sys/conf.h>
+#include <sys/systm.h>
#include <sys/device.h>
#include <sys/proc.h>
+#include <sys/conf.h>
#include <machine/autoconf.h>
#include <machine/fbio.h>
+#include <machine/kbd.h>
#include <machine/fbvar.h>
+#include <machine/conf.h>
#if defined(SUN4)
#include <machine/eeprom.h>
+#include <sparc/dev/pfourreg.h>
#endif
static struct fbdevice *devfb;
+
void
fb_unblank()
{
@@ -72,13 +77,71 @@ fb_unblank()
}
void
-fb_attach(fb)
+fb_attach(fb, isconsole)
struct fbdevice *fb;
+ int isconsole;
{
+ static int no_replace, seen_force;
+
+ /*
+ * We've already had a framebuffer forced into /dev/fb. Don't
+ * allow any more, even if this is the console.
+ */
+ if (seen_force) {
+ if (devfb) { /* sanity */
+ printf("%s: /dev/fb already full\n",
+ fb->fb_device->dv_xname);
+ return;
+ } else
+ seen_force = 0;
+ }
+
+ /*
+ * Check to see if we're being forced into /dev/fb.
+ */
+ if (fb->fb_flags & FB_FORCE) {
+ if (devfb)
+ printf("%s: forcefully replacing %s\n",
+ fb->fb_device->dv_xname,
+ devfb->fb_device->dv_xname);
+ devfb = fb;
+ seen_force = no_replace = 1;
+ goto attached;
+ }
+
+ /*
+ * Check to see if we're the console. If we are, then replace
+ * any currently existing framebuffer.
+ */
+ if (isconsole) {
+ if (devfb)
+ printf("%s: replacing %s\n", fb->fb_device->dv_xname,
+ devfb->fb_device->dv_xname);
+ devfb = fb;
+ no_replace = 1;
+ goto attached;
+ }
+
+ /*
+ * For the final case, we check to see if we can replace an
+ * existing framebuffer, if not, say so and return.
+ */
+ if (no_replace) {
+ if (devfb) { /* sanity */
+ printf("%s: /dev/fb already full\n",
+ fb->fb_device->dv_xname);
+ return;
+ } else
+ no_replace = 0;
+ }
if (devfb)
- printf("warning: multiple /dev/fb declarers\n");
+ printf("%s: replacing %s\n", fb->fb_device->dv_xname,
+ devfb->fb_device->dv_xname);
devfb = fb;
+
+ attached:
+ printf("%s: attached to /dev/fb\n", devfb->fb_device->dv_xname);
}
int
@@ -138,25 +201,18 @@ fb_setsize(fb, depth, def_width, def_height, node, bustype)
* to be correct as defaults go...
*/
switch (bustype) {
- case BUS_PFOUR:
- fb->fb_type.fb_width = def_width;
- fb->fb_type.fb_height = def_height;
- fb->fb_linebytes = (fb->fb_type.fb_width * depth) / 8;
- break;
case BUS_VME16:
case BUS_VME32:
case BUS_OBIO:
-#if defined(SUN4M)
- if (cputyp == CPU_SUN4M) { /* 4m has framebuffer on obio */
- fb->fb_type.fb_width = getpropint(node, "width",
- def_width);
+ if (CPU_ISSUN4M) { /* 4m has framebuffer on obio */
+ fb->fb_type.fb_width = getpropint(node, "width",
+ def_width);
fb->fb_type.fb_height = getpropint(node, "height",
- def_height);
+ def_height);
fb->fb_linebytes = getpropint(node, "linebytes",
(fb->fb_type.fb_width * depth) / 8);
break;
}
-#endif
/* Set up some defaults. */
fb->fb_type.fb_width = def_width;
fb->fb_type.fb_height = def_height;
@@ -165,39 +221,104 @@ fb_setsize(fb, depth, def_width, def_height, node, bustype)
* This is not particularly useful on Sun 4 VME framebuffers.
* The EEPROM only contains info about the built-in.
*/
- if (cputyp == CPU_SUN4 && (bustype == BUS_VME16 ||
+ if (CPU_ISSUN4 && (bustype == BUS_VME16 ||
bustype == BUS_VME32))
goto donesize;
#if defined(SUN4)
- if (cputyp==CPU_SUN4) {
+ if (CPU_ISSUN4) {
struct eeprom *eep = (struct eeprom *)eeprom_va;
- if (eep != NULL) {
- switch (eep->ee_diag.eed_scrsize) {
- case EED_SCR_1152X900:
+
+ if (fb->fb_flags & FB_PFOUR) {
+ volatile u_int32_t pfour;
+
+ /*
+ * Some pfour framebuffers, e.g. the
+ * cgsix, don't encode resolution the
+ * same, so the driver handles that.
+ * The driver can let us know that it
+ * needs to do this by not mapping in
+ * the pfour register by the time this
+ * routine is called.
+ */
+ if (fb->fb_pfour == NULL)
+ goto donesize;
+
+ pfour = *fb->fb_pfour;
+
+ /*
+ * Use the pfour register to determine
+ * the size. Note that the cgsix and
+ * cgeight don't use this size encoding.
+ * In this case, we have to settle
+ * for the defaults we were provided
+ * with.
+ */
+ if ((PFOUR_ID(pfour) == PFOUR_ID_COLOR24) ||
+ (PFOUR_ID(pfour) == PFOUR_ID_FASTCOLOR))
+ goto donesize;
+
+ switch (PFOUR_SIZE(pfour)) {
+ case PFOUR_SIZE_1152X900:
fb->fb_type.fb_width = 1152;
fb->fb_type.fb_height = 900;
break;
- case EED_SCR_1024X1024:
+
+ case PFOUR_SIZE_1024X1024:
fb->fb_type.fb_width = 1024;
fb->fb_type.fb_height = 1024;
break;
- case EED_SCR_1600X1280:
+
+ case PFOUR_SIZE_1280X1024:
+ fb->fb_type.fb_width = 1280;
+ fb->fb_type.fb_height = 1024;
+ break;
+
+ case PFOUR_SIZE_1600X1280:
fb->fb_type.fb_width = 1600;
fb->fb_type.fb_height = 1280;
break;
- case EED_SCR_1440X1440:
+
+ case PFOUR_SIZE_1440X1440:
fb->fb_type.fb_width = 1440;
fb->fb_type.fb_height = 1440;
break;
- case EED_SCR_640X480:
+
+ case PFOUR_SIZE_640X480:
fb->fb_type.fb_width = 640;
fb->fb_type.fb_height = 480;
break;
- case EED_SCR_1280X1024:
- fb->fb_type.fb_width = 1280;
+
+ default:
+ /*
+ * XXX: Do nothing, I guess.
+ * Should we print a warning about
+ * an unknown value? --thorpej
+ */
+ break;
+ }
+ } else if (eep != NULL) {
+ switch (eep->eeScreenSize) {
+ case EE_SCR_1152X900:
+ fb->fb_type.fb_width = 1152;
+ fb->fb_type.fb_height = 900;
+ break;
+
+ case EE_SCR_1024X1024:
+ fb->fb_type.fb_width = 1024;
fb->fb_type.fb_height = 1024;
break;
+
+ case EE_SCR_1600X1280:
+ fb->fb_type.fb_width = 1600;
+ fb->fb_type.fb_height = 1280;
+ break;
+
+ case EE_SCR_1440X1440:
+ fb->fb_type.fb_width = 1440;
+ fb->fb_type.fb_height = 1440;
+ break;
+
default:
/*
* XXX: Do nothing, I guess.
@@ -210,20 +331,22 @@ fb_setsize(fb, depth, def_width, def_height, node, bustype)
}
#endif /* SUN4 */
#if defined(SUN4M)
- if (cputyp==CPU_SUN4M) {
+ if (CPU_ISSUN4M) {
/* XXX: need code to find 4/600 vme screen size */
}
#endif /* SUN4M */
-donesize:
+ donesize:
fb->fb_linebytes = (fb->fb_type.fb_width * depth) / 8;
break;
+
case BUS_SBUS:
- fb->fb_type.fb_width = getpropint(node, "width", 1152);
- fb->fb_type.fb_height = getpropint(node, "height", 900);
+ fb->fb_type.fb_width = getpropint(node, "width", def_width);
+ fb->fb_type.fb_height = getpropint(node, "height", def_height);
fb->fb_linebytes = getpropint(node, "linebytes",
(fb->fb_type.fb_width * depth) / 8);
break;
+
default:
panic("fb_setsize: inappropriate bustype");
/* NOTREACHED */
@@ -233,7 +356,8 @@ donesize:
#ifdef RASTERCONSOLE
#include <machine/kbd.h>
-extern int (*v_putc) __P((int));
+static int a2int __P((char *, int));
+static void fb_bell __P((int));
static int
a2int(cp, deflt)
@@ -278,40 +402,103 @@ fbrcons_init(fb)
#if defined(RASTERCONS_FULLSCREEN) || defined(RASTERCONS_SMALLFONT)
rc->rc_maxcol = rc->rc_width / rc->rc_font->width;
- rc->rc_maxrow = rc->rc_height / rc->rc_font->height;
+ rc->rc_maxrow = rc->rc_height / rc->rc_font->height;
#else
#if defined(SUN4)
- if (cputyp == CPU_SUN4) {
+ if (CPU_ISSUN4) {
struct eeprom *eep = (struct eeprom *)eeprom_va;
if (eep == NULL) {
rc->rc_maxcol = 80;
rc->rc_maxrow = 34;
} else {
- rc->rc_maxcol = eep->ee_diag.eed_colsize;
- rc->rc_maxrow = eep->ee_diag.eed_rowsize;
+ rc->rc_maxcol = eep->eeTtyCols;
+ rc->rc_maxrow = eep->eeTtyRows;
}
}
#endif /* SUN4 */
-#if defined(SUN4C) || defined(SUN4M)
- if (cputyp != CPU_SUN4) {
- rc->rc_maxcol = a2int(getpropstring(optionsnode,
- "screen-#columns"), 80);
- rc->rc_maxrow = a2int(getpropstring(optionsnode,
- "screen-#rows"), 34);
+
+ if (!CPU_ISSUN4) {
+ rc->rc_maxcol =
+ a2int(getpropstring(optionsnode, "screen-#columns"), 80);
+ rc->rc_maxrow =
+ a2int(getpropstring(optionsnode, "screen-#rows"), 34);
}
-#endif /* SUN4C || SUN4M */
#endif /* RASTERCONS_FULLSCREEN || RASTERCONS_SMALLFONT */
#if !(defined(RASTERCONS_FULLSCREEN) || defined(RASTERCONS_SMALLFONT))
/* Determine addresses of prom emulator row and column */
- if (cputyp == CPU_SUN4 || romgetcursoraddr(&rc->rc_row, &rc->rc_col))
+ if (CPU_ISSUN4 ||
+ romgetcursoraddr(&rc->rc_row, &rc->rc_col))
#endif
rc->rc_row = rc->rc_col = NULL;
rc->rc_bell = fb_bell;
rcons_init(rc);
/* Hook up virtual console */
- v_putc = (int (*) __P((int)))rcons_cnputc;
+ v_putc = rcons_cnputc;
}
#endif
+
+#if defined(SUN4)
+/*
+ * Support routines for pfour framebuffers.
+ */
+
+/*
+ * Probe for a pfour framebuffer. Return values:
+ *
+ * PFOUR_NOTPFOUR framebuffer is not a pfour
+ * framebuffer
+ *
+ * otherwise returns pfour ID
+ */
+int
+fb_pfour_id(va)
+ void *va;
+{
+ volatile u_int32_t val, save, *pfour = va;
+
+ /* Read the pfour register. */
+ save = *pfour;
+
+ /*
+ * Try to modify the type code. If it changes, put the
+ * original value back, and notify the caller that it's
+ * not a pfour framebuffer.
+ */
+ val = save & ~PFOUR_REG_RESET;
+ *pfour = (val ^ PFOUR_FBTYPE_MASK);
+ if ((*pfour ^ val) & PFOUR_FBTYPE_MASK) {
+ *pfour = save;
+ return (PFOUR_NOTPFOUR);
+ }
+
+ return (PFOUR_ID(val));
+}
+
+/*
+ * Return the status of the video enable.
+ */
+int
+fb_pfour_get_video(fb)
+ struct fbdevice *fb;
+{
+
+ return ((*fb->fb_pfour & PFOUR_REG_VIDEO) != 0);
+}
+
+/*
+ * Enable or disable the framebuffer.
+ */
+void
+fb_pfour_set_video(fb, enable)
+ struct fbdevice *fb;
+ int enable;
+{
+ volatile u_int32_t pfour;
+
+ pfour = *fb->fb_pfour & ~(PFOUR_REG_INTCLR|PFOUR_REG_VIDEO);
+ *fb->fb_pfour = pfour | (enable ? PFOUR_REG_VIDEO : 0);
+}
+#endif /* SUN4 */
diff --git a/sys/arch/sparc/dev/fd.c b/sys/arch/sparc/dev/fd.c
index fa729ce26ea..d4dee591b72 100644
--- a/sys/arch/sparc/dev/fd.c
+++ b/sys/arch/sparc/dev/fd.c
@@ -1,4 +1,4 @@
-/* $NetBSD: fd.c,v 1.13 1995/10/09 22:33:07 pk Exp $ */
+/* $NetBSD: fd.c,v 1.33.4.1 1996/06/12 20:52:25 pk Exp $ */
/*-
* Copyright (c) 1993, 1994, 1995 Charles Hannum.
@@ -43,7 +43,6 @@
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/kernel.h>
-#include <sys/conf.h>
#include <sys/file.h>
#include <sys/ioctl.h>
#include <sys/device.h>
@@ -52,12 +51,17 @@
#include <sys/disk.h>
#include <sys/buf.h>
#include <sys/uio.h>
-#include <sys/mtio.h>
+#include <sys/stat.h>
#include <sys/syslog.h>
#include <sys/queue.h>
+#include <sys/conf.h>
+
+#include <dev/cons.h>
#include <machine/cpu.h>
#include <machine/autoconf.h>
+#include <machine/conf.h>
+
#include <sparc/sparc/auxreg.h>
#include <sparc/dev/fdreg.h>
#include <sparc/dev/fdvar.h>
@@ -128,10 +132,16 @@ extern struct fdcio *fdciop;
int fdcmatch __P((struct device *, void *, void *));
void fdcattach __P((struct device *, struct device *, void *));
-struct cfdriver fdccd = {
- NULL, "fdc", fdcmatch, fdcattach, DV_DULL, sizeof(struct fdc_softc)
+struct cfattach fdc_ca = {
+ sizeof(struct fdc_softc), fdcmatch, fdcattach
};
+struct cfdriver fdc_cd = {
+ NULL, "fdc", DV_DULL
+};
+
+__inline struct fd_type *fd_dev_to_type __P((struct fd_softc *, dev_t));
+
/*
* Floppies come in various flavors, e.g., 1.2MB vs 1.44MB; here is how
* we tell them apart.
@@ -184,6 +194,8 @@ struct fd_softc {
#define FD_MOTOR_WAIT 0x04 /* motor coming up */
int sc_cylin; /* where we think the head is */
+ void *sc_sdhook; /* shutdownhook cookie */
+
TAILQ_ENTRY(fd_softc) sc_drivechain;
int sc_ops; /* I/O ops since last switch */
struct buf sc_q; /* head of buf chain */
@@ -193,14 +205,19 @@ struct fd_softc {
int fdmatch __P((struct device *, void *, void *));
void fdattach __P((struct device *, struct device *, void *));
-struct cfdriver fdcd = {
- NULL, "fd", fdmatch, fdattach, DV_DISK, sizeof(struct fd_softc)
+struct cfattach fd_ca = {
+ sizeof(struct fd_softc), fdmatch, fdattach
};
-void fdgetdisklabel __P((struct fd_softc *));
+struct cfdriver fd_cd = {
+ NULL, "fd", DV_DISK
+};
+
+void fdgetdisklabel __P((dev_t));
int fd_get_parms __P((struct fd_softc *));
void fdstrategy __P((struct buf *));
void fdstart __P((struct fd_softc *));
+int fdprint __P((void *, char *));
struct dkdriver fddkdriver = { fdstrategy };
@@ -223,6 +240,9 @@ void fdchwintr __P((void));
int fdcswintr __P((struct fdc_softc *));
void fdcretry __P((struct fdc_softc *fdc));
void fdfinish __P((struct fd_softc *fd, struct buf *bp));
+void fd_do_eject __P((void));
+void fd_mountroot_hook __P((struct device *));
+static void fdconf __P((struct fdc_softc *));
#if PIL_FDSOFT == 4
#define IE_FDSOFT IE_L4
@@ -230,29 +250,44 @@ void fdfinish __P((struct fd_softc *fd, struct buf *bp));
#error 4
#endif
-#define OBP_FDNAME (cputyp == CPU_SUN4M ? "SUNW,fdtwo" : "fd")
+#define OBP_FDNAME (CPU_ISSUN4M ? "SUNW,fdtwo" : "fd")
int
fdcmatch(parent, match, aux)
struct device *parent;
void *match, *aux;
{
- struct cfdata *cf = match;
register struct confargs *ca = aux;
register struct romaux *ra = &ca->ca_ra;
+ /*
+ * Floppy doesn't exist on sun4.
+ */
+ if (CPU_ISSUN4)
+ return (0);
+
+ /*
+ * Floppy controller is on mainbus on sun4c.
+ */
+ if ((CPU_ISSUN4C) && (ca->ca_bustype != BUS_MAIN))
+ return (0);
+
+ /*
+ * Floppy controller is on obio on sun4m.
+ */
+ if ((CPU_ISSUN4M) && (ca->ca_bustype != BUS_OBIO))
+ return (0);
+
/* Sun PROMs call the controller an "fd" or "SUNW,fdtwo" */
if (strcmp(OBP_FDNAME, ra->ra_name))
return (0);
- if (ca->ca_bustype == BUS_MAIN) {
- if (ca->ca_ra.ra_vaddr &&
- probeget(ca->ca_ra.ra_vaddr, 1) == -1) {
- return (0);
- }
- return (1);
+
+ if (ca->ca_ra.ra_vaddr &&
+ probeget(ca->ca_ra.ra_vaddr, 1) == -1) {
+ return (0);
}
- return (0);
+ return (1);
}
/*
@@ -260,7 +295,7 @@ fdcmatch(parent, match, aux)
*/
struct fdc_attach_args {
int fa_drive;
- int fa_bootdev;
+ struct bootpath *fa_bootpath;
struct fd_type *fa_deftype;
};
@@ -315,7 +350,8 @@ fdcattach(parent, self, aux)
register struct confargs *ca = aux;
struct fdc_softc *fdc = (void *)self;
struct fdc_attach_args fa;
- int n, pri;
+ struct bootpath *bp;
+ int pri;
char code;
if (ca->ca_ra.ra_vaddr)
@@ -398,13 +434,36 @@ fdcattach(parent, self, aux)
* Controller and drives are represented by one and the same
* Openprom node, so we can as well check for the floppy boots here.
*/
- if (ca->ca_ra.ra_bp &&
- strcmp(ca->ca_ra.ra_bp->name, OBP_FDNAME) == 0 &&
- ca->ca_ra.ra_bp->val[0] == 0 &&
- ca->ca_ra.ra_bp->val[1] == 0)
- fa.fa_bootdev = 1;
- else
- fa.fa_bootdev = 0;
+ fa.fa_bootpath = 0;
+ if ((bp = ca->ca_ra.ra_bp) && strcmp(bp->name, OBP_FDNAME) == 0) {
+ /*
+ * WOAH THERE! It looks like we can get the bootpath
+ * in several different formats!! The faked
+ * bootpath (and some v2?) looks like /fd@0,0
+ * but the real bootpath on some v2 OpenPROM
+ * systems looks like /fd0. In the case of
+ * a floppy controller on obio (such as on the sun4m),
+ * we use "slot, offset" to determine if this is the
+ * right one. --thorpej
+ */
+ switch (ca->ca_bustype) {
+ case BUS_MAIN:
+ if (((bp->val[0] == 0) && /* /fd@0,0 */
+ (bp->val[1] == 0)) ||
+ ((bp->val[0] == -1) && /* /fd0 */
+ (bp->val[1] == 0)))
+ fa.fa_bootpath = bp;
+ break;
+
+ case BUS_OBIO:
+ /* /obio0/SUNW,fdtwo@0,700000 */
+ if ((bp->val[0] == ca->ca_slot) &&
+ (bp->val[1] == ca->ca_offset))
+ fa.fa_bootpath = bp;
+ break;
+ }
+
+ }
/* physical limit: four drives per controller. */
for (fa.fa_drive = 0; fa.fa_drive < 4; fa.fa_drive++) {
@@ -422,10 +481,9 @@ fdmatch(parent, match, aux)
void *match, *aux;
{
struct fdc_softc *fdc = (void *)parent;
- struct cfdata *cf = match;
struct fdc_attach_args *fa = aux;
int drive = fa->fa_drive;
- int n;
+ int n, ok;
if (drive > 0)
/* XXX - for now, punt > 1 drives */
@@ -439,13 +497,12 @@ fdmatch(parent, match, aux)
} else {
auxregbisc(AUXIO_FDS, 0);
}
-
fdc->sc_nstat = 0;
out_fdc(fdc, NE7CMD_RECAL);
out_fdc(fdc, drive);
/* wait for recalibrate */
- for (n = 0; n < 100000; n++) {
- delay(10);
+ for (n = 0; n < 10000; n++) {
+ delay(1000);
if ((*fdc->sc_reg_msr & (NE7_RQM|NE7_DIO|NE7_CB)) == NE7_RQM) {
/* wait a bit longer till device *really* is ready */
delay(100000);
@@ -471,15 +528,17 @@ fdmatch(parent, match, aux)
printf("\n");
}
#endif
- if (n != 2 || (fdc->sc_status[0] & 0xf8) != 0x20)
- return 0;
+ ok = (n == 2 && (fdc->sc_status[0] & 0xf8) == 0x20) ? 1 : 0;
+
/* turn off motor */
- if (fdc->sc_flags & FDC_82077)
+ if (fdc->sc_flags & FDC_82077) {
+ /* select drive and turn on motor */
*fdc->sc_reg_dor = FDO_FRST;
- else
+ } else {
auxregbisc(0, AUXIO_FDS);
+ }
- return 1;
+ return ok;
}
/*
@@ -499,7 +558,7 @@ fdattach(parent, self, aux)
/* XXX Allow `flags' to override device type? */
if (type)
- printf(": %s, %d cyl, %d head, %d sec\n", type->name,
+ printf(": %s %d cyl, %d head, %d sec\n", type->name,
type->tracks, type->heads, type->sectrac);
else
printf(": density unknown\n");
@@ -519,14 +578,23 @@ fdattach(parent, self, aux)
/*
* We're told if we're the boot device in fdcattach().
*/
- if (fa->fa_bootdev)
- bootdv = &fd->sc_dv;
+ if (fa->fa_bootpath)
+ fa->fa_bootpath->dev = &fd->sc_dv;
+
+ /*
+ * Establish a mountroot_hook anyway in case we booted
+ * with RB_ASKNAME and get selected as the boot device.
+ */
+ mountroot_hook_establish(fd_mountroot_hook, &fd->sc_dv);
+
+ /* Make sure the drive motor gets turned off at shutdown time. */
+ fd->sc_sdhook = shutdownhook_establish(fd_motor_off, fd);
/* XXX Need to do some more fiddling with sc_dk. */
dk_establish(&fd->sc_dk, &fd->sc_dv);
}
-inline struct fd_type *
+__inline struct fd_type *
fd_dev_to_type(fd, dev)
struct fd_softc *fd;
dev_t dev;
@@ -548,8 +616,8 @@ fdstrategy(bp)
int s;
/* Valid unit, controller, and request? */
- if (unit >= fdcd.cd_ndevs ||
- (fd = fdcd.cd_devs[unit]) == 0 ||
+ if (unit >= fd_cd.cd_ndevs ||
+ (fd = fd_cd.cd_devs[unit]) == 0 ||
bp->b_blkno < 0 ||
(bp->b_bcount % FDC_BSIZE) != 0) {
bp->b_error = EINVAL;
@@ -582,7 +650,7 @@ fdstrategy(bp)
#ifdef FD_DEBUG
if (fdc_debug > 1)
- printf("fdstrategy: b_blkno %d b_bcount %d blkno %d cylin %d\n",
+ printf("fdstrategy: b_blkno %d b_bcount %ld blkno %d cylin %ld\n",
bp->b_blkno, bp->b_bcount, fd->sc_blkno, bp->b_cylin);
#endif
@@ -685,7 +753,7 @@ fd_set_motor(fdc)
if (fdc->sc_flags & FDC_82077) {
status = FDO_FRST | FDO_FDMAEN;
- if (fd = fdc->sc_drives.tqh_first)
+ if ((fd = fdc->sc_drives.tqh_first) != NULL)
status |= fd->sc_drive;
for (n = 0; n < 4; n++)
@@ -774,18 +842,19 @@ out_fdc(fdc, x)
}
int
-Fdopen(dev, flags)
+fdopen(dev, flags, fmt, p)
dev_t dev;
- int flags;
+ int flags, fmt;
+ struct proc *p;
{
- int unit;
+ int unit, pmask;
struct fd_softc *fd;
struct fd_type *type;
unit = FDUNIT(dev);
- if (unit >= fdcd.cd_ndevs)
+ if (unit >= fd_cd.cd_ndevs)
return ENXIO;
- fd = fdcd.cd_devs[unit];
+ fd = fd_cd.cd_devs[unit];
if (fd == 0)
return ENXIO;
type = fd_dev_to_type(fd, dev);
@@ -800,33 +869,70 @@ Fdopen(dev, flags)
fd->sc_cylin = -1;
fd->sc_flags |= FD_OPEN;
+ /*
+ * Only update the disklabel if we're not open anywhere else.
+ */
+ if (fd->sc_dk.dk_openmask == 0)
+ fdgetdisklabel(dev);
+
+ pmask = (1 << DISKPART(dev));
+
+ switch (fmt) {
+ case S_IFCHR:
+ fd->sc_dk.dk_copenmask |= pmask;
+ break;
+
+ case S_IFBLK:
+ fd->sc_dk.dk_bopenmask |= pmask;
+ break;
+ }
+ fd->sc_dk.dk_openmask =
+ fd->sc_dk.dk_copenmask | fd->sc_dk.dk_bopenmask;
+
return 0;
}
int
-fdclose(dev, flags)
+fdclose(dev, flags, fmt, p)
dev_t dev;
- int flags;
+ int flags, fmt;
+ struct proc *p;
{
- struct fd_softc *fd = fdcd.cd_devs[FDUNIT(dev)];
+ struct fd_softc *fd = fd_cd.cd_devs[FDUNIT(dev)];
+ int pmask = (1 << DISKPART(dev));
fd->sc_flags &= ~FD_OPEN;
+
+ switch (fmt) {
+ case S_IFCHR:
+ fd->sc_dk.dk_copenmask &= ~pmask;
+ break;
+
+ case S_IFBLK:
+ fd->sc_dk.dk_bopenmask &= ~pmask;
+ break;
+ }
+ fd->sc_dk.dk_openmask =
+ fd->sc_dk.dk_copenmask | fd->sc_dk.dk_bopenmask;
+
return 0;
}
int
-fdread(dev, uio)
+fdread(dev, uio, flag)
dev_t dev;
struct uio *uio;
+ int flag;
{
return (physio(fdstrategy, NULL, dev, B_READ, minphys, uio));
}
int
-fdwrite(dev, uio)
+fdwrite(dev, uio, flag)
dev_t dev;
struct uio *uio;
+ int flag;
{
return (physio(fdstrategy, NULL, dev, B_WRITE, minphys, uio));
@@ -855,7 +961,6 @@ fdcstatus(dv, n, s)
char *s;
{
struct fdc_softc *fdc = (void *)dv->dv_parent;
-
#if 0
/*
* A 82072 seems to return <invalid command> on
@@ -949,7 +1054,7 @@ fdchwintr(fdc)
fdcresult(fdc);
fdc->sc_istate = ISTATE_IDLE;
ienab_bis(IE_FDSOFT);
- goto done;
+ return 1;
case ISTATE_IDLE:
case ISTATE_SPURIOUS:
auxregbisc(0, AUXIO_FDS); /* Does this help? */
@@ -957,12 +1062,12 @@ fdchwintr(fdc)
fdc->sc_istate = ISTATE_SPURIOUS;
printf("fdc: stray hard interrupt... ");
ienab_bis(IE_FDSOFT);
- goto done;
+ return 1;
case ISTATE_DMA:
break;
default:
printf("fdc: goofed ...\n");
- goto done;
+ return 1;
}
read = bp->b_flags & B_READ;
@@ -1005,9 +1110,7 @@ fdchwintr(fdc)
break;
}
}
-done:
- sc->sc_intrcnt.ev_count++;
- return (1);
+ return 1;
}
#endif
@@ -1023,7 +1126,7 @@ fdcswintr(fdc)
struct fd_softc *fd;
struct buf *bp;
- int read, head, trac, sec, i, s, nblks;
+ int read, head, sec, nblks;
struct fd_type *type;
@@ -1246,11 +1349,12 @@ loop:
printf("fdc: %d -> threshold\n", thr);
#endif
fdconf(fdc);
- fdc->sc_state = DOIO;
fdc->sc_overruns = 0;
}
- if (++fdc->sc_overruns < 3)
+ if (++fdc->sc_overruns < 3) {
+ fdc->sc_state = DOIO;
goto loop;
+ }
}
fdcretry(fdc);
goto loop;
@@ -1367,7 +1471,7 @@ fdcretry(fdc)
case 0:
/* try again */
fdc->sc_state =
- (fdc->sc_flags & FDC_EIS) ? DOIO : SEEKCOMPLETE;
+ (fdc->sc_flags & FDC_EIS) ? DOIO : DOSEEK;
break;
case 1: case 2: case 3:
@@ -1383,6 +1487,7 @@ fdcretry(fdc)
default:
diskerr(bp, "fd", "hard error", LOG_PRINTF,
fd->sc_skip / FDC_BSIZE, (struct disklabel *)NULL);
+
printf(" (st0 %b st1 %b st2 %b cyl %d head %d sec %d)\n",
fdc->sc_status[0], NE7_ST0BITS,
fdc->sc_status[1], NE7_ST1BITS,
@@ -1406,7 +1511,11 @@ fdsize(dev)
}
int
-fddump()
+fddump(dev, blkno, va, size)
+ dev_t dev;
+ daddr_t blkno;
+ caddr_t va;
+ size_t size;
{
/* Not implemented. */
@@ -1414,29 +1523,18 @@ fddump()
}
int
-fdioctl(dev, cmd, addr, flag)
+fdioctl(dev, cmd, addr, flag, p)
dev_t dev;
u_long cmd;
caddr_t addr;
int flag;
+ struct proc *p;
{
- struct fd_softc *fd = fdcd.cd_devs[FDUNIT(dev)];
- struct disklabel buffer;
+ struct fd_softc *fd = fd_cd.cd_devs[FDUNIT(dev)];
int error;
switch (cmd) {
case DIOCGDINFO:
- bzero(fd->sc_dk.dk_label, sizeof(struct disklabel));
-
- fd->sc_dk.dk_label->d_secpercyl = fd->sc_type->seccyl;
- fd->sc_dk.dk_label->d_type = DTYPE_FLOPPY;
- fd->sc_dk.dk_label->d_secsize = FDC_BSIZE;
-
- if (readdisklabel(dev, fdstrategy,
- fd->sc_dk.dk_label,
- fd->sc_dk.dk_cpulabel) != NULL)
- return EINVAL;
-
*(struct disklabel *)addr = *(fd->sc_dk.dk_label);
return 0;
@@ -1461,16 +1559,16 @@ fdioctl(dev, cmd, addr, flag)
fd->sc_dk.dk_cpulabel);
return error;
- case MTIOCTOP:
- if (((struct mtop *)addr)->mt_op != MTOFFL)
- return EIO;
-#ifdef COMPAT_SUNOS
- case SUNOS_FDIOCEJECT:
-#endif
- auxregbisc(AUXIO_FDS, AUXIO_FEJ);
- delay(10);
- auxregbisc(AUXIO_FEJ, AUXIO_FDS);
+ case DIOCLOCK:
+ /*
+ * Nothing to do here, really.
+ */
+ return 0;
+
+ case DIOCEJECT:
+ fd_do_eject();
return 0;
+
#ifdef DEBUG
case _IO('f', 100):
{
@@ -1492,7 +1590,7 @@ fdioctl(dev, cmd, addr, flag)
~CFG_THRHLD_MASK;
((struct fdc_softc *)fd->sc_dv.dv_parent)->sc_cfg |=
(*(int *)addr & CFG_THRHLD_MASK);
- fdconf(fd->sc_dv.dv_parent);
+ fdconf((struct fdc_softc *) fd->sc_dv.dv_parent);
return 0;
case _IO('f', 102):
{
@@ -1516,3 +1614,92 @@ fdioctl(dev, cmd, addr, flag)
panic("fdioctl: impossible");
#endif
}
+
+void
+fdgetdisklabel(dev)
+ dev_t dev;
+{
+ int unit = FDUNIT(dev), i;
+ struct fd_softc *fd = fd_cd.cd_devs[unit];
+ struct disklabel *lp = fd->sc_dk.dk_label;
+ struct cpu_disklabel *clp = fd->sc_dk.dk_cpulabel;
+
+ bzero(lp, sizeof(struct disklabel));
+ bzero(lp, sizeof(struct cpu_disklabel));
+
+ lp->d_type = DTYPE_FLOPPY;
+ lp->d_secsize = FDC_BSIZE;
+ lp->d_secpercyl = fd->sc_type->seccyl;
+ lp->d_nsectors = fd->sc_type->sectrac;
+ lp->d_ncylinders = fd->sc_type->tracks;
+ lp->d_ntracks = fd->sc_type->heads; /* Go figure... */
+ lp->d_rpm = 3600; /* XXX like it matters... */
+
+ strncpy(lp->d_typename, "floppy", sizeof(lp->d_typename));
+ strncpy(lp->d_packname, "fictitious", sizeof(lp->d_packname));
+ lp->d_interleave = 1;
+
+ lp->d_partitions[RAW_PART].p_offset = 0;
+ lp->d_partitions[RAW_PART].p_size = lp->d_secpercyl * lp->d_ncylinders;
+ lp->d_partitions[RAW_PART].p_fstype = FS_UNUSED;
+ lp->d_npartitions = RAW_PART + 1;
+
+ lp->d_magic = DISKMAGIC;
+ lp->d_magic2 = DISKMAGIC;
+ lp->d_checksum = dkcksum(lp);
+
+ /*
+ * Call the generic disklabel extraction routine. If there's
+ * not a label there, fake it.
+ */
+ if (readdisklabel(dev, fdstrategy, lp, clp) != NULL) {
+ strncpy(lp->d_packname, "default label",
+ sizeof(lp->d_packname));
+ /*
+ * Reset the partition info; it might have gotten
+ * trashed in readdisklabel().
+ *
+ * XXX Why do we have to do this? readdisklabel()
+ * should be safe...
+ */
+ for (i = 0; i < MAXPARTITIONS; ++i) {
+ lp->d_partitions[i].p_offset = 0;
+ if (i == RAW_PART) {
+ lp->d_partitions[i].p_size =
+ lp->d_secpercyl * lp->d_ncylinders;
+ lp->d_partitions[i].p_fstype = FS_BSDFFS;
+ } else {
+ lp->d_partitions[i].p_size = 0;
+ lp->d_partitions[i].p_fstype = FS_UNUSED;
+ }
+ }
+ lp->d_npartitions = RAW_PART + 1;
+ }
+}
+
+void
+fd_do_eject()
+{
+
+ auxregbisc(AUXIO_FDS, AUXIO_FEJ);
+ delay(10);
+ auxregbisc(AUXIO_FEJ, AUXIO_FDS);
+}
+
+/* ARGSUSED */
+void
+fd_mountroot_hook(dev)
+ struct device *dev;
+{
+ int c;
+
+ fd_do_eject();
+ printf("Insert filesystem floppy and press return.");
+ for (;;) {
+ c = cngetc();
+ if ((c == '\r') || (c == '\n')) {
+ printf("\n");
+ return;
+ }
+ }
+}
diff --git a/sys/arch/sparc/dev/fdreg.h b/sys/arch/sparc/dev/fdreg.h
index a522eb2d3c2..4166d58b0ba 100644
--- a/sys/arch/sparc/dev/fdreg.h
+++ b/sys/arch/sparc/dev/fdreg.h
@@ -1,4 +1,4 @@
-/* $NetBSD: fdreg.h,v 1.4 1995/06/28 04:31:19 cgd Exp $ */
+/* $NetBSD: fdreg.h,v 1.5 1996/02/01 22:32:27 mycroft Exp $ */
/*-
* Copyright (c) 1991 The Regents of the University of California.
@@ -42,7 +42,7 @@
/* uses NEC765 controller */
#include <dev/ic/nec765reg.h>
-#ifndef LOCORE
+#ifndef _LOCORE
struct fdreg_77 {
u_int8_t fd_statusA;
u_int8_t fd_statusB;
diff --git a/sys/arch/sparc/dev/fdvar.h b/sys/arch/sparc/dev/fdvar.h
index f74d37ce41d..9a74ebcf162 100644
--- a/sys/arch/sparc/dev/fdvar.h
+++ b/sys/arch/sparc/dev/fdvar.h
@@ -1,5 +1,5 @@
/*
- * $NetBSD: fdvar.h,v 1.3 1995/04/07 19:46:15 pk Exp $
+ * $NetBSD: fdvar.h,v 1.4 1996/02/01 22:32:29 mycroft Exp $
*
* Copyright (c) 1995 Paul Kranenburg
* All rights reserved.
@@ -36,7 +36,7 @@
#define FDC_NSTATUS 10
-#if !defined(LOCORE)
+#ifndef _LOCORE
struct fdcio {
/*
* 82072 (sun4c) and 82077 (sun4m) controllers have different
@@ -62,9 +62,9 @@ struct fdcio {
/*
* Statictics.
*/
- struct evcnt fdcio_intrcnt;
+ struct evcnt fdcio_intrcnt;
};
-#endif /* LOCORE */
+#endif /* _LOCORE */
/* istate values */
#define ISTATE_IDLE 0 /* No HW interrupt expected */
@@ -72,5 +72,5 @@ struct fdcio {
#define ISTATE_SENSEI 2 /* Do SENSEI on next HW interrupt */
#define ISTATE_DMA 3 /* Pseudo-DMA in progress */
-#define SUNOS_FDIOCEJECT _IO('f', 24)
+#define FDIOCEJECT _IO('f', 24)
diff --git a/sys/arch/sparc/dev/if_ie.c b/sys/arch/sparc/dev/if_ie.c
index 20aa5204d26..3bd3defb91a 100644
--- a/sys/arch/sparc/dev/if_ie.c
+++ b/sys/arch/sparc/dev/if_ie.c
@@ -1,4 +1,4 @@
-/* $NetBSD: if_ie.c,v 1.15 1995/04/11 09:18:09 pk Exp $ */
+/* $NetBSD: if_ie.c,v 1.24 1996/05/07 01:28:28 thorpej Exp $ */
/*-
* Copyright (c) 1993, 1994, 1995 Charles Hannum.
@@ -57,7 +57,7 @@
*
* Majorly cleaned up and 3C507 code merged by Charles Hannum.
*
- * Converted to SUN ie driver by Charles D. Cranor,
+ * Converted to SUN ie driver by Charles D. Cranor,
* October 1994, January 1995.
* This sun version based on i386 version 1.30.
*/
@@ -129,6 +129,11 @@ Mode of operation:
#include <netinet/if_ether.h>
#endif
+#ifdef NS
+#include <netns/ns.h>
+#include <netns/ns_if.h>
+#endif
+
#include <vm/vm.h>
/*
@@ -221,7 +226,6 @@ const char *ie_hardware_names[] = {
struct ie_softc {
struct device sc_dev; /* device structure */
struct intrhand sc_ih; /* interrupt info */
- struct evcnt sc_intrcnt; /* # of interrupts, per ie */
caddr_t sc_iobase; /* KVA of base of 24 bit addr space */
caddr_t sc_maddr; /* KVA of base of chip's RAM (16bit addr sp.)*/
@@ -230,15 +234,18 @@ struct ie_softc {
struct arpcom sc_arpcom;/* system arpcom structure */
- void (*reset_586)(); /* card dependent reset function */
- void (*chan_attn)(); /* card dependent attn function */
- void (*run_586)(); /* card depenent "go on-line" function */
+ void (*reset_586) __P((struct ie_softc *));
+ /* card dependent reset function */
+ void (*chan_attn) __P((struct ie_softc *));
+ /* card dependent attn function */
+ void (*run_586) __P((struct ie_softc *));
+ /* card depenent "go on-line" function */
void (*memcopy) __P((const void *, void *, u_int));
/* card dependent memory copy function */
void (*memzero) __P((void *, u_int));
/* card dependent memory zero function */
-
+
enum ie_hardware hard_type; /* card type */
int want_mcsetup; /* mcsetup flag */
@@ -253,10 +260,10 @@ struct ie_softc {
volatile struct ie_sys_ctl_block *scb;
/*
- * pointer and size of a block of KVA where the buffers
+ * pointer and size of a block of KVA where the buffers
* are to be allocated from
*/
-
+
caddr_t buf_area;
int buf_area_sz;
@@ -294,7 +301,7 @@ static void ie_vmereset __P((struct ie_softc *));
static void ie_vmeattend __P((struct ie_softc *));
static void ie_vmerun __P((struct ie_softc *));
-void iewatchdog __P((/* short */));
+void iewatchdog __P((struct ifnet *));
int ieintr __P((void *));
int ieinit __P((struct ie_softc *));
int ieioctl __P((struct ifnet *, u_long, caddr_t));
@@ -302,6 +309,7 @@ void iestart __P((struct ifnet *));
void iereset __P((struct ie_softc *));
static void ie_readframe __P((struct ie_softc *, int));
static void ie_drop_packet_buffer __P((struct ie_softc *));
+int ie_setupram __P((struct ie_softc *));
static int command_and_wait __P((struct ie_softc *, int,
void volatile *, int));
/*static*/ void ierint __P((struct ie_softc *));
@@ -311,6 +319,20 @@ static int ieget __P((struct ie_softc *, struct mbuf **,
static void setup_bufs __P((struct ie_softc *));
static int mc_setup __P((struct ie_softc *, void *));
static void mc_reset __P((struct ie_softc *));
+static __inline int ether_equal __P((u_char *, u_char *));
+static __inline void ie_ack __P((struct ie_softc *, u_int));
+static __inline void ie_setup_config __P((volatile struct ie_config_cmd *,
+ int, int));
+static __inline int check_eh __P((struct ie_softc *, struct ether_header *,
+ int *));
+static __inline int ie_buflen __P((struct ie_softc *, int));
+static __inline int ie_packet_len __P((struct ie_softc *));
+static __inline void iexmit __P((struct ie_softc *));
+static __inline caddr_t Align __P((caddr_t));
+
+static void chan_attn_timeout __P((void *));
+static void run_tdr __P((struct ie_softc *, struct ie_tdr_cmd *));
+static void iestop __P((struct ie_softc *));
#ifdef IEDEBUG
void print_rbd __P((volatile struct ie_recv_buf_desc *));
@@ -319,11 +341,15 @@ int in_ierint = 0;
int in_ietint = 0;
#endif
-int iematch();
-void ieattach();
+int iematch __P((struct device *, void *, void *));
+void ieattach __P((struct device *, struct device *, void *));
-struct cfdriver iecd = {
- NULL, "ie", iematch, ieattach, DV_IFNET, sizeof(struct ie_softc)
+struct cfattach ie_ca = {
+ sizeof(struct ie_softc), iematch, ieattach
+};
+
+struct cfdriver ie_cd = {
+ NULL, "ie", DV_IFNET
};
/*
@@ -344,7 +370,7 @@ struct cfdriver iecd = {
* Here are a few useful functions. We could have done these as macros, but
* since we have the inline facility, it makes sense to use that instead.
*/
-static inline void
+static __inline void
ie_setup_config(cmd, promiscuous, manchester)
volatile struct ie_config_cmd *cmd;
int promiscuous, manchester;
@@ -364,7 +390,7 @@ ie_setup_config(cmd, promiscuous, manchester)
cmd->ie_junk = 0xff;
}
-static inline void
+static __inline void
ie_ack(sc, mask)
struct ie_softc *sc;
u_int mask;
@@ -375,12 +401,13 @@ ie_ack(sc, mask)
}
-int
-iematch(parent, cf, aux)
+int
+iematch(parent, vcf, aux)
struct device *parent;
- struct cfdata *cf;
+ void *vcf;
void *aux;
{
+ struct cfdata *cf = vcf;
struct confargs *ca = aux;
struct romaux *ra = &ca->ca_ra;
@@ -389,7 +416,7 @@ iematch(parent, cf, aux)
if (ca->ca_bustype == BUS_SBUS)
return (0);
- if (cputyp == CPU_SUN4) {
+ if (CPU_ISSUN4) {
/*
* XXX need better probe here so we can figure out what we've got
*/
@@ -408,7 +435,7 @@ iematch(parent, cf, aux)
/*
* MULTIBUS/VME support
*/
-void
+void
ie_vmereset(sc)
struct ie_softc *sc;
{
@@ -418,7 +445,7 @@ ie_vmereset(sc)
iev->status = 0;
}
-void
+void
ie_vmeattend(sc)
struct ie_softc *sc;
{
@@ -428,7 +455,7 @@ ie_vmeattend(sc)
iev->status &= ~IEVME_ATTEN; /* down. */
}
-void
+void
ie_vmerun(sc)
struct ie_softc *sc;
{
@@ -473,7 +500,8 @@ ie_obrun(sc)
*/
void
ieattach(parent, self, aux)
- struct device *parent, *self;
+ struct device *parent;
+ struct device *self;
void *aux;
{
struct ie_softc *sc = (void *) self;
@@ -511,13 +539,13 @@ ieattach(parent, self, aux)
* XXX
*/
- ie_map = vm_map_create(pmap_kernel(), (vm_offset_t)IEOB_ADBASE,
+ ie_map = vm_map_create(pmap_kernel(), (vm_offset_t)IEOB_ADBASE,
(vm_offset_t)IEOB_ADBASE + sc->sc_msize, 1);
if (ie_map == NULL) panic("ie_map");
sc->sc_maddr = (caddr_t) kmem_alloc(ie_map, sc->sc_msize);
if (sc->sc_maddr == NULL) panic("ie kmem_alloc");
kvm_uncache(sc->sc_maddr, sc->sc_msize >> PGSHIFT);
- if (((u_long)sc->sc_maddr & ~(NBPG-1)) != (u_long)sc->sc_maddr)
+ if (((u_long)sc->sc_maddr & ~(NBPG-1)) != (u_long)sc->sc_maddr)
panic("unaligned dvmamalloc breaks");
sc->sc_iobase = (caddr_t)IEOB_ADBASE; /* 24 bit base addr */
(sc->memzero)(sc->sc_maddr, sc->sc_msize);
@@ -539,7 +567,7 @@ ieattach(parent, self, aux)
pa = pmap_extract(pmap_kernel(), (vm_offset_t)sc->sc_maddr);
if (pa == 0) panic("ie pmap_extract");
pmap_enter(pmap_kernel(), trunc_page(IEOB_ADBASE+IE_SCP_ADDR),
- (vm_offset_t)pa | PMAP_NC,
+ (vm_offset_t)pa | PMAP_NC /*| PMAP_IOC*/,
VM_PROT_READ | VM_PROT_WRITE, 1);
sc->scp = (volatile struct ie_sys_conf_ptr *)
@@ -574,8 +602,8 @@ ieattach(parent, self, aux)
/* 4 more */
rampaddr = rampaddr | ((iev->status & IEVME_HADDR) << 16);
rampaddr -= (u_long)ca->ca_ra.ra_paddr;
- sc->sc_maddr = mapiodev(ca->ca_ra.ra_reg, rampaddr, sc->sc_msize,
- ca->ca_bustype);
+ sc->sc_maddr = mapiodev(ca->ca_ra.ra_reg, rampaddr,
+ sc->sc_msize, ca->ca_bustype);
sc->sc_iobase = sc->sc_maddr;
iev->pectrl = iev->pectrl | IEVME_PARACK; /* clear to start */
@@ -623,8 +651,8 @@ ieattach(parent, self, aux)
/* XXX should reclaim resources? */
return;
}
- ifp->if_unit = sc->sc_dev.dv_unit;
- ifp->if_name = iecd.cd_name;
+ bcopy(sc->sc_dev.dv_xname, ifp->if_xname, IFNAMSIZ);
+ ifp->if_softc = sc;
ifp->if_start = iestart;
ifp->if_ioctl = ieioctl;
ifp->if_watchdog = iewatchdog;
@@ -651,6 +679,7 @@ ieattach(parent, self, aux)
intr_establish(pri, &sc->sc_ih);
break;
case BUS_VME16:
+ case BUS_VME32:
sc->sc_ih.ih_fun = ieintr;
sc->sc_ih.ih_arg = sc;
vmeintr_establish(ca->ca_ra.ra_intr[0].int_vec, pri,
@@ -659,12 +688,10 @@ ieattach(parent, self, aux)
#endif /* SUN4 */
}
- evcnt_attach(&sc->sc_dev, "intr", &sc->sc_intrcnt);
-
bp = ca->ca_ra.ra_bp;
if (bp != NULL && strcmp(bp->name, "ie") == 0 &&
sc->sc_dev.dv_unit == bp->val[1])
- bootdv = &sc->sc_dev;
+ bp->dev = &sc->sc_dev;
}
@@ -674,10 +701,10 @@ ieattach(parent, self, aux)
* an interrupt after a transmit has been started on it.
*/
void
-iewatchdog(unit)
- short unit;
+iewatchdog(ifp)
+ struct ifnet *ifp;
{
- struct ie_softc *sc = iecd.cd_devs[unit];
+ struct ie_softc *sc = ifp->if_softc;
log(LOG_ERR, "%s: device timeout\n", sc->sc_dev.dv_xname);
++sc->sc_arpcom.ac_if.if_oerrors;
@@ -701,12 +728,12 @@ void *v;
* check for parity error
*/
if (sc->hard_type == IE_VME) {
- volatile struct ievme *iev = (volatile struct ievme *)sc->sc_reg;
-
+ volatile struct ievme *iev = (volatile struct ievme *)sc->sc_reg
+;
if (iev->status & IEVME_PERR) {
printf("%s: parity error (ctrl %x @ %02x%04x)\n",
- iev->pectrl, iev->pectrl & IEVME_HADDR,
- iev->peaddr);
+ sc->sc_dev.dv_xname, iev->pectrl,
+ iev->pectrl & IEVME_HADDR, iev->peaddr);
iev->pectrl = iev->pectrl | IEVME_PARACK;
}
}
@@ -754,7 +781,6 @@ loop:
if ((status = sc->scb->ie_status) & IE_ST_WHENCE)
goto loop;
- sc->sc_intrcnt.ev_count++;
return 1;
}
@@ -777,9 +803,9 @@ ierint(sc)
sc->sc_arpcom.ac_if.if_ipackets++;
if (!--timesthru) {
sc->sc_arpcom.ac_if.if_ierrors +=
- SWAP(scb->ie_err_crc) +
+ SWAP(scb->ie_err_crc) +
SWAP(scb->ie_err_align) +
- SWAP(scb->ie_err_resource) +
+ SWAP(scb->ie_err_resource) +
SWAP(scb->ie_err_overrun);
scb->ie_err_crc = scb->ie_err_align =
scb->ie_err_resource = scb->ie_err_overrun =
@@ -792,7 +818,7 @@ ierint(sc)
(scb->ie_status & IE_RU_READY) == 0) {
sc->rframes[0]->ie_fd_buf_desc =
MK_16(sc->sc_maddr, sc->rbuffs[0]);
- scb->ie_recv_list =
+ scb->ie_recv_list =
MK_16(sc->sc_maddr, sc->rframes[0]);
command_and_wait(sc, IE_RU_START, 0, 0);
}
@@ -823,7 +849,7 @@ ietint(sc)
if (status & IE_STAT_OK) {
sc->sc_arpcom.ac_if.if_opackets++;
- sc->sc_arpcom.ac_if.if_collisions +=
+ sc->sc_arpcom.ac_if.if_collisions +=
SWAP(status & IE_XS_MAXCOLL);
} else if (status & IE_STAT_ABORT) {
printf("%s: send aborted\n", sc->sc_dev.dv_xname);
@@ -865,7 +891,7 @@ ietint(sc)
* Compare two Ether/802 addresses for equality, inlined and unrolled for
* speed. I'd love to have an inline assembler version of this...
*/
-static inline int
+static __inline int
ether_equal(one, two)
u_char *one, *two;
{
@@ -888,7 +914,7 @@ ether_equal(one, two)
* only client which will fiddle with IFF_PROMISC is BPF. This is
* probably a good assumption, but we do not make it here. (Yet.)
*/
-static inline int
+static __inline int
check_eh(sc, eh, to_bpf)
struct ie_softc *sc;
struct ether_header *eh;
@@ -991,7 +1017,7 @@ check_eh(sc, eh, to_bpf)
* IE_RBUF_SIZE is an even power of two. If somehow the act_len exceeds
* the size of the buffer, then we are screwed anyway.
*/
-static inline int
+static __inline int
ie_buflen(sc, head)
struct ie_softc *sc;
int head;
@@ -1001,7 +1027,7 @@ ie_buflen(sc, head)
& (IE_RBUF_SIZE | (IE_RBUF_SIZE - 1)));
}
-static inline int
+static __inline int
ie_packet_len(sc)
struct ie_softc *sc;
{
@@ -1034,7 +1060,7 @@ ie_packet_len(sc)
* command to the chip to be executed. On the way, if we have a BPF listener
* also give him a copy.
*/
-inline static void
+static __inline void
iexmit(sc)
struct ie_softc *sc;
{
@@ -1063,7 +1089,7 @@ iexmit(sc)
sc->xmit_cmds[sc->xctail]->ie_xmit_desc =
MK_16(sc->sc_maddr, sc->xmit_buffs[sc->xctail]);
- sc->scb->ie_command_list =
+ sc->scb->ie_command_list =
MK_16(sc->sc_maddr, sc->xmit_cmds[sc->xctail]);
command_and_wait(sc, IE_CU_START, 0, 0);
@@ -1079,7 +1105,7 @@ iexmit(sc)
* length of the data available. This enables us to allocate mbuf
* clusters in many situations where before we would have had a long
* chain of partially-full mbufs. This should help to speed up the
- * operation considerably. (Provided that it works, of course.)
+ * operation considerably. (Provided that it works, of course.)
*/
static inline int
ieget(sc, mp, ehp, to_bpf)
@@ -1385,7 +1411,7 @@ void
iestart(ifp)
struct ifnet *ifp;
{
- struct ie_softc *sc = iecd.cd_devs[ifp->if_unit];
+ struct ie_softc *sc = ifp->if_softc;
struct mbuf *m0, *m;
u_char *buffer;
u_short len;
@@ -1408,7 +1434,7 @@ iestart(ifp)
len = 0;
buffer = sc->xmit_cbuffs[sc->xchead];
- for (m0 = m; m && (len +m->m_len) < IE_TBUF_SIZE;
+ for (m0 = m; m && (len +m->m_len) < IE_TBUF_SIZE;
m = m->m_next) {
bcopy(mtod(m, caddr_t), buffer, m->m_len);
buffer += m->m_len;
@@ -1435,7 +1461,7 @@ iestart(ifp)
/*
* set up IE's ram space
*/
-int
+int
ie_setupram(sc)
struct ie_softc *sc;
{
@@ -1517,7 +1543,7 @@ iereset(sc)
*/
static void
chan_attn_timeout(rock)
- caddr_t rock;
+ void *rock;
{
*(int *)rock = 1;
@@ -1528,9 +1554,9 @@ chan_attn_timeout(rock)
* or be accepted, depending on the command. If the command pointer
* is null, then pretend that the command is not an action command.
* If the command pointer is not null, and the command is an action
- * command, wait for
- * ((volatile struct ie_cmd_common *)pcmd)->ie_cmd_status & MASK
- * to become true.
+ * command, wait for
+ * ((volatile struct ie_cmd_common *)pcmd)->ie_cmd_status & MASK
+ * to become true.
*/
static int
command_and_wait(sc, cmd, pcmd, mask)
@@ -1638,7 +1664,7 @@ run_tdr(sc, cmd)
#define ALLOC(p, n) _ALLOC(p, ALIGN(n)) /* XXX convert to this? */
#endif
-static inline caddr_t
+static __inline caddr_t
Align(ptr)
caddr_t ptr;
{
@@ -1661,13 +1687,11 @@ Align(ptr)
* note: this function was written to be easy to understand, rather than
* highly efficient (it isn't in the critical path).
*/
-static void
+static void
setup_bufs(sc)
struct ie_softc *sc;
{
caddr_t ptr = sc->buf_area; /* memory pool */
- volatile struct ie_recv_frame_desc *rfd = (void *) ptr;
- volatile struct ie_recv_buf_desc *rbd;
int n, r;
/*
@@ -1677,14 +1701,15 @@ setup_bufs(sc)
(sc->memzero)(ptr, sc->buf_area_sz);
ptr = Align(ptr); /* set alignment and stick with it */
- n = (int)Align(sizeof(struct ie_xmit_cmd)) +
- (int)Align(sizeof(struct ie_xmit_buf)) + IE_TBUF_SIZE;
+ n = (int)Align((caddr_t) sizeof(struct ie_xmit_cmd)) +
+ (int)Align((caddr_t) sizeof(struct ie_xmit_buf)) + IE_TBUF_SIZE;
n *= NTXBUF; /* n = total size of xmit area */
n = sc->buf_area_sz - n;/* n = free space for recv stuff */
- r = (int)Align(sizeof(struct ie_recv_frame_desc)) +
- (((int)Align(sizeof(struct ie_recv_buf_desc)) + IE_RBUF_SIZE) * B_PER_F);
+ r = (int)Align((caddr_t) sizeof(struct ie_recv_frame_desc)) +
+ (((int)Align((caddr_t) sizeof(struct ie_recv_buf_desc)) +
+ IE_RBUF_SIZE) * B_PER_F);
/* r = size of one R frame */
@@ -1804,7 +1829,7 @@ mc_setup(sc, ptr)
(sc->memcopy)((caddr_t)sc->mcast_addrs, (caddr_t)cmd->ie_mcast_addrs,
sc->mcast_count * sizeof *sc->mcast_addrs);
- cmd->ie_mcast_bytes =
+ cmd->ie_mcast_bytes =
SWAP(sc->mcast_count * ETHER_ADDR_LEN); /* grrr... */
sc->scb->ie_command_list = MK_16(sc->sc_maddr, cmd);
@@ -1831,7 +1856,6 @@ ieinit(sc)
{
volatile struct ie_sys_ctl_block *scb = sc->scb;
void *ptr;
- int n;
ptr = sc->buf_area;
@@ -1867,7 +1891,7 @@ ieinit(sc)
cmd->com.ie_cmd_cmd = IE_CMD_IASETUP | IE_CMD_LAST;
cmd->com.ie_cmd_link = SWAP(0xffff);
- (sc->memcopy)(sc->sc_arpcom.ac_enaddr,
+ (sc->memcopy)(sc->sc_arpcom.ac_enaddr,
(caddr_t)&cmd->ie_address, sizeof cmd->ie_address);
if (command_and_wait(sc, IE_CU_START, cmd, IE_STAT_COMPL) ||
@@ -1920,18 +1944,13 @@ ieioctl(ifp, cmd, data)
u_long cmd;
caddr_t data;
{
- struct ie_softc *sc = iecd.cd_devs[ifp->if_unit];
+ struct ie_softc *sc = ifp->if_softc;
struct ifaddr *ifa = (struct ifaddr *)data;
struct ifreq *ifr = (struct ifreq *)data;
int s, error = 0;
s = splnet();
- if ((error = ether_ioctl(ifp, &sc->sc_arpcom, cmd, data)) > 0) {
- splx(s);
- return error;
- }
-
switch(cmd) {
case SIOCSIFADDR:
@@ -1944,6 +1963,24 @@ ieioctl(ifp, cmd, data)
arp_ifinit(&sc->sc_arpcom, ifa);
break;
#endif
+#ifdef NS
+ /* XXX - This code is probably wrong. */
+ case AF_NS:
+ {
+ struct ns_addr *ina = &IA_SNS(ifa)->sns_addr;
+
+ if (ns_nullhost(*ina))
+ ina->x_host =
+ *(union ns_host *)(sc->sc_arpcom.ac_enaddr);
+ else
+ bcopy(ina->x_host.c_host,
+ sc->sc_arpcom.ac_enaddr,
+ sizeof(sc->sc_arpcom.ac_enaddr));
+ /* Set new address. */
+ ieinit(sc);
+ break;
+ }
+#endif /* NS */
default:
ieinit(sc);
break;
diff --git a/sys/arch/sparc/dev/if_le.c b/sys/arch/sparc/dev/if_le.c
index 055e2a31a5c..4db240f0f62 100644
--- a/sys/arch/sparc/dev/if_le.c
+++ b/sys/arch/sparc/dev/if_le.c
@@ -1,6 +1,8 @@
-/* $NetBSD: if_le.c,v 1.24 1995/12/11 12:43:28 pk Exp $ */
+/* $NetBSD: if_le.c,v 1.35.4.1 1996/07/17 01:46:00 jtc Exp $ */
/*-
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
* Copyright (c) 1995 Charles M. Hannum. All rights reserved.
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
@@ -18,6 +20,8 @@
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
+ * This product includes software developed by Aaron Brown and
+ * Harvard University.
* This product includes software developed by the University of
* California, Berkeley and its contributors.
* 4. Neither the name of the University nor the names of its contributors
@@ -62,46 +66,92 @@
#include <sparc/dev/sbusvar.h>
#include <sparc/dev/dmareg.h>
#include <sparc/dev/dmavar.h>
-#include <sparc/dev/if_lereg.h>
-#include <sparc/dev/if_levar.h>
+
#include <dev/ic/am7990reg.h>
-#define LE_NEED_BUF_CONTIG
#include <dev/ic/am7990var.h>
-#define LE_SOFTC(unit) lecd.cd_devs[unit]
-#define LE_DELAY(x) DELAY(x)
+#include <sparc/dev/if_lereg.h>
+#include <sparc/dev/if_levar.h>
int lematch __P((struct device *, void *, void *));
void leattach __P((struct device *, struct device *, void *));
-int leintr __P((void *));
-struct cfdriver lecd = {
- NULL, "le", lematch, leattach, DV_IFNET, sizeof(struct le_softc)
+#if defined(SUN4M) /* XXX */
+int myleintr __P((void *));
+int ledmaintr __P((struct dma_softc *));
+
+int
+myleintr(arg)
+ void *arg;
+{
+ register struct le_softc *lesc = arg;
+
+ if (lesc->sc_dma->sc_regs->csr & D_ERR_PEND)
+ return ledmaintr(lesc->sc_dma);
+
+ /*
+ * XXX There is a bug somewhere in the interrupt code that causes stray
+ * ethernet interrupts under high network load. This bug has been
+ * impossible to locate, so until it is found, we just ignore stray
+ * interrupts, as they do not in fact correspond to dropped packets.
+ */
+
+ /* return */ am7990_intr(arg);
+ return 1;
+}
+#endif
+
+struct cfattach le_ca = {
+ sizeof(struct le_softc), lematch, leattach
};
-integrate void
+hide void lewrcsr __P((struct am7990_softc *, u_int16_t, u_int16_t));
+hide u_int16_t lerdcsr __P((struct am7990_softc *, u_int16_t));
+hide void lehwinit __P((struct am7990_softc *));
+
+hide void
lewrcsr(sc, port, val)
- struct le_softc *sc;
+ struct am7990_softc *sc;
u_int16_t port, val;
{
- register struct lereg1 *ler1 = sc->sc_r1;
+ register struct lereg1 *ler1 = ((struct le_softc *)sc)->sc_r1;
ler1->ler1_rap = port;
ler1->ler1_rdp = val;
}
-integrate u_int16_t
+hide u_int16_t
lerdcsr(sc, port)
- struct le_softc *sc;
+ struct am7990_softc *sc;
u_int16_t port;
{
- register struct lereg1 *ler1 = sc->sc_r1;
+ register struct lereg1 *ler1 = ((struct le_softc *)sc)->sc_r1;
u_int16_t val;
ler1->ler1_rap = port;
val = ler1->ler1_rdp;
return (val);
-}
+}
+
+hide void
+lehwinit(sc)
+ struct am7990_softc *sc;
+{
+#if defined(SUN4M)
+ struct le_softc *lesc = (struct le_softc *)sc;
+
+ if (CPU_ISSUN4M && lesc->sc_dma) {
+ struct ifnet *ifp = &sc->sc_arpcom.ac_if;
+
+ if (ifp->if_flags & IFF_LINK0)
+ lesc->sc_dma->sc_regs->csr |= DE_AUI_TP;
+ else if (ifp->if_flags & IFF_LINK1)
+ lesc->sc_dma->sc_regs->csr &= ~DE_AUI_TP;
+
+ delay(20000); /* must not touch le for 20ms */
+ }
+#endif
+}
int
lematch(parent, match, aux)
@@ -125,15 +175,18 @@ leattach(parent, self, aux)
struct device *parent, *self;
void *aux;
{
- struct le_softc *sc = (void *)self;
+ struct le_softc *lesc = (struct le_softc *)self;
+ struct am7990_softc *sc = &lesc->sc_am7990;
struct confargs *ca = aux;
int pri;
struct bootpath *bp;
u_long laddr;
- int dmachild = strncmp(parent->dv_xname, "ledma", 5) == 0;
+#if defined(SUN4C) || defined(SUN4M)
+ int sbuschild = strncmp(parent->dv_xname, "sbus", 4) == 0;
+#endif
/* XXX the following declarations should be elsewhere */
- extern void myetheraddr(u_char *);
+ extern void myetheraddr __P((u_char *));
if (ca->ca_ra.ra_nintr != 1) {
printf(": expected 1 interrupt, got %d\n", ca->ca_ra.ra_nintr);
@@ -142,13 +195,13 @@ leattach(parent, self, aux)
pri = ca->ca_ra.ra_intr[0].int_pri;
printf(" pri %d", pri);
- sc->sc_r1 = (struct lereg1 *)mapiodev(ca->ca_ra.ra_reg, 0,
+ lesc->sc_r1 = (struct lereg1 *)mapiodev(ca->ca_ra.ra_reg, 0,
sizeof(struct lereg1),
ca->ca_bustype);
sc->sc_conf3 = LE_C3_BSWP | LE_C3_ACON | LE_C3_BCON;
laddr = (u_long)dvma_malloc(MEMSIZE, &sc->sc_mem, M_NOWAIT);
#if defined (SUN4M)
- if ((laddr & 0xffffff) >= (laddr & 0xffffff) + MEMSIZE)
+ if ((laddr & 0xffffff) >= (laddr & 0xffffff) + MEMSIZE)
panic("if_le: Lance buffer crosses 16MB boundary");
#endif
sc->sc_addr = laddr & 0xffffff;
@@ -156,14 +209,17 @@ leattach(parent, self, aux)
myetheraddr(sc->sc_arpcom.ac_enaddr);
- sc->sc_copytodesc = copytobuf_contig;
- sc->sc_copyfromdesc = copyfrombuf_contig;
- sc->sc_copytobuf = copytobuf_contig;
- sc->sc_copyfrombuf = copyfrombuf_contig;
- sc->sc_zerobuf = zerobuf_contig;
+ sc->sc_copytodesc = am7990_copytobuf_contig;
+ sc->sc_copyfromdesc = am7990_copyfrombuf_contig;
+ sc->sc_copytobuf = am7990_copytobuf_contig;
+ sc->sc_copyfrombuf = am7990_copyfrombuf_contig;
+ sc->sc_zerobuf = am7990_zerobuf_contig;
- sc->sc_arpcom.ac_if.if_name = lecd.cd_name;
- leconfig(sc);
+ sc->sc_rdcsr = lerdcsr;
+ sc->sc_wrcsr = lewrcsr;
+ sc->sc_hwinit = lehwinit;
+
+ am7990_config(sc);
bp = ca->ca_ra.ra_bp;
switch (ca->ca_bustype) {
@@ -173,44 +229,41 @@ leattach(parent, self, aux)
(bp->val[0] == -1 && bp->val[1] == sc->sc_dev.dv_unit))
case BUS_SBUS:
- sc->sc_sd.sd_reset = (void *)lereset;
- if (dmachild) {
-#ifdef notyet
- sc->sc_dma = (struct dma_softc *)parent;
- sc->sc_dma->sc_le = sc;
- sc->sc_dma->sc_regs->en_bar = laddr & 0xff000000;
- sbus_establish(&sc->sc_sd, parent);
-#endif
+ lesc->sc_sd.sd_reset = (void *)am7990_reset;
+ if (sbuschild) {
+ lesc->sc_dma = NULL;
+ sbus_establish(&lesc->sc_sd, &sc->sc_dev);
} else {
- sc->sc_dma = NULL;
- sbus_establish(&sc->sc_sd, &sc->sc_dev);
+ lesc->sc_dma = (struct dma_softc *)parent;
+ lesc->sc_dma->sc_le = lesc;
+ lesc->sc_dma->sc_regs->en_bar = laddr & 0xff000000;
+ /* Assume SBus is grandparent */
+ sbus_establish(&lesc->sc_sd, parent);
}
- if (bp != NULL && strcmp(bp->name, lecd.cd_name) == 0 &&
+ if (bp != NULL && strcmp(bp->name, le_cd.cd_name) == 0 &&
SAME_LANCE(bp, ca))
- bootdv = &sc->sc_dev;
+ bp->dev = &sc->sc_dev;
break;
#endif /* SUN4C || SUN4M */
default:
- if (bp != NULL && strcmp(bp->name, lecd.cd_name) == 0 &&
+ if (bp != NULL && strcmp(bp->name, le_cd.cd_name) == 0 &&
sc->sc_dev.dv_unit == bp->val[1])
- bootdv = &sc->sc_dev;
+ bp->dev = &sc->sc_dev;
break;
}
- sc->sc_ih.ih_fun = leintr;
-#if defined(SUN4M) && 0
- if (cputyp == CPU_SUN4M)
- sc->sc_ih.ih_fun = myleintr;
+ lesc->sc_ih.ih_fun = am7990_intr;
+#if defined(SUN4M) /*XXX*/
+ if (CPU_ISSUN4M)
+ lesc->sc_ih.ih_fun = myleintr;
#endif
- sc->sc_ih.ih_arg = sc;
- intr_establish(pri, &sc->sc_ih);
+ lesc->sc_ih.ih_arg = sc;
+ intr_establish(pri, &lesc->sc_ih);
/* now initialize DMA */
- if (sc->sc_dma) {
- dmaenintr(sc->sc_dma);
+ if (lesc->sc_dma) {
+ DMA_ENINTR(lesc->sc_dma);
}
}
-
-#include <dev/ic/am7990.c>
diff --git a/sys/arch/sparc/dev/if_levar.h b/sys/arch/sparc/dev/if_levar.h
index 7361c4eae7a..934d74c932e 100644
--- a/sys/arch/sparc/dev/if_levar.h
+++ b/sys/arch/sparc/dev/if_levar.h
@@ -1,4 +1,4 @@
-/* $NetBSD: if_levar.h,v 1.2 1995/12/11 12:43:29 pk Exp $ */
+/* $NetBSD: if_levar.h,v 1.5 1996/05/07 01:27:32 thorpej Exp $ */
/*-
* Copyright (c) 1995 Charles M. Hannum. All rights reserved.
@@ -47,45 +47,10 @@
* This structure contains the output queue for the interface, its address, ...
*/
struct le_softc {
- struct device sc_dev; /* base structure */
- struct arpcom sc_arpcom; /* Ethernet common part */
-
- void (*sc_copytodesc)(); /* Copy to descriptor */
- void (*sc_copyfromdesc)(); /* Copy from descriptor */
-
- void (*sc_copytobuf)(); /* Copy to buffer */
- void (*sc_copyfrombuf)(); /* Copy from buffer */
- void (*sc_zerobuf)(); /* and Zero bytes in buffer */
-
- u_int16_t sc_conf3; /* CSR3 value */
-
- void *sc_mem; /* base address of RAM -- CPU's view */
- u_long sc_addr; /* base address of RAM -- LANCE's view */
- u_long sc_memsize; /* size of RAM */
-
- int sc_nrbuf; /* number of receive buffers */
- int sc_ntbuf; /* number of transmit buffers */
- int sc_last_rd;
- int sc_first_td, sc_last_td, sc_no_td;
-
- int sc_initaddr;
- int sc_rmdaddr;
- int sc_tmdaddr;
- int sc_rbufaddr;
- int sc_tbufaddr;
-
-#ifdef LEDEBUG
- int sc_debug;
-#endif
+ struct am7990_softc sc_am7990; /* glue to MI code */
struct sbusdev sc_sd; /* sbus device */
struct intrhand sc_ih; /* interrupt vectoring */
struct lereg1 *sc_r1; /* LANCE registers */
struct dma_softc *sc_dma; /* pointer to my dma */
};
-
-/* DMA macros for ledma */
-#define DMA_ENINTR(r) ((r->enintr)(r))
-#define DMA_ISINTR(r) ((r->isintr)(r))
-#define DMA_RESET(r) ((r->reset)(r))
-
diff --git a/sys/arch/sparc/dev/kbd.c b/sys/arch/sparc/dev/kbd.c
index 194da8fdd6e..f65228f0848 100644
--- a/sys/arch/sparc/dev/kbd.c
+++ b/sys/arch/sparc/dev/kbd.c
@@ -1,4 +1,4 @@
-/* $NetBSD: kbd.c,v 1.19 1995/07/06 05:35:34 pk Exp $ */
+/* $NetBSD: kbd.c,v 1.23 1996/04/01 17:34:34 christos Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -51,7 +51,6 @@
*/
#include <sys/param.h>
-#include <sys/conf.h>
#include <sys/device.h>
#include <sys/ioctl.h>
#include <sys/kernel.h>
@@ -59,8 +58,11 @@
#include <sys/syslog.h>
#include <sys/systm.h>
#include <sys/tty.h>
+#include <sys/signalvar.h>
+#include <sys/conf.h>
#include <machine/autoconf.h>
+#include <machine/conf.h>
#include <machine/vuid_event.h>
#include <sparc/dev/event_var.h>
@@ -217,19 +219,10 @@ struct kbd_softc {
} kbd_softc;
/* Prototypes */
-void kbd_ascii(struct tty *);
-void kbd_serial(struct tty *, void (*)(), void (*)());
-static void kbd_getid(void *);
-void kbd_reset(struct kbd_state *);
-static int kbd_translate(int, struct kbd_state *);
-void kbd_rint(int);
-int kbdopen(dev_t, int, int, struct proc *);
-int kbdclose(dev_t, int, int, struct proc *);
-int kbdread(dev_t, struct uio *, int);
-int kbdwrite(dev_t, struct uio *, int);
-int kbdioctl(dev_t, u_long, caddr_t, int, struct proc *);
-int kbdselect(dev_t, int, struct proc *);
-int kbd_docmd(int, int);
+void kbd_reset __P((struct kbd_state *));
+static int kbd_translate __P((int, struct kbd_state *));
+void kbdattach __P((int));
+void kbd_repeat __P((void *arg));
/* set in kbdattach() */
int kbd_repeat_start;
@@ -240,7 +233,8 @@ int kbd_repeat_step;
* This happens before kbd_serial.
*/
void
-kbd_ascii(struct tty *tp)
+kbd_ascii(tp)
+ struct tty *tp;
{
kbd_softc.k_cons = tp;
@@ -251,7 +245,10 @@ kbd_ascii(struct tty *tp)
* We pick up the initial keyboard click state here as well.
*/
void
-kbd_serial(struct tty *tp, void (*iopen)(), void (*iclose)())
+kbd_serial(tp, iopen, iclose)
+ struct tty *tp;
+ void (*iopen) __P((struct tty *));
+ void (*iclose) __P((struct tty *));
{
register struct kbd_softc *k;
register char *cp;
@@ -261,13 +258,11 @@ kbd_serial(struct tty *tp, void (*iopen)(), void (*iclose)())
k->k_open = iopen;
k->k_close = iclose;
-#if defined(SUN4C) || defined(SUN4M)
- if (cputyp != CPU_SUN4) {
+ if (!CPU_ISSUN4) {
cp = getpropstring(optionsnode, "keyboard-click?");
if (cp && strcmp(cp, "true") == 0)
k->k_state.kbd_click = 1;
}
-#endif /* SUN4C || SUN4M */
}
/*
@@ -276,7 +271,8 @@ kbd_serial(struct tty *tp, void (*iopen)(), void (*iclose)())
* send a RESET, so that we can find out what kind of keyboard it is.
*/
void
-kbdattach(int nkbd)
+kbdattach(kbd)
+ int kbd;
{
register struct kbd_softc *k;
register struct tty *tp;
@@ -295,7 +291,8 @@ kbdattach(int nkbd)
}
void
-kbd_reset(register struct kbd_state *ks)
+kbd_reset(ks)
+ register struct kbd_state *ks;
{
/*
* On first identification, wake up anyone waiting for type
@@ -328,7 +325,6 @@ kbd_reset(register struct kbd_state *ks)
break;
default:
printf("Unknown keyboard type %d\n", ks->kbd_id);
- break;
}
ks->kbd_leds = 0;
@@ -338,7 +334,9 @@ kbd_reset(register struct kbd_state *ks)
* Turn keyboard up/down codes into ASCII.
*/
static int
-kbd_translate(register int c, register struct kbd_state *ks)
+kbd_translate(c, ks)
+ register int c;
+ register struct kbd_state *ks;
{
register int down;
@@ -401,7 +399,8 @@ kbd_translate(register int c, register struct kbd_state *ks)
void
-kbd_repeat(void *arg)
+kbd_repeat(arg)
+ void *arg;
{
struct kbd_softc *k = (struct kbd_softc *)arg;
int s = spltty();
@@ -414,7 +413,8 @@ kbd_repeat(void *arg)
}
void
-kbd_rint(register int c)
+kbd_rint(c)
+ register int c;
{
register struct kbd_softc *k = &kbd_softc;
register struct firm_event *fe;
@@ -512,7 +512,11 @@ kbd_rint(register int c)
}
int
-kbdopen(dev_t dev, int flags, int mode, struct proc *p)
+kbdopen(dev, flags, mode, p)
+ dev_t dev;
+ int flags;
+ int mode;
+ struct proc *p;
{
int s, error;
struct tty *tp;
@@ -548,7 +552,11 @@ kbdopen(dev_t dev, int flags, int mode, struct proc *p)
}
int
-kbdclose(dev_t dev, int flags, int mode, struct proc *p)
+kbdclose(dev, flags, mode, p)
+ dev_t dev;
+ int flags;
+ int mode;
+ struct proc *p;
{
/*
@@ -564,7 +572,10 @@ kbdclose(dev_t dev, int flags, int mode, struct proc *p)
}
int
-kbdread(dev_t dev, struct uio *uio, int flags)
+kbdread(dev, uio, flags)
+ dev_t dev;
+ struct uio *uio;
+ int flags;
{
return (ev_read(&kbd_softc.k_events, uio, flags));
@@ -572,14 +583,22 @@ kbdread(dev_t dev, struct uio *uio, int flags)
/* this routine should not exist, but is convenient to write here for now */
int
-kbdwrite(dev_t dev, struct uio *uio, int flags)
+kbdwrite(dev, uio, flags)
+ dev_t dev;
+ struct uio *uio;
+ int flags;
{
return (EOPNOTSUPP);
}
int
-kbdioctl(dev_t dev, u_long cmd, register caddr_t data, int flag, struct proc *p)
+kbdioctl(dev, cmd, data, flag, p)
+ dev_t dev;
+ u_long cmd;
+ register caddr_t data;
+ int flag;
+ struct proc *p;
{
register struct kbd_softc *k = &kbd_softc;
register struct kiockey *kmp;
@@ -602,10 +621,11 @@ kbdioctl(dev_t dev, u_long cmd, register caddr_t data, int flag, struct proc *p)
case KIOCGETKEY:
if (((struct okiockey *)data)->kio_station == 118) {
/*
- * This is X11 asking if a type 3 keyboard is
- * really a type 3 keyboard. Say yes.
+ * This is X11 asking (in an inappropriate fashion)
+ * if a type 3 keyboard is really a type 3 keyboard.
+ * Say yes (inappropriately).
*/
- ((struct okiockey *)data)->kio_entry = (u_char) HOLE;
+ ((struct okiockey *)data)->kio_entry = (u_char)HOLE;
return (0);
}
break;
@@ -680,7 +700,6 @@ kbdioctl(dev_t dev, u_long cmd, register caddr_t data, int flag, struct proc *p)
int s;
char leds = *(char *)data;
struct tty *tp = kbd_softc.k_kbd;
-
s = spltty();
if (tp->t_outq.c_cc > 120)
(void) tsleep((caddr_t)&lbolt, TTIPRI,
@@ -699,6 +718,7 @@ kbdioctl(dev_t dev, u_long cmd, register caddr_t data, int flag, struct proc *p)
*(char *)data = k->k_state.kbd_leds;
return (0);
+
case FIONBIO: /* we will remove this someday (soon???) */
return (0);
@@ -722,7 +742,10 @@ kbdioctl(dev_t dev, u_long cmd, register caddr_t data, int flag, struct proc *p)
}
int
-kbdselect(dev_t dev, int rw, struct proc *p)
+kbdselect(dev, rw, p)
+ dev_t dev;
+ int rw;
+ struct proc *p;
{
return (ev_select(&kbd_softc.k_events, rw, p));
@@ -734,7 +757,9 @@ kbdselect(dev_t dev, int rw, struct proc *p)
* is flooding. (The keyboard runs at 1200 baud, or 120 cps.)
*/
int
-kbd_docmd(int cmd, int isuser)
+kbd_docmd(cmd, isuser)
+ int cmd;
+ int isuser;
{
register struct tty *tp = kbd_softc.k_kbd;
register struct kbd_softc *k = &kbd_softc;
diff --git a/sys/arch/sparc/dev/ms.c b/sys/arch/sparc/dev/ms.c
index fb71f4ecb36..365d1c476b2 100644
--- a/sys/arch/sparc/dev/ms.c
+++ b/sys/arch/sparc/dev/ms.c
@@ -1,4 +1,4 @@
-/* $NetBSD: ms.c,v 1.5 1995/08/29 22:15:35 pk Exp $ */
+/* $NetBSD: ms.c,v 1.8 1996/04/01 17:29:52 christos Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -49,15 +49,20 @@
*/
#include <sys/param.h>
-#include <sys/conf.h>
#include <sys/ioctl.h>
#include <sys/kernel.h>
#include <sys/proc.h>
#include <sys/syslog.h>
#include <sys/systm.h>
#include <sys/tty.h>
+#include <sys/signalvar.h>
+#include <sys/conf.h>
#include <machine/vuid_event.h>
+#include <machine/cpu.h>
+#include <machine/kbd.h>
+#include <machine/conf.h>
+
#include <sparc/dev/event_var.h>
/*
@@ -92,7 +97,8 @@ struct ms_softc {
void
ms_serial(tp, iopen, iclose)
struct tty *tp;
- void (*iopen)(), (*iclose)();
+ void (*iopen) __P((struct tty *));
+ void (*iclose) __P((struct tty *));
{
ms_softc.ms_mouse = tp;
@@ -242,13 +248,18 @@ msopen(dev, flags, mode, p)
int flags, mode;
struct proc *p;
{
- int s, error;
if (ms_softc.ms_events.ev_io)
return (EBUSY);
ms_softc.ms_events.ev_io = p;
ev_init(&ms_softc.ms_events); /* may cause sleep */
+ if (CPU_ISSUN4) {
+ /* We need to set the baud rate on the mouse. */
+ ms_softc.ms_mouse->t_ispeed =
+ ms_softc.ms_mouse->t_ospeed = 1200;
+ }
+
(*ms_softc.ms_open)(ms_softc.ms_mouse);
ms_softc.ms_ready = 1; /* start accepting events */
return (0);
@@ -297,8 +308,6 @@ msioctl(dev, cmd, data, flag, p)
int flag;
struct proc *p;
{
- int s;
-
switch (cmd) {
case FIONBIO: /* we will remove this someday (soon???) */
diff --git a/sys/arch/sparc/dev/obio.c b/sys/arch/sparc/dev/obio.c
index f78bbcc2f37..37cf369fe48 100644
--- a/sys/arch/sparc/dev/obio.c
+++ b/sys/arch/sparc/dev/obio.c
@@ -1,7 +1,8 @@
-/* $NetBSD: obio.c,v 1.15 1995/05/27 08:12:51 pk Exp $ */
+/* $NetBSD: obio.c,v 1.24 1996/05/18 12:22:49 mrg Exp $ */
/*
* Copyright (c) 1993, 1994 Theo de Raadt
+ * Copyright (c) 1995 Paul Kranenburg
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
@@ -31,6 +32,7 @@
*/
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/device.h>
#include <sys/malloc.h>
@@ -48,10 +50,14 @@
#include <machine/ctlreg.h>
#include <sparc/sparc/asm.h>
#include <sparc/sparc/vaddrs.h>
+#include <sparc/dev/sbusvar.h>
struct bus_softc {
- struct device sc_dev; /* base device */
- int nothing;
+ union {
+ struct device scu_dev; /* base device */
+ struct sbus_softc scu_sbus; /* obio is another sbus slot */
+ } bu;
+#define sc_dev bu.scu_dev
};
/* autoconfiguration driver */
@@ -60,17 +66,40 @@ static void obioattach __P((struct device *, struct device *, void *));
static void vmesattach __P((struct device *, struct device *, void *));
static void vmelattach __P((struct device *, struct device *, void *));
-struct cfdriver obiocd = { NULL, "obio", busmatch, obioattach,
- DV_DULL, sizeof(struct bus_softc)
+int busprint __P((void *, char *));
+static int busattach __P((struct device *, void *, void *, int));
+void * bus_map __P((struct rom_reg *, int, int));
+int obio_scan __P((struct device *, void *, void *));
+int vmes_scan __P((struct device *, void *, void *));
+int vmel_scan __P((struct device *, void *, void *));
+int vmeintr __P((void *));
+
+struct cfattach obio_ca = {
+ sizeof(struct bus_softc), busmatch, obioattach
+};
+
+struct cfdriver obio_cd = {
+ NULL, "obio", DV_DULL
};
-struct cfdriver vmelcd = { NULL, "vmel", busmatch, vmelattach,
- DV_DULL, sizeof(struct bus_softc)
+
+struct cfattach vmel_ca = {
+ sizeof(struct bus_softc), busmatch, vmelattach
};
-struct cfdriver vmescd = { NULL, "vmes", busmatch, vmesattach,
- DV_DULL, sizeof(struct bus_softc)
+
+struct cfdriver vmel_cd = {
+ NULL, "vmel", DV_DULL
};
-static int busattach __P((struct device *, void *, void *, int));
+struct cfattach vmes_ca = {
+ sizeof(struct bus_softc), busmatch, vmesattach
+};
+
+struct cfdriver vmes_cd = {
+ NULL, "vmes", DV_DULL
+};
+
+struct intrhand **vmeints;
+
int
busmatch(parent, vcf, aux)
@@ -81,8 +110,12 @@ busmatch(parent, vcf, aux)
register struct confargs *ca = aux;
register struct romaux *ra = &ca->ca_ra;
- if (cputyp != CPU_SUN4)
+ if (CPU_ISSUN4M)
+ return (strcmp(cf->cf_driver->cd_name, ra->ra_name) == 0);
+
+ if (!CPU_ISSUN4)
return (0);
+
return (strcmp(cf->cf_driver->cd_name, ra->ra_name) == 0);
}
@@ -95,27 +128,180 @@ busprint(args, obio)
if (ca->ca_ra.ra_name == NULL)
ca->ca_ra.ra_name = "<unknown>";
+
if (obio)
printf("[%s at %s]", ca->ca_ra.ra_name, obio);
- printf(" addr 0x%x", ca->ca_ra.ra_paddr);
- if (ca->ca_ra.ra_intr[0].int_vec != -1)
+
+ printf(" addr %p", ca->ca_ra.ra_paddr);
+
+ if (CPU_ISSUN4 && ca->ca_ra.ra_intr[0].int_vec != -1)
printf(" vec 0x%x", ca->ca_ra.ra_intr[0].int_vec);
+
return (UNCONF);
}
+
+void
+obioattach(parent, self, args)
+ struct device *parent, *self;
+ void *args;
+{
+#if defined(SUN4M)
+ register struct bus_softc *sc = (struct bus_softc *)self;
+ struct confargs oca, *ca = args;
+ register struct romaux *ra = &ca->ca_ra;
+ register int node0, node;
+ register char *name;
+ register const char *sp;
+ const char *const *ssp;
+ extern int autoconf_nzs;
+
+ static const char *const special4m[] = {
+ /* find these first */
+ "eeprom",
+ "counter",
+ "auxio",
+ "",
+ /* place device to ignore here */
+ "interrupt",
+ NULL
+ };
+#endif
+
+ if (CPU_ISSUN4) {
+ if (self->dv_unit > 0) {
+ printf(" unsupported\n");
+ return;
+ }
+ printf("\n");
+
+ (void)config_search(obio_scan, self, args);
+ bus_untmp();
+ }
+
+#if defined(SUN4M)
+ if (!CPU_ISSUN4M)
+ return;
+
+ /*
+ * There is only one obio bus (it is in fact one of the Sbus slots)
+ * How about VME?
+ */
+ if (sc->sc_dev.dv_unit > 0) {
+ printf(" unsupported\n");
+ return;
+ }
+
+ printf("\n");
+
+ if (ra->ra_bp != NULL && strcmp(ra->ra_bp->name, "obio") == 0)
+ oca.ca_ra.ra_bp = ra->ra_bp + 1;
+ else
+ oca.ca_ra.ra_bp = NULL;
+
+ sc->bu.scu_sbus.sc_range = ra->ra_range;
+ sc->bu.scu_sbus.sc_nrange = ra->ra_nrange;
+
+ /*
+ * Loop through ROM children, fixing any relative addresses
+ * and then configuring each device.
+ * We first do the crucial ones, such as eeprom, etc.
+ */
+ node0 = firstchild(ra->ra_node);
+ for (ssp = special4m ; *(sp = *ssp) != 0; ssp++) {
+ if ((node = findnode(node0, sp)) == 0) {
+ printf("could not find %s amongst obio devices\n", sp);
+ panic(sp);
+ }
+ if (!romprop(&oca.ca_ra, sp, node))
+ continue;
+
+ sbus_translate(self, &oca);
+ oca.ca_bustype = BUS_OBIO;
+ (void) config_found(&sc->sc_dev, (void *)&oca, busprint);
+ }
+
+ for (node = node0; node; node = nextsibling(node)) {
+ name = getpropstring(node, "name");
+ for (ssp = special4m ; (sp = *ssp) != NULL; ssp++)
+ if (strcmp(name, sp) == 0)
+ break;
+
+ if (sp != NULL || !romprop(&oca.ca_ra, name, node))
+ continue;
+
+ if (strcmp(name, "zs") == 0)
+ /* XXX - see autoconf.c for this hack */
+ autoconf_nzs++;
+
+ /* Translate into parent address spaces */
+ sbus_translate(self, &oca);
+ oca.ca_bustype = BUS_OBIO;
+ (void) config_found(&sc->sc_dev, (void *)&oca, busprint);
+ }
+#endif
+}
+
+void
+vmesattach(parent, self, args)
+ struct device *parent, *self;
+ void *args;
+{
+ if (CPU_ISSUN4M || self->dv_unit > 0) {
+ printf(" unsupported\n");
+ return;
+ }
+ printf("\n");
+
+ if (vmeints == NULL) {
+ vmeints = (struct intrhand **)malloc(256 *
+ sizeof(struct intrhand *), M_TEMP, M_NOWAIT);
+ bzero(vmeints, 256 * sizeof(struct intrhand *));
+ }
+ (void)config_search(vmes_scan, self, args);
+ bus_untmp();
+}
+
+void
+vmelattach(parent, self, args)
+ struct device *parent, *self;
+ void *args;
+{
+ if (CPU_ISSUN4M || self->dv_unit > 0) {
+ printf(" unsupported\n");
+ return;
+ }
+ printf("\n");
+
+ if (vmeints == NULL) {
+ vmeints = (struct intrhand **)malloc(256 *
+ sizeof(struct intrhand *), M_TEMP, M_NOWAIT);
+ bzero(vmeints, 256 * sizeof(struct intrhand *));
+ }
+ (void)config_search(vmel_scan, self, args);
+ bus_untmp();
+}
+
int
busattach(parent, child, args, bustype)
struct device *parent;
- void *child, *args;
+ void *args, *child;
int bustype;
{
+#if defined(SUN4)
struct cfdata *cf = child;
- register struct bus_softc *sc = (struct bus_softc *)parent;
register struct confargs *ca = args;
struct confargs oca;
caddr_t tmp;
- if (bustype == BUS_OBIO && cputyp == CPU_SUN4) {
+ if (bustype == BUS_OBIO && CPU_ISSUN4) {
+
+ /*
+ * avoid sun4m entries which don't have valid PA's.
+ * no point in even probing them.
+ */
+ if (cf->cf_loc[0] == -1) return 0;
+
/*
* On the 4/100 obio addresses must be mapped at
* 0x0YYYYYYY, but alias higher up (we avoid the
@@ -123,64 +309,69 @@ busattach(parent, child, args, bustype)
* XXX: We also assume that 4/[23]00 obio addresses
* must be 0xZYYYYYYY, where (Z != 0)
*/
- if (cpumod==SUN4_100 && (cf->cf_loc[0] & 0xf0000000))
- return (0);
- if (cpumod!=SUN4_100 && !(cf->cf_loc[0] & 0xf0000000))
- return (0);
+ if (cpumod == SUN4_100 && (cf->cf_loc[0] & 0xf0000000))
+ return 0;
+ if (cpumod != SUN4_100 && !(cf->cf_loc[0] & 0xf0000000))
+ return 0;
}
if (parent->dv_cfdata->cf_driver->cd_indirect) {
printf(" indirect devices not supported\n");
- return (0);
+ return 0;
}
oca.ca_ra.ra_iospace = -1;
oca.ca_ra.ra_paddr = (void *)cf->cf_loc[0];
oca.ca_ra.ra_len = 0;
oca.ca_ra.ra_nreg = 1;
- tmp = NULL;
- if (oca.ca_ra.ra_paddr != (void *)-1)
- tmp = bus_tmp(oca.ca_ra.ra_paddr, bustype);
+ if (oca.ca_ra.ra_paddr)
+ tmp = (caddr_t)bus_tmp(oca.ca_ra.ra_paddr,
+ bustype);
+ else
+ tmp = NULL;
oca.ca_ra.ra_vaddr = tmp;
oca.ca_ra.ra_intr[0].int_pri = cf->cf_loc[1];
if (bustype == BUS_VME16 || bustype == BUS_VME32)
oca.ca_ra.ra_intr[0].int_vec = cf->cf_loc[2];
else
oca.ca_ra.ra_intr[0].int_vec = -1;
- oca.ca_ra.ra_nintr = 0;
- if (oca.ca_ra.ra_intr[0].int_pri != -1)
- oca.ca_ra.ra_nintr = 1;
+ oca.ca_ra.ra_nintr = 1;
oca.ca_ra.ra_name = cf->cf_driver->cd_name;
- oca.ca_ra.ra_bp = NULL;
if (ca->ca_ra.ra_bp != NULL &&
- ((bustype == BUS_VME16 && strcmp(ca->ca_ra.ra_bp->name, "vmes") == 0) ||
- (bustype == BUS_VME32 && strcmp(ca->ca_ra.ra_bp->name, "vmel") == 0) ||
- (bustype == BUS_OBIO && strcmp(ca->ca_ra.ra_bp->name, "obio") == 0)))
+ ((bustype == BUS_VME16 && strcmp(ca->ca_ra.ra_bp->name,"vmes") ==0) ||
+ (bustype == BUS_VME32 && strcmp(ca->ca_ra.ra_bp->name,"vmel") ==0) ||
+ (bustype == BUS_OBIO && strcmp(ca->ca_ra.ra_bp->name,"obio") == 0)))
oca.ca_ra.ra_bp = ca->ca_ra.ra_bp + 1;
+ else
+ oca.ca_ra.ra_bp = NULL;
oca.ca_bustype = bustype;
- if ((*cf->cf_driver->cd_match)(parent, cf, &oca) == 0)
- return (0);
+ if ((*cf->cf_attach->ca_match)(parent, cf, &oca) == 0)
+ return 0;
/*
- * check if XXmatch() replaced the temporary mapping with
- * a real mapping. If not, then make sure we don't pass the
- * tmp mapping to the attach routine.
+ * check if XXmatch routine replaced the temporary mapping with
+ * a real mapping. If not, then make sure we don't pass the
+ * tmp mapping to the attach routine.
*/
if (oca.ca_ra.ra_vaddr == tmp)
oca.ca_ra.ra_vaddr = NULL; /* wipe out tmp address */
/*
- * the match routine will set "ra_len" if it wants us to
+ * the match routine will set "ra_len" if it wants us to
* establish a mapping for it.
* (which won't be seen on future XXmatch calls,
* so not as useful as it seems.)
*/
if (oca.ca_ra.ra_len)
- oca.ca_ra.ra_vaddr = bus_map(oca.ca_ra.ra_paddr,
+ oca.ca_ra.ra_vaddr =
+ bus_map(oca.ca_ra.ra_reg,
oca.ca_ra.ra_len, oca.ca_bustype);
config_attach(parent, cf, &oca, busprint);
- return (1);
+ return 1;
+#else
+ return 0;
+#endif
}
int
@@ -191,23 +382,6 @@ obio_scan(parent, child, args)
return busattach(parent, child, args, BUS_OBIO);
}
-void
-obioattach(parent, self, args)
- struct device *parent, *self;
- void *args;
-{
- if (self->dv_unit > 0) {
- printf(" unsupported\n");
- return;
- }
- printf("\n");
-
- (void)config_search(obio_scan, self, args);
- bus_untmp();
-}
-
-struct intrhand **vmeints;
-
int
vmes_scan(parent, child, args)
struct device *parent;
@@ -216,26 +390,6 @@ vmes_scan(parent, child, args)
return busattach(parent, child, args, BUS_VME16);
}
-void
-vmesattach(parent, self, args)
- struct device *parent, *self;
- void *args;
-{
- if (self->dv_unit > 0) {
- printf(" unsupported\n");
- return;
- }
- printf("\n");
-
- if (vmeints == NULL) {
- vmeints = (struct intrhand **)malloc(256 *
- sizeof(struct intrhand *), M_TEMP, M_NOWAIT);
- bzero(vmeints, 256 * sizeof(struct intrhand *));
- }
- (void)config_search(vmes_scan, self, args);
- bus_untmp();
-}
-
int
vmel_scan(parent, child, args)
struct device *parent;
@@ -244,26 +398,6 @@ vmel_scan(parent, child, args)
return busattach(parent, child, args, BUS_VME32);
}
-void
-vmelattach(parent, self, args)
- struct device *parent, *self;
- void *args;
-{
- if (self->dv_unit > 0) {
- printf(" unsupported\n");
- return;
- }
- printf("\n");
-
- if (vmeints == NULL) {
- vmeints = (struct intrhand **)malloc(256 *
- sizeof(struct intrhand *), M_TEMP, M_NOWAIT);
- bzero(vmeints, 256 * sizeof(struct intrhand *));
- }
- (void)config_search(vmel_scan, self, args);
- bus_untmp();
-}
-
int pil_to_vme[] = {
-1, /* pil 0 */
-1, /* pil 1 */
@@ -291,10 +425,17 @@ vmeintr(arg)
struct intrhand *ih;
int i = 0;
- vec = ldcontrolb(AC_VMEINTVEC | (pil_to_vme[level] << 1) | 1);
+#ifdef DIAGNOSTIC
+ if (!CPU_ISSUN4) {
+ panic("vme: spurious interrupt");
+ }
+#endif
+
+ vec = ldcontrolb((caddr_t)
+ (AC_VMEINTVEC | (pil_to_vme[level] << 1) | 1));
if (vec == -1) {
printf("vme: spurious interrupt\n");
- return (0);
+ return 0;
}
for (ih = vmeints[vec]; ih; ih = ih->ih_next)
@@ -307,9 +448,14 @@ void
vmeintr_establish(vec, level, ih)
int vec, level;
struct intrhand *ih;
-{
+{
struct intrhand *ihs;
+ if (!CPU_ISSUN4) {
+ panic("vmeintr_establish: not supported on cpu-type %d",
+ cputyp);
+ }
+
if (vec == -1)
panic("vmeintr_establish: uninitialized vec\n");
@@ -344,14 +490,13 @@ vmeintr_establish(vec, level, ih)
*/
void *
bus_map(pa, len, bustype)
- void *pa;
+ struct rom_reg *pa;
int len;
int bustype;
{
- struct rom_reg rr;
- u_long pf = (u_long)pa >> PGSHIFT;
+ u_long pf = (u_long)(pa->rr_paddr) >> PGSHIFT;
u_long va, pte;
- int pgtype;
+ int pgtype = -1;
switch (bt2pmt[bustype]) {
case PMAP_OBIO:
@@ -370,12 +515,12 @@ bus_map(pa, len, bustype)
pte = getpte(va);
if ((pte & PG_V) != 0 && (pte & PG_TYPE) == pgtype &&
(pte & PG_PFNUM) == pf)
- return ((void *)(va | ((u_long)pa & PGOFSET)));
+ return ((void *)
+ (va | ((u_long)pa->rr_paddr & PGOFSET)) );
/* note: preserve page offset */
}
}
- rr.rr_paddr = pa;
- return mapiodev(&rr, 0, len, bustype);
+ return mapiodev(pa, 0, len, bustype);
}
void *
@@ -387,8 +532,8 @@ bus_tmp(pa, bustype)
int pmtype = bt2pmt[bustype];
pmap_enter(pmap_kernel(), TMPMAP_VA,
- addr | pmtype | PMAP_NC,
- VM_PROT_READ | VM_PROT_WRITE, 1);
+ addr | pmtype | PMAP_NC,
+ VM_PROT_READ | VM_PROT_WRITE, 1);
return ((void *)(TMPMAP_VA | ((u_long) pa & PGOFSET)) );
}
diff --git a/sys/arch/sparc/dev/pfourreg.h b/sys/arch/sparc/dev/pfourreg.h
index de3f656dc35..8ec422c2019 100644
--- a/sys/arch/sparc/dev/pfourreg.h
+++ b/sys/arch/sparc/dev/pfourreg.h
@@ -1,6 +1,7 @@
-/* $Id: pfourreg.h,v 1.2 1995/11/09 21:26:14 deraadt Exp $ */
+/* $NetBSD: pfourreg.h,v 1.1 1996/02/27 22:09:36 thorpej Exp $ */
/*
+ * Copyright (c) 1996 Jason R. Thorpe. All rights reserved.
* Copyright (c) 1995 Theo de Raadt
* All rights reserved.
*
@@ -31,11 +32,18 @@
*/
/*
- * pfour bus registers.
+ * pfour framebuffer registers.
*/
-/* offsets */
-#define PFOUR_REG 0x300000 /* offset from 0x[0f]b000000 */
+/* Offset of bwtwo framebuffer from pfour register */
+#define PFOUR_BW_OFF 0x00100000
+
+/* Offsets for color framebuffers */
+#define PFOUR_COLOR_OFF_OVERLAY 0x00100000
+#define PFOUR_COLOR_OFF_ENABLE 0x00300000
+#define PFOUR_COLOR_OFF_COLOR 0x00500000
+#define PFOUR_COLOR_OFF_END 0x00700000
+#define PFOUR_COLOR_OFF_CMAP 0xfff00000 /* (-0x00100000) */
#define PFOUR_REG_DIAG 0x80
#define PFOUR_REG_READBACKCLR 0x40
@@ -48,6 +56,7 @@
#define PFOUR_REG_FIRSTHALF 0x01
#define PFOUR_REG_RESET 0x01
+#define PFOUR_FBTYPE_MASK 0x7f000000
#define PFOUR_FBTYPE(x) ((x) >> 24)
#define PFOUR_ID_MASK 0xf0
@@ -68,6 +77,4 @@
#define PFOUR_SIZE_1440X1440 0x04
#define PFOUR_SIZE_640X480 0x05
-int pfour_videosize __P((int reg, int *xp, int *yp));
-void pfour_reset __P((void));
-
+#define PFOUR_NOTPFOUR -1
diff --git a/sys/arch/sparc/dev/power.c b/sys/arch/sparc/dev/power.c
new file mode 100644
index 00000000000..02308765505
--- /dev/null
+++ b/sys/arch/sparc/dev/power.c
@@ -0,0 +1,103 @@
+/* $NetBSD: power.c,v 1.2 1996/05/16 15:56:56 abrown Exp $ */
+
+/*
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
+ *
+ * All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by Aaron Brown and
+ * Harvard University.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by Harvard University
+ * and its contributors.
+ * 4. Neither the name of the University nor the names of its contributors
+ * may be used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ *
+ * $Id: power.c,v 1.1 1996/08/11 05:34:25 deraadt Exp $
+ */
+
+#include <sys/param.h>
+#include <sys/device.h>
+#include <sys/kernel.h>
+#include <sys/systm.h>
+
+#include <machine/autoconf.h>
+
+#include <sparc/dev/power.h>
+
+static int powermatch __P((struct device *, void *, void *));
+static void powerattach __P((struct device *, struct device *, void *));
+
+struct cfattach power_ca = {
+ sizeof(struct device), powermatch, powerattach
+};
+
+struct cfdriver power_cd = {
+ NULL, "power", DV_DULL
+};
+
+/*
+ * This is the driver for the "power" register available on some Sun4m
+ * machines. This allows the machine to remove power automatically when
+ * shutdown or halted or whatever.
+ *
+ * XXX: this capability is not utilized in the current kernel.
+ */
+
+static int
+powermatch(parent, vcf, aux)
+ struct device *parent;
+ void *aux, *vcf;
+{
+ register struct confargs *ca = aux;
+
+ if (CPU_ISSUN4M)
+ return (strcmp("power", ca->ca_ra.ra_name) == 0);
+
+ return (0);
+}
+
+/* ARGSUSED */
+static void
+powerattach(parent, self, aux)
+ struct device *parent, *self;
+ void *aux;
+{
+ struct confargs *ca = aux;
+ struct romaux *ra = &ca->ca_ra;
+
+ power_reg = mapdev(ra->ra_reg, 0, 0, sizeof(long), ca->ca_bustype);
+
+ printf("\n");
+}
+
+void
+powerdown()
+{
+ *POWER_REG |= POWER_OFF;
+}
diff --git a/sys/arch/sparc/dev/power.h b/sys/arch/sparc/dev/power.h
new file mode 100644
index 00000000000..252195b210c
--- /dev/null
+++ b/sys/arch/sparc/dev/power.h
@@ -0,0 +1,60 @@
+/* $NetBSD: power.h,v 1.2 1996/05/16 15:56:57 abrown Exp $ */
+
+/*
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
+ *
+ * All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by Aaron Brown and
+ * Harvard University.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by Harvard University
+ * and its contributors.
+ * 4. Neither the name of the University nor the names of its contributors
+ * may be used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ *
+ * $Id: power.h,v 1.1 1996/08/11 05:34:25 deraadt Exp $
+ *
+ */
+
+/*
+ * Sun-4m power register. This register allows the computer to power itself
+ * down.
+ */
+
+#define POWER_OFF 0x1 /* remove power */
+
+#define POWER_REG ((volatile u_char *)(power_reg))
+
+#define POWER_BITS "\20\1POWEROFF"
+
+#ifndef _LOCORE
+volatile u_char *power_reg;
+#endif
+
+void powerdown __P((void)); /* power off function */
diff --git a/sys/arch/sparc/dev/rcons_font.h b/sys/arch/sparc/dev/rcons_font.h
index 0583e458e03..3cbc847ea53 100644
--- a/sys/arch/sparc/dev/rcons_font.h
+++ b/sys/arch/sparc/dev/rcons_font.h
@@ -1,4 +1,4 @@
-/* $NetBSD: rcons_font.h,v 1.2 1995/10/08 19:31:44 pk Exp $ */
+/* $NetBSD: rcons_font.h,v 1.3 1995/11/29 22:03:53 pk Exp $ */
/*
* Raster Console font definition; this file exports `console_font',
diff --git a/sys/arch/sparc/dev/sbus.c b/sys/arch/sparc/dev/sbus.c
index 77ba2349467..32e52e31533 100644
--- a/sys/arch/sparc/dev/sbus.c
+++ b/sys/arch/sparc/dev/sbus.c
@@ -1,4 +1,4 @@
-/* $NetBSD: sbus.c,v 1.6 1995/02/01 12:37:28 pk Exp $ */
+/* $NetBSD: sbus.c,v 1.10 1996/04/22 02:35:03 abrown Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -49,6 +49,7 @@
*/
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/device.h>
#include <machine/autoconf.h>
@@ -56,12 +57,19 @@
#include <sparc/dev/sbusreg.h>
#include <sparc/dev/sbusvar.h>
+int sbus_print __P((void *, char *));
+void sbusreset __P((int));
+
/* autoconfiguration driver */
void sbus_attach __P((struct device *, struct device *, void *));
int sbus_match __P((struct device *, void *, void *));
-struct cfdriver sbuscd = {
- NULL, "sbus", sbus_match, sbus_attach,
- DV_DULL, sizeof(struct sbus_softc)
+
+struct cfattach sbus_ca = {
+ sizeof(struct sbus_softc), sbus_match, sbus_attach
+};
+
+struct cfdriver sbus_cd = {
+ NULL, "sbus", DV_DULL
};
/*
@@ -92,8 +100,9 @@ sbus_match(parent, vcf, aux)
register struct confargs *ca = aux;
register struct romaux *ra = &ca->ca_ra;
- if (cputyp==CPU_SUN4)
+ if (CPU_ISSUN4)
return (0);
+
return (strcmp(cf->cf_driver->cd_name, ra->ra_name) == 0);
}
@@ -109,8 +118,7 @@ sbus_attach(parent, self, aux)
register struct sbus_softc *sc = (struct sbus_softc *)self;
struct confargs *ca = aux;
register struct romaux *ra = &ca->ca_ra;
- register int base, node, slot;
- register int i;
+ register int node;
register char *name;
struct confargs oca;
@@ -131,11 +139,19 @@ sbus_attach(parent, self, aux)
sc->sc_clockfreq = getpropint(node, "clock-frequency", 25*1000*1000);
printf(": clock = %s MHz\n", clockfreq(sc->sc_clockfreq));
+ /*
+ * Get the SBus burst transfer size if burst transfers are supported
+ */
+ sc->sc_burst = getpropint(node, "burst-sizes", 0);
+
if (ra->ra_bp != NULL && strcmp(ra->ra_bp->name, "sbus") == 0)
oca.ca_ra.ra_bp = ra->ra_bp + 1;
else
oca.ca_ra.ra_bp = NULL;
+ sc->sc_range = ra->ra_range;
+ sc->sc_nrange = ra->ra_nrange;
+
/*
* Loop through ROM children, fixing any relative addresses
* and then configuring each device.
@@ -144,23 +160,60 @@ sbus_attach(parent, self, aux)
name = getpropstring(node, "name");
if (!romprop(&oca.ca_ra, name, node))
continue;
- base = (int)oca.ca_ra.ra_paddr;
+
+ sbus_translate(self, &oca);
+ oca.ca_bustype = BUS_SBUS;
+ (void) config_found(&sc->sc_dev, (void *)&oca, sbus_print);
+ }
+}
+
+void
+sbus_translate(dev, ca)
+ struct device *dev;
+ struct confargs *ca;
+{
+ struct sbus_softc *sc = (struct sbus_softc *)dev;
+ register int base, slot;
+ register int i;
+
+ if (sc->sc_nrange == 0) {
+ /* Old-style Sbus configuration */
+ base = (int)ca->ca_ra.ra_paddr;
if (SBUS_ABS(base)) {
- oca.ca_slot = SBUS_ABS_TO_SLOT(base);
- oca.ca_offset = SBUS_ABS_TO_OFFSET(base);
+ ca->ca_slot = SBUS_ABS_TO_SLOT(base);
+ ca->ca_offset = SBUS_ABS_TO_OFFSET(base);
} else {
- oca.ca_slot = slot = oca.ca_ra.ra_iospace;
- oca.ca_offset = base;
- oca.ca_ra.ra_paddr = (void *)SBUS_ADDR(slot, base);
+ ca->ca_slot = slot = ca->ca_ra.ra_iospace;
+ ca->ca_offset = base;
+ ca->ca_ra.ra_paddr =
+ (void *)SBUS_ADDR(slot, base);
/* Fix any remaining register banks */
- for (i = 1; i < oca.ca_ra.ra_nreg; i++) {
- base = (int)oca.ca_ra.ra_reg[i].rr_paddr;
- oca.ca_ra.ra_reg[i].rr_paddr =
+ for (i = 1; i < ca->ca_ra.ra_nreg; i++) {
+ base = (int)ca->ca_ra.ra_reg[i].rr_paddr;
+ ca->ca_ra.ra_reg[i].rr_paddr =
(void *)SBUS_ADDR(slot, base);
}
}
- oca.ca_bustype = BUS_SBUS;
- (void) config_found(&sc->sc_dev, (void *)&oca, sbus_print);
+
+ } else {
+
+ ca->ca_slot = ca->ca_ra.ra_iospace;
+ ca->ca_offset = (int)ca->ca_ra.ra_paddr;
+
+ /* Translate into parent address spaces */
+ for (i = 0; i < ca->ca_ra.ra_nreg; i++) {
+ int j, cspace = ca->ca_ra.ra_reg[i].rr_iospace;
+
+ for (j = 0; j < sc->sc_nrange; j++) {
+ if (sc->sc_range[j].cspace == cspace) {
+ (int)ca->ca_ra.ra_reg[i].rr_paddr +=
+ sc->sc_range[j].poffset;
+ (int)ca->ca_ra.ra_reg[i].rr_iospace =
+ sc->sc_range[j].pspace;
+ break;
+ }
+ }
+ }
}
}
@@ -173,7 +226,25 @@ sbus_establish(sd, dev)
register struct sbusdev *sd;
register struct device *dev;
{
- register struct sbus_softc *sc = (struct sbus_softc *)dev->dv_parent;
+ register struct sbus_softc *sc;
+ register struct device *curdev;
+
+ /*
+ * We have to look for the sbus by name, since it is not necessarily
+ * our immediate parent (i.e. sun4m /iommu/sbus/espdma/esp)
+ * We don't just use the device structure of the above-attached
+ * sbus, since we might (in the future) support multiple sbus's.
+ */
+ for (curdev = dev->dv_parent; ; curdev = curdev->dv_parent) {
+ if (!curdev || !curdev->dv_xname)
+ panic("sbus_establish: can't find sbus parent for %s",
+ (sd->sd_dev->dv_xname ? sd->sd_dev->dv_xname :
+ "<unknown>"));
+
+ if (strncmp(curdev->dv_xname, "sbus", 4) == 0)
+ break;
+ }
+ sc = (struct sbus_softc *) curdev;
sd->sd_dev = dev;
sd->sd_bchain = sc->sc_sbdev;
@@ -188,7 +259,7 @@ sbusreset(sbus)
int sbus;
{
register struct sbusdev *sd;
- struct sbus_softc *sc = sbuscd.cd_devs[sbus];
+ struct sbus_softc *sc = sbus_cd.cd_devs[sbus];
struct device *dev;
printf("reset %s:", sc->sc_dev.dv_xname);
@@ -200,24 +271,3 @@ sbusreset(sbus)
}
}
}
-
-/*
- * Returns true if this device is in a slave slot, so that drivers
- * can bail rather than fail.
- */
-int
-sbus_slavecheck(self, ca)
- struct device *self;
- struct confargs *ca;
-{
- int slave_only;
-
- slave_only = getpropint(findroot(), "slave-only", 8);
-
- if (slave_only & (1<<ca->ca_slot)) {
- printf("%s: slave sbus slot -- not supported\n",
- self->dv_xname);
- return (1);
- }
- return (0);
-}
diff --git a/sys/arch/sparc/dev/sbusvar.h b/sys/arch/sparc/dev/sbusvar.h
index 72badef900a..72fa1025523 100644
--- a/sys/arch/sparc/dev/sbusvar.h
+++ b/sys/arch/sparc/dev/sbusvar.h
@@ -1,4 +1,4 @@
-/* $NetBSD: sbusvar.h,v 1.2 1994/11/20 20:52:27 deraadt Exp $ */
+/* $NetBSD: sbusvar.h,v 1.4 1996/04/22 02:35:05 abrown Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -67,8 +67,11 @@ struct sbus_softc {
struct device sc_dev; /* base device */
int sc_clockfreq; /* clock frequency (in Hz) */
struct sbusdev *sc_sbdev; /* list of all children */
+ struct rom_range *sc_range;
+ int sc_nrange;
+ int sc_burst; /* burst transfer sizes supported */
};
int sbusdev_match __P((struct cfdata *, void *));
void sbus_establish __P((struct sbusdev *, struct device *));
-int sbus_slavecheck __P((struct device *self, struct confargs *ca));
+void sbus_translate __P((struct device *, struct confargs *));
diff --git a/sys/arch/sparc/dev/si.c b/sys/arch/sparc/dev/si.c
index ff0f4e400da..7908739ae4d 100644
--- a/sys/arch/sparc/dev/si.c
+++ b/sys/arch/sparc/dev/si.c
@@ -1,4 +1,4 @@
-/* $NetBSD: si.c,v 1.8 1996/01/01 22:40:56 thorpej Exp $ */
+/* $NetBSD: si.c,v 1.24 1996/05/13 01:53:45 thorpej Exp $ */
/*
* Copyright (c) 1995 Jason R. Thorpe
@@ -67,7 +67,8 @@
* for lots of helpful tips and suggestions. Thanks also to Paul Kranenburg
* and Chris Torek for bits of insight needed along the way. Thanks to
* David Gilbert and Andrew Gillham who risked filesystem life-and-limb
- * for the sake of testing.
+ * for the sake of testing. Andrew Gillham helped work out the bugs
+ * the the 4/100 DMA code.
*/
/*
@@ -116,6 +117,8 @@
#define DEBUG XXX
#endif
+#define COUNT_SW_LEFTOVERS XXX /* See sw DMA completion code */
+
#include <dev/ic/ncr5380reg.h>
#include <dev/ic/ncr5380var.h>
@@ -135,10 +138,6 @@
*/
#define MAX_DMA_LEN 0xE000
-#ifndef DEBUG
-#define DEBUG XXX
-#endif
-
#ifdef DEBUG
int si_debug = 0;
static int si_link_flags = 0 /* | SDEV_DB2 */ ;
@@ -180,6 +179,11 @@ struct si_softc {
* Alternatively, you can patch your kernel with DDB or some other
* mechanism. The sc_options member of the softc is OR'd with
* the value in si_options.
+ *
+ * On the "sw", interrupts (and thus) reselection don't work, so they're
+ * disabled by default. DMA is still a little dangerous, too.
+ *
+ * Note, there's a separate sw_options to make life easier.
*/
#define SI_ENABLE_DMA 0x01 /* Use DMA (maybe polled) */
#define SI_DMA_INTR 0x02 /* DMA completion interrupts */
@@ -187,6 +191,7 @@ struct si_softc {
#define SI_OPTIONS_MASK (SI_ENABLE_DMA|SI_DMA_INTR|SI_DO_RESELECT)
#define SI_OPTIONS_BITS "\10\3RESELECT\2DMA_INTR\1DMA"
int si_options = SI_ENABLE_DMA;
+int sw_options = SI_ENABLE_DMA;
/* How long to wait for DMA before declaring an error. */
int si_dma_intr_timo = 500; /* ticks (sec. X 100) */
@@ -196,6 +201,7 @@ static void si_attach __P((struct device *, struct device *, void *));
static int si_intr __P((void *));
static void si_reset_adapter __P((struct ncr5380_softc *));
static void si_minphys __P((struct buf *));
+static int si_print __P((void *, char *));
void si_dma_alloc __P((struct ncr5380_softc *));
void si_dma_free __P((struct ncr5380_softc *));
@@ -235,15 +241,21 @@ static struct scsi_device si_dev = {
/* The Sun SCSI-3 VME controller. */
-struct cfdriver sicd = {
- NULL, "si", si_match, si_attach,
- DV_DULL, sizeof(struct si_softc), NULL, 0,
+struct cfattach si_ca = {
+ sizeof(struct si_softc), si_match, si_attach
+};
+
+struct cfdriver si_cd = {
+ NULL, "si", DV_DULL
};
/* The Sun "SCSI Weird" 4/100 obio controller. */
-struct cfdriver swcd = {
- NULL, "sw", si_match, si_attach,
- DV_DULL, sizeof(struct si_softc), NULL, 0,
+struct cfattach sw_ca = {
+ sizeof(struct si_softc), si_match, si_attach
+};
+
+struct cfdriver sw_cd = {
+ NULL, "sw", DV_DULL
};
static int
@@ -263,14 +275,14 @@ si_match(parent, vcf, args)
{
struct cfdata *cf = vcf;
struct confargs *ca = args;
- struct romaux *ra = &ca->ca_ra;
+ struct romaux *ra = &ca->ca_ra;
/* Are we looking for the right thing? */
if (strcmp(cf->cf_driver->cd_name, ra->ra_name))
return (0);
/* Nothing but a Sun 4 is going to have these devices. */
- if (cputyp != CPU_SUN4)
+ if (!CPU_ISSUN4)
return (0);
/*
@@ -283,13 +295,20 @@ si_match(parent, vcf, args)
/* Figure out the bus type and look for the appropriate adapter. */
switch (ca->ca_bustype) {
case BUS_VME16:
+ /* AFAIK, the `si' can only exist on the vmes. */
+ if (strcmp(ra->ra_name, "si") || cpumod == SUN4_100)
+ return (0);
break;
case BUS_OBIO:
/* AFAIK, an `sw' can only exist on the obio. */
- if (cpumod != SUN4_100)
+ if (strcmp(ra->ra_name, "sw") || cpumod != SUN4_100)
return (0);
break;
+
+ default:
+ /* Don't know what we ended up with ... */
+ return (0);
}
/* Make sure there is something there... */
@@ -323,17 +342,12 @@ si_attach(parent, self, args)
int i;
/* Pull in the options flags. */
- if (ca->ca_bustype == BUS_OBIO) {
- /*
- * XXX Interrupts and reselect don't work on the "sw".
- * I don't know why (yet). Disable DMA by default, too.
- * It's still a little dangerous.
- */
- sc->sc_options = ncr_sc->sc_dev.dv_cfdata->cf_flags &
- SI_OPTIONS_MASK;
- } else
- sc->sc_options =
- ((ncr_sc->sc_dev.dv_cfdata->cf_flags | si_options) & SI_OPTIONS_MASK);
+ if (ca->ca_bustype == BUS_OBIO)
+ sc->sc_options = sw_options;
+ else
+ sc->sc_options = si_options;
+ sc->sc_options |=
+ (ncr_sc->sc_dev.dv_cfdata->cf_flags & SI_OPTIONS_MASK);
/* Map the controller registers. */
regs = (struct si_regs *)mapiodev(ra->ra_reg, 0,
@@ -392,6 +406,10 @@ si_attach(parent, self, args)
ncr_sc->sc_intr_on = si_obio_intr_on;
ncr_sc->sc_intr_off = si_obio_intr_off;
break;
+
+ default:
+ panic("\nsi_attach: impossible bus type 0x%x", ca->ca_bustype);
+ /* NOTREACHED */
}
ncr_sc->sc_flags = 0;
@@ -442,15 +460,19 @@ si_attach(parent, self, args)
sc->sc_adapter_iv_am =
VME_SUPV_DATA_24 | (ra->ra_intr[0].int_vec & 0xFF);
break;
+
+ default:
+ /* Impossible case handled above. */
+ break;
}
printf(" pri %d\n", ra->ra_intr[0].int_pri);
- if (sc->sc_options)
+ if (sc->sc_options) {
printf("%s: options=%b\n", ncr_sc->sc_dev.dv_xname,
- sc->sc_options, SI_OPTIONS_BITS);
-
+ sc->sc_options, SI_OPTIONS_BITS);
+ }
#ifdef DEBUG
if (si_debug)
- printf("si: Set TheSoftC=%x TheRegs=%x\n", sc, regs);
+ printf("si: Set TheSoftC=%p TheRegs=%p\n", sc, regs);
ncr_sc->sc_link.flags |= si_link_flags;
#endif
@@ -679,7 +701,7 @@ found:
dh->dh_dvma = (long)kdvma_mapin((caddr_t)addr, xlen, 0);
if (dh->dh_dvma == 0) {
/* Can't remap segment */
- printf("si_dma_alloc: can't remap %x/%x, doing PIO\n",
+ printf("si_dma_alloc: can't remap %p/%x, doing PIO\n",
dh->dh_addr, dh->dh_maplen);
dh->dh_flags = 0;
return;
@@ -733,7 +755,6 @@ si_dma_poll(ncr_sc)
{
struct si_softc *sc = (struct si_softc *)ncr_sc;
struct sci_req *sr = ncr_sc->sc_current;
- struct si_dma_handle *dh = sr->sr_dma_hand;
volatile struct si_regs *si = sc->sc_regs;
int tmo, csr_mask, csr;
@@ -864,14 +885,14 @@ si_vme_dma_start(ncr_sc)
*/
data_pa = (u_long)(dh->dh_dvma - DVMA_BASE);
if (data_pa & 1)
- panic("si_dma_start: bad pa=0x%x", data_pa);
+ panic("si_dma_start: bad pa=0x%lx", data_pa);
xlen = ncr_sc->sc_datalen;
xlen &= ~1;
sc->sc_xlen = xlen; /* XXX: or less... */
#ifdef DEBUG
if (si_debug & 2) {
- printf("si_dma_start: dh=0x%x, pa=0x%x, xlen=%d\n",
+ printf("si_dma_start: dh=%p, pa=0x%lx, xlen=%d\n",
dh, data_pa, xlen);
}
#endif
@@ -895,7 +916,7 @@ si_vme_dma_start(ncr_sc)
} else {
si->si_csr &= ~SI_CSR_BPCON;
}
-
+
si->dma_addrh = (u_short)(data_pa >> 16);
si->dma_addrl = (u_short)(data_pa & 0xFFFF);
@@ -997,12 +1018,15 @@ si_vme_dma_stop(ncr_sc)
*
* SCSI-3 VME interface is a little funny on writes:
* if we have a disconnect, the dma has overshot by
- * one byte and needs to be incremented. This is
- * true if we have not transferred either all data
- * or no data. XXX - from Matt Jacob
+ * one byte and the resid needs to be incremented.
+ * Only happens for partial transfers.
+ * (Thanks to Matt Jacob)
*/
resid = si->fifo_count & 0xFFFF;
+ if (dh->dh_flags & SIDH_OUT)
+ if ((resid > 0) && (resid < sc->sc_xlen))
+ resid++;
ntrans = sc->sc_xlen - resid;
#ifdef DEBUG
@@ -1162,14 +1186,14 @@ si_obio_dma_start(ncr_sc)
*/
data_pa = (u_long)(dh->dh_dvma - DVMA_BASE);
if (data_pa & 1)
- panic("si_dma_start: bad pa=0x%x", data_pa);
+ panic("si_dma_start: bad pa=0x%lx", data_pa);
xlen = ncr_sc->sc_datalen;
xlen &= ~1;
sc->sc_xlen = xlen; /* XXX: or less... */
#ifdef DEBUG
if (si_debug & 2) {
- printf("si_dma_start: dh=0x%x, pa=0x%x, xlen=%d\n",
+ printf("si_dma_start: dh=%p, pa=0x%lx, xlen=%d\n",
dh, data_pa, xlen);
}
#endif
@@ -1260,6 +1284,19 @@ si_obio_dma_eop(ncr_sc)
/* Not needed - DMA was stopped prior to examining sci_csr */
}
+#if (defined(DEBUG) || defined(DIAGNOSTIC)) && !defined(COUNT_SW_LEFTOVERS)
+#define COUNT_SW_LEFTOVERS
+#endif
+#ifdef COUNT_SW_LEFTOVERS
+/*
+ * Let's find out how often these occur. Read these with DDB from time
+ * to time.
+ */
+int sw_3_leftover = 0;
+int sw_2_leftover = 0;
+int sw_1_leftover = 0;
+int sw_0_leftover = 0;
+#endif
void
si_obio_dma_stop(ncr_sc)
@@ -1282,7 +1319,23 @@ si_obio_dma_stop(ncr_sc)
/* First, halt the DMA engine. */
si->sw_csr &= ~SI_CSR_DMA_EN;
+ /*
+ * XXX HARDWARE BUG!
+ * Apparently, some early 4/100 SCSI controllers had a hardware
+ * bug that caused the controller to do illegal memory access.
+ * We see this as SI_CSR_DMA_BUS_ERR (makes sense). To work around
+ * this, we simply need to clean up after ourselves ... there will
+ * be as many as 3 bytes left over. Since we clean up "left-over"
+ * bytes on every read anyway, we just continue to chug along
+ * if SI_CSR_DMA_BUS_ERR is asserted. (This was probably worked
+ * around in hardware later with the "left-over byte" indicator
+ * in the VME controller.)
+ */
+#if 0
if (si->sw_csr & (SI_CSR_DMA_CONFLICT | SI_CSR_DMA_BUS_ERR)) {
+#else
+ if (si->sw_csr & (SI_CSR_DMA_CONFLICT)) {
+#endif
printf("sw: DMA error, csr=0x%x, reset\n", si->sw_csr);
sr->sr_xs->error = XS_DRIVER_STUFFUP;
ncr_sc->sc_state |= NCR_ABORTING;
@@ -1334,19 +1387,34 @@ si_obio_dma_stop(ncr_sc)
switch (Dma_addr & 3) {
case 3:
- cp[-3] = (si->sw_bpr & 0xff000000) >> 24;
- cp[-2] = (si->sw_bpr & 0x00ff0000) >> 16;
- cp[-1] = (si->sw_bpr & 0x0000ff00) >> 8;
+ cp[0] = (si->sw_bpr & 0xff000000) >> 24;
+ cp[1] = (si->sw_bpr & 0x00ff0000) >> 16;
+ cp[2] = (si->sw_bpr & 0x0000ff00) >> 8;
+#ifdef COUNT_SW_LEFTOVERS
+ ++sw_3_leftover;
+#endif
break;
case 2:
- cp[-2] = (si->sw_bpr & 0xff000000) >> 24;
- cp[-1] = (si->sw_bpr & 0x00ff0000) >> 16;
+ cp[0] = (si->sw_bpr & 0xff000000) >> 24;
+ cp[1] = (si->sw_bpr & 0x00ff0000) >> 16;
+#ifdef COUNT_SW_LEFTOVERS
+ ++sw_2_leftover;
+#endif
break;
case 1:
- cp[-1] = (si->sw_bpr & 0xff000000) >> 24;
+ cp[0] = (si->sw_bpr & 0xff000000) >> 24;
+#ifdef COUNT_SW_LEFTOVERS
+ ++sw_1_leftover;
+#endif
+ break;
+
+#ifdef COUNT_SW_LEFTOVERS
+ default:
+ ++sw_0_leftover;
break;
+#endif
}
}
diff --git a/sys/arch/sparc/dev/xd.c b/sys/arch/sparc/dev/xd.c
index b2fc2b6bfd7..dd5df1fa016 100644
--- a/sys/arch/sparc/dev/xd.c
+++ b/sys/arch/sparc/dev/xd.c
@@ -1,4 +1,4 @@
-/* $NetBSD: xd.c,v 1.11 1996/01/07 22:03:02 thorpej Exp $ */
+/* $NetBSD: xd.c,v 1.25 1996/04/22 02:42:06 christos Exp $ */
/*
*
@@ -36,7 +36,7 @@
* x d . c x y l o g i c s 7 5 3 / 7 0 5 3 v m e / s m d d r i v e r
*
* author: Chuck Cranor <chuck@ccrc.wustl.edu>
- * id: $Id: xd.c,v 1.7 1996/03/04 20:35:23 chuck Exp $
+ * id: $NetBSD: xd.c,v 1.25 1996/04/22 02:42:06 christos Exp $
* started: 27-Feb-95
* references: [1] Xylogics Model 753 User's Manual
* part number: 166-753-001, Revision B, May 21, 1988.
@@ -61,7 +61,6 @@
#include <sys/proc.h>
#include <sys/systm.h>
#include <sys/kernel.h>
-#include <sys/conf.h>
#include <sys/file.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
@@ -73,11 +72,14 @@
#include <sys/disk.h>
#include <sys/syslog.h>
#include <sys/dkbad.h>
+#include <sys/conf.h>
+
#include <vm/vm.h>
#include <vm/vm_kern.h>
#include <machine/autoconf.h>
#include <machine/sun_disklabel.h>
+#include <machine/conf.h>
#include <sparc/dev/xdreg.h>
#include <sparc/dev/xdvar.h>
@@ -215,25 +217,15 @@ inline void xdc_rqinit __P((struct xd_iorq *, struct xdc_softc *,
struct xd_softc *, int, u_long, int,
caddr_t, struct buf *));
void xdc_rqtopb __P((struct xd_iorq *, struct xd_iopb *, int, int));
-int xdc_start __P((struct xdc_softc *, int));
+void xdc_start __P((struct xdc_softc *, int));
int xdc_startbuf __P((struct xdc_softc *, struct xd_softc *, struct buf *));
int xdc_submit_iorq __P((struct xdc_softc *, int, int));
void xdc_tick __P((void *));
-int xdc_xdreset __P((struct xdc_softc *, struct xd_softc *));
+void xdc_xdreset __P((struct xdc_softc *, struct xd_softc *));
/* machine interrupt hook */
int xdcintr __P((void *));
-/* {b,c}devsw */
-int xdclose __P((dev_t, int, int));
-int xddump __P((dev_t));
-int xdioctl __P((dev_t, u_long, caddr_t, int, struct proc *));
-int xdopen __P((dev_t, int, int));
-int xdread __P((dev_t, struct uio *));
-int xdwrite __P((dev_t, struct uio *));
-int xdsize __P((dev_t));
-void xdstrategy __P((struct buf *));
-
/* autoconf */
int xdcmatch __P((struct device *, void *, void *));
void xdcattach __P((struct device *, struct device *, void *));
@@ -247,12 +239,21 @@ int xdgetdisklabel __P((struct xd_softc *, void *));
* cfdrivers: device driver interface to autoconfig
*/
-struct cfdriver xdccd = {
- NULL, "xdc", xdcmatch, xdcattach, DV_DULL, sizeof(struct xdc_softc)
+struct cfattach xdc_ca = {
+ sizeof(struct xdc_softc), xdcmatch, xdcattach
+};
+
+
+struct cfdriver xdc_cd = {
+ NULL, "xdc", DV_DULL
+};
+
+struct cfattach xd_ca = {
+ sizeof(struct xd_softc), xdmatch, xdattach
};
-struct cfdriver xdcd = {
- NULL, "xd", xdmatch, xdattach, DV_DISK, sizeof(struct xd_softc)
+struct cfdriver xd_cd = {
+ NULL, "xd", DV_DISK
};
struct xdc_attach_args { /* this is the "aux" args to xdattach */
@@ -313,11 +314,11 @@ xdgetdisklabel(xd, b)
if (sdl->sl_magic == SUN_DKMAGIC)
xd->pcyl = sdl->sl_pcylinders;
else {
- printf("%s: WARNING: no `pcyl' in disk label.\n",
+ printf("%s: WARNING: no `pcyl' in disk label.\n",
xd->sc_dev.dv_xname);
xd->pcyl = xd->sc_dk.dk_label->d_ncylinders +
xd->sc_dk.dk_label->d_acylinders;
- printf("%s: WARNING: guessing pcyl=%d (ncyl+acyl)\n",
+ printf("%s: WARNING: guessing pcyl=%d (ncyl+acyl)\n",
xd->sc_dev.dv_xname, xd->pcyl);
}
@@ -357,9 +358,9 @@ int xdcmatch(parent, match, aux)
if (strcmp(cf->cf_driver->cd_name, ra->ra_name))
return (0);
- if (cputyp == CPU_SUN4) {
+ if (CPU_ISSUN4) {
xdc = (struct xdc *) ra->ra_vaddr;
- if (probeget(&xdc->xdc_csr, 1) == -1)
+ if (probeget((caddr_t) &xdc->xdc_csr, 1) == -1)
return (0);
xdc->xdc_csr = XDC_RESET;
XDC_WAIT(xdc, del, XDC_RESETUSEC, XDC_RESET);
@@ -372,7 +373,7 @@ int xdcmatch(parent, match, aux)
/*
* xdcattach: attach controller
*/
-void
+void
xdcattach(parent, self, aux)
struct device *parent, *self;
void *aux;
@@ -400,7 +401,7 @@ xdcattach(parent, self, aux)
xdc->sc_drives[lcv] = (struct xd_softc *) 0;
/* allocate and zero buffers
- *
+ *
* note: we simplify the code by allocating the max number of iopbs and
* iorq's up front. thus, we avoid linked lists and the costs
* associated with them in exchange for wasting a little memory. */
@@ -511,13 +512,12 @@ xdcattach(parent, self, aux)
* spin up and configure a disk after the system is booted (we can
* call xdattach!).
*/
-int
+int
xdmatch(parent, match, aux)
struct device *parent;
void *match, *aux;
{
- struct xdc_softc *xdc = (void *) parent;
struct cfdata *cf = match;
struct xdc_attach_args *xa = aux;
@@ -534,7 +534,7 @@ xdmatch(parent, match, aux)
* xdattach: attach a disk. this can be called from autoconf and also
* from xdopen/xdstrategy.
*/
-void
+void
xdattach(parent, self, aux)
struct device *parent, *self;
void *aux;
@@ -543,7 +543,7 @@ xdattach(parent, self, aux)
struct xd_softc *xd = (void *) self;
struct xdc_softc *xdc = (void *) parent;
struct xdc_attach_args *xa = aux;
- int rqno, err, spt, mb, blk, lcv, fmode, s, newstate;
+ int rqno, err, spt = 0, mb, blk, lcv, fmode, s = 0, newstate;
struct xd_iopb_drive *driopb;
struct dkbad *dkb;
struct bootpath *bp;
@@ -717,9 +717,9 @@ xdattach(parent, self, aux)
if (xa->booting) {
/* restore bootpath! (do this via attach_args again?)*/
bp = bootpath_store(0, NULL);
- if (bp && strcmp("xd", bp->name) == 0 &&
+ if (bp && strcmp("xd", bp->name) == 0 &&
xd->xd_drive == bp->val[0])
- bootdv = &xd->sc_dev;
+ bp->dev = &xd->sc_dev;
}
dk_establish(&xd->sc_dk, &xd->sc_dev); /* XXX */
@@ -743,13 +743,13 @@ done:
/*
* xdclose: close device
*/
-int
-xdclose(dev, flag, fmt)
+int
+xdclose(dev, flag, fmt, p)
dev_t dev;
int flag, fmt;
-
+ struct proc *p;
{
- struct xd_softc *xd = xdcd.cd_devs[DISKUNIT(dev)];
+ struct xd_softc *xd = xd_cd.cd_devs[DISKUNIT(dev)];
int part = DISKPART(dev);
/* clear mask bits */
@@ -770,20 +770,22 @@ xdclose(dev, flag, fmt)
/*
* xddump: crash dump system
*/
-int
-xddump(dev)
- dev_t dev;
-
+int
+xddump(dev, blkno, va, size)
+ dev_t dev;
+ daddr_t blkno;
+ caddr_t va;
+ size_t size;
{
int unit, part;
struct xd_softc *xd;
unit = DISKUNIT(dev);
- if (unit >= xdcd.cd_ndevs)
+ if (unit >= xd_cd.cd_ndevs)
return ENXIO;
part = DISKPART(dev);
- xd = xdcd.cd_devs[unit];
+ xd = xd_cd.cd_devs[unit];
printf("%s%c: crash dump not supported (yet)\n", xd->sc_dev.dv_xname,
'a' + part);
@@ -795,11 +797,11 @@ xddump(dev)
* "dumpsize" == size of dump in clicks "physmem" == size of physical
* memory (clicks, ctob() to get bytes) (normal case: dumpsize ==
* physmem)
- *
+ *
* dump a copy of physical memory to the dump device starting at sector
* "dumplo" in the swap partition (make sure > 0). map in pages as
* we go. use polled I/O.
- *
+ *
* XXX how to handle NON_CONTIG? */
}
@@ -807,7 +809,7 @@ xddump(dev)
/*
* xdioctl: ioctls on XD drives. based on ioctl's of other netbsd disks.
*/
-int
+int
xdioctl(dev, command, addr, flag, p)
dev_t dev;
u_long command;
@@ -822,7 +824,7 @@ xdioctl(dev, command, addr, flag, p)
unit = DISKUNIT(dev);
- if (unit >= xdcd.cd_ndevs || (xd = xdcd.cd_devs[unit]) == NULL)
+ if (unit >= xd_cd.cd_ndevs || (xd = xd_cd.cd_devs[unit]) == NULL)
return (ENXIO);
/* switch on ioctl type */
@@ -901,11 +903,11 @@ xdioctl(dev, command, addr, flag, p)
* xdopen: open drive
*/
-int
-xdopen(dev, flag, fmt)
+int
+xdopen(dev, flag, fmt, p)
dev_t dev;
int flag, fmt;
-
+ struct proc *p;
{
int unit, part;
struct xd_softc *xd;
@@ -914,7 +916,7 @@ xdopen(dev, flag, fmt)
/* first, could it be a valid target? */
unit = DISKUNIT(dev);
- if (unit >= xdcd.cd_ndevs || (xd = xdcd.cd_devs[unit]) == NULL)
+ if (unit >= xd_cd.cd_ndevs || (xd = xd_cd.cd_devs[unit]) == NULL)
return (ENXIO);
part = DISKPART(dev);
@@ -954,18 +956,20 @@ xdopen(dev, flag, fmt)
}
int
-xdread(dev, uio)
+xdread(dev, uio, flags)
dev_t dev;
struct uio *uio;
+ int flags;
{
return (physio(xdstrategy, NULL, dev, B_READ, minphys, uio));
}
int
-xdwrite(dev, uio)
+xdwrite(dev, uio, flags)
dev_t dev;
struct uio *uio;
+ int flags;
{
return (physio(xdstrategy, NULL, dev, B_WRITE, minphys, uio));
@@ -976,28 +980,28 @@ xdwrite(dev, uio)
* xdsize: return size of a partition for a dump
*/
-int
+int
xdsize(dev)
dev_t dev;
{
struct xd_softc *xdsc;
- int unit, part, size;
+ int part, size;
/* valid unit? try an open */
- if (xdopen(dev, 0, S_IFBLK) != 0)
+ if (xdopen(dev, 0, S_IFBLK, NULL) != 0)
return (-1);
/* do it */
- xdsc = xdcd.cd_devs[DISKUNIT(dev)];
+ xdsc = xd_cd.cd_devs[DISKUNIT(dev)];
part = DISKPART(dev);
if (xdsc->sc_dk.dk_label->d_partitions[part].p_fstype != FS_SWAP)
size = -1; /* only give valid size for swap partitions */
else
size = xdsc->sc_dk.dk_label->d_partitions[part].p_size;
- if (xdclose(dev, 0, S_IFBLK) != 0)
+ if (xdclose(dev, 0, S_IFBLK, NULL) != 0)
return -1;
return size;
}
@@ -1005,7 +1009,7 @@ xdsize(dev)
* xdstrategy: buffering system interface to xd.
*/
-void
+void
xdstrategy(bp)
struct buf *bp;
@@ -1020,7 +1024,7 @@ xdstrategy(bp)
/* check for live device */
- if (unit >= xdcd.cd_ndevs || (xd = xdcd.cd_devs[unit]) == 0 ||
+ if (unit >= xd_cd.cd_ndevs || (xd = xd_cd.cd_devs[unit]) == 0 ||
bp->b_blkno < 0 ||
(bp->b_bcount % xd->sc_dk.dk_label->d_secsize) != 0) {
bp->b_error = EINVAL;
@@ -1062,7 +1066,7 @@ xdstrategy(bp)
/*
* now we know we have a valid buf structure that we need to do I/O
* on.
- *
+ *
* note that we don't disksort because the controller has a sorting
* algorithm built into the hardware.
*/
@@ -1115,14 +1119,12 @@ done: /* tells upper layers we are done with this
*
* xdcintr: hardware interrupt.
*/
-int
+int
xdcintr(v)
void *v;
{
struct xdc_softc *xdcsc = v;
- struct xd_softc *xd;
- struct buf *bp;
/* kick the event counter */
@@ -1156,7 +1158,7 @@ xdcintr(v)
* xdc_rqinit: fill out the fields of an I/O request
*/
-inline void
+inline void
xdc_rqinit(rq, xdc, xd, md, blk, cnt, db, bp)
struct xd_iorq *rq;
struct xdc_softc *xdc;
@@ -1181,7 +1183,7 @@ xdc_rqinit(rq, xdc, xd, md, blk, cnt, db, bp)
* xdc_rqtopb: load up an IOPB based on an iorq
*/
-void
+void
xdc_rqtopb(iorq, iopb, cmd, subfun)
struct xd_iorq *iorq;
struct xd_iopb *iopb;
@@ -1218,7 +1220,7 @@ xdc_rqtopb(iorq, iopb, cmd, subfun)
XDPC_RBC | XDPC_ECC2;
ctrl->throttle = XDC_THROTTLE;
#ifdef sparc
- if (cputyp == CPU_SUN4 && cpumod == SUN4_300)
+ if (CPU_ISSUN4 && cpumod == SUN4_300)
ctrl->delay = XDC_DELAY_4_300;
else
ctrl->delay = XDC_DELAY_SPARC;
@@ -1297,7 +1299,7 @@ xdc_rqtopb(iorq, iopb, cmd, subfun)
* If you've already got an IORQ, you can call submit directly (currently
* there is no need to do this). NORM requests are handled seperately.
*/
-int
+int
xdc_cmd(xdcsc, cmd, subfn, unit, block, scnt, dptr, fullmode)
struct xdc_softc *xdcsc;
int cmd, subfn, unit, block, scnt;
@@ -1306,7 +1308,6 @@ xdc_cmd(xdcsc, cmd, subfn, unit, block, scnt, dptr, fullmode)
{
int rqno, submode = XD_STATE(fullmode), retry;
- u_long dp;
struct xd_iorq *iorq;
struct xd_iopb *iopb;
@@ -1364,7 +1365,7 @@ xdc_cmd(xdcsc, cmd, subfn, unit, block, scnt, dptr, fullmode)
* start a buffer running, assumes nfree > 0
*/
-int
+int
xdc_startbuf(xdcsc, xdsc, bp)
struct xdc_softc *xdcsc;
struct xd_softc *xdsc;
@@ -1375,7 +1376,7 @@ xdc_startbuf(xdcsc, xdsc, bp)
struct xd_iorq *iorq;
struct xd_iopb *iopb;
struct buf *wq;
- u_long block, dp;
+ u_long block;
caddr_t dbuf;
if (!xdcsc->nfree)
@@ -1409,7 +1410,7 @@ xdc_startbuf(xdcsc, xdsc, bp)
/*
* load request. we have to calculate the correct block number based
* on partition info.
- *
+ *
* note that iorq points to the buffer as mapped into DVMA space,
* where as the bp->b_data points to its non-DVMA mapping.
*/
@@ -1479,7 +1480,7 @@ xdc_startbuf(xdcsc, xdsc, bp)
*/
-int
+int
xdc_submit_iorq(xdcsc, iorqno, type)
struct xdc_softc *xdcsc;
int iorqno;
@@ -1560,7 +1561,7 @@ xdc_submit_iorq(xdcsc, iorqno, type)
* the caller is interesting in. if freeone is true, then it returns
* when there is a free iorq.
*/
-int
+int
xdc_piodriver(xdcsc, iorqno, freeone)
struct xdc_softc *xdcsc;
char iorqno;
@@ -1638,7 +1639,7 @@ xdc_piodriver(xdcsc, iorqno, freeone)
* xdc_reset: reset one drive. NOTE: assumes xdc was just reset.
* we steal iopb[0] for this, but we put it back when we are done.
*/
-int
+void
xdc_xdreset(xdcsc, xdsc)
struct xdc_softc *xdcsc;
struct xd_softc *xdsc;
@@ -1672,14 +1673,14 @@ xdc_xdreset(xdcsc, xdsc)
* xdc_reset: reset everything: requests are marked as errors except
* a polled request (which is resubmitted)
*/
-int
+int
xdc_reset(xdcsc, quiet, blastmode, error, xdsc)
struct xdc_softc *xdcsc;
int quiet, blastmode, error;
struct xd_softc *xdsc;
{
- int del = 0, lcv, poll = -1, retval = XD_ERR_AOK;
+ int del = 0, lcv, retval = XD_ERR_AOK;
int oldfree = xdcsc->nfree;
/* soft reset hardware */
@@ -1722,7 +1723,7 @@ xdc_reset(xdcsc, quiet, blastmode, error, xdsc)
(vm_offset_t)iorq->buf->b_un.b_addr,
iorq->buf->b_bcount);
disk_unbusy(&xdcsc->reqs[lcv].xd->sc_dk,
- (xdcsc->reqs[lcv].buf->b_bcount -
+ (xdcsc->reqs[lcv].buf->b_bcount -
xdcsc->reqs[lcv].buf->b_resid));
biodone(iorq->buf);
XDC_FREE(xdcsc, lcv); /* add to free list */
@@ -1770,7 +1771,7 @@ xdc_reset(xdcsc, quiet, blastmode, error, xdsc)
* xdc_start: start all waiting buffers
*/
-int
+void
xdc_start(xdcsc, maxio)
struct xdc_softc *xdcsc;
int maxio;
@@ -1790,14 +1791,13 @@ xdc_start(xdcsc, maxio)
* xdc_remove_iorq: remove "done" IOPB's.
*/
-int
+int
xdc_remove_iorq(xdcsc)
struct xdc_softc *xdcsc;
{
int errno, rqno, comm, errs;
struct xdc *xdc = xdcsc->xdc;
- u_long addr;
struct xd_iopb *iopb;
struct xd_iorq *iorq;
struct buf *bp;
@@ -1821,7 +1821,7 @@ xdc_remove_iorq(xdcsc)
/*
* get iopb that is done
- *
+ *
* hmm... I used to read the address of the done IOPB off the VME
* registers and calculate the rqno directly from that. that worked
* until I started putting a load on the controller. when loaded, i
@@ -1889,7 +1889,7 @@ xdc_remove_iorq(xdcsc)
* at the bad144 sector. to exit bad144 mode, we
* must advance the pointers 1 sector and issue a new
* request if there are still sectors left to process
- *
+ *
*/
XDC_ADVANCE(iorq, 1); /* advance 1 sector */
@@ -1957,7 +1957,7 @@ xdc_remove_iorq(xdcsc)
* is in lasterror. also, if iorq->errno == 0, then we recovered
* from that error (otherwise iorq->errno == iorq->lasterror).
*/
-void
+void
xdc_perror(iorq, iopb, still_trying)
struct xd_iorq *iorq;
struct xd_iopb *iopb;
@@ -1990,7 +1990,7 @@ xdc_perror(iorq, iopb, still_trying)
* xdc_error: non-fatal error encountered... recover.
* return AOK if resubmitted, return FAIL if this iopb is done
*/
-int
+int
xdc_error(xdcsc, iorq, iopb, rqno, comm)
struct xdc_softc *xdcsc;
struct xd_iorq *iorq;
@@ -2062,7 +2062,7 @@ xdc_error(xdcsc, iorq, iopb, rqno, comm)
/*
* xdc_tick: make sure xd is still alive and ticking (err, kicking).
*/
-void
+void
xdc_tick(arg)
void *arg;
@@ -2070,7 +2070,7 @@ xdc_tick(arg)
struct xdc_softc *xdcsc = arg;
int lcv, s, reset = 0;
#ifdef XDC_DIAG
- int wait, run, free, done, whd;
+ int wait, run, free, done, whd = 0;
u_char fqc[XDC_MAXIOPB], wqc[XDC_MAXIOPB], mark[XDC_MAXIOPB];
s = splbio();
wait = xdcsc->nwait;
@@ -2100,7 +2100,7 @@ xdc_tick(arg)
printf("\n");
for (lcv = 0; lcv < XDC_MAXIOPB; lcv++) {
if (mark[lcv] == 0)
- printf("MARK: running %d: mode %d done %d errs %d errno 0x%x ttl %d buf %x\n",
+ printf("MARK: running %d: mode %d done %d errs %d errno 0x%x ttl %d buf %p\n",
lcv, xdcsc->reqs[lcv].mode,
xdcsc->iopbase[lcv].done,
xdcsc->iopbase[lcv].errs,
@@ -2154,7 +2154,7 @@ xdc_tick(arg)
* in user code, and is also useful for some debugging. we return
* an error code. called at user priority.
*/
-int
+int
xdc_ioctlcmd(xd, dev, xio)
struct xd_softc *xd;
dev_t dev;
@@ -2248,7 +2248,7 @@ xdc_ioctlcmd(xd, dev, xio)
if (xio->dlen) {
dvmabuf = dvma_malloc(xio->dlen, &buf, M_WAITOK);
if (xio->cmd == XDCMD_WR || xio->cmd == XDCMD_XWR) {
- if (err = copyin(xio->dptr, buf, xio->dlen)) {
+ if ((err = copyin(xio->dptr, buf, xio->dlen)) != 0) {
dvma_free(dvmabuf, xio->dlen, &buf);
return (err);
}
diff --git a/sys/arch/sparc/dev/xdreg.h b/sys/arch/sparc/dev/xdreg.h
index addbe589534..c723d018d0d 100644
--- a/sys/arch/sparc/dev/xdreg.h
+++ b/sys/arch/sparc/dev/xdreg.h
@@ -1,4 +1,4 @@
-/* $NetBSD: xdreg.h,v 1.1 1995/06/26 23:07:11 pk Exp $ */
+/* $NetBSD: xdreg.h,v 1.3 1996/03/31 22:38:54 pk Exp $ */
/*
*
@@ -203,9 +203,9 @@ struct xd_iopb {
/*
* some commands overload bytes 6 to 0x13 of the iopb with different meanings.
- * these commands include:
+ * these commands include:
* section 4.2: controller parameters
- * section 4.3: drive parameters
+ * section 4.3: drive parameters
* sectino 4.4: format parameters
*
* note that the commands that overload the iopb are not part of the
@@ -287,7 +287,7 @@ struct xd_iopb_ctrl {
#define XDC_DELAY_SPARC 0
/*
- * drive parameters iopb: redefines bytes: 6, 8, 9, a, b, c, d, e
+ * drive parameters iopb: redefines bytes: 6, 8, 9, a, b, c, d, e
*/
struct xd_iopb_drive {
@@ -311,7 +311,7 @@ struct xd_iopb_drive {
};
/*
- * format parameters iopb: redefines bytes: 6, 8, 9, a, b, c, d, 0x10, 0x11
+ * format parameters iopb: redefines bytes: 6, 8, 9, a, b, c, d, 0x10, 0x11
*/
struct xd_iopb_format {
diff --git a/sys/arch/sparc/dev/xdvar.h b/sys/arch/sparc/dev/xdvar.h
index 5417e3953ac..294f3a8c84b 100644
--- a/sys/arch/sparc/dev/xdvar.h
+++ b/sys/arch/sparc/dev/xdvar.h
@@ -1,4 +1,4 @@
-/* $NetBSD: xdvar.h,v 1.2 1996/01/07 22:03:04 thorpej Exp $ */
+/* $NetBSD: xdvar.h,v 1.5 1996/03/31 22:38:56 pk Exp $ */
/*
*
@@ -32,9 +32,9 @@
*/
/*
- * x d v a r . h
+ * x d v a r . h
*
- * this file defines the software structure we use to control the
+ * this file defines the software structure we use to control the
* 753/7053.
*
* author: Chuck Cranor <chuck@ccrc.wustl.edu>
@@ -67,7 +67,7 @@ struct xd_iorq {
#define XD_SUB_MASK 0xf0 /* mask bits for state */
#define XD_SUB_FREE 0x00 /* free */
#define XD_SUB_NORM 0x10 /* normal I/O request */
-#define XD_SUB_WAIT 0x20 /* normal I/O request in the
+#define XD_SUB_WAIT 0x20 /* normal I/O request in the
context of a process */
#define XD_SUB_POLL 0x30 /* polled mode */
#define XD_SUB_DONE 0x40 /* not active, but can't be free'd yet */
@@ -151,12 +151,12 @@ struct xdc_softc {
struct buf sc_wq; /* queue'd IOPBs for this controller */
char freereq[XDC_MAXIOPB]; /* free list (stack) */
char waitq[XDC_MAXIOPB]; /* wait queue */
- char nfree; /* number of iopbs free */
- char nrun; /* number running */
- char nwait; /* number of waiting iopbs */
- char ndone; /* number of done IORQs */
- char waithead; /* head of queue */
- char waitend; /* end of queue */
+ u_char nfree; /* number of iopbs free */
+ u_char nrun; /* number running */
+ u_char nwait; /* number of waiting iopbs */
+ u_char ndone; /* number of done IORQs */
+ u_char waithead; /* head of queue */
+ u_char waitend; /* end of queue */
};
/*
diff --git a/sys/arch/sparc/dev/xio.h b/sys/arch/sparc/dev/xio.h
index e9e22859542..1ad5662b6be 100644
--- a/sys/arch/sparc/dev/xio.h
+++ b/sys/arch/sparc/dev/xio.h
@@ -1,4 +1,4 @@
-/* $NetBSD: xio.h,v 1.1 1995/06/26 23:07:26 pk Exp $ */
+/* $NetBSD: xio.h,v 1.2 1996/03/31 22:38:58 pk Exp $ */
/*
*
@@ -32,9 +32,9 @@
*/
/*
- * x i o . h
+ * x i o . h
*
- * this file defines the software structure we use to ioctl the
+ * this file defines the software structure we use to ioctl the
* 753/7053. this interface isn't set in stone and may (or may not)
* need adjustment.
*
diff --git a/sys/arch/sparc/dev/xy.c b/sys/arch/sparc/dev/xy.c
index bd85ae48a25..2841bd9ba84 100644
--- a/sys/arch/sparc/dev/xy.c
+++ b/sys/arch/sparc/dev/xy.c
@@ -1,4 +1,4 @@
-/* $NetBSD: xy.c,v 1.3 1996/01/07 22:03:05 thorpej Exp $ */
+/* $NetBSD: xy.c,v 1.17 1996/04/22 02:42:04 christos Exp $ */
/*
*
@@ -36,7 +36,7 @@
* x y . c x y l o g i c s 4 5 0 / 4 5 1 s m d d r i v e r
*
* author: Chuck Cranor <chuck@ccrc.wustl.edu>
- * id: $Id: xy.c,v 1.7 1996/03/04 20:35:25 chuck Exp $
+ * id: $NetBSD: xy.c,v 1.17 1996/04/22 02:42:04 christos Exp $
* started: 14-Sep-95
* references: [1] Xylogics Model 753 User's Manual
* part number: 166-753-001, Revision B, May 21, 1988.
@@ -44,7 +44,7 @@
* [2] other NetBSD disk device drivers
* [3] Xylogics Model 450 User's Manual
* part number: 166-017-001, Revision B, 1983.
- * [4] Addendum to Xylogics Model 450 Disk Controller User's
+ * [4] Addendum to Xylogics Model 450 Disk Controller User's
* Manual, Jan. 1985.
* [5] The 451 Controller, Rev. B3, September 2, 1986.
* [6] David Jones <dej@achilles.net>'s unfinished 450/451 driver
@@ -61,7 +61,6 @@
#include <sys/proc.h>
#include <sys/systm.h>
#include <sys/kernel.h>
-#include <sys/conf.h>
#include <sys/file.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
@@ -73,11 +72,14 @@
#include <sys/disk.h>
#include <sys/syslog.h>
#include <sys/dkbad.h>
+#include <sys/conf.h>
+
#include <vm/vm.h>
#include <vm/vm_kern.h>
#include <machine/autoconf.h>
#include <machine/sun_disklabel.h>
+#include <machine/conf.h>
#include <sparc/dev/xyreg.h>
#include <sparc/dev/xyvar.h>
@@ -151,32 +153,22 @@ int xyc_ioctlcmd __P((struct xy_softc *, dev_t dev, struct xd_iocmd *));
void xyc_perror __P((struct xy_iorq *, struct xy_iopb *, int));
int xyc_piodriver __P((struct xyc_softc *, struct xy_iorq *));
int xyc_remove_iorq __P((struct xyc_softc *));
-int xyc_reset __P((struct xyc_softc *, int, struct xy_iorq *, int,
+int xyc_reset __P((struct xyc_softc *, int, struct xy_iorq *, int,
struct xy_softc *));
inline void xyc_rqinit __P((struct xy_iorq *, struct xyc_softc *,
struct xy_softc *, int, u_long, int,
caddr_t, struct buf *));
void xyc_rqtopb __P((struct xy_iorq *, struct xy_iopb *, int, int));
-int xyc_start __P((struct xyc_softc *, struct xy_iorq *));
+void xyc_start __P((struct xyc_softc *, struct xy_iorq *));
int xyc_startbuf __P((struct xyc_softc *, struct xy_softc *, struct buf *));
int xyc_submit_iorq __P((struct xyc_softc *, struct xy_iorq *, int));
void xyc_tick __P((void *));
int xyc_unbusy __P((struct xyc *, int));
-int xyc_xyreset __P((struct xyc_softc *, struct xy_softc *));
+void xyc_xyreset __P((struct xyc_softc *, struct xy_softc *));
/* machine interrupt hook */
int xycintr __P((void *));
-/* {b,c}devsw */
-int xyclose __P((dev_t, int, int));
-int xydump __P((dev_t));
-int xyioctl __P((dev_t, u_long, caddr_t, int, struct proc *));
-int xyopen __P((dev_t, int, int));
-int xyread __P((dev_t, struct uio *));
-int xywrite __P((dev_t, struct uio *));
-int xysize __P((dev_t));
-void xystrategy __P((struct buf *));
-
/* autoconf */
int xycmatch __P((struct device *, void *, void *));
void xycattach __P((struct device *, struct device *, void *));
@@ -190,12 +182,20 @@ int xygetdisklabel __P((struct xy_softc *, void *));
* cfdrivers: device driver interface to autoconfig
*/
-struct cfdriver xyccd = {
- NULL, "xyc", xycmatch, xycattach, DV_DULL, sizeof(struct xyc_softc)
+struct cfattach xyc_ca = {
+ sizeof(struct xyc_softc), xycmatch, xycattach
+};
+
+struct cfdriver xyc_cd = {
+ NULL, "xyc", DV_DULL
};
-struct cfdriver xycd = {
- NULL, "xy", xymatch, xyattach, DV_DISK, sizeof(struct xy_softc)
+struct cfattach xy_ca = {
+ sizeof(struct xy_softc), xymatch, xyattach
+};
+
+struct cfdriver xy_cd = {
+ NULL, "xy", DV_DISK
};
struct xyc_attach_args { /* this is the "aux" args to xyattach */
@@ -256,11 +256,11 @@ xygetdisklabel(xy, b)
if (sdl->sl_magic == SUN_DKMAGIC)
xy->pcyl = sdl->sl_pcylinders;
else {
- printf("%s: WARNING: no `pcyl' in disk label.\n",
- xy->sc_dev.dv_xname);
+ printf("%s: WARNING: no `pcyl' in disk label.\n",
+ xy->sc_dev.dv_xname);
xy->pcyl = xy->sc_dk.dk_label->d_ncylinders +
xy->sc_dk.dk_label->d_acylinders;
- printf("%s: WARNING: guessing pcyl=%d (ncyl+acyl)\n",
+ printf("%s: WARNING: guessing pcyl=%d (ncyl+acyl)\n",
xy->sc_dev.dv_xname, xy->pcyl);
}
@@ -295,14 +295,13 @@ int xycmatch(parent, match, aux)
struct confargs *ca = aux;
struct romaux *ra = &ca->ca_ra;
struct xyc *xyc;
- int del = 0;
if (strcmp(cf->cf_driver->cd_name, ra->ra_name))
return (0);
- if (cputyp == CPU_SUN4) {
+ if (CPU_ISSUN4) {
xyc = (struct xyc *) ra->ra_vaddr;
- if (probeget(&xyc->xyc_rsetup, 1) == -1)
+ if (probeget((caddr_t) &xyc->xyc_rsetup, 1) == -1)
return (0);
if (xyc_unbusy(xyc, XYC_RESETUSEC) == XY_ERR_FAIL)
return(0);
@@ -313,7 +312,7 @@ int xycmatch(parent, match, aux)
/*
* xycattach: attach controller
*/
-void
+void
xycattach(parent, self, aux)
struct device *parent, *self;
void *aux;
@@ -343,10 +342,10 @@ xycattach(parent, self, aux)
for (lcv = 0; lcv < XYC_MAXDEV; lcv++)
xyc->sc_drives[lcv] = (struct xy_softc *) 0;
- /*
+ /*
* allocate and zero buffers
- * check boundaries of the KVA's ... all IOPBs must reside in
- * the same 64K region.
+ * check boundaries of the KVA's ... all IOPBs must reside in
+ * the same 64K region.
*/
pbsz = XYC_MAXIOPB * sizeof(struct xy_iopb);
@@ -359,7 +358,7 @@ xycattach(parent, self, aux)
dvma_free(dtmp2, pbsz, &tmp2);
ultmp = (u_long) dtmp;
if ((ultmp & 0xffff0000) != ((ultmp + pbsz) & 0xffff0000)) {
- printf("%s: can't alloc IOPB mem in 64K\n",
+ printf("%s: can't alloc IOPB mem in 64K\n",
xyc->sc_dev.dv_xname);
return;
}
@@ -374,9 +373,9 @@ xycattach(parent, self, aux)
panic("xyc malloc");
bzero(xyc->reqs, XYC_MAXIOPB * sizeof(struct xy_iorq));
- /*
+ /*
* init iorq to iopb pointers, and non-zero fields in the
- * iopb which never change.
+ * iopb which never change.
*/
for (lcv = 0; lcv < XYC_MAXIOPB; lcv++) {
@@ -414,8 +413,8 @@ xycattach(parent, self, aux)
return;
}
if ((xyc->xyc->xyc_csr & XYC_ADRM) == 0) {
- printf("%s: 24 bit addressing turned off\n",
- xyc->sc_dev.dv_xname);
+ printf("%s: 24 bit addressing turned off\n",
+ xyc->sc_dev.dv_xname);
printf("please set hardware jumpers JM1-JM2=in, JM3-JM4=out\n");
printf("to enable 24 bit mode and this driver\n");
return;
@@ -458,13 +457,12 @@ xycattach(parent, self, aux)
* spin up and configure a disk after the system is booted (we can
* call xyattach!).
*/
-int
+int
xymatch(parent, match, aux)
struct device *parent;
void *match, *aux;
{
- struct xyc_softc *xyc = (void *) parent;
struct cfdata *cf = match;
struct xyc_attach_args *xa = aux;
@@ -481,7 +479,7 @@ xymatch(parent, match, aux)
* xyattach: attach a disk. this can be called from autoconf and also
* from xyopen/xystrategy.
*/
-void
+void
xyattach(parent, self, aux)
struct device *parent, *self;
void *aux;
@@ -490,7 +488,7 @@ xyattach(parent, self, aux)
struct xy_softc *xy = (void *) self, *oxy;
struct xyc_softc *xyc = (void *) parent;
struct xyc_attach_args *xa = aux;
- int res, err, spt, mb, blk, lcv, fmode, s, newstate;
+ int err, spt, mb, blk, lcv, fmode, s = 0, newstate;
struct dkbad *dkb;
struct bootpath *bp;
@@ -565,13 +563,13 @@ xyattach(parent, self, aux)
xy->nsect = 1;
xy->sectpercyl = 1;
for (lcv = 0; lcv < 126; lcv++) /* init empty bad144 table */
- xy->dkb.bt_bad[lcv].bt_cyl =
+ xy->dkb.bt_bad[lcv].bt_cyl =
xy->dkb.bt_bad[lcv].bt_trksec = 0xffff;
/* read disk label */
- for (xy->drive_type = 0 ; xy->drive_type <= XYC_MAXDT ;
+ for (xy->drive_type = 0 ; xy->drive_type <= XYC_MAXDT ;
xy->drive_type++) {
- err = xyc_cmd(xyc, XYCMD_RD, 0, xy->xy_drive, 0, 1,
+ err = xyc_cmd(xyc, XYCMD_RD, 0, xy->xy_drive, 0, 1,
xa->dvmabuf, fmode);
XYC_DONE(xyc, err);
if (err == XY_ERR_AOK) break;
@@ -605,9 +603,9 @@ xyattach(parent, self, aux)
* 450/451 stupidity: the drive type is encoded into the format
* of the disk. the drive type in the IOPB must match the drive
* type in the format, or you will not be able to do I/O to the
- * disk (you get header not found errors). if you have two drives
- * of different sizes that have the same drive type in their
- * formatting then you are out of luck.
+ * disk (you get header not found errors). if you have two drives
+ * of different sizes that have the same drive type in their
+ * formatting then you are out of luck.
*
* this problem was corrected in the 753/7053.
*/
@@ -624,7 +622,7 @@ xyattach(parent, self, aux)
panic("xy drive size mismatch");
}
}
-
+
/* now set the real drive parameters! */
@@ -645,10 +643,10 @@ xyattach(parent, self, aux)
* last track of the disk (i.e. second cyl of "acyl" area).
*/
- blk = (xy->ncyl + xy->acyl - 1) * (xy->nhead * xy->nsect) +
+ blk = (xy->ncyl + xy->acyl - 1) * (xy->nhead * xy->nsect) +
/* last cyl */
(xy->nhead - 1) * xy->nsect; /* last head */
- err = xyc_cmd(xyc, XYCMD_RD, 0, xy->xy_drive, blk, 1,
+ err = xyc_cmd(xyc, XYCMD_RD, 0, xy->xy_drive, blk, 1,
xa->dvmabuf, fmode);
XYC_DONE(xyc, err);
if (err) {
@@ -681,9 +679,9 @@ xyattach(parent, self, aux)
if (xa->booting) {
/* restore bootpath! (do this via attach_args again?)*/
bp = bootpath_store(0, NULL);
- if (bp && strcmp("xy", bp->name) == 0 &&
- xy->xy_drive == bp->val[0])
- bootdv = &xy->sc_dev;
+ if (bp && strcmp("xy", bp->name) == 0 &&
+ xy->xy_drive == bp->val[0])
+ bp->dev = &xy->sc_dev;
}
dk_establish(&xy->sc_dk, &xy->sc_dev); /* XXX */
@@ -707,13 +705,14 @@ done:
/*
* xyclose: close device
*/
-int
-xyclose(dev, flag, fmt)
+int
+xyclose(dev, flag, fmt, p)
dev_t dev;
int flag, fmt;
+ struct proc *p;
{
- struct xy_softc *xy = xycd.cd_devs[DISKUNIT(dev)];
+ struct xy_softc *xy = xy_cd.cd_devs[DISKUNIT(dev)];
int part = DISKPART(dev);
/* clear mask bits */
@@ -734,20 +733,22 @@ xyclose(dev, flag, fmt)
/*
* xydump: crash dump system
*/
-int
-xydump(dev)
- dev_t dev;
-
+int
+xydump(dev, blkno, va, size)
+ dev_t dev;
+ daddr_t blkno;
+ caddr_t va;
+ size_t size;
{
int unit, part;
struct xy_softc *xy;
unit = DISKUNIT(dev);
- if (unit >= xycd.cd_ndevs)
+ if (unit >= xy_cd.cd_ndevs)
return ENXIO;
part = DISKPART(dev);
- xy = xycd.cd_devs[unit];
+ xy = xy_cd.cd_devs[unit];
printf("%s%c: crash dump not supported (yet)\n", xy->sc_dev.dv_xname,
'a' + part);
@@ -759,11 +760,11 @@ xydump(dev)
* "dumpsize" == size of dump in clicks "physmem" == size of physical
* memory (clicks, ctob() to get bytes) (normal case: dumpsize ==
* physmem)
- *
+ *
* dump a copy of physical memory to the dump device starting at sector
* "dumplo" in the swap partition (make sure > 0). map in pages as
* we go. use polled I/O.
- *
+ *
* XXX how to handle NON_CONTIG? */
}
@@ -771,7 +772,7 @@ xydump(dev)
/*
* xyioctl: ioctls on XY drives. based on ioctl's of other netbsd disks.
*/
-int
+int
xyioctl(dev, command, addr, flag, p)
dev_t dev;
u_long command;
@@ -786,7 +787,7 @@ xyioctl(dev, command, addr, flag, p)
unit = DISKUNIT(dev);
- if (unit >= xycd.cd_ndevs || (xy = xycd.cd_devs[unit]) == NULL)
+ if (unit >= xy_cd.cd_ndevs || (xy = xy_cd.cd_devs[unit]) == NULL)
return (ENXIO);
/* switch on ioctl type */
@@ -866,11 +867,11 @@ xyioctl(dev, command, addr, flag, p)
* xyopen: open drive
*/
-int
-xyopen(dev, flag, fmt)
+int
+xyopen(dev, flag, fmt, p)
dev_t dev;
int flag, fmt;
-
+ struct proc *p;
{
int unit, part;
struct xy_softc *xy;
@@ -879,7 +880,7 @@ xyopen(dev, flag, fmt)
/* first, could it be a valid target? */
unit = DISKUNIT(dev);
- if (unit >= xycd.cd_ndevs || (xy = xycd.cd_devs[unit]) == NULL)
+ if (unit >= xy_cd.cd_ndevs || (xy = xy_cd.cd_devs[unit]) == NULL)
return (ENXIO);
part = DISKPART(dev);
@@ -890,7 +891,7 @@ xyopen(dev, flag, fmt)
xa.dvmabuf = (char *)dvma_malloc(XYFM_BPS, &xa.buf, M_NOWAIT);
xa.fullmode = XY_SUB_WAIT;
xa.booting = 0;
- xyattach((struct device *) xy->parent,
+ xyattach((struct device *) xy->parent,
(struct device *) xy, &xa);
dvma_free(xa.dvmabuf, XYFM_BPS, &xa.buf);
if (xy->state == XY_DRIVE_UNKNOWN) {
@@ -920,18 +921,20 @@ xyopen(dev, flag, fmt)
}
int
-xyread(dev, uio)
+xyread(dev, uio, flags)
dev_t dev;
struct uio *uio;
+ int flags;
{
return (physio(xystrategy, NULL, dev, B_READ, minphys, uio));
}
int
-xywrite(dev, uio)
+xywrite(dev, uio, flags)
dev_t dev;
struct uio *uio;
+ int flags;
{
return (physio(xystrategy, NULL, dev, B_WRITE, minphys, uio));
@@ -942,28 +945,28 @@ xywrite(dev, uio)
* xysize: return size of a partition for a dump
*/
-int
+int
xysize(dev)
dev_t dev;
{
struct xy_softc *xysc;
- int unit, part, size;
+ int part, size;
/* valid unit? try an open */
- if (xyopen(dev, 0, S_IFBLK) != 0)
+ if (xyopen(dev, 0, S_IFBLK, NULL) != 0)
return (-1);
/* do it */
- xysc = xycd.cd_devs[DISKUNIT(dev)];
+ xysc = xy_cd.cd_devs[DISKUNIT(dev)];
part = DISKPART(dev);
if (xysc->sc_dk.dk_label->d_partitions[part].p_fstype != FS_SWAP)
size = -1; /* only give valid size for swap partitions */
else
size = xysc->sc_dk.dk_label->d_partitions[part].p_size;
- if (xyclose(dev, 0, S_IFBLK) != 0)
+ if (xyclose(dev, 0, S_IFBLK, NULL) != 0)
return -1;
return size;
}
@@ -972,14 +975,12 @@ xysize(dev)
* xystrategy: buffering system interface to xy.
*/
-void
+void
xystrategy(bp)
struct buf *bp;
{
struct xy_softc *xy;
- struct xyc_softc *parent;
- struct buf *wq;
int s, unit;
struct xyc_attach_args xa;
@@ -987,7 +988,7 @@ xystrategy(bp)
/* check for live device */
- if (unit >= xycd.cd_ndevs || (xy = xycd.cd_devs[unit]) == 0 ||
+ if (unit >= xy_cd.cd_ndevs || (xy = xy_cd.cd_devs[unit]) == 0 ||
bp->b_blkno < 0 ||
(bp->b_bcount % xy->sc_dk.dk_label->d_secsize) != 0) {
bp->b_error = EINVAL;
@@ -1059,14 +1060,12 @@ done: /* tells upper layers we are done with this
*
* xycintr: hardware interrupt.
*/
-int
+int
xycintr(v)
void *v;
{
struct xyc_softc *xycsc = v;
- struct xy_softc *xy;
- struct buf *bp;
/* kick the event counter */
@@ -1094,7 +1093,7 @@ xycintr(v)
* xyc_rqinit: fill out the fields of an I/O request
*/
-inline void
+inline void
xyc_rqinit(rq, xyc, xy, md, blk, cnt, db, bp)
struct xy_iorq *rq;
struct xyc_softc *xyc;
@@ -1120,7 +1119,7 @@ xyc_rqinit(rq, xyc, xy, md, blk, cnt, db, bp)
* xyc_rqtopb: load up an IOPB based on an iorq
*/
-void
+void
xyc_rqtopb(iorq, iopb, cmd, subfun)
struct xy_iorq *iorq;
struct xy_iopb *iopb;
@@ -1190,7 +1189,7 @@ int del;
* xyc_cmd: front end for POLL'd and WAIT'd commands. Returns 0 or error.
* note that NORM requests are handled seperately.
*/
-int
+int
xyc_cmd(xycsc, cmd, subfn, unit, block, scnt, dptr, fullmode)
struct xyc_softc *xycsc;
int cmd, subfn, unit, block, scnt;
@@ -1198,8 +1197,7 @@ xyc_cmd(xycsc, cmd, subfn, unit, block, scnt, dptr, fullmode)
int fullmode;
{
- int submode = XY_STATE(fullmode), retry;
- u_long dp;
+ int submode = XY_STATE(fullmode);
struct xy_iorq *iorq = xycsc->ciorq;
struct xy_iopb *iopb = xycsc->ciopb;
@@ -1208,7 +1206,7 @@ xyc_cmd(xycsc, cmd, subfn, unit, block, scnt, dptr, fullmode)
*/
start:
if (submode == XY_SUB_WAIT && XY_STATE(iorq->mode) != XY_SUB_FREE) {
- if (tsleep(iorq, PRIBIO, "xyc_cmd", 0))
+ if (tsleep(iorq, PRIBIO, "xyc_cmd", 0))
return(XY_ERR_FAIL);
goto start;
}
@@ -1241,7 +1239,7 @@ start:
* start a buffer for running
*/
-int
+int
xyc_startbuf(xycsc, xysc, bp)
struct xyc_softc *xycsc;
struct xy_softc *xysc;
@@ -1251,7 +1249,7 @@ xyc_startbuf(xycsc, xysc, bp)
int partno;
struct xy_iorq *iorq;
struct xy_iopb *iopb;
- u_long block, dp;
+ u_long block;
caddr_t dbuf;
iorq = xysc->xyrq;
@@ -1259,7 +1257,7 @@ xyc_startbuf(xycsc, xysc, bp)
/* get buf */
- if (bp == NULL)
+ if (bp == NULL)
panic("xyc_startbuf null buf");
partno = DISKPART(bp->b_dev);
@@ -1273,7 +1271,7 @@ xyc_startbuf(xycsc, xysc, bp)
/*
* load request. we have to calculate the correct block number based
* on partition info.
- *
+ *
* note that iorq points to the buffer as mapped into DVMA space,
* where as the bp->b_data points to its non-DVMA mapping.
*/
@@ -1283,7 +1281,7 @@ xyc_startbuf(xycsc, xysc, bp)
dbuf = kdvma_mapin(bp->b_data, bp->b_bcount, 0);
if (dbuf == NULL) { /* out of DVMA space */
- printf("%s: warning: out of DVMA space\n",
+ printf("%s: warning: out of DVMA space\n",
xycsc->sc_dev.dv_xname);
return (XY_ERR_FAIL); /* XXX: need some sort of
* call-back scheme here? */
@@ -1327,9 +1325,9 @@ xyc_startbuf(xycsc, xysc, bp)
* time, where as a 451 can take them in chains. [the 450 claims it
* can handle chains, but is appears to be buggy...] iopb are allocated
* in DVMA space at boot up time. each disk gets one iopb, and the
- * controller gets one (for POLL and WAIT commands). what happens if
- * the iopb is busy? for i/o type [1], the buffers are queued at the
- * "buff" layer and * picked up later by the interrupt routine. for case
+ * controller gets one (for POLL and WAIT commands). what happens if
+ * the iopb is busy? for i/o type [1], the buffers are queued at the
+ * "buff" layer and * picked up later by the interrupt routine. for case
* [2] we can only be blocked if there is a WAIT type I/O request being
* run. since this can only happen when we are crashing, we wait a sec
* and then steal the IOPB. for case [3] the process can sleep
@@ -1337,7 +1335,7 @@ xyc_startbuf(xycsc, xysc, bp)
*/
-int
+int
xyc_submit_iorq(xycsc, iorq, type)
struct xyc_softc *xycsc;
struct xy_iorq *iorq;
@@ -1348,7 +1346,7 @@ xyc_submit_iorq(xycsc, iorq, type)
u_long iopbaddr;
#ifdef XYC_DEBUG
- printf("xyc_submit_iorq(%s, addr=0x%x, type=%d)\n",
+ printf("xyc_submit_iorq(%s, addr=0x%x, type=%d)\n",
xycsc->sc_dev.dv_xname, iorq, type);
#endif
@@ -1381,12 +1379,12 @@ xyc_submit_iorq(xycsc, iorq, type)
iopb = xyc_chain(xycsc, iorq); /* build chain */
if (iopb == NULL) { /* nothing doing? */
- if (type == XY_SUB_NORM || type == XY_SUB_NOQ)
+ if (type == XY_SUB_NORM || type == XY_SUB_NOQ)
return(XY_ERR_AOK);
panic("xyc_submit_iorq: xyc_chain failed!\n");
}
iopbaddr = (u_long) iopb - DVMA_BASE;
-
+
XYC_GO(xycsc->xyc, iopbaddr);
/* command now running, wrap it up */
@@ -1434,11 +1432,11 @@ struct xy_iorq *iorq;
xycsc->iopbase[XYC_CTLIOPB].done == 0)
iorq = &xycsc->reqs[XYC_CTLIOPB];
}
- /*
+ /*
* special case: if iorq != NULL then we have a POLL or WAIT request.
* we let these take priority and do them first.
*/
- if (iorq) {
+ if (iorq) {
xycsc->xy_chain[0] = iorq;
iorq->iopb->chen = 0;
return(iorq->iopb);
@@ -1479,9 +1477,9 @@ struct xy_iorq *iorq;
*
* programmed i/o driver. this function takes over the computer
* and drains off the polled i/o request. it returns the status of the iorq
- * the caller is interesting in.
+ * the caller is interesting in.
*/
-int
+int
xyc_piodriver(xycsc, iorq)
struct xyc_softc *xycsc;
struct xy_iorq *iorq;
@@ -1490,7 +1488,6 @@ xyc_piodriver(xycsc, iorq)
int nreset = 0;
int retval = 0;
u_long res;
- struct xyc *xyc = xycsc->xyc;
#ifdef XYC_DEBUG
printf("xyc_piodriver(%s, 0x%x)\n", xycsc->sc_dev.dv_xname, iorq);
#endif
@@ -1534,7 +1531,7 @@ xyc_piodriver(xycsc, iorq)
/* start up any bufs that have queued */
- xyc_start(xycsc, NULL);
+ xyc_start(xycsc, NULL);
return (retval);
}
@@ -1543,7 +1540,7 @@ xyc_piodriver(xycsc, iorq)
* xyc_xyreset: reset one drive. NOTE: assumes xyc was just reset.
* we steal iopb[XYC_CTLIOPB] for this, but we put it back when we are done.
*/
-int
+void
xyc_xyreset(xycsc, xysc)
struct xyc_softc *xycsc;
struct xy_softc *xysc;
@@ -1558,7 +1555,7 @@ xyc_xyreset(xycsc, xysc)
xycsc->ciopb->com = XYCMD_RST;
xycsc->ciopb->unit = xysc->xy_drive;
addr = (u_long) xycsc->ciopb - DVMA_BASE;
-
+
XYC_GO(xycsc->xyc, addr);
del = XYC_RESETUSEC;
@@ -1585,7 +1582,7 @@ xyc_xyreset(xycsc, xysc)
* xyc_reset: reset everything: requests are marked as errors except
* a polled request (which is resubmitted)
*/
-int
+int
xyc_reset(xycsc, quiet, blastmode, error, xysc)
struct xyc_softc *xycsc;
int quiet, error;
@@ -1593,7 +1590,7 @@ xyc_reset(xycsc, quiet, blastmode, error, xysc)
struct xy_softc *xysc;
{
- int del = 0, lcv, poll = -1, retval = XY_ERR_AOK;
+ int del = 0, lcv, retval = XY_ERR_AOK;
/* soft reset hardware */
@@ -1619,7 +1616,7 @@ xyc_reset(xycsc, quiet, blastmode, error, xysc)
/* is it active? */
continue;
- if (blastmode == XY_RSET_ALL ||
+ if (blastmode == XY_RSET_ALL ||
blastmode != iorq) {
/* failed */
iorq->errno = error;
@@ -1666,7 +1663,7 @@ xyc_reset(xycsc, quiet, blastmode, error, xysc)
* xyc_start: start waiting buffers
*/
-int
+void
xyc_start(xycsc, iorq)
struct xyc_softc *xycsc;
struct xy_iorq *iorq;
@@ -1687,10 +1684,10 @@ xyc_start(xycsc, iorq)
}
/*
- * xyc_remove_iorq: remove "done" IOPB's.
+ * xyc_remove_iorq: remove "done" IOPB's.
*/
-int
+int
xyc_remove_iorq(xycsc)
struct xyc_softc *xycsc;
@@ -1770,7 +1767,7 @@ xyc_remove_iorq(xycsc)
* at the bad144 sector. to exit bad144 mode, we
* must advance the pointers 1 sector and issue a new
* request if there are still sectors left to process
- *
+ *
*/
XYC_ADVANCE(iorq, 1); /* advance 1 sector */
@@ -1838,7 +1835,7 @@ xyc_remove_iorq(xycsc)
* is in lasterror. also, if iorq->errno == 0, then we recovered
* from that error (otherwise iorq->errno == iorq->lasterror).
*/
-void
+void
xyc_perror(iorq, iopb, still_trying)
struct xy_iorq *iorq;
struct xy_iopb *iopb;
@@ -1871,7 +1868,7 @@ xyc_perror(iorq, iopb, still_trying)
* xyc_error: non-fatal error encountered... recover.
* return AOK if resubmitted, return FAIL if this iopb is done
*/
-int
+int
xyc_error(xycsc, iorq, iopb, comm)
struct xyc_softc *xycsc;
struct xy_iorq *iorq;
@@ -1939,7 +1936,7 @@ xyc_error(xycsc, iorq, iopb, comm)
/*
* xyc_tick: make sure xy is still alive and ticking (err, kicking).
*/
-void
+void
xyc_tick(arg)
void *arg;
@@ -1976,14 +1973,14 @@ xyc_tick(arg)
*
* XXX missing a few commands (see the 7053 driver for ideas)
*/
-int
+int
xyc_ioctlcmd(xy, dev, xio)
struct xy_softc *xy;
dev_t dev;
struct xd_iocmd *xio;
{
- int s, err, rqno, dummy;
+ int s, err, rqno, dummy = 0;
caddr_t dvmabuf = NULL, buf = NULL;
struct xyc_softc *xycsc;
@@ -2018,7 +2015,7 @@ xyc_ioctlcmd(xy, dev, xio)
if (xio->dlen) {
dvmabuf = dvma_malloc(xio->dlen, &buf, M_WAITOK);
if (xio->cmd == XYCMD_WR) {
- if (err = copyin(xio->dptr, buf, xio->dlen)) {
+ if ((err = copyin(xio->dptr, buf, xio->dlen)) != 0) {
dvma_free(dvmabuf, xio->dlen, &buf);
return (err);
}
@@ -2147,7 +2144,7 @@ int errno;
return(XY_ERA_SOFT); /* an FYI error */
- case XY_ERR_WPRO:
+ case XY_ERR_WPRO:
return(XY_ERA_WPRO); /* write protect */
}
diff --git a/sys/arch/sparc/dev/xyreg.h b/sys/arch/sparc/dev/xyreg.h
index 757e73cfd9c..2508bb97a8e 100644
--- a/sys/arch/sparc/dev/xyreg.h
+++ b/sys/arch/sparc/dev/xyreg.h
@@ -1,4 +1,4 @@
-/* $NetBSD: xyreg.h,v 1.1 1995/09/25 20:35:15 chuck Exp $ */
+/* $NetBSD: xyreg.h,v 1.3 1996/03/31 22:39:02 pk Exp $ */
/*
*
@@ -90,9 +90,9 @@ struct xyc {
* add iopb to the chain, and clear AREQ to resume I/O
*
* when the controller is done with a command it may interrupt (if you
- * ask it to) and it will set the XYC_IPND bit in the csr. clear
+ * ask it to) and it will set the XYC_IPND bit in the csr. clear
* the interrupt by writing one to this bit.
- *
+ *
* the format of the iopb is described in section 2.4 of the manual.
* note that it is byte-swapped on the sun.
*/
diff --git a/sys/arch/sparc/dev/xyvar.h b/sys/arch/sparc/dev/xyvar.h
index 5296d9fa202..e7ec8305980 100644
--- a/sys/arch/sparc/dev/xyvar.h
+++ b/sys/arch/sparc/dev/xyvar.h
@@ -1,4 +1,4 @@
-/* $NetBSD: xyvar.h,v 1.2 1996/01/07 22:03:06 thorpej Exp $ */
+/* $NetBSD: xyvar.h,v 1.4 1996/03/31 22:39:04 pk Exp $ */
/*
*
@@ -32,9 +32,9 @@
*/
/*
- * x y v a r . h
+ * x y v a r . h
*
- * this file defines the software structure we use to control the
+ * this file defines the software structure we use to control the
* 450/451.
*
* author: Chuck Cranor <chuck@ccrc.wustl.edu>
@@ -67,7 +67,7 @@ struct xy_iorq {
#define XY_SUB_MASK 0xf0 /* mask bits for state */
#define XY_SUB_FREE 0x00 /* free */
#define XY_SUB_NORM 0x10 /* normal I/O request */
-#define XY_SUB_WAIT 0x20 /* normal I/O request in the
+#define XY_SUB_WAIT 0x20 /* normal I/O request in the
context of a process */
#define XY_SUB_POLL 0x30 /* polled mode */
#define XY_SUB_DONE 0x40 /* not active, but can't be free'd yet */
diff --git a/sys/arch/sparc/dev/zs.c b/sys/arch/sparc/dev/zs.c
index c37e8f97041..ee99cb25b5a 100644
--- a/sys/arch/sparc/dev/zs.c
+++ b/sys/arch/sparc/dev/zs.c
@@ -1,4 +1,4 @@
-/* $NetBSD: zs.c,v 1.28 1995/04/21 15:51:26 pk Exp $ */
+/* $NetBSD: zs.c,v 1.37.4.1 1996/06/02 09:07:55 mrg Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -48,17 +48,14 @@
* Zilog Z8530 (ZSCC) driver.
*
* Runs two tty ports (ttya and ttyb) on zs0,
- * and runs a keyboard and mouse on zs1, and
- * possibly two more tty ports (ttyc and ttyd) on zs2.
+ * and runs a keyboard and mouse on zs1.
*
* This driver knows far too much about chip to usage mappings.
*/
-#include "zs.h"
-
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/proc.h>
#include <sys/device.h>
-#include <sys/conf.h>
#include <sys/file.h>
#include <sys/ioctl.h>
#include <sys/malloc.h>
@@ -66,22 +63,26 @@
#include <sys/time.h>
#include <sys/kernel.h>
#include <sys/syslog.h>
+#include <sys/conf.h>
#include <machine/autoconf.h>
+#include <machine/conf.h>
#include <machine/cpu.h>
#include <sparc/sparc/vaddrs.h>
#include <sparc/sparc/auxreg.h>
+
#include <machine/kbd.h>
#include <dev/ic/z8530reg.h>
+
#include <sparc/dev/zsvar.h>
#ifdef KGDB
#include <machine/remote-sl.h>
#endif
-#define ZSMAJOR 12 /* XXX */
+#define ZSMAJOR 12 /* XXX */
#define ZS_KBD 2 /* XXX */
#define ZS_MOUSE 3 /* XXX */
@@ -103,20 +104,25 @@
#endif
/*
- * Software state per found chip.
+ * Software state per found chip. This would be called `zs_softc',
+ * but the previous driver had a rather different zs_softc....
*/
-struct zs_softc {
- struct device sc_dev; /* base device */
- volatile struct zsdevice *sc_zs; /* chip registers */
- struct evcnt sc_intrcnt;
- struct zs_chanstate sc_cs[2]; /* chan A/B software state */
+struct zsinfo {
+ struct device zi_dev; /* base device */
+ volatile struct zsdevice *zi_zs;/* chip registers */
+ struct zs_chanstate zi_cs[2]; /* channel A and B software state */
};
/* Definition of the driver for autoconfig. */
static int zsmatch __P((struct device *, void *, void *));
static void zsattach __P((struct device *, struct device *, void *));
-struct cfdriver zscd = {
- NULL, "zs", zsmatch, zsattach, DV_TTY, sizeof(struct zs_softc)
+
+struct cfattach zs_ca = {
+ sizeof(struct zsinfo), zsmatch, zsattach
+};
+
+struct cfdriver zs_cd = {
+ NULL, "zs", DV_TTY
};
/* Interrupt handlers. */
@@ -131,31 +137,41 @@ struct zs_chanstate *zslist;
static void zsiopen __P((struct tty *));
static void zsiclose __P((struct tty *));
static void zsstart __P((struct tty *));
-void zsstop __P((struct tty *, int));
static int zsparam __P((struct tty *, struct termios *));
/* Routines purely local to this driver. */
static int zs_getspeed __P((volatile struct zschan *));
+#ifdef KGDB
static void zs_reset __P((volatile struct zschan *, int, int));
+#endif
static void zs_modem __P((struct zs_chanstate *, int));
static void zs_loadchannelregs __P((volatile struct zschan *, u_char *));
/* Console stuff. */
static struct tty *zs_ctty; /* console `struct tty *' */
static int zs_consin = -1, zs_consout = -1;
-static int zscnputc __P((int)); /* console putc function */
+static void zscnputc __P((int)); /* console putc function */
static volatile struct zschan *zs_conschan;
-static struct tty *zs_checkcons __P((struct zs_softc *, int,
- struct zs_chanstate *));
+static struct tty *zs_checkcons __P((struct zsinfo *, int, struct zs_chanstate *));
#ifdef KGDB
/* KGDB stuff. Must reboot to change zs_kgdbunit. */
extern int kgdb_dev, kgdb_rate;
static int zs_kgdb_savedspeed;
static void zs_checkkgdb __P((int, struct zs_chanstate *, struct tty *));
+void zskgdb __P((int));
+static int zs_kgdb_getc __P((void *));
+static void zs_kgdb_putc __P((void *, int));
#endif
-extern void *findzs __P((int));
+static int zsrint __P((struct zs_chanstate *, volatile struct zschan *));
+static int zsxint __P((struct zs_chanstate *, volatile struct zschan *));
+static int zssint __P((struct zs_chanstate *, volatile struct zschan *));
+
+void zsabort __P((void));
+static void zsoverrun __P((int, long *, char *));
+
+#include "zs.h" /* XXX: */
static volatile struct zsdevice *zsaddr[NZS]; /* XXX, but saves work */
/*
@@ -174,10 +190,13 @@ int zshardscope;
int zsshortcuts; /* number of "shortcut" software interrupts */
#ifdef SUN4
-static u_char
+static u_int zs_read __P((volatile struct zschan *, u_int reg));
+static u_int zs_write __P((volatile struct zschan *, u_int, u_int));
+
+static u_int
zs_read(zc, reg)
volatile struct zschan *zc;
- u_char reg;
+ u_int reg;
{
u_char val;
@@ -188,10 +207,10 @@ zs_read(zc, reg)
return val;
}
-static u_char
+static u_int
zs_write(zc, reg, val)
volatile struct zschan *zc;
- u_char reg, val;
+ u_int reg, val;
{
zc->zc_csr = reg;
ZS_DELAY();
@@ -216,7 +235,8 @@ zsmatch(parent, vcf, aux)
if (strcmp(cf->cf_driver->cd_name, ra->ra_name))
return (0);
- if (ca->ca_bustype==BUS_MAIN && cputyp!=CPU_SUN4)
+ if ((ca->ca_bustype == BUS_MAIN && !CPU_ISSUN4) ||
+ (ca->ca_bustype == BUS_OBIO && CPU_ISSUN4M))
return (getpropint(ra->ra_node, "slave", -2) == cf->cf_unit);
ra->ra_len = NBPG;
return (probeget(ra->ra_vaddr, 1) != -1);
@@ -235,7 +255,7 @@ zsattach(parent, dev, aux)
void *aux;
{
register int zs = dev->dv_unit, unit;
- register struct zs_softc *sc;
+ register struct zsinfo *zi;
register struct zs_chanstate *cs;
register volatile struct zsdevice *addr;
register struct tty *tp, *ctp;
@@ -246,7 +266,7 @@ zsattach(parent, dev, aux)
int ringsize;
if ((addr = zsaddr[zs]) == NULL)
- addr = zsaddr[zs] = (volatile struct zsdevice *)findzs(zs);
+ addr = zsaddr[zs] = findzs(zs);
if (ca->ca_bustype==BUS_MAIN)
if ((void *)addr != ra->ra_vaddr)
panic("zsattach");
@@ -263,36 +283,31 @@ zsattach(parent, dev, aux)
intr_establish(PIL_TTY, &levelsoft);
} else if (pri != prevpri)
panic("broken zs interrupt scheme");
- sc = (struct zs_softc *)dev;
- evcnt_attach(&sc->sc_dev, "intr", &sc->sc_intrcnt);
- sc->sc_zs = addr;
+ zi = (struct zsinfo *)dev;
+ zi->zi_zs = addr;
unit = zs * 2;
- cs = sc->sc_cs;
+ cs = zi->zi_cs;
/* link into interrupt list with order (A,B) (B=A+1) */
cs[0].cs_next = &cs[1];
- cs[0].cs_sc = sc;
cs[1].cs_next = zslist;
- cs[1].cs_sc = sc;
zslist = cs;
cs->cs_unit = unit;
cs->cs_speed = zs_getspeed(&addr->zs_chan[ZS_CHAN_A]);
cs->cs_zc = &addr->zs_chan[ZS_CHAN_A];
- if ((ctp = zs_checkcons(sc, unit, cs)) != NULL)
+ if ((ctp = zs_checkcons(zi, unit, cs)) != NULL)
tp = ctp;
else {
tp = ttymalloc();
tp->t_dev = makedev(ZSMAJOR, unit);
tp->t_oproc = zsstart;
tp->t_param = zsparam;
- }
- cs->cs_ttyp = tp;
#ifdef KGDB
- if (ctp == NULL)
zs_checkkgdb(unit, cs, tp);
#endif
- ringsize = 4096;
+ }
+ cs->cs_ttyp = tp;
if (unit == ZS_KBD) {
/*
* Keyboard: tell /dev/kbd driver how to talk to us.
@@ -302,45 +317,51 @@ zsattach(parent, dev, aux)
kbd_serial(tp, zsiopen, zsiclose);
cs->cs_conk = 1; /* do L1-A processing */
ringsize = 128;
+ } else {
+ if (tp != ctp)
+ tty_attach(tp);
+ ringsize = 4096;
}
+
cs->cs_ringmask = ringsize - 1;
cs->cs_rbuf = malloc((u_long)ringsize * sizeof(*cs->cs_rbuf),
- M_DEVBUF, M_NOWAIT);
+ M_DEVBUF, M_NOWAIT);
unit++;
cs++;
-
cs->cs_unit = unit;
cs->cs_speed = zs_getspeed(&addr->zs_chan[ZS_CHAN_B]);
cs->cs_zc = &addr->zs_chan[ZS_CHAN_B];
- if ((ctp = zs_checkcons(sc, unit, cs)) != NULL)
+ if ((ctp = zs_checkcons(zi, unit, cs)) != NULL)
tp = ctp;
else {
tp = ttymalloc();
tp->t_dev = makedev(ZSMAJOR, unit);
tp->t_oproc = zsstart;
tp->t_param = zsparam;
- }
- cs->cs_ttyp = tp;
-
#ifdef KGDB
- if (ctp == NULL)
zs_checkkgdb(unit, cs, tp);
#endif
- ringsize = 4096;
+ }
+ cs->cs_ttyp = tp;
if (unit == ZS_MOUSE) {
/*
* Mouse: tell /dev/mouse driver how to talk to us.
*/
- tp->t_ispeed = tp->t_ospeed = B1200;
+ tp->t_ispeed = tp->t_ospeed = cs->cs_speed;
tp->t_cflag = CS8;
ms_serial(tp, zsiopen, zsiclose);
ringsize = 128;
+ } else {
+ if (tp != ctp)
+ tty_attach(tp);
+ ringsize = 4096;
}
cs->cs_ringmask = ringsize - 1;
cs->cs_rbuf = malloc((u_long)ringsize * sizeof(*cs->cs_rbuf),
- M_DEVBUF, M_NOWAIT);
+ M_DEVBUF, M_NOWAIT);
}
+#ifdef KGDB
/*
* Put a channel in a known state. Interrupts may be left disabled
* or enabled, as desired.
@@ -376,6 +397,7 @@ zs_reset(zc, inten, speed)
reg[13] = tconst >> 8;
zs_loadchannelregs(zc, reg);
}
+#endif
/*
* Declare the given tty (which is in fact &cons) as a console input
@@ -391,9 +413,8 @@ zsconsole(tp, unit, out, fnstop)
register struct tty *tp;
register int unit;
int out;
- void (**fnstop) __P((struct tty *, int));
+ int (**fnstop) __P((struct tty *, int));
{
- extern int (*v_putc)();
int zs;
volatile struct zsdevice *addr;
@@ -403,7 +424,7 @@ zsconsole(tp, unit, out, fnstop)
zs_consout = unit;
zs = unit >> 1;
if ((addr = zsaddr[zs]) == NULL)
- addr = zsaddr[zs] = (volatile struct zsdevice *)findzs(zs);
+ addr = zsaddr[zs] = findzs(zs);
zs_conschan = (unit & 1) == 0 ? &addr->zs_chan[ZS_CHAN_A] :
&addr->zs_chan[ZS_CHAN_B];
v_putc = zscnputc;
@@ -417,7 +438,7 @@ zsconsole(tp, unit, out, fnstop)
/*
* Polled console output putchar.
*/
-static int
+static void
zscnputc(c)
int c;
{
@@ -431,10 +452,8 @@ zscnputc(c)
* lowering current ipl. Need a better way.
*/
s = splhigh();
-#ifdef SUN4C /* XXX */
- if (cputyp==CPU_SUN4C && s <= (12 << 8))
+ if (CPU_ISSUN4C && s <= (12 << 8)) /* XXX */
(void) splzs();
-#endif
while ((zc->zc_csr & ZSRR0_TX_READY) == 0)
ZS_DELAY();
zc->zc_data = c;
@@ -447,15 +466,15 @@ zscnputc(c)
* needed. Return console tty if it is to receive console input.
*/
static struct tty *
-zs_checkcons(sc, unit, cs)
- struct zs_softc *sc;
+zs_checkcons(zi, unit, cs)
+ struct zsinfo *zi;
int unit;
struct zs_chanstate *cs;
{
register struct tty *tp;
char *i, *o;
- if ((tp = zs_ctty) == NULL)
+ if ((tp = zs_ctty) == NULL) /* XXX */
return (0);
i = zs_consin == unit ? "input" : NULL;
o = zs_consout == unit ? "output" : NULL;
@@ -483,7 +502,7 @@ zs_checkcons(sc, unit, cs)
tp->t_oproc = zsstart;
}
printf("%s%c: console %s\n",
- sc->sc_dev.dv_xname, (unit & 1) + 'a', i ? (o ? "i/o" : i) : o);
+ zi->zi_dev.dv_xname, (unit & 1) + 'a', i ? (o ? "i/o" : i) : o);
cs->cs_consio = 1;
cs->cs_brkabort = 1;
return (tp);
@@ -568,13 +587,13 @@ zsopen(dev, flags, mode, p)
{
register struct tty *tp;
register struct zs_chanstate *cs;
- struct zs_softc *sc;
+ struct zsinfo *zi;
int unit = minor(dev), zs = unit >> 1, error, s;
- if (zs >= zscd.cd_ndevs || (sc = zscd.cd_devs[zs]) == NULL ||
+ if (zs >= zs_cd.cd_ndevs || (zi = zs_cd.cd_devs[zs]) == NULL ||
unit == ZS_KBD || unit == ZS_MOUSE)
return (ENXIO);
- cs = &sc->sc_cs[unit & 1];
+ cs = &zi->zi_cs[unit & 1];
if (cs->cs_consio)
return (ENXIO); /* ??? */
tp = cs->cs_ttyp;
@@ -595,7 +614,7 @@ zsopen(dev, flags, mode, p)
return (EBUSY);
}
error = 0;
- while (1) {
+ for (;;) {
register int rr0;
/* loop, turning on the device, until carrier present */
@@ -609,8 +628,9 @@ zsopen(dev, flags, mode, p)
tp->t_state & TS_CARR_ON)
break;
tp->t_state |= TS_WOPEN;
- if (error = ttysleep(tp, (caddr_t)&tp->t_rawq, TTIPRI | PCATCH,
- ttopen, 0)) {
+ error = ttysleep(tp, (caddr_t)&tp->t_rawq, TTIPRI | PCATCH,
+ ttopen, 0);
+ if (error) {
if (!(tp->t_state & TS_ISOPEN)) {
zs_modem(cs, 0);
tp->t_state &= ~TS_WOPEN;
@@ -640,11 +660,11 @@ zsclose(dev, flags, mode, p)
{
register struct zs_chanstate *cs;
register struct tty *tp;
- struct zs_softc *sc;
+ struct zsinfo *zi;
int unit = minor(dev), s;
- sc = zscd.cd_devs[unit >> 1];
- cs = &sc->sc_cs[unit & 1];
+ zi = zs_cd.cd_devs[unit >> 1];
+ cs = &zi->zi_cs[unit & 1];
tp = cs->cs_ttyp;
linesw[tp->t_line].l_close(tp, flags);
if (tp->t_cflag & HUPCL || tp->t_state & TS_WOPEN ||
@@ -682,15 +702,16 @@ zsread(dev, uio, flags)
int flags;
{
register struct zs_chanstate *cs;
- register struct zs_softc *sc;
+ register struct zsinfo *zi;
register struct tty *tp;
int unit = minor(dev);
- sc = zscd.cd_devs[unit >> 1];
- cs = &sc->sc_cs[unit & 1];
+ zi = zs_cd.cd_devs[unit >> 1];
+ cs = &zi->zi_cs[unit & 1];
tp = cs->cs_ttyp;
return (linesw[tp->t_line].l_read(tp, uio, flags));
+
}
int
@@ -700,12 +721,12 @@ zswrite(dev, uio, flags)
int flags;
{
register struct zs_chanstate *cs;
- register struct zs_softc *sc;
+ register struct zsinfo *zi;
register struct tty *tp;
int unit = minor(dev);
- sc = zscd.cd_devs[unit >> 1];
- cs = &sc->sc_cs[unit & 1];
+ zi = zs_cd.cd_devs[unit >> 1];
+ cs = &zi->zi_cs[unit & 1];
tp = cs->cs_ttyp;
return (linesw[tp->t_line].l_write(tp, uio, flags));
@@ -716,18 +737,15 @@ zstty(dev)
dev_t dev;
{
register struct zs_chanstate *cs;
- register struct zs_softc *sc;
+ register struct zsinfo *zi;
int unit = minor(dev);
- sc = zscd.cd_devs[unit >> 1];
- cs = &sc->sc_cs[unit & 1];
+ zi = zs_cd.cd_devs[unit >> 1];
+ cs = &zi->zi_cs[unit & 1];
return (cs->cs_ttyp);
-}
-static int zsrint __P((struct zs_chanstate *, volatile struct zschan *));
-static int zsxint __P((struct zs_chanstate *, volatile struct zschan *));
-static int zssint __P((struct zs_chanstate *, volatile struct zschan *));
+}
/*
* ZS hardware interrupt. Scan all ZS channels. NB: we know here that
@@ -746,64 +764,57 @@ zshard(intrarg)
void *intrarg;
{
register struct zs_chanstate *a;
- struct zs_softc *sc;
#define b (a + 1)
register volatile struct zschan *zc;
register int rr3, intflags = 0, v, i, ringmask;
-
-#define ZSHARD_NEED_SOFTINTR 1
-#define ZSHARD_WAS_SERVICED 2
-#define ZSHARD_CHIP_GOTINTR 4
+ static int zsrint(struct zs_chanstate *, volatile struct zschan *);
+ static int zsxint(struct zs_chanstate *, volatile struct zschan *);
+ static int zssint(struct zs_chanstate *, volatile struct zschan *);
for (a = zslist; a != NULL; a = b->cs_next) {
ringmask = a->cs_ringmask;
rr3 = ZS_READ(a->cs_zc, 3);
if (rr3 & (ZSRR3_IP_A_RX|ZSRR3_IP_A_TX|ZSRR3_IP_A_STAT)) {
- intflags |= (ZSHARD_CHIP_GOTINTR|ZSHARD_WAS_SERVICED);
+ intflags |= 2;
zc = a->cs_zc;
i = a->cs_rbput;
if (rr3 & ZSRR3_IP_A_RX && (v = zsrint(a, zc)) != 0) {
a->cs_rbuf[i++ & ringmask] = v;
- intflags |= ZSHARD_NEED_SOFTINTR;
+ intflags |= 1;
}
if (rr3 & ZSRR3_IP_A_TX && (v = zsxint(a, zc)) != 0) {
a->cs_rbuf[i++ & ringmask] = v;
- intflags |= ZSHARD_NEED_SOFTINTR;
+ intflags |= 1;
}
if (rr3 & ZSRR3_IP_A_STAT && (v = zssint(a, zc)) != 0) {
a->cs_rbuf[i++ & ringmask] = v;
- intflags |= ZSHARD_NEED_SOFTINTR;
+ intflags |= 1;
}
a->cs_rbput = i;
}
if (rr3 & (ZSRR3_IP_B_RX|ZSRR3_IP_B_TX|ZSRR3_IP_B_STAT)) {
- intflags |= (ZSHARD_CHIP_GOTINTR|ZSHARD_WAS_SERVICED);
+ intflags |= 2;
zc = b->cs_zc;
i = b->cs_rbput;
if (rr3 & ZSRR3_IP_B_RX && (v = zsrint(b, zc)) != 0) {
b->cs_rbuf[i++ & ringmask] = v;
- intflags |= ZSHARD_NEED_SOFTINTR;
+ intflags |= 1;
}
if (rr3 & ZSRR3_IP_B_TX && (v = zsxint(b, zc)) != 0) {
b->cs_rbuf[i++ & ringmask] = v;
- intflags |= ZSHARD_NEED_SOFTINTR;
+ intflags |= 1;
}
if (rr3 & ZSRR3_IP_B_STAT && (v = zssint(b, zc)) != 0) {
b->cs_rbuf[i++ & ringmask] = v;
- intflags |= ZSHARD_NEED_SOFTINTR;
+ intflags |= 1;
}
b->cs_rbput = i;
}
- if (intflags & ZSHARD_CHIP_GOTINTR) {
- a->cs_sc->sc_intrcnt.ev_count++;
- intflags &= ~ZSHARD_CHIP_GOTINTR;
- }
}
#undef b
- if (intflags & ZSHARD_NEED_SOFTINTR) {
-#if defined(SUN4C) || defined(SUN4M)
- if (cputyp==CPU_SUN4M || cputyp==CPU_SUN4C) {
+ if (intflags & 1) {
+ if (CPU_ISSUN4COR4M) {
/* XXX -- but this will go away when zshard moves to locore.s */
struct clockframe *p = intrarg;
@@ -817,10 +828,15 @@ zshard(intrarg)
return (zssoft(intrarg));
}
}
+
+#if defined(SUN4M)
+ if (CPU_ISSUN4M)
+ raise(0, PIL_TTY);
+ else
#endif
- ienab_bis(IE_ZSSOFT);
+ ienab_bis(IE_ZSSOFT);
}
- return (intflags & ZSHARD_WAS_SERVICED);
+ return (intflags & 2);
}
static int
@@ -828,11 +844,9 @@ zsrint(cs, zc)
register struct zs_chanstate *cs;
register volatile struct zschan *zc;
{
- register u_int c;
+ register int c = zc->zc_data;
- c = zc->zc_data;
ZS_DELAY();
-
if (cs->cs_conk) {
register struct conk_state *conk = &zsconk_state;
@@ -857,7 +871,7 @@ zsrint(cs, zc)
}
}
#ifdef KGDB
- if (c == FRAME_START && cs->cs_kgdb &&
+ if (c == FRAME_START && cs->cs_kgdb &&
(cs->cs_ttyp->t_state & TS_ISOPEN) == 0) {
zskgdb(cs->cs_unit);
goto clearit;
@@ -910,7 +924,7 @@ zssint(cs, zc)
register struct zs_chanstate *cs;
register volatile struct zschan *zc;
{
- register u_int rr0;
+ register int rr0;
rr0 = zc->zc_csr;
ZS_DELAY();
@@ -938,22 +952,21 @@ zssint(cs, zc)
}
}
if ((rr0 & ZSRR0_BREAK) && cs->cs_brkabort) {
-#ifdef SUN4
/*
* XXX This might not be necessary. Test and
* delete if it isn't.
*/
- if (cputyp==CPU_SUN4) {
+ if (CPU_ISSUN4) {
while (zc->zc_csr & ZSRR0_BREAK)
ZS_DELAY();
}
-#endif
zsabort();
return (0);
}
return (ZRING_MAKE(ZRING_SINT, rr0));
}
+void
zsabort()
{
@@ -970,6 +983,7 @@ zsabort()
* KGDB framing character received: enter kernel debugger. This probably
* should time out after a few seconds to avoid hanging on spurious input.
*/
+void
zskgdb(unit)
int unit;
{
@@ -1133,10 +1147,10 @@ zsioctl(dev, cmd, data, flag, p)
struct proc *p;
{
int unit = minor(dev);
- struct zs_softc *sc = zscd.cd_devs[unit >> 1];
- register struct tty *tp = sc->sc_cs[unit & 1].cs_ttyp;
+ struct zsinfo *zi = zs_cd.cd_devs[unit >> 1];
+ register struct zs_chanstate *cs = &zi->zi_cs[unit & 1];
+ register struct tty *tp = cs->cs_ttyp;
register int error, s;
- register struct zs_chanstate *cs = &sc->sc_cs[unit & 1];
error = linesw[tp->t_line].l_ioctl(tp, cmd, data, flag, p);
if (error >= 0)
@@ -1173,7 +1187,7 @@ zsioctl(dev, cmd, data, flag, p)
break;
}
case TIOCSFLAGS: {
- int userbits, driverbits = 0;
+ int userbits;
error = suser(p->p_ucred, &p->p_acflag);
if (error != 0)
@@ -1230,16 +1244,10 @@ zsioctl(dev, cmd, data, flag, p)
case TIOCCDTR:
zs_modem(cs, 0);
break;
- case TIOCMGET:
- /* XXX: fixme */
- *(int *)data = TIOCM_CAR | TIOCM_CTS | TIOCM_DTR | TIOCM_RTS;
- return (0);
case TIOCMSET:
- /* XXX: fixme */
- zs_modem(cs, *(int *)data & (TIOCM_DTR|TIOCM_RTS));
- return (0);
case TIOCMBIS:
case TIOCMBIC:
+ case TIOCMGET:
default:
return (ENOTTY);
}
@@ -1256,9 +1264,9 @@ zsstart(tp)
register struct zs_chanstate *cs;
register int s, nch;
int unit = minor(tp->t_dev);
- struct zs_softc *sc = zscd.cd_devs[unit >> 1];
+ struct zsinfo *zi = zs_cd.cd_devs[unit >> 1];
- cs = &sc->sc_cs[unit & 1];
+ cs = &zi->zi_cs[unit & 1];
s = spltty();
/*
@@ -1310,16 +1318,16 @@ out:
/*
* Stop output, e.g., for ^S or output flush.
*/
-void
+int
zsstop(tp, flag)
register struct tty *tp;
int flag;
{
register struct zs_chanstate *cs;
register int s, unit = minor(tp->t_dev);
- struct zs_softc *sc = zscd.cd_devs[unit >> 1];
+ struct zsinfo *zi = zs_cd.cd_devs[unit >> 1];
- cs = &sc->sc_cs[unit & 1];
+ cs = &zi->zi_cs[unit & 1];
s = splzs();
if (tp->t_state & TS_BUSY) {
/*
@@ -1330,6 +1338,7 @@ zsstop(tp, flag)
tp->t_state |= TS_FLUSH;
}
splx(s);
+ return 0;
}
/*
@@ -1344,8 +1353,8 @@ zsparam(tp, t)
register struct termios *t;
{
int unit = minor(tp->t_dev);
- struct zs_softc *sc = zscd.cd_devs[unit >> 1];
- register struct zs_chanstate *cs = &sc->sc_cs[unit & 1];
+ struct zsinfo *zi = zs_cd.cd_devs[unit >> 1];
+ register struct zs_chanstate *cs = &zi->zi_cs[unit & 1];
register int tmp, tmp5, cflag, s;
/*
@@ -1519,13 +1528,10 @@ zs_kgdb_getc(arg)
void *arg;
{
register volatile struct zschan *zc = (volatile struct zschan *)arg;
- u_char c;
while ((zc->zc_csr & ZSRR0_RX_READY) == 0)
ZS_DELAY();
- c = zc->zc_data;
- ZS_DELAY();
- return c;
+ return (zc->zc_data);
}
/*
@@ -1567,7 +1573,7 @@ zs_kgdb_init()
}
zs = unit >> 1;
if ((addr = zsaddr[zs]) == NULL)
- addr = zsaddr[zs] = (volatile struct zsdevice *)findzs(zs);
+ addr = zsaddr[zs] = findzs(zs);
unit &= 1;
zc = unit == 0 ? &addr->zs_chan[ZS_CHAN_A] : &addr->zs_chan[ZS_CHAN_B];
zs_kgdb_savedspeed = zs_getspeed(zc);
diff --git a/sys/arch/sparc/dev/zsvar.h b/sys/arch/sparc/dev/zsvar.h
index 3fef4d3981e..3234fbc7537 100644
--- a/sys/arch/sparc/dev/zsvar.h
+++ b/sys/arch/sparc/dev/zsvar.h
@@ -1,4 +1,4 @@
-/* $NetBSD: zsvar.h,v 1.6 1996/01/24 19:52:57 gwr Exp $ */
+/* $NetBSD: zsvar.h,v 1.8 1996/03/31 22:39:08 pk Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -91,8 +91,7 @@ struct zsdevice {
#define ZRING_MAKE(t, v) ((t) | (v) << 8)
struct zs_chanstate {
- struct zs_chanstate *cs_next; /* linked list for zshard() */
- struct zs_softc *cs_sc; /* points to my softc */
+ struct zs_chanstate *cs_next; /* linked list for zshard() */
volatile struct zschan *cs_zc; /* points to hardware regs */
int cs_unit; /* unit number */
struct tty *cs_ttyp; /* ### */
@@ -163,17 +162,22 @@ struct zs_chanstate {
* On the SparcStation the 1.6 microsecond recovery time is
* handled in hardware. On the older Sun4 machine it isn't, and
* software must deal with the problem.
+ *
+ * However, it *is* a problem on some Sun4m's (i.e. the SS20) (XXX: why?).
+ * Thus we leave in the delay.
+ *
+ * XXX: (ABB) Think about this more.
*/
-#ifdef SUN4
+#if defined(SUN4)
+
#define ZS_READ(c, r) zs_read(c, r)
#define ZS_WRITE(c, r, v) zs_write(c, r, v)
-#if defined(SUN4C) || defined(SUN4M)
-#define ZS_DELAY() (cputyp == CPU_SUN4 ? delay(2) : 0)
-#else
-#define ZS_DELAY() delay(2)
-#endif
-#else
+#define ZS_DELAY() (CPU_ISSUN4C ? (0) : delay(1))
+
+#else /* SUN4 */
+
#define ZS_READ(c, r) ((c)->zc_csr = (r), (c)->zc_csr)
#define ZS_WRITE(c, r, v) ((c)->zc_csr = (r), (c)->zc_csr = (v))
-#define ZS_DELAY()
-#endif
+#define ZS_DELAY() (CPU_ISSUN4M ? delay(1) : 0)
+
+#endif /* SUN4 */
diff --git a/sys/arch/sparc/fpu/fpu.c b/sys/arch/sparc/fpu/fpu.c
index 4aa8fba2a29..f5212b9ae2d 100644
--- a/sys/arch/sparc/fpu/fpu.c
+++ b/sys/arch/sparc/fpu/fpu.c
@@ -1,4 +1,4 @@
-/* $NetBSD: fpu.c,v 1.2 1994/11/20 20:52:33 deraadt Exp $ */
+/* $NetBSD: fpu.c,v 1.3 1996/03/14 19:41:49 christos Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -49,11 +49,13 @@
#include <sys/signal.h>
#include <sys/systm.h>
#include <sys/syslog.h>
+#include <sys/signalvar.h>
#include <machine/instr.h>
#include <machine/reg.h>
#include <sparc/fpu/fpu_emu.h>
+#include <sparc/fpu/fpu_extern.h>
/*
* fpu_execute returns the following error numbers (0 = no error):
@@ -93,6 +95,7 @@ static u_char fpu_codes[] = {
* nor FBfcc instructions. Experiments with `crashme' prove that
* unknown FPops do enter the queue, however.
*/
+void
fpu_cleanup(p, fs)
register struct proc *p;
register struct fpstate *fs;
diff --git a/sys/arch/sparc/fpu/fpu_add.c b/sys/arch/sparc/fpu/fpu_add.c
index d85bb57b3a3..1aa9225c01e 100644
--- a/sys/arch/sparc/fpu/fpu_add.c
+++ b/sys/arch/sparc/fpu/fpu_add.c
@@ -1,4 +1,4 @@
-/* $NetBSD: fpu_add.c,v 1.2 1994/11/20 20:52:34 deraadt Exp $ */
+/* $NetBSD: fpu_add.c,v 1.3 1996/03/14 19:41:52 christos Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -51,11 +51,16 @@
*/
#include <sys/types.h>
+#ifdef DIAGNOSTIC
+#include <sys/systm.h>
+#endif
#include <machine/reg.h>
+#include <machine/instr.h>
#include <sparc/fpu/fpu_arith.h>
#include <sparc/fpu/fpu_emu.h>
+#include <sparc/fpu/fpu_extern.h>
struct fpn *
fpu_add(fe)
diff --git a/sys/arch/sparc/fpu/fpu_explode.c b/sys/arch/sparc/fpu/fpu_explode.c
index cc992fdad89..ceaa896c02d 100644
--- a/sys/arch/sparc/fpu/fpu_explode.c
+++ b/sys/arch/sparc/fpu/fpu_explode.c
@@ -1,4 +1,4 @@
-/* $NetBSD: fpu_explode.c,v 1.2 1994/11/20 20:52:41 deraadt Exp $ */
+/* $NetBSD: fpu_explode.c,v 1.3 1996/03/14 19:41:54 christos Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -50,6 +50,7 @@
*/
#include <sys/types.h>
+#include <sys/systm.h>
#include <machine/ieee.h>
#include <machine/instr.h>
@@ -57,6 +58,7 @@
#include <sparc/fpu/fpu_arith.h>
#include <sparc/fpu/fpu_emu.h>
+#include <sparc/fpu/fpu_extern.h>
/*
* N.B.: in all of the following, we assume the FP format is
diff --git a/sys/arch/sparc/fpu/fpu_extern.h b/sys/arch/sparc/fpu/fpu_extern.h
new file mode 100644
index 00000000000..0f2ed79668d
--- /dev/null
+++ b/sys/arch/sparc/fpu/fpu_extern.h
@@ -0,0 +1,76 @@
+/* $NetBSD: fpu_extern.h,v 1.1 1996/03/14 19:41:56 christos Exp $ */
+
+/*
+ * Copyright (c) 1995 Christos Zoulas. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by Christos Zoulas.
+ * 4. The name of the author may not be used to endorse or promote products
+ * derived from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
+ * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
+ * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
+ * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
+ * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
+ * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+struct proc;
+struct fpstate;
+struct trapframe;
+union instr;
+struct fpemu;
+struct fpn;
+
+/* fpu.c */
+void fpu_cleanup __P((struct proc *, struct fpstate *));
+int fpu_emulate __P((struct proc *, struct trapframe *, struct fpstate *));
+int fpu_execute __P((struct fpemu *, union instr));
+
+/* fpu_add.c */
+struct fpn *fpu_add __P((struct fpemu *));
+
+/* fpu_compare.c */
+void fpu_compare __P((struct fpemu *, int));
+
+/* fpu_div.c */
+struct fpn *fpu_div __P((struct fpemu *));
+
+/* fpu_explode.c */
+int fpu_itof __P((struct fpn *, u_int));
+int fpu_stof __P((struct fpn *, u_int));
+int fpu_dtof __P((struct fpn *, u_int, u_int ));
+int fpu_xtof __P((struct fpn *, u_int, u_int , u_int , u_int ));
+void fpu_explode __P((struct fpemu *, struct fpn *, int, int ));
+
+/* fpu_implode.c */
+u_int fpu_ftoi __P((struct fpemu *, struct fpn *));
+u_int fpu_ftos __P((struct fpemu *, struct fpn *));
+u_int fpu_ftod __P((struct fpemu *, struct fpn *, u_int *));
+u_int fpu_ftox __P((struct fpemu *, struct fpn *, u_int *));
+void fpu_implode __P((struct fpemu *, struct fpn *, int, u_int *));
+
+/* fpu_mul.c */
+struct fpn *fpu_mul __P((struct fpemu *));
+
+/* fpu_sqrt.c */
+struct fpn *fpu_sqrt __P((struct fpemu *));
+
+/* fpu_subr.c */
+int fpu_shr __P((register struct fpn *, register int));
+void fpu_norm __P((register struct fpn *));
+struct fpn *fpu_newnan __P((register struct fpemu *));
diff --git a/sys/arch/sparc/fpu/fpu_implode.c b/sys/arch/sparc/fpu/fpu_implode.c
index 492b308f89a..753b0ba9973 100644
--- a/sys/arch/sparc/fpu/fpu_implode.c
+++ b/sys/arch/sparc/fpu/fpu_implode.c
@@ -1,4 +1,4 @@
-/* $NetBSD: fpu_implode.c,v 1.2 1994/11/20 20:52:42 deraadt Exp $ */
+/* $NetBSD: fpu_implode.c,v 1.3 1996/03/14 19:41:59 christos Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -50,6 +50,7 @@
*/
#include <sys/types.h>
+#include <sys/systm.h>
#include <machine/ieee.h>
#include <machine/instr.h>
@@ -57,6 +58,10 @@
#include <sparc/fpu/fpu_arith.h>
#include <sparc/fpu/fpu_emu.h>
+#include <sparc/fpu/fpu_extern.h>
+
+static int round __P((register struct fpemu *, register struct fpn *));
+static int toinf __P((struct fpemu *, int));
/*
* Round a number (algorithm from Motorola MC68882 manual, modified for
@@ -74,7 +79,7 @@ static int
round(register struct fpemu *fe, register struct fpn *fp)
{
register u_int m0, m1, m2, m3;
- register int gr, s, ret;
+ register int gr, s;
m0 = fp->fp_mant[0];
m1 = fp->fp_mant[1];
diff --git a/sys/arch/sparc/fpu/fpu_subr.c b/sys/arch/sparc/fpu/fpu_subr.c
index 685ae2de34f..45b80c219ba 100644
--- a/sys/arch/sparc/fpu/fpu_subr.c
+++ b/sys/arch/sparc/fpu/fpu_subr.c
@@ -1,4 +1,4 @@
-/* $NetBSD: fpu_subr.c,v 1.2 1994/11/20 20:52:47 deraadt Exp $ */
+/* $NetBSD: fpu_subr.c,v 1.3 1996/03/14 19:42:01 christos Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -49,11 +49,16 @@
*/
#include <sys/types.h>
+#ifdef DIAGNOSTIC
+#include <sys/systm.h>
+#endif
#include <machine/reg.h>
+#include <machine/instr.h>
#include <sparc/fpu/fpu_arith.h>
#include <sparc/fpu/fpu_emu.h>
+#include <sparc/fpu/fpu_extern.h>
/*
* Shift the given number right rsh bits. Any bits that `fall off' will get
diff --git a/sys/arch/sparc/include/ansi.h b/sys/arch/sparc/include/ansi.h
index 6e90c691b78..a74e468804f 100644
--- a/sys/arch/sparc/include/ansi.h
+++ b/sys/arch/sparc/include/ansi.h
@@ -1,4 +1,4 @@
-/* $NetBSD: ansi.h,v 1.3 1994/11/20 20:52:50 deraadt Exp $ */
+/* $NetBSD: ansi.h,v 1.5 1996/03/31 22:21:21 pk Exp $ */
/*-
* Copyright (c) 1990, 1993
@@ -63,12 +63,13 @@
* chosen over a long is that the is*() and to*() routines take ints (says
* ANSI C), but they use _RUNE_T_ instead of int. By changing it here, you
* lose a bit of ANSI conformance, but your programs will still work.
- *
+ *
* Note that _WCHAR_T_ and _RUNE_T_ must be of the same type. When wchar_t
* and rune_t are typedef'd, _WCHAR_T_ will be undef'd, but _RUNE_T remains
* defined for ctype.h.
*/
#define _BSD_WCHAR_T_ int /* wchar_t */
+#define _BSD_WINT_T_ int /* wint_t */
#define _BSD_RUNE_T_ int /* rune_t */
#endif /* _ANSI_H_ */
diff --git a/sys/arch/sparc/include/autoconf.h b/sys/arch/sparc/include/autoconf.h
index 4cb7f4b25f6..1caa73db7ab 100644
--- a/sys/arch/sparc/include/autoconf.h
+++ b/sys/arch/sparc/include/autoconf.h
@@ -1,4 +1,4 @@
-/* $NetBSD: autoconf.h,v 1.9 1995/03/08 15:51:03 pk Exp $ */
+/* $NetBSD: autoconf.h,v 1.16 1996/04/10 20:33:38 pk Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -56,15 +56,19 @@
* (which `is' the CPU, in some sense) gets just the node, with a
* fake name ("mainbus").
*/
-#define RA_MAXVADDR 4 /* max (virtual) addresses per device */
-#define RA_MAXREG 2 /* max # of register banks per device */
+
+#define RA_MAXVADDR 8 /* max (virtual) addresses per device */
+#define RA_MAXREG 16 /* max # of register banks per device */
#define RA_MAXINTR 8 /* max interrupts per device */
+#define RA_MAXRANGE 10 /* max # of bus translations */
+
struct romaux {
const char *ra_name; /* name from FORTH PROM */
int ra_node; /* FORTH PROM node ID */
void *ra_vaddrs[RA_MAXVADDR];/* ROM mapped virtual addresses */
int ra_nvaddrs; /* # of ra_vaddrs[]s, may be 0 */
#define ra_vaddr ra_vaddrs[0] /* compatibility */
+
struct rom_reg {
int rr_iospace; /* register space (obio, etc) */
void *rr_paddr; /* register physical address */
@@ -74,13 +78,23 @@ struct romaux {
#define ra_iospace ra_reg[0].rr_iospace
#define ra_paddr ra_reg[0].rr_paddr
#define ra_len ra_reg[0].rr_len
+
struct rom_intr { /* interrupt information: */
int int_pri; /* priority (IPL) */
int int_vec; /* vector (always 0?) */
} ra_intr[RA_MAXINTR];
int ra_nintr; /* number of interrupt info elements */
+
+ struct rom_range { /* Only used on v3 PROMs */
+ u_int32_t cspace; /* Client space */
+ u_int32_t coffset; /* Client offset */
+ u_int32_t pspace; /* Parent space */
+ u_int32_t poffset; /* Parent offset */
+ u_int32_t size; /* Size in bytes of this range */
+ } ra_range[RA_MAXRANGE];
+ int ra_nrange;
+
struct bootpath *ra_bp; /* used for locating boot device */
- int ra_pfour; /* p4 register, for BUS_PFOUR devices */
};
@@ -95,11 +109,35 @@ struct confargs {
#define BUS_VME16 2
#define BUS_VME32 3
#define BUS_SBUS 4
-#define BUS_PFOUR 5
extern int bt2pmt[];
/*
+ * mapiodev maps an I/O device to a virtual address, returning the address.
+ * mapdev does the real work: you can supply a special virtual address and
+ * it will use that instead of creating one, but you must only do this if
+ * you get it from ../sparc/vaddrs.h.
+ */
+void *mapdev __P((struct rom_reg *pa, int va,
+ int offset, int size, int bustype));
+#define mapiodev(pa, offset, size, bustype) \
+ mapdev(pa, 0, offset, size, bustype)
+/*
+ * REG2PHYS is provided for drivers with a `d_mmap' function.
+ */
+#define REG2PHYS(rr, offset, bt) \
+ (((u_int)(rr)->rr_paddr + (offset)) | \
+ ((CPU_ISSUN4M) \
+ ? ((rr)->rr_iospace << PMAP_SHFT4M) \
+ : bt2pmt[bt]) \
+ )
+
+/* For VME and sun4/obio busses */
+void *bus_map __P((struct rom_reg *, int, int));
+void *bus_tmp __P((void *, int));
+void bus_untmp __P((void));
+
+/*
* The various getprop* functions obtain `properties' from the ROMs.
* getprop() obtains a property as a byte-sequence, and returns its
* length; the others convert or make some other guarantee.
@@ -124,6 +162,7 @@ int romprop __P((struct romaux *ra, const char *name, int node));
* its aux pointer to point to a pointer to the name (the address of
* a romaux structure suffices, for instance).
*/
+struct device;
int matchbyname __P((struct device *, void *cf, void *aux));
/*
@@ -133,36 +172,6 @@ int matchbyname __P((struct device *, void *cf, void *aux));
char *clockfreq __P((int freq));
/*
- * mapiodev maps an I/O device to a virtual address, returning the address.
- * mapdev does the real work: you can supply a special virtual address and
- * it will use that instead of creating one, but you must only do this if
- * you get it from ../sparc/vaddrs.h.
- */
-void *mapdev __P((struct rom_reg *rr, int va, int offset,
- int size, int bustype));
-#define mapiodev(rr, offset, size, bustype) mapdev(rr, 0, offset, size, bustype)
-
-void *bus_map __P((void *pa, int len, int bustype));
-void *bus_tmp __P((void *pa, int bustype));
-void bus_untmp __P((void));
-
-#ifdef notyet
-/*
- * REG2PHYS is provided for drivers with a `d_mmap' function.
- */
-#define REG2PHYS(rr, offset, bt) \
- (((u_int)(rr)->rr_paddr + (offset)) | \
- ((cputyp == CPU_SUN4M) \
- ? ((rr)->rr_iospace << PMAP_SHFT4M) \
- : bt2pmt[bt]) \
- )
-#else
-#define REG2PHYS(rr, offset, bt) \
- (((u_int)(rr)->rr_paddr + (offset)) | (bt2pmt[bt]) \
- )
-#endif
-
-/*
* Memory description arrays. Shared between pmap.c and autoconf.c; no
* one else should use this (except maybe mem.c, e.g., if we fix the VM to
* handle discontiguous physical memory).
@@ -180,14 +189,30 @@ void rominterpret __P((char *));
/* Openprom V2 style boot path */
struct bootpath {
- char name[8]; /* name of this node */
- int val[2]; /* up to two optional values */
+ char name[16]; /* name of this node */
+ int val[3]; /* up to three optional values */
+ struct device *dev; /* device that recognised this component */
};
-struct device *bootdv; /* found during autoconfiguration */
-
-struct bootpath *bootpath_store __P((int, struct bootpath *));
-int sd_crazymap __P((int));
+struct bootpath *bootpath_store __P((int, struct bootpath *));
+int sd_crazymap __P((int));
/* Parse a disk string into a dev_t, return device struct pointer */
struct device *parsedisk __P((char *, int, int, dev_t *));
+
+/* Establish a mountroot_hook, for benefit of floppy drive, mostly. */
+void mountroot_hook_establish __P((void (*) __P((struct device *)),
+ struct device *));
+
+void configure __P((void));
+void bootstrap __P((void));
+int firstchild __P((int));
+int nextsibling __P((int));
+void callrom __P((void));
+struct device *getdevunit __P((char *, int));
+void *findzs __P((int));
+int romgetcursoraddr __P((int **, int **));
+int findroot __P((void));
+int findnode __P((int, const char *));
+int opennode __P((char *));
+int node_has_property __P((int, const char *));
diff --git a/sys/arch/sparc/include/bsd_openprom.h b/sys/arch/sparc/include/bsd_openprom.h
index 1879ea5909d..35a9b3a7c13 100644
--- a/sys/arch/sparc/include/bsd_openprom.h
+++ b/sys/arch/sparc/include/bsd_openprom.h
@@ -1,4 +1,4 @@
-/* $NetBSD: bsd_openprom.h,v 1.3 1995/09/04 09:53:53 pk Exp $ */
+/* $NetBSD: bsd_openprom.h,v 1.11 1996/05/18 12:27:43 mrg Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -39,6 +39,12 @@
*/
/*
+ * Sun4m support by Aaron Brown, Harvard University.
+ * Changes Copyright (c) 1995 The President and Fellows of Harvard College.
+ * All rights reserved.
+ */
+
+/*
* This file defines the interface between the kernel and the Openboot PROM.
* N.B.: this has been tested only on interface versions 0 and 2 (we have
* never seen interface version 1).
@@ -70,12 +76,12 @@
struct v0devops {
int (*v0_open) __P((char *dev));
int (*v0_close) __P((int d));
- int (*v0_rbdev) __P((int d, int nblks, int blkno, caddr_t addr));
- int (*v0_wbdev) __P((int d, int nblks, int blkno, caddr_t addr));
- int (*v0_wnet) __P((int d, int nbytes, caddr_t addr));
- int (*v0_rnet) __P((int d, int nbytes, caddr_t addr));
- int (*v0_rcdev) __P((int d, int nbytes, int, caddr_t addr));
- int (*v0_wcdev) __P((int d, int nbytes, int, caddr_t addr));
+ int (*v0_rbdev) __P((int d, int nblks, int blkno, void *addr));
+ int (*v0_wbdev) __P((int d, int nblks, int blkno, void *addr));
+ int (*v0_wnet) __P((int d, int nbytes, void *addr));
+ int (*v0_rnet) __P((int d, int nbytes, void *addr));
+ int (*v0_rcdev) __P((int d, int nbytes, int, void *addr));
+ int (*v0_wcdev) __P((int d, int nbytes, int, void *addr));
int (*v0_seek) __P((int d, long offset, int whence));
};
@@ -97,7 +103,7 @@ struct v2devops {
int (*v2_fd_phandle) __P((int d));
/* Memory allocation and release. */
- caddr_t (*v2_malloc) __P((caddr_t va, u_int sz));
+ void *(*v2_malloc) __P((caddr_t va, u_int sz));
void (*v2_free) __P((caddr_t va, u_int sz));
/* Device memory mapper. */
@@ -107,12 +113,12 @@ struct v2devops {
/* Device open, close, etc. */
int (*v2_open) __P((char *devpath));
void (*v2_close) __P((int d));
- int (*v2_read) __P((int d, caddr_t buf, int nbytes));
- int (*v2_write) __P((int d, caddr_t buf, int nbytes));
+ int (*v2_read) __P((int d, void *buf, int nbytes));
+ int (*v2_write) __P((int d, void *buf, int nbytes));
void (*v2_seek) __P((int d, int hi, int lo));
- void (*v2_xxx2) __P(()); /* ??? */
- void (*v2_xxx3) __P(()); /* ??? */
+ void (*v2_chain) __P((void)); /* ??? */
+ void (*v2_release) __P((void)); /* ??? */
};
/*
@@ -217,7 +223,7 @@ struct promvec {
void (*pv_printf) __P((const char *fmt, ...));
void (*pv_abort) __P((void)); /* L1-A abort */
int *pv_ticks; /* Ticks since last reset */
- __dead void (*pv_halt) __P((void)); /* Halt! */
+ __dead void (*pv_halt) __P((void)) __attribute__((noreturn));/* Halt! */
void (**pv_synchook) __P((void)); /* "sync" command hook */
/*
@@ -251,13 +257,17 @@ struct promvec {
* easily.
*/
void (*pv_setctxt) __P((int ctxt, caddr_t va, int pmeg));
-
- /* OpenPROM V3 SMP cpu-control directives. */
- int (*pv_cpustart) __P((int cpu_node, caddr_t ctxttbl_paddr,
- int ctxt, caddr_t pc));
- int (*pv_cpustop) __P((int cpu_node));
- int (*pv_cpuidle) __P((int cpu_node));
- int (*pv_cpuresume) __P((int cpu_node));
+#if defined(SUN4M) && defined(notyet)
+ /*
+ * The following are V3 ROM functions to handle MP machines in the
+ * Sun4m series. They have undefined results when run on a uniprocessor!
+ */
+ int (*pv_v3cpustart) __P((u_int module, u_int ctxtbl,
+ int context, caddr_t pc));
+ int (*pv_v3cpustop) __P((u_int module));
+ int (*pv_v3cpuidle) __P((u_int module));
+ int (*pv_v3cpuresume) __P((u_int module));
+#endif
};
/*
@@ -304,6 +314,14 @@ struct nodeops {
*/
int (*no_proplen) __P((int node, caddr_t name));
int (*no_getprop) __P((int node, caddr_t name, caddr_t val));
- int (*no_setprop) __P((int node, caddr_t name, caddr_t val, int len));
+ int (*no_setprop) __P((int node, caddr_t name, caddr_t val,
+ int len));
caddr_t (*no_nextprop) __P((int node, caddr_t name));
};
+
+void romhalt __P((void))
+ __attribute__((__noreturn__));
+void romboot __P((char *))
+ __attribute__((__noreturn__));
+
+extern struct promvec *promvec;
diff --git a/sys/arch/sparc/include/cdefs.h b/sys/arch/sparc/include/cdefs.h
index 286edf32784..36f4990a9cc 100644
--- a/sys/arch/sparc/include/cdefs.h
+++ b/sys/arch/sparc/include/cdefs.h
@@ -30,8 +30,6 @@
__asm__(".stabs msg,30,0,0,0"); \
__asm__(".stabs \"_/**/sym\",1,0,0,0")
#endif
-#else
-#define __warn_references(sym,msg) /* nothing */
#endif
#endif /* !_MACHINE_CDEFS_H_ */
diff --git a/sys/arch/sparc/include/conf.h b/sys/arch/sparc/include/conf.h
new file mode 100644
index 00000000000..fbb10a56edc
--- /dev/null
+++ b/sys/arch/sparc/include/conf.h
@@ -0,0 +1,81 @@
+/* $NetBSD: conf.h,v 1.1 1996/03/30 21:17:55 christos Exp $ */
+
+/*
+ * Copyright (c) 1996 Christos Zoulas. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by Christos Zoulas.
+ * 4. The name of the author may not be used to endorse or promote products
+ * derived from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
+ * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
+ * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
+ * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
+ * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
+ * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#define mmread mmrw
+#define mmwrite mmrw
+cdev_decl(mm);
+
+/* open, close, ioctl */
+#define cdev_openprom_init(c,n) { \
+ dev_init(c,n,open), dev_init(c,n,close), (dev_type_read((*))) enodev, \
+ (dev_type_write((*))) enodev, dev_init(c,n,ioctl), \
+ (dev_type_stop((*))) nullop, 0, (dev_type_select((*))) enodev, \
+ (dev_type_mmap((*))) enodev }
+
+cdev_decl(openprom);
+
+cdev_decl(cn);
+
+cdev_decl(zs);
+
+bdev_decl(fd);
+cdev_decl(fd);
+
+cdev_decl(fb);
+
+/* open, close, read, write, ioctl, select */
+#define cdev_gen_init(c,n) { \
+ dev_init(c,n,open), dev_init(c,n,close), dev_init(c,n,read), \
+ dev_init(c,n,write), dev_init(c,n,ioctl), (dev_type_stop((*))) nullop, \
+ 0, dev_init(c,n,select), (dev_type_mmap((*))) enodev }
+
+cdev_decl(ms);
+
+cdev_decl(kbd);
+
+cdev_decl(bwtwo);
+
+cdev_decl(cgthree);
+
+cdev_decl(cgfour);
+
+cdev_decl(cgsix);
+
+cdev_decl(cgeight);
+
+bdev_decl(xd);
+cdev_decl(xd);
+
+bdev_decl(xy);
+cdev_decl(xy);
+
+bdev_decl(sw);
+cdev_decl(sw);
diff --git a/sys/arch/sparc/include/cpu.h b/sys/arch/sparc/include/cpu.h
index 1dccb549d5d..88c05b45a98 100644
--- a/sys/arch/sparc/include/cpu.h
+++ b/sys/arch/sparc/include/cpu.h
@@ -1,4 +1,4 @@
-/* $NetBSD: cpu.h,v 1.17 1995/06/28 02:56:05 cgd Exp $ */
+/* $NetBSD: cpu.h,v 1.21 1996/03/31 22:17:14 pk Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -48,7 +48,7 @@
#define _CPU_H_
/*
- * CTL_MACHDEP definitinos.
+ * CTL_MACHDEP definitions.
*/
#define CPU_MAXID 1 /* no valid machdep ids */
@@ -73,15 +73,6 @@
#define cpu_wait(p) /* nothing */
/*
- * See syscall() for an explanation of the following. Note that the
- * locore bootstrap code follows the syscall stack protocol. The
- * framep argument is unused.
- */
-#define cpu_set_init_frame(p, fp) \
- (p)->p_md.md_tf = (struct trapframe *) \
- ((caddr_t)(p)->p_addr + USPACE - sizeof(struct trapframe))
-
-/*
* Arguments to hardclock, softclock and gatherstats encapsulate the
* previous machine state in an opaque clockframe. The ipl is here
* as well for strayintr (see locore.s:interrupt and intr.c:strayintr).
@@ -114,7 +105,17 @@ union sir {
#define SIR_NET 0
#define SIR_CLOCK 1
-#define setsoftint() ienab_bis(IE_L1)
+#if defined(SUN4M)
+extern void raise __P((int, int));
+#if !(defined(SUN4) || defined(SUN4C))
+#define setsoftint() raise(0,1)
+#else /* both defined */
+#define setsoftint() (cputyp == CPU_SUN4M ? raise(0,1) : ienab_bis(IE_L1))
+#endif /* !4,!4c */
+#else /* 4m not defined */
+#define setsoftint() ienab_bis(IE_L1)
+#endif /* SUN4M */
+
#define setsoftnet() (sir.sir_which[SIR_NET] = 1, setsoftint())
#define setsoftclock() (sir.sir_which[SIR_CLOCK] = 1, setsoftint())
@@ -129,7 +130,7 @@ int want_resched; /* resched() was called */
/*
* Give a profiling tick to the current process when the user profiling
- * buffer pages are invalid. On the sparc, request an ast to send us
+ * buffer pages are invalid. On the sparc, request an ast to send us
* through trap(), marking the proc as needing a profiling tick.
*/
#define need_proftick(p) ((p)->p_flag |= P_OWEUPC, want_ast = 1)
@@ -170,6 +171,65 @@ void vmeintr_establish __P((int vec, int level, struct intrhand *));
*/
void intr_fasttrap __P((int level, void (*vec)(void)));
+/* disksubr.c */
+struct dkbad;
+int isbad __P((struct dkbad *bt, int, int, int));
+/* machdep.c */
+int ldcontrolb __P((caddr_t));
+void dumpconf __P((void));
+caddr_t reserve_dumppages __P((caddr_t));
+/* clock.c */
+struct timeval;
+void lo_microtime __P((struct timeval *));
+int statintr __P((void *));
+int clockintr __P((void *));/* level 10 (clock) interrupt code */
+int statintr __P((void *)); /* level 14 (statclock) interrupt code */
+/* locore.s */
+struct fpstate;
+void savefpstate __P((struct fpstate *));
+void loadfpstate __P((struct fpstate *));
+int probeget __P((caddr_t, int));
+void write_all_windows __P((void));
+void write_user_windows __P((void));
+void proc_trampoline __P((void));
+struct pcb;
+void snapshot __P((struct pcb *));
+struct frame *getfp __P((void));
+int xldcontrolb __P((caddr_t, struct pcb *));
+void copywords __P((const void *, void *, size_t));
+void qcopy __P((const void *, void *, size_t));
+void qzero __P((void *, size_t));
+/* locore2.c */
+void remrq __P((struct proc *));
+/* trap.c */
+void kill_user_windows __P((struct proc *));
+int rwindow_save __P((struct proc *));
+void child_return __P((struct proc *));
+/* amd7930intr.s */
+void amd7930_trap __P((void));
+/* cons.c */
+int cnrom __P((void));
+/* zs.c */
+void zsconsole __P((struct tty *, int, int, int (**)(struct tty *, int)));
+#ifdef KGDB
+void zs_kgdb_init __P((void));
+#endif
+/* fb.c */
+void fb_unblank __P((void));
+/* cache.c */
+void cache_flush __P((caddr_t, u_int));
+/* kgdb_stub.c */
+#ifdef KGDB
+void kgdb_attach __P((int (*)(void *), void (*)(void *, int), void *));
+void kgdb_connect __P((int));
+void kgdb_panic __P((void));
+#endif
+/* vm_machdep.c */
+void cpu_set_kpc __P((struct proc *, void (*)(struct proc *)));
+/* iommu.c */
+void iommu_enter __P((u_int, u_int));
+void iommu_remove __P((u_int, u_int));
+
/*
*
* The SPARC has a Trap Base Register (TBR) which holds the upper 20 bits
@@ -187,7 +247,7 @@ void intr_fasttrap __P((int level, void (*vec)(void)));
struct trapvec {
int tv_instr[4]; /* the four instructions */
};
-extern struct trapvec trapbase[256]; /* the 256 vectors */
+extern struct trapvec *trapbase; /* the 256 vectors */
extern void wzero __P((void *, u_int));
extern void wcopy __P((const void *, void *, u_int));
diff --git a/sys/arch/sparc/include/ctlreg.h b/sys/arch/sparc/include/ctlreg.h
index 154f358b985..1f5d8519aef 100644
--- a/sys/arch/sparc/include/ctlreg.h
+++ b/sys/arch/sparc/include/ctlreg.h
@@ -1,6 +1,8 @@
-/* $NetBSD: ctlreg.h,v 1.8 1995/06/25 21:35:05 pk Exp $ */
+/* $NetBSD: ctlreg.h,v 1.12 1996/05/16 15:57:00 abrown Exp $ */
/*
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
*
@@ -10,6 +12,7 @@
*
* All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
+ * This product includes software developed by Harvard University.
* This product includes software developed by the University of
* California, Lawrence Berkeley Laboratory.
*
@@ -45,95 +48,85 @@
*/
/*
- * Sun-4, 4c, and 4m control registers. (includes address space definitions
+ * Sun4m support by Aaron Brown, Harvard University.
+ * Changes Copyright (c) 1995 The President and Fellows of Harvard College.
+ * All rights reserved.
+ */
+
+/*
+ * Sun 4, 4c, and 4m control registers. (includes address space definitions
* and some registers in control space).
*/
+/*
+ * The Alternate address spaces.
+ */
+
/* 0x00 unused */
/* 0x01 unused */
-#if defined(SUN4C) || defined(SUN4)
#define ASI_CONTROL 0x02 /* cache enable, context reg, etc */
-#define ASI_SEGMAP 0x03 /* segment maps (so we can reach each pmeg) */
-#define ASI_PTE 0x04 /* PTE space (pmegs) */
-#define ASI_REGMAP 0x06 /* region maps (3 level MMUs only) */
-#define ASI_HWFLUSHSEG 0x05 /* hardware assisted version of FLUSHSEG */
-#define ASI_HWFLUSHPG 0x06 /* hardware assisted version of FLUSHPG */
-#define ASI_HWFLUSHCTX 0x07 /* hardware assisted version of FLUSHCTX */
-#endif
-#if defined(SUN4M)
-#define ASI_SRMMUFP 0x03 /* ref mmu flush/probe */
-#define ASI_SRMMUFP_L3 (0<<8) /* probe L3 | flush L3 PTE */
-#define ASI_SRMMUFP_L2 (1<<8) /* probe L2 | flush L2/L3 PTE/PTD's */
-#define ASI_SRMMUFP_L1 (2<<8) /* probe L1 | flush L1/L2/L3 PTE/PTD's*/
-#define ASI_SRMMUFP_L0 (3<<8) /* probe L0 | flush L0/L1/L2/L3 PTE/PTD's */
-#define ASI_SRMMUFP_LN (4<<8) /* probe all | flush all levels */
-
-#define ASI_SRMMU 0x04 /* ref mmu registers */
-#define ASI_SRMMUDIAG 0x06
-#endif
+#define ASI_SEGMAP 0x03 /* [4/4c] segment maps */
+#define ASI_SRMMUFP 0x03 /* [4m] ref mmu flush/probe */
+#define ASI_PTE 0x04 /* [4/4c] PTE space (pmegs) */
+#define ASI_SRMMU 0x04 /* [4m] ref mmu registers */
+#define ASI_REGMAP 0x06 /* [4/3-level MMU ] region maps */
+#define ASI_HWFLUSHSEG 0x05 /* [4/4c] hardware assisted version of FLUSHSEG */
+#define ASI_HWFLUSHPG 0x06 /* [4/4c] hardware assisted version of FLUSHPG */
+#define ASI_SRMMUDIAG 0x06 /* [4m] */
+#define ASI_HWFLUSHCTX 0x07 /* [4/4c] hardware assisted version of FLUSHCTX */
#define ASI_USERI 0x08 /* I-space (user) */
#define ASI_KERNELI 0x09 /* I-space (kernel) */
#define ASI_USERD 0x0a /* D-space (user) */
#define ASI_KERNELD 0x0b /* D-space (kernel) */
-#if defined(SUN4C) || defined(SUN4)
-#define ASI_FLUSHREG 0x7 /* causes hardware to flush cache region */
-#define ASI_FLUSHSEG 0x0c /* causes hardware to flush cache segment */
-#define ASI_FLUSHPG 0x0d /* causes hardware to flush cache page */
-#define ASI_FLUSHCTX 0x0e /* causes hardware to flush cache context */
-#endif
-#if defined(SUN4)
-#define ASI_DCACHE 0x0f /* flush data cache; not used on 4c */
-#endif
+#define ASI_FLUSHREG 0x7 /* [4/4c] flush cache by region */
+#define ASI_FLUSHSEG 0x0c /* [4/4c] flush cache by segment */
+#define ASI_FLUSHPG 0x0d /* [4/4c] flush cache by page */
+#define ASI_FLUSHCTX 0x0e /* [4/4c] flush cache by context */
-#if defined(SUN4M)
-#define ASI_ICACHETAG 0x0c /* instruction cache tag */
-#define ASI_ICACHEDATA 0x0d /* instruction cache data */
-#define ASI_DCACHETAG 0x0e /* data cache tag */
-#define ASI_DCACHEDATA 0x0f /* data cache data */
-#define ASI_IDCACHELFP 0x10 /* ms2 only: flush i&d cache line (page) */
-#define ASI_IDCACHELFS 0x11 /* ms2 only: flush i&d cache line (seg) */
-#define ASI_IDCACHELFR 0x12 /* ms2 only: flush i&d cache line (reg) */
-#define ASI_IDCACHELFC 0x13 /* ms2 only: flush i&d cache line (ctxt) */
-#define ASI_IDCACHELFU 0x14 /* ms2 only: flush i&d cache line (user) */
-#define ASI_BYPASS 0x20 /* sun ref mmu bypass, ie. direct phys access */
-#define ASI_ICACHECLR 0x36 /* ms1 only: instruction cache flash clear */
-#define ASI_DCACHECLR 0x37 /* ms1 only: data cache clear */
-#define ASI_DCACHEDIAG 0x39 /* data cache diagnostic register access */
-#endif
+#define ASI_DCACHE 0x0f /* [4] flush data cache */
-#if defined(SUN4C) || defined(SUN4)
-/* registers in the control space */
-#define AC_CONTEXT 0x30000000 /* context register (byte) */
-#define AC_SYSENABLE 0x40000000 /* system enable register (byte) */
-#define AC_CACHETAGS 0x80000000 /* cache tag base address */
-#define AC_SERIAL 0xf0000000 /* special serial port sneakiness */
- /* AC_SERIAL is not used in the kernel (it is for the PROM) */
-#endif
+#define ASI_ICACHETAG 0x0c /* [4m] instruction cache tag */
+#define ASI_ICACHEDATA 0x0d /* [4m] instruction cache data */
+#define ASI_DCACHETAG 0x0e /* [4m] data cache tag */
+#define ASI_DCACHEDATA 0x0f /* [4m] data cache data */
+#define ASI_IDCACHELFP 0x10 /* [4m] ms2 only: flush i&d cache line (page) */
+#define ASI_IDCACHELFS 0x11 /* [4m] ms2 only: flush i&d cache line (seg) */
+#define ASI_IDCACHELFR 0x12 /* [4m] ms2 only: flush i&d cache line (reg) */
+#define ASI_IDCACHELFC 0x13 /* [4m] ms2 only: flush i&d cache line (ctxt) */
+#define ASI_IDCACHELFU 0x14 /* [4m] ms2 only: flush i&d cache line (user) */
+#define ASI_BYPASS 0x20 /* [4m] sun ref mmu bypass,
+ ie. direct phys access */
+#define ASI_ICACHECLR 0x36 /* [4m] ms1 only: I-cache flash clear */
+#define ASI_DCACHECLR 0x37 /* [4m] ms1 only: D-cache flash clear */
+#define ASI_DCACHEDIAG 0x39 /* [4m] data cache diagnostic register access */
-#if defined(SUN4)
-#define AC_IDPROM 0x00000000 /* ID PROM */
-#define AC_DVMA_ENABLE 0x50000000 /* enable user dvma */
-#define AC_BUS_ERR 0x60000000 /* bus error register */
-#define AC_DIAG_REG 0x70000000 /* diagnostic reg */
-#define AC_DVMA_MAP 0xd0000000 /* user dvma map entries */
-#define AC_VMEINTVEC 0xe0000000 /* vme interrupt vector */
+/*
+ * [4/4c] Registers in the control space (ASI_CONTROL).
+ */
+#define AC_IDPROM 0x00000000 /* [4] ID PROM */
+#define AC_CONTEXT 0x30000000 /* [4/4c] context register (byte) */
+#define AC_SYSENABLE 0x40000000 /* [4/4c] system enable register (byte) */
+#define AC_DVMA_ENABLE 0x50000000 /* [4] enable user dvma */
+#define AC_BUS_ERR 0x60000000 /* [4] bus error register */
+#define AC_SYNC_ERR 0x60000000 /* [4c] sync (memory) error reg */
+#define AC_SYNC_VA 0x60000004 /* [4c] sync error virtual addr */
+#define AC_ASYNC_ERR 0x60000008 /* [4c] async error reg */
+#define AC_ASYNC_VA 0x6000000c /* [4c] async error virtual addr */
+#define AC_DIAG_REG 0x70000000 /* [4] diagnostic reg */
+#define AC_CACHETAGS 0x80000000 /* [4/4c?] cache tag base address */
+#define AC_CACHEDATA 0x90000000 /* [4] cached data [sun4/400?] */
+#define AC_DVMA_MAP 0xd0000000 /* [4] user dvma map entries */
+#define AC_VMEINTVEC 0xe0000000 /* [4] vme interrupt vector */
+#define AC_SERIAL 0xf0000000 /* [4/4c] special serial port sneakiness */
+ /* AC_SERIAL is not used in the kernel (it is for the PROM) */
/* XXX: does not belong here */
#define ME_REG_IERR 0x80 /* memory err ctrl reg error intr pending bit */
-#endif
-
-#if defined(SUN4C)
-#define AC_SYNC_ERR 0x60000000 /* sync (memory) error reg */
-#define AC_SYNC_VA 0x60000004 /* sync error virtual addr */
-#define AC_ASYNC_ERR 0x60000008 /* async error reg */
-#define AC_ASYNC_VA 0x6000000c /* async error virtual addr */
-#define AC_CACHEDATA 0x90000000 /* cached data */
-#endif
-#if defined(SUN4C) || defined(SUN4)
/*
+ * [4/4c]
* Bits in sync error register. Reading the register clears these;
* otherwise they accumulate. The error(s) occurred at the virtual
* address stored in the sync error address register, and may have
@@ -147,13 +140,14 @@
#define SER_TIMEOUT 0x20 /* bus timeout (non-existent mem) */
#define SER_SBUSERR 0x10 /* S-Bus bus error */
#define SER_MEMERR 0x08 /* memory ecc/parity error */
-#define SER_SZERR 0x02 /* size error, whatever that is */
+#define SER_SZERR 0x02 /* [4/vme?] size error, whatever that is */
#define SER_WATCHDOG 0x01 /* watchdog reset (never see this) */
#define SER_BITS \
"\20\20WRITE\10INVAL\7PROT\6TIMEOUT\5SBUSERR\4MEMERR\2SZERR\1WATCHDOG"
/*
+ * [4/4c]
* Bits in async error register (errors from DVMA or Sun-4 cache
* writeback). The corresponding bit is also set in the sync error reg.
*
@@ -168,7 +162,7 @@
#define AER_BITS "\20\10WBINVAL\6TIMEOUT\5DVMAERR"
/*
- * Bits in system enable register.
+ * [4/4c] Bits in system enable register.
*/
#define SYSEN_DVMA 0x20 /* Enable dvma */
#define SYSEN_CACHE 0x10 /* Enable cache */
@@ -176,19 +170,74 @@
#define SYSEN_VIDEO 0x08 /* Enable on-board video */
#define SYSEN_RESET 0x04 /* Reset the hardware */
#define SYSEN_RESETVME 0x02 /* Reset the VME bus */
-#endif
-#if defined(SUN4M)
+
+/*
+ * [4m] Bits in ASI_CONTROL? space, sun4m only.
+ */
+#define MXCC_ENABLE_ADDR 0x1c00a00 /* Enable register for MXCC */
+#define MXCC_ENABLE_BIT 0x4 /* Enable bit for MXCC */
+
+/*
+ * Bits in ASI_SRMMUFP space.
+ * Bits 8-11 determine the type of flush/probe.
+ * Address bits 12-31 hold the page frame.
+ */
+#define ASI_SRMMUFP_L3 (0<<8) /* probe L3 | flush L3 PTE */
+#define ASI_SRMMUFP_L2 (1<<8) /* probe L2 | flush L2/L3 PTE/PTD's */
+#define ASI_SRMMUFP_L1 (2<<8) /* probe L1 | flush L1/L2/L3 PTE/PTD's*/
+#define ASI_SRMMUFP_L0 (3<<8) /* probe L0 | flush L0/L1/L2/L3 PTE/PTD's */
+#define ASI_SRMMUFP_LN (4<<8) /* probe all | flush all levels */
+
+/*
+ * [4m] Registers and bits in the SPARC Reference MMU (ASI_SRMMU).
+ */
#define SRMMU_PCR 0x00000000 /* Processor control register */
#define SRMMU_CXTPTR 0x00000100 /* Context table pointer register */
#define SRMMU_CXR 0x00000200 /* Context register */
#define SRMMU_SFSTAT 0x00000300 /* Synchronous fault status reg */
#define SRMMU_SFADDR 0x00000400 /* Synchronous fault address reg */
+#define SRMMU_AFSTAT 0x00000500 /* Asynchronous fault status reg (HS) */
+#define SRMMU_AFADDR 0x00000600 /* Asynchronous fault address reg (HS)*/
#define SRMMU_TLBCTRL 0x00001000 /* TLB replacement control reg */
-/* Synchronous Fault Status Register bits */
+/* [4m] Bits in SRMMU control register */
+#define SRMMU_PCR_ME 0x00000001 /* MMU Enable */
+#define SRMMU_PCR_NF 0x00000002 /* Fault inhibit bit */
+#define SRMMU_PCR_PSO 0x00000080 /* Partial Store Ordering enable */
+#define SRMMU_PCR_CE 0x00000100 /* HS: Cache enable bit (HyperSPARC) */
+#define SRMMU_PCR_DCE 0x00000100 /* SS: Data cache enable bit */
+#define SRMMU_PCR_ICE 0x00000200 /* SS: SuperSPARC instr. cache enable */
+#define SRMMU_PCR_CM 0x00000400 /* HS: Cache mode: 1 == write-back */
+#define SRMMU_PCR_SB 0x00000400 /* SS: Store buffer enable bit */
+#define SRMMU_PCR_MR 0x00000800 /* HS: Memory reflection: 1 == on */
+#define SRMMU_PCR_MB 0x00000800 /* SS: MBus mode: 0=MXCC, 1=no MXCC */
+#define SRMMU_PCR_CS 0x00001000 /* HS: cache size: 1==256k, 0==128k */
+#define SRMMU_PCR_PE 0x00001000 /* SS: Enable memory parity checking */
+#define SRMMU_PCR_C 0x00002000 /* HS: enable cache when MMU off */
+#define SRMMU_PCR_SSBM 0x00002000 /* SS: 1 iff booting */
+#define SRMMU_PCR_HSBM 0x00004000 /* HS: 1 iff booting */
+#define SRMMU_PCR_SSSE 0x00004000 /* SS: Coherent bus snoop enable */
+#define SRMMU_PCR_AC 0x00008000 /* SS: 1=cache non-MMU accesses */
+#define SRMMU_PCR_TC 0x00010000 /* SS: 1=cache table walks */
+#define SRMMU_PCR_MID 0x00078000 /* HS: MBus module ID MID<3:0> */
+#define SRMMU_PCR_WBE 0x00080000 /* HS: Write buffer enable */
+#define SRMMU_PCR_HSSE 0x00100000 /* HS: Coherent bus snoop enable */
+#define SRMMU_PCR_CWR 0x00200000 /* HS: Cache wrap enable */
+#define SRMMU_PCR_VER 0x0f000000 /* Version of MMU implementation */
+#define SRMMU_PCR_IMPL 0xf0000000 /* Implementation number of MMU */
+
+#ifdef notyet
+#define SRMMU_PCR_INITIAL (SRMMU_PCR_ME | SRMMU_PCR_TC)
+#endif
+
+/* [4m] Bits in the Synchronous Fault Status Register */
+#define SFSR_EM 0x00020000 /* Error mode watchdog reset occurred */
#define SFSR_CS 0x00010000 /* Control Space error */
#define SFSR_PERR 0x00006000 /* Parity error code */
+#define SFSR_SB 0x00008000 /* SS: Store Buffer Error */
+#define SFSR_P 0x00004000 /* SS: Parity error */
+#define SFSR_UC 0x00001000 /* Uncorrectable error */
#define SFSR_TO 0x00000800 /* S-Bus timeout */
#define SFSR_BE 0x00000400 /* S-Bus bus error */
#define SFSR_LVL 0x00000300 /* Pagetable level causing the fault */
@@ -197,7 +246,50 @@
#define SFSR_FAV 0x00000002 /* Fault Address is valid */
#define SFSR_OW 0x00000001 /* Overwritten with new fault */
-/* TLB Replacement Control Register bits */
+#define SFSR_BITS \
+"\20\21CSERR\17PARITY\16SYSERR\15UNCORR\14TIMEOUT\13BUSERR\2FAV\1OW"
+
+/* [4m] Synchronous Fault Types */
+#define SFSR_FT_NONE (0 << 2) /* no fault */
+#define SFSR_FT_INVADDR (1 << 2) /* invalid address fault */
+#define SFSR_FT_PROTERR (2 << 2) /* protection fault */
+#define SFSR_FT_PRIVERR (3 << 2) /* privelege violation */
+#define SFSR_FT_TRANSERR (4 << 2) /* translation fault */
+#define SFSR_FT_BUSERR (5 << 2) /* access bus error */
+#define SFSR_FT_INTERR (6 << 2) /* internal error */
+#define SFSR_FT_RESERVED (7 << 2) /* reserved */
+
+/* [4m] Synchronous Fault Access Types */
+#define SFSR_AT_LDUDATA (0 << 5) /* Load user data */
+#define SFSR_AT_LDSDATA (1 << 5) /* Load supervisor data */
+#define SFSR_AT_LDUTEXT (2 << 5) /* Load user text */
+#define SFSR_AT_LDSTEXT (3 << 5) /* Load supervisor text */
+#define SFSR_AT_STUDATA (4 << 5) /* Store user data */
+#define SFSR_AT_STSDATA (5 << 5) /* Store supervisor data */
+#define SFSR_AT_STUTEXT (6 << 5) /* Store user text */
+#define SFSR_AT_STSTEXT (7 << 5) /* Store supervisor text */
+#define SFSR_AT_SUPERVISOR (1 << 5) /* Set iff supervisor */
+#define SFSR_AT_TEXT (2 << 5) /* Set iff text */
+#define SFSR_AT_STORE (4 << 5) /* Set iff store */
+
+/* [4m] Synchronous Fault PT Levels */
+#define SFSR_LVL_0 (0 << 8) /* Context table entry */
+#define SFSR_LVL_1 (1 << 8) /* Region table entry */
+#define SFSR_LVL_2 (2 << 8) /* Segment table entry */
+#define SFSR_LVL_3 (3 << 8) /* Page table entry */
+
+/* [4m] Asynchronous Fault Status Register bits */
+#define AFSR_AFO 0x00000001 /* Async. fault occurred */
+#define AFSR_AFA 0x000000f0 /* Bits <35:32> of faulting phys addr */
+#define AFSR_AFA_RSHIFT 4 /* Shift to get AFA to bit 0 */
+#define AFSR_AFA_LSHIFT 28 /* Shift to get AFA to bit 32 */
+#define AFSR_BE 0x00000400 /* Bus error */
+#define AFSR_TO 0x00000800 /* Bus timeout */
+#define AFSR_UC 0x00001000 /* Uncorrectable error */
+#define AFSR_SE 0x00002000 /* System error */
+
+#define AFSR_BITS "\20\16SYSERR\15UNCORR\14TIMEOUT\13BUSERR\1AFO"
+
+/* [4m] TLB Replacement Control Register bits */
#define TLBC_DISABLE 0x00000020 /* Disable replacement counter */
#define TLBC_RCNTMASK 0x0000001f /* Replacement counter (0-31) */
-#endif
diff --git a/sys/arch/sparc/include/db_machdep.h b/sys/arch/sparc/include/db_machdep.h
index 1464a723c35..b6028854359 100644
--- a/sys/arch/sparc/include/db_machdep.h
+++ b/sys/arch/sparc/include/db_machdep.h
@@ -1,27 +1,27 @@
-/* $NetBSD: db_machdep.h,v 1.3 1995/02/09 10:34:21 pk Exp $ */
+/* $NetBSD: db_machdep.h,v 1.7 1996/03/31 22:21:28 pk Exp $ */
-/*
+/*
* Mach Operating System
* Copyright (c) 1991,1990 Carnegie Mellon University
* All Rights Reserved.
- *
+ *
* Permission to use, copy, modify and distribute this software and its
* documentation is hereby granted, provided that both the copyright
* notice and this permission notice appear in all copies of the
* software, derivative works or modified versions, and any portions
* thereof, and that both notices appear in supporting documentation.
- *
+ *
* CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
* CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
* ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
- *
+ *
* Carnegie Mellon requests users of this software to return to
- *
+ *
* Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
* School of Computer Science
* Carnegie Mellon University
* Pittsburgh PA 15213-3890
- *
+ *
* any improvements or extensions that they make and grant Carnegie Mellon
* the rights to redistribute these changes.
*/
@@ -61,8 +61,8 @@ db_regs_t ddb_regs; /* register state */
#define BKPT_SIZE (4) /* size of breakpoint inst */
#define BKPT_SET(inst) (BKPT_INST)
-#define db_clear_single_step(regs) (0)
-#define db_set_single_step(regs) (0)
+#define db_clear_single_step(regs) (void) (0)
+#define db_set_single_step(regs) (void) (0)
#define IS_BREAKPOINT_TRAP(type, code) ((type) == T_BREAKPOINT)
#define IS_WATCHPOINT_TRAP(type, code) (0)
@@ -75,4 +75,8 @@ db_regs_t ddb_regs; /* register state */
#define DB_MACHINE_COMMANDS
+void db_machine_init __P((void));
+int kdb_trap __P((int, struct trapframe *));
+
+
#endif /* _SPARC_DB_MACHDEP_H_ */
diff --git a/sys/arch/sparc/include/eeprom.h b/sys/arch/sparc/include/eeprom.h
index 0f47f5b41f2..902153d3f04 100644
--- a/sys/arch/sparc/include/eeprom.h
+++ b/sys/arch/sparc/include/eeprom.h
@@ -1,5 +1,7 @@
+/* $NetBSD: eeprom.h,v 1.2 1995/08/29 22:11:58 pk Exp $ */
+
/*
- * Copyright (c) 1995 Theo de Raadt
+ * Copyright (c) 1994 Gordon W. Ross
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
@@ -10,10 +12,7 @@
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
- * 3. All advertising materials mentioning features or use of this software
- * must display the following acknowledgement:
- * This product includes software developed by Theo de Raadt.
- * 4. The name of the author may not be used to endorse or promote products
+ * 3. The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
@@ -29,325 +28,156 @@
*/
/*
- * This structure defines the contents of the EEPROM for all Sun
- * Workstations with the SunMon monitor.
+ * Structure/definitions for the Sun3/Sun4 EEPROM.
*
- * The EEPROM is divided into a diagnostic section, a reserved section,
- * a ROM section, and a software section (defined in detail elsewhere).
+ * This information is published in the Sun document:
+ * "PROM User's Manual", part number 800-1736010.
*/
+
/*
* Note that most places where the PROM stores a "true/false" flag,
* the true value is 0x12 and false is the usual zero. Such flags
* all take the values EE_TRUE or EE_FALSE so this file does not
* need to define so many value macros.
*/
-#define EE_TRUE 0x12
-#define EE_FALSE 0x00
-
-struct ee_keymap {
- u_char keymap[128]; /* PROM/EEPROM are 7 bit */
-};
-
-#define PW_SIZE 8 /* max size of a password entry */
-
-struct password_inf {
- u_short bad_counter; /* illegal password count */
- char pw_mode; /* mode */
- char pw_bytes[PW_SIZE]; /* password */
-};
-#define NON_SECURE_MODE 0x00 /* non-secure */
-#define COMMAND_SECURE_MODE 0x01 /* command secure */
-#define FULLY_SECURE_MODE 0x5e /* fully secure */
- /* 0, 2-0x5d, 0x5f-0xff: non secure*/
-
-
-#define MAX_SLOTS 12
-#define CONFIG_SIZE 16 /* bytes of config data for each slot */
+#define EE_TRUE 0x12
+#define EE_FALSE 0
struct eeprom {
- struct ee_diag {
- u_int eed_test; /* 0x000: diagnostic test write area */
- u_short eed_wrcnt[3]; /* 0x004: diag area write count (3 copies) */
- short eed_nu1;
-
- u_char eed_chksum[3]; /* 0x00c: diag area checksum (3 copies) */
- char eed_nu2;
- time_t eed_hwupdate; /* 0x010: date of last hardware update */
-
- char eed_memsize; /* 0x014: MB's of memory in system */
- char eed_memtest; /* 0x015: MB's of memory to test */
-
- char eed_scrsize; /* 0x016: screen size */
-#define EED_SCR_1152X900 0x00
-#define EED_SCR_1024X1024 0x12
-#define EED_SCR_1600X1280 0x13
-#define EED_SCR_1440X1440 0x14
-#define EED_SCR_640X480 0x15
-#define EED_SCR_1280X1024 0x16
-
- char eed_dogaction; /* 0x017: watchdog reset action */
-#define EED_DOG_MONITOR 0x00 /* return to monitor command level */
-#define EED_DOG_REBOOT 0x12 /* perform power on reset and reboot */
-
- char eed_defboot; /* 0x018: default boot? */
-#define EED_DEFBOOT 0x00 /* do default boot */
-#define EED_NODEFBOOT 0x12 /* don't do default boot */
-
- char eed_bootdev[2]; /* 0x019: boot device name (e.g. xy, ie) */
-
- u_char eed_bootctrl; /* 0x01b: controller number */
- u_char eed_bootunit; /* 0x01c: unit number */
- u_char eed_bootpart; /* 0x01d: partition number */
-
- char eed_kbdtype; /* 0x01f: non-Sun keyboard type - for OEM's */
-#define EED_KBD_SUN 0 /* one of the Sun keyboards */
-
- char eed_console; /* 0x01f: console device */
-#define EED_CONS_BW 0x00 /* use b&w monitor */
-#define EED_CONS_TTYA 0x10 /* use tty A port */
-#define EED_CONS_TTYB 0x11 /* use tty B port */
-#define EED_CONS_COLOR 0x12 /* use color monitor */
-#define EED_CONS_P4 0x20 /* use the P4 monitor */
-
- char eed_showlogo; /* 0x020: display Sun logo? */
-#define EED_LOGO 0x00
-#define EED_NOLOGO 0x12
-
- char eed_keyclick; /* 0x021: keyboard click? */
-#define EED_NOKEYCLICK 0x00
-#define EED_KEYCLICK 0x12
-
- char eed_diagdev[2]; /* 0x022: boot device name (e.g. xy, ie) */
- u_char eed_diagctrl; /* 0x024: controller number */
- u_char eed_diagunit; /* 0x025: unit number */
- u_char eed_diagpart; /* 0x026: partition number */
-#define EED_WOB 0x12 /* bring system up white on black */
-
- u_char eed_videobg;
- char eed_diagpath[40]; /* 0x028: boot path of diagnostic */
-#define EED_TERM_34x80 00
-#define EED_TERM_48x120 0x12 /* for large scrn size 1600x1280 */
- char eed_colsize; /* 0x050: number of columns */
- char eed_rowsize; /* 0x051: number of rows */
-
- char eed_nu5[6];
-
- struct eed_tty_def { /* 0x058: tty port defaults */
- char eet_sel_baud; /* user specifies baud rate */
-#define EET_DEFBAUD 0x00
-#define EET_SELBAUD 0x12
- u_char eet_hi_baud; /* upper byte of baud rate */
- u_char eet_lo_baud; /* lower byte of baud rate */
- u_char eet_rtsdtr; /* flag for dtr and rts */
-#define NO_RTSDTR 0x12
- char eet_unused[4];
- } eed_ttya_def, eed_ttyb_def;
-
- char eed_banner[80]; /* 0x068: replacement banner */
- /* last two chars must be \r\n (XXX - why not \0?) */
-
- u_short eed_pattern; /* 0x0b8: test pattern - must contain 0xAA55 */
- short eed_nu6;
- struct eed_conf { /* 0x0bc: system configuration, by slot */
- union {
- struct eec_gen {
- u_char eec_type; /* board type code */
- char eec_size[CONFIG_SIZE - 1];
- } eec_gen;
- char conf_byte[CONFIG_SIZE];
- u_char eec_type; /* type of this board */
-#define EEC_TYPE_NONE 0 /* no board this slot */
-#define EEC_TYPE_CPU 0x01 /* cpu */
- struct eec_cpu {
- u_char eec_type; /* board type */
- u_char eec_cpu_memsize; /* MB's on cpu */
- int eec_cpu_unused:6;
- int eec_cpu_dcp:1; /* dcp? */
- int eec_cpu_68881:1; /* 68881? */
- u_char eec_cpu_cachesize; /* KB's cache */
- } eec_cpu;
- struct eec_cpu_alt {
- u_char eec_type; /* board type */
- u_char memsize; /* MB's on cpu */
- u_char option; /* option flags */
-#define CPU_HAS_DCP 0x02
-#define CPU_HAS_68881 0x01
- u_char cachesize; /* KB's in cache */
- } eec_cpu_alt;
-
-#define EEC_TYPE_MEM 0x02 /* memory board */
- struct eec_mem {
- u_char eec_type; /* board type */
- u_char eec_mem_size; /* MB's on card */
- } eec_mem;
-
-#define EEC_TYPE_COLOR 0x03 /* color frame buffer */
- struct eec_color {
- u_char eec_type; /* board type */
- char eec_color_type;
-#define EEC_COLOR_TYPE_CG2 2 /* cg2 color board */
-#define EEC_COLOR_TYPE_CG3 3 /* cg3 color board */
-#define EEC_COLOR_TYPE_CG5 5 /* cg5 color board */
- } eec_color;
-
-#define EEC_TYPE_BW 0x04 /* b&w frame buffer */
-
-#define EEC_TYPE_FPA 0x05 /* floating point accelerator */
-
-#define EEC_TYPE_DISK 0x06 /* SMD disk controller */
- struct eec_disk {
- u_char eec_type; /* board type */
- char eec_disk_type; /* controller type */
-#define EEC_DISK_TYPE_X450 1
-#define EEC_DISK_TYPE_X451 2
- char eec_disk_ctlr; /* controller number */
- char eec_disk_disks; /* number of disks */
- char eec_disk_cap[4]; /* capacity */
-#define EEC_DISK_NONE 0
-#define EEC_DISK_130 1
-#define EEC_DISK_280 2
-#define EEC_DISK_380 3
-#define EEC_DISK_575 4
-#define EEC_DISK_900 5
- } eec_disk;
-
-#define EEC_TYPE_TAPE 0x07 /* 1/2" tape controller */
- struct eec_tape {
- u_char eec_type; /* board type */
- char eec_tape_type; /* controller type */
-#define EEC_TAPE_TYPE_XT 1 /* Xylogics 472 */
-#define EEC_TAPE_TYPE_MT 2 /* TapeMaster */
- char eec_tape_ctlr; /* controller number */
- char eec_tape_drives; /* number of drives */
- } eec_tape;
-
-#define EEC_TYPE_ETHER 0x08 /* Ethernet controller */
-
-#define EEC_TYPE_TTY 0x09 /* terminal multiplexer */
- struct eec_tty {
- u_char eec_type; /* board type */
- char eec_tty_lines; /* number of lines */
-#define MAX_TTY_LINES 16
- char manufacturer; /* code for maker */
-#define EEC_TTY_UNKNOWN 0
-#define EEC_TTY_SYSTECH 1
-#define EEC_TTY_SUN 2
- } eec_tty;
-
-#define EEC_TYPE_GP 0x0a /* graphics processor/buffer */
- struct eec_gp {
- u_char eec_type; /* board type */
- char eec_gp_type;
-#define EEC_GP_TYPE_GPPLUS 1 /* GP+ graphic processor board */
-#define EEC_GP_TYPE_GP2 2 /* GP2 graphic processor board */
- } eec_gp;
-
-#define EEC_TYPE_DCP 0x0b /* DCP ??? XXX */
-
-#define EEC_TYPE_SCSI 0x0c /* SCSI controller */
- struct eec_scsi {
- u_char eec_type; /* board type */
- char eec_scsi_type; /* host adaptor type */
-#define EEC_SCSI_SUN2 2
-#define EEC_SCSI_SUN3 3
- char eec_scsi_tapes; /* number of tapes */
- char eec_scsi_disks; /* number of disks */
- char eec_scsi_tape_type;
-#define EEC_SCSI_SYSG 1
-#define EEC_SCSI_MT02 2
- char eec_scsi_disk_type;
-#define EEC_SCSI_MD21 1
-#define EEC_SCSI_ADAPT 2
- char eec_scsi_driv_code[2];
-#define EEC_SCSI_D71 1
-#define EEC_SCSI_D141 2
-#define EEC_SCSI_D327 3
- } eec_scsi;
-#define EEC_TYPE_IPC 0x0d
-
-#define EEC_TYPE_GB 0x0e
-
-#define EEC_TYPE_SCSI375 0x0f
-
-#define EEC_TYPE_MAPKIT 0x10
- struct eec_mapkit {
- u_char eec_type; /* board type */
- char eec_mapkit_hw; /* whether INI */
-#define EEC_TYPE_MAPKIT_INI 1
- } eec_mapkit;
-
-#define EEC_TYPE_CHANNEL 0x11
-#define EEC_TYPE_ALM2 0x12
-
-
- struct eec_generic_net {
-#define EEC_TYPE_GENERIC_NET 0x13
- u_char eec_devtype; /* board type */
- u_char eec_mem_type; /* Memory type */
- u_char gn_nu1;
- u_char gn_nu2;
-#define ID_VME 0x0 /* Data buffers are in VME space(Shared) */
-#define ID_DVMA 0x1 /* or DVMA buffer memory type */
- char *dev_reg_ptr;
- } eec_generic_net;
-
-
-#define EEC_TYPE_END 0xff /* end of card cage */
- } eec_un;
- } eed_conf[MAX_SLOTS + 1];
-#define EEPROM_TABLE 0x58 /* 1 indicates alternate key table. */
- u_char which_kbt; /* 0x18C: which keytable? */
-
- /*
- * Note. The following contains the full keyboard id
- * returned by the type4 transmit layout command. It is not
- * necessarily anything to do with locale of operation.
- */
- u_char ee_kb_type;
- u_char ee_kb_id; /* 0x18E: keyboard id in case of EEPROM table */
-#define EEPROM_LOGO 0x12
- u_char otherlogo; /* 0x18F: True if eeprom logo needed */
- struct ee_keymap ee_keytab_lc[1]; /* 0x190: */
- struct ee_keymap ee_keytab_uc[1]; /* 0x210: */
- u_char ee_logo[64][8]; /* 0x290: A 64X64 bit space for custom/OEM designed logo icon */
- struct password_inf ee_password_inf; /* 0x490: reserved */
- char eed_resv[0x500 - 0x49c]; /* 0x49c: reserved */
- } ee_diag;
-
- struct ee_resv { /* reserved area of EEPROM */
- u_short eev_wrcnt[3]; /* 0x500: write count (3 copies) */
- short eev_nu1;
- u_char eev_chksum[3]; /* 0x508: reserved area checksum (3 copies) */
- char eev_nu2;
- char eev_resv[0x600 - 0x50c]; /* 0x50c: */
- } ee_resv;
-
- struct ee_rom { /* ROM area of EEPROM */
- u_short eer_wrcnt[3]; /* 0x600: write count (3 copies) */
- short eer_nu1;
- u_char eer_chksum[3]; /* 0x608: ROM area checksum (3 copies) */
- char eer_nu2;
- char eer_resv[0x700 - 0x60c]; /* 0x60c: */
- } ee_rom;
-
- /*
- * The following area is reserved for software (i.e. UNIX).
- * The actual contents of this area are defined elsewhere.
- */
- struct ee_softresv { /* software area of EEPROM */
- u_short ees_wrcnt[3]; /* 0x700: write count (3 copies) */
- short ees_nu1;
- u_char ees_chksum[3]; /* 0x708: software area checksum (3 copies) */
- char ees_nu2;
- char ees_resv[0x800 - 0x70c]; /* 0x70c: */
- } ee_soft;
+ /* 0x00 */
+ u_char eeTestArea[4]; /* Factory Defined */
+ u_short eeWriteCount[4]; /* || || */
+ u_char eeChecksum[4]; /* || || */
+ time_t eeLastHwUpdate; /* || || */
+
+ /* 0x14 */
+ u_char eeInstalledMem; /* Megabytes */
+ u_char eeMemTestSize; /* || */
+
+ /* 0x16 */
+ u_char eeScreenSize;
+#define EE_SCR_1152X900 0x00
+#define EE_SCR_1024X1024 0x12
+#define EE_SCR_1600X1280 0x13
+#define EE_SCR_1440X1440 0x14
+
+ u_char eeWatchDogDoesReset; /* Watchdog timeout action:
+ * true: reset/reboot
+ * false: return to monitor
+ */
+ /* 0x18 */
+ u_char eeBootDevStored; /* Is the boot device stored:
+ * true: use stored device spec.
+ * false: use default (try all)
+ */
+ /* Stored boot device spec. i.e.: "sd(Ctlr,Unit,Part)" */
+ u_char eeBootDevName[2]; /* xy,xd,sd,ie,le,st,xt,mt,... */
+ u_char eeBootDevCtlr;
+ u_char eeBootDevUnit;
+ u_char eeBootDevPart;
+
+ /* 0x1E */
+ u_char eeKeyboardType; /* zero for sun keyboards */
+ u_char eeConsole; /* What to use for the console */
+#define EE_CONS_BW 0x00 /* - On-board B&W / keyboard */
+#define EE_CONS_TTYA 0x10 /* - serial port A */
+#define EE_CONS_TTYB 0x11 /* - serial port B */
+#define EE_CONS_COLOR 0x12 /* - Color FB / keyboard */
+#define EE_CONS_P4OPT 0x20 /* - Option board on P4 */
+
+ /* 0x20 */
+ u_char eeCustomBanner; /* Is there a custom banner:
+ * true: use text at 0x68
+ * false: use Sun banner
+ */
+
+ u_char eeKeyClick; /* true/false */
+
+ /* Boot device with "Diag" switch in Diagnostic mode: */
+ u_char eeDiagDevName[2];
+ u_char eeDiagDevCtlr;
+ u_char eeDiagDevUnit;
+ u_char eeDiagDevPart;
+
+ /* Video white-on-black (not implemented) */
+ u_char eeWhiteOnBlack; /* true/false */
+
+ /* 0x28 */
+ char eeDiagPath[40]; /* path name of diag program */
+
+ /* 0x50 */
+ u_char eeTtyCols; /* normally 80 */
+ u_char eeTtyRows; /* normally 34 */
+ u_char ee_x52[6]; /* unused */
+
+ /* 0x58 */
+ /* Default parameters for tty A and tty B: */
+ struct eeTtyDef {
+ u_char eetBaudSet; /* Is the baud rate set?
+ * true: use values here
+ * false: use default (9600)
+ */
+ u_char eetBaudHi; /* i.e. 96.. */
+ u_char eetBaudLo; /* ..00 */
+ u_char eetNoRtsDtr; /* true: disable H/W flow
+ * false: enable H/W flow */
+ u_char eet_pad[4];
+ } eeTtyDefA, eeTtyDefB;
+
+ /* 0x68 */
+ char eeBannerString[80]; /* see eeCustomBanner above */
+
+ /* 0xB8 */
+ u_short eeTestPattern; /* must be 0xAA55 */
+ u_short ee_xBA; /* unused */
+
+ /* 0xBC */
+ /* Configuration data. Hopefully we don't need it. */
+ struct eeConf {
+ u_char eecData[16];
+ } eeConf[12+1];
+
+ /* 0x18c */
+ u_char eeAltKeyTable; /* What Key table to use:
+ * 0x58: EEPROM tables
+ * else: PROM key tables
+ */
+ u_char eeKeyboardLocale; /* extended keyboard type */
+ u_char eeKeyboardID; /* for EEPROM key tables */
+ u_char eeCustomLogo; /* true: use eeLogoBitmap */
+
+ /* 0x190 */
+ u_char eeKeymapLC[0x80];
+ u_char eeKeymapUC[0x80];
+
+ /* 0x290 */
+ u_char eeLogoBitmap[64][8]; /* 64x64 bit custom logo */
+
+ /* 0x490 */
+ u_char ee_x490[0x500-0x490]; /* unused */
+
+ /* Other stuff we don't care about... */
+ /* 0x500 */
+ u_char eeReserved[0x100];
+ /* 0x600 */
+ u_char eeROM_Area[0x100];
+ /* 0x700 */
+ u_char eeUnixArea[0x100];
};
-#define EEPROM_BASE 0xffd04000
+/*
+ * The size of the eeprom on machines with the old clock is 2k. However,
+ * on machines with the new clock (and the `eeprom' in the nvram area)
+ * there are only 2040 bytes available. (???). Since we really only
+ * care about the `diagnostic' area, we'll use it's size when dealing
+ * with the eeprom in general.
+ */
+#define EEPROM_SIZE 0x500
-#ifdef _KERNEL
+#ifdef _KERNEL
extern char *eeprom_va;
int eeprom_uio __P((struct uio *));
-#endif /* _KERNEL */
+#endif /* _KERNEL */
+
diff --git a/sys/arch/sparc/include/endian.h b/sys/arch/sparc/include/endian.h
index a7004e09ea6..3d4297321bf 100644
--- a/sys/arch/sparc/include/endian.h
+++ b/sys/arch/sparc/include/endian.h
@@ -1,4 +1,4 @@
-/* $NetBSD: endian.h,v 1.2 1994/11/20 20:53:01 deraadt Exp $ */
+/* $NetBSD: endian.h,v 1.3 1996/02/13 17:04:58 christos Exp $ */
/*
* Copyright (c) 1987, 1991 Regents of the University of California.
@@ -69,10 +69,10 @@ __END_DECLS
#define htonl(x) (x)
#define htons(x) (x)
-#define NTOHL(x) (x)
-#define NTOHS(x) (x)
-#define HTONL(x) (x)
-#define HTONS(x) (x)
+#define NTOHL(x) (void) (x)
+#define NTOHS(x) (void) (x)
+#define HTONL(x) (void) (x)
+#define HTONS(x) (void) (x)
#else
diff --git a/sys/arch/sparc/include/exec.h b/sys/arch/sparc/include/exec.h
index 10c47d28aaa..beac6897c05 100644
--- a/sys/arch/sparc/include/exec.h
+++ b/sys/arch/sparc/include/exec.h
@@ -57,11 +57,4 @@ struct relocation_info_sparc {
};
#define relocation_info relocation_info_sparc
-#define ELF_TARG_CLASS ELFCLASS32
-#define ELF_TARG_DATA ELFDATA2MSB
-#define ELF_TARG_MACH EM_SPARC
-
-#define DO_AOUT /* support a.out */
-#define DO_ELF /* support ELF */
-
#endif /* _SPARC_EXEC_H_ */
diff --git a/sys/arch/sparc/include/fbio.h b/sys/arch/sparc/include/fbio.h
index a80ec853cb2..2eddc2b5d69 100644
--- a/sys/arch/sparc/include/fbio.h
+++ b/sys/arch/sparc/include/fbio.h
@@ -1,4 +1,4 @@
-/* $NetBSD: fbio.h,v 1.3 1994/11/20 20:53:03 deraadt Exp $ */
+/* $NetBSD: fbio.h,v 1.4 1996/03/31 22:21:31 pk Exp $ */
/*
* Copyright (c) 1992 Regents of the University of California.
@@ -154,7 +154,7 @@ struct fbcurpos {
short x;
short y;
};
-
+
#define FB_CUR_SETCUR 0x01
#define FB_CUR_SETPOS 0x02
#define FB_CUR_SETHOT 0x04
@@ -172,14 +172,14 @@ struct fbcursor {
char *image; /* cursor's image bits */
char *mask; /* cursor's mask bits */
};
-
+
/* set/get cursor attributes/shape */
#define FBIOSCURSOR _IOW('F', 24, struct fbcursor)
#define FBIOGCURSOR _IOWR('F', 25, struct fbcursor)
-
+
/* set/get cursor position */
#define FBIOSCURPOS _IOW('F', 26, struct fbcurpos)
#define FBIOGCURPOS _IOW('F', 27, struct fbcurpos)
-
+
/* get max cursor size */
#define FBIOGCURMAX _IOR('F', 28, struct fbcurpos)
diff --git a/sys/arch/sparc/include/fbvar.h b/sys/arch/sparc/include/fbvar.h
index 1b00249ae50..d5f3e8855c5 100644
--- a/sys/arch/sparc/include/fbvar.h
+++ b/sys/arch/sparc/include/fbvar.h
@@ -1,4 +1,4 @@
-/* $NetBSD: fbvar.h,v 1.5 1995/10/08 01:40:25 pk Exp $ */
+/* $NetBSD: fbvar.h,v 1.7 1996/02/27 22:09:39 thorpej Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -76,14 +76,26 @@ struct fbdevice {
struct fbdriver *fb_driver; /* pointer to driver */
struct device *fb_device; /* parameter for fbd_unblank */
+ int fb_flags; /* misc. flags */
+#define FB_FORCE 0x00000001 /* force device into /dev/fb */
+#define FB_PFOUR 0x00010000 /* indicates fb is a pfour fb */
+#define FB_USERMASK (FB_FORCE) /* flags that the user can set */
+
+ volatile u_int32_t *fb_pfour; /* pointer to pfour register */
+
#ifdef RASTERCONSOLE
/* Raster console emulator state */
struct rconsole fb_rcons;
#endif
};
-void fbattach __P((struct fbdevice *));
+void fb_attach __P((struct fbdevice *, int));
void fb_setsize __P((struct fbdevice *, int, int, int, int, int));
#ifdef RASTERCONSOLE
void fbrcons_init __P((struct fbdevice *));
#endif
+#if defined(SUN4)
+int fb_pfour_id __P((void *));
+int fb_pfour_get_video __P((struct fbdevice *));
+void fb_pfour_set_video __P((struct fbdevice *, int));
+#endif
diff --git a/sys/arch/sparc/include/ieeefp.h b/sys/arch/sparc/include/ieeefp.h
index 87c19132039..a1e67846b83 100644
--- a/sys/arch/sparc/include/ieeefp.h
+++ b/sys/arch/sparc/include/ieeefp.h
@@ -1,4 +1,4 @@
-/*
+/*
* Written by J.T. Conklin, Apr 6, 1995
* Public domain.
*/
diff --git a/sys/arch/sparc/include/kbd.h b/sys/arch/sparc/include/kbd.h
index 7fe3c4e2653..4cabc471287 100644
--- a/sys/arch/sparc/include/kbd.h
+++ b/sys/arch/sparc/include/kbd.h
@@ -1,4 +1,4 @@
-/* $NetBSD: kbd.h,v 1.4 1995/07/06 05:36:29 pk Exp $ */
+/* $NetBSD: kbd.h,v 1.6 1996/03/31 22:21:35 pk Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -86,3 +86,12 @@
#define LED_COMPOSE 0x2
#define LED_SCROLL_LOCK 0x4
#define LED_CAPS_LOCK 0x8
+
+void kbd_serial __P((struct tty *,
+ void (*)(struct tty *), void (*)(struct tty *)));
+void ms_serial __P((struct tty *,
+ void (*)(struct tty *), void (*)(struct tty *)));
+void kbd_rint __P((int));
+void ms_rint __P((int));
+void kbd_ascii __P((struct tty *));
+int kbd_docmd __P((int, int));
diff --git a/sys/arch/sparc/include/kbio.h b/sys/arch/sparc/include/kbio.h
index 2a2c3b427aa..049bfb79fed 100644
--- a/sys/arch/sparc/include/kbio.h
+++ b/sys/arch/sparc/include/kbio.h
@@ -66,7 +66,7 @@
* Keyboard commands and types are defined in kbd.h as they are actually
* real hardware commands and type numbers.
*/
-struct okiockey { /* Out-dated key translation structure */
+struct okiockey { /* Out-dated key translation structure */
int kio_tablemask; /* whatever */
u_char kio_station; /* key number */
u_char kio_entry; /* HOLE if not present */
@@ -85,12 +85,12 @@ struct kiockey {
* in KIOC[SG]KEY ioctls. Currently, we only have "non-shift" and "shift"
* tables.
*/
-#define KIOC_NOMASK 0x0
-#define KIOC_CAPSMASK 0x1
-#define KIOC_SHIFTMASK 0xe
-#define KIOC_CTRLMASK 0x30
-#define KIOC_ALTGMASK 0x200
-#define KIOC_NUMLMASK 0x800
+#define KIOC_NOMASK 0x0
+#define KIOC_CAPSMASK 0x1
+#define KIOC_SHIFTMASK 0xe
+#define KIOC_CTRLMASK 0x30
+#define KIOC_ALTGMASK 0x200
+#define KIOC_NUMLMASK 0x800
#define HOLE 0x302 /* value for kio_entry to say `really type 3' */
@@ -101,8 +101,8 @@ struct kiockey {
#define KIOCCMD _IOW('k', 8, int) /* X uses this to ring bell */
#define KIOCTYPE _IOR('k', 9, int) /* get keyboard type */
#define KIOCSDIRECT _IOW('k', 10, int) /* keys to console? */
-#define KIOCSKEY _IOW('k', 12, struct kiockey) /* set xlat mode */
-#define KIOCGKEY _IOWR('k', 13, struct kiockey) /* get xlat mode */
+#define KIOCSKEY _IOW('k', 12, struct kiockey) /* set xlat mode */
+#define KIOCGKEY _IOWR('k', 13, struct kiockey) /* get xlat mode */
#define KIOCLAYOUT _IOR('k', 20, int) /* get keyboard layout */
#define KIOCSLED _IOW('k', 14, char) /* set LED state */
#define KIOCGLED _IOR('k', 15, char) /* get LED state */
diff --git a/sys/arch/sparc/include/oldmon.h b/sys/arch/sparc/include/oldmon.h
index 02cbf052656..d18a256f05c 100644
--- a/sys/arch/sparc/include/oldmon.h
+++ b/sys/arch/sparc/include/oldmon.h
@@ -1,4 +1,4 @@
-/* $NetBSD: oldmon.h,v 1.5 1995/09/03 21:36:06 pk Exp $ */
+/* $NetBSD: oldmon.h,v 1.11 1996/03/31 22:21:38 pk Exp $ */
/*
* Copyright (C) 1985 Regents of the University of California
@@ -74,24 +74,56 @@ struct devinfo {
};
/*
+ * A "stand alone I/O request".
+ * This is passed as the main argument to the PROM I/O routines
+ * in the `om_boottable' structure.
+ */
+struct saioreq {
+ char si_flgs;
+ struct om_boottable *si_boottab;/* Points to boottab entry if any */
+ char *si_devdata; /* Device-specific data pointer */
+ int si_ctlr; /* Controller number or address */
+ int si_unit; /* Unit number within controller */
+ long si_boff; /* Partition number within unit */
+ long si_cyloff;
+ long si_offset;
+ long si_bn; /* Block number to R/W */
+ char *si_ma; /* Memory address to R/W */
+ int si_cc; /* Character count to R/W */
+ struct saif *si_sif; /* net if. pointer (set by b_open) */
+ char *si_devaddr; /* Points to mapped in device */
+ char *si_dmaaddr; /* Points to allocated DMA space */
+};
+#define SAIO_F_READ 0x01
+#define SAIO_F_WRITE 0x02
+#define SAIO_F_ALLOC 0x04
+#define SAIO_F_FILE 0x08
+#define SAIO_F_EOF 0x10 /* EOF on device */
+#define SAIO_F_AJAR 0x20 /* Descriptor "ajar" (stopped but not closed) */
+
+
+/*
* The table entry that describes a device. It exists in the PROM; a
* pointer to it is passed in MachMonBootParam. It can be used to locate
* PROM subroutines for opening, reading, and writing the device.
*
* When using this interface, only one device can be open at once.
*
- * NOTE: I am not sure what arguments boot, open, close, and strategy take.
- * What is here is just translated verbatim from the sun monitor code. We
+ * NOTE: I am not sure what arguments boot, open, close, and strategy take.
+ * What is here is just translated verbatim from the sun monitor code. We
* should figure this out eventually if we need it.
*/
struct om_boottable {
char b_devname[2]; /* The name of the device */
- int (*b_probe)(); /* probe() --> -1 or found controller
+ int (*b_probe) __P((void)); /* probe() --> -1 or found controller
number */
- int (*b_boot)(); /* boot(bp) --> -1 or start address */
- int (*b_open)(); /* open(iobp) --> -1 or 0 */
- int (*b_close)(); /* close(iobp) --> -1 or 0 */
- int (*b_strategy)(); /* strategy(iobp,rw) --> -1 or 0 */
+ int (*b_boot) __P((void)); /* boot(bp) --> -1 or start address */
+ int (*b_open)
+ __P((struct saioreq *));/* open(iobp) --> -1 or 0 */
+ int (*b_close)
+ __P((struct saioreq *));/* close(iobp) --> -1 or 0 */
+ int (*b_strategy)
+ __P((struct saioreq *, int));/* strategy(iobp,rw) --> -1 or 0 */
char *b_desc; /* Printable string describing dev */
struct devinfo *b_devinfo; /* info to configure device. */
};
@@ -121,7 +153,7 @@ struct om_bootparam {
*/
struct om_vector {
char *initSp; /* Initial system stack ptr for hardware */
- int (*startMon)(); /* Initial PC for hardware */
+ int (*startMon) __P((void));/* Initial PC for hardware */
int *diagberr; /* Bus err handler for diags */
/* Monitor and hardware revision and identification */
@@ -142,8 +174,8 @@ struct om_vector {
#define PROMDEV_TTYB 2 /* in/out to ttyb */
/* Keyboard input (scanned by monitor nmi routine) */
- int (*getKey)(); /* Get next key if one exists */
- int (*initGetKey)(); /* Initialize get key */
+ int (*getKey) __P((void)); /* Get next key if one exists */
+ int (*initGetKey) __P((void));/* Initialize get key */
u_int *translation; /* Kbd translation selector */
u_char *keyBid; /* Keyboard ID byte */
int *screen_x; /* V2: Screen x pos (R/O) */
@@ -154,10 +186,11 @@ struct om_vector {
char *monId;
/* Frame buffer output and terminal emulation */
- int (*fbWriteChar)(); /* Write a character to FB */
+ int (*fbWriteChar) __P((void));/* Write a character to FB */
int *fbAddr; /* Address of frame buffer */
char **font; /* Font table for FB */
- void (*fbWriteStr) __P((char *, int)); /* Quickly write string to FB */
+ void (*fbWriteStr) __P((char *, int));
+ /* Quickly write string to FB */
/* Reboot interface routine -- resets and reboots system. */
void (*reBoot) __P((char *)); /* e.g. reBoot("xy()vmunix") */
@@ -166,30 +199,30 @@ struct om_vector {
u_char *lineBuf; /* The line input buffer */
u_char **linePtr; /* Cur pointer into linebuf */
int *lineSize; /* length of line in linebuf */
- int (*getLine)(); /* Get line from user */
- u_char (*getNextChar)(); /* Get next char from linebuf */
- u_char (*peekNextChar)(); /* Peek at next char */
+ int (*getLine) __P((void)); /* Get line from user */
+ u_char (*getNextChar) __P((void));/* Get next char from linebuf */
+ u_char (*peekNextChar) __P((void));/* Peek at next char */
int *fbThere; /* =1 if frame buffer there */
- int (*getNum)(); /* Grab hex num from line */
+ int (*getNum) __P((void)); /* Grab hex num from line */
/* Print formatted output to current output sink */
- int (*printf)(); /* Similar to "Kernel printf" */
- int (*printHex)(); /* Format N digits in hex */
+ int (*printf) __P((void)); /* Similar to "Kernel printf" */
+ int (*printHex) __P((void));/* Format N digits in hex */
/* Led stuff */
u_char *leds; /* RAM copy of LED register */
- int (*setLeds)(); /* Sets LED's and RAM copy */
+ int (*setLeds) __P((void)); /* Sets LED's and RAM copy */
- /* Non-maskable interrupt (nmi) information */
- int (*nmiAddr)(); /* Addr for level 7 vector */
- void (*abortEntry) __P((void)); /* Entry for keyboard abort */
+ /* Non-maskable interrupt (nmi) information */
+ int (*nmiAddr) __P((void)); /* Addr for level 7 vector */
+ void (*abortEntry) __P((void));/* Entry for keyboard abort */
int *nmiClock; /* Counts up in msec */
/* Frame buffer type: see <machine/fbio.h> */
int *fbType;
/* Assorted other things */
- u_long romvecVersion; /* Version # of Romvec */
+ u_long romvecVersion; /* Version # of Romvec */
struct globram *globRam; /* monitor global variables */
caddr_t kbdZscc; /* Addr of keyboard in use */
@@ -200,10 +233,12 @@ struct om_vector {
long *resetMap; /* pgmap entry for resetaddr */
/* Really struct pgmapent * */
- __dead void (*exitToMon) __P((void)); /* Exit from user program */
+ __dead void (*exitToMon)
+ __P((void)) __attribute__((noreturn));/* Exit from user program */
u_char **memorybitmap; /* V1: &{0 or &bits} */
- void (*setcxsegmap)(); /* Set seg in any context */
- void (**vector_cmd)(); /* V2: Handler for 'v' cmd */
+ void (*setcxsegmap) /* Set seg in any context */
+ __P((int, caddr_t, int));
+ void (**vector_cmd) __P((u_long, char *));/* V2: Handler for 'v' cmd */
u_long *ExpectedTrapSig;
u_long *TrapVectorTable;
int dummy1z;
@@ -227,9 +262,9 @@ struct om_vector {
/*
* OLDMON_STARTVADDR and OLDMON_ENDVADDR denote the range of the damn monitor.
- *
+ *
* supposedly you can steal pmegs within this range that do not contain
- * valid pages.
+ * valid pages.
*/
#define OLDMON_STARTVADDR 0xFFD00000
#define OLDMON_ENDVADDR 0xFFF00000
@@ -248,33 +283,6 @@ struct om_vector {
#define MONSHORTPAGE 0x0FFFE000
#define MONSHORTSEG 0x0FFE0000
-/*
- * A "stand alone I/O request".
- * This is passed as the main argument to the PROM I/O routines
- * in the `om_boottable' structure.
- */
-struct saioreq {
- char si_flgs;
- struct om_boottable *si_boottab;/* Points to boottab entry if any */
- char *si_devdata; /* Device-specific data pointer */
- int si_ctlr; /* Controller number or address */
- int si_unit; /* Unit number within controller */
- long si_boff; /* Partition number within unit */
- long si_cyloff;
- long si_offset;
- long si_bn; /* Block number to R/W */
- char *si_ma; /* Memory address to R/W */
- int si_cc; /* Character count to R/W */
- struct saif *si_sif; /* net if. pointer (set by b_open) */
- char *si_devaddr; /* Points to mapped in device */
- char *si_dmaaddr; /* Points to allocated DMA space */
-};
-#define SAIO_F_READ 0x01
-#define SAIO_F_WRITE 0x02
-#define SAIO_F_ALLOC 0x04
-#define SAIO_F_FILE 0x08
-#define SAIO_F_EOF 0x10 /* EOF on device */
-#define SAIO_F_AJAR 0x20 /* Descriptor "ajar" (stopped but not closed) */
/*
@@ -288,11 +296,18 @@ struct saioreq {
*/
struct saif {
/* transmit packet, returns zero on success. */
- int (*sif_xmit) __P((void *devdata, char *buf, int len));
+ int (*sif_xmit)(void *devdata, char *buf, int len);
/* wait for packet, zero if none arrived */
- int (*sif_poll) __P((void *devdata, char *buf));
+ int (*sif_poll)(void *devdata, char *buf);
/* reset interface, set addresses, etc. */
- int (*sif_reset) __P((void *devdata, struct saioreq *sip));
+ int (*sif_reset)(void *devdata, struct saioreq *sip);
/* Later (sun4 only) proms have more stuff here. */
};
-#endif /* MACHINE_OLDMON_H */
+
+
+#if defined(SUN4)
+void oldmon_w_trace __P((u_long));
+void oldmon_w_cmd __P((u_long, char *));
+#endif
+
+#endif /* _MACHINE_OLDMON_H */
diff --git a/sys/arch/sparc/include/param.h b/sys/arch/sparc/include/param.h
index da5231519df..ce866af3ca1 100644
--- a/sys/arch/sparc/include/param.h
+++ b/sys/arch/sparc/include/param.h
@@ -1,4 +1,4 @@
-/* $NetBSD: param.h,v 1.16 1995/06/28 02:43:50 cgd Exp $ */
+/* $NetBSD: param.h,v 1.24 1996/05/15 02:13:48 mrg Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -43,17 +43,22 @@
*
* @(#)param.h 8.1 (Berkeley) 6/11/93
*/
-
/*
- * Machine dependent constants for Sun-4c (SPARCstation)
+ * Sun4M support by Aaron Brown, Harvard University.
+ * Changes Copyright (c) 1995 The President and Fellows of Harvard College.
+ * All rights reserved.
*/
+#define _MACHINE sparc
#define MACHINE "sparc"
+#define _MACHINE_ARCH sparc
#define MACHINE_ARCH "sparc"
#define MID_MACHINE MID_SPARC
#ifdef _KERNEL /* XXX */
+#ifndef _LOCORE /* XXX */
#include <machine/cpu.h> /* XXX */
#endif /* XXX */
+#endif /* XXX */
/*
* Round p (pointer or byte index) up to a correctly-aligned value for
@@ -71,7 +76,7 @@
* sun4 only 8192 bytes/page
* sun4c/sun4m only 4096 bytes/page
* sun4/sun4c/sun4m either of the above
- *
+ *
* In the later case NBPG, PGOFSET, and PGSHIFT are encoded in variables
* initialized early in locore.s. Since they are variables, rather than
* simple constants, the kernel will not perform slighly worse.
@@ -87,7 +92,7 @@
#define PGSHIFT SUN4CM_PGSHIFT /* log2(NBPG) */
#endif
#if defined(SUN4) && (defined(SUN4C) || defined(SUN4M))
-#if defined(_KERNEL) && !defined(LOCORE)
+#if defined(_KERNEL) && !defined(_LOCORE)
extern int nbpg, pgofset, pgshift;
#endif
#define NBPG nbpg /* bytes/page */
@@ -124,9 +129,9 @@ extern int nbpg, pgofset, pgshift;
#ifndef NMBCLUSTERS
#ifdef GATEWAY
-#define NMBCLUSTERS 2048 /* map size, max cluster allocation */
+#define NMBCLUSTERS 512 /* map size, max cluster allocation */
#else
-#define NMBCLUSTERS 1024 /* map size, max cluster allocation */
+#define NMBCLUSTERS 256 /* map size, max cluster allocation */
#endif
#endif
@@ -168,11 +173,10 @@ extern int nbpg, pgofset, pgshift;
* in DVMA space.
*/
#ifdef _KERNEL
-#ifndef LOCORE
+#ifndef _LOCORE
extern vm_offset_t dvma_base;
extern vm_offset_t dvma_end;
extern struct map *dvmamap;
-#endif
/*
* The dvma resource map is defined in page units, which are numbered 1 to N.
* Use these macros to convert to/from virtual addresses.
@@ -183,21 +187,17 @@ extern struct map *dvmamap;
extern caddr_t kdvma_mapin __P((caddr_t, int, int));
extern caddr_t dvma_malloc __P((size_t, void *, int));
extern void dvma_free __P((caddr_t, size_t, void *));
-#endif
-
-#ifdef _KERNEL
-#ifndef LOCORE
+extern void delay __P((unsigned int));
#define DELAY(n) delay(n)
-#endif
-#endif
-#ifdef _KERNEL
-#ifndef LOCORE
extern int cputyp;
extern int cpumod;
-#endif
-#endif
+extern int mmumod;
+
+#endif /* _LOCORE */
+#endif /* _KERNEL */
+
/*
* Values for the cputyp variable.
*/
@@ -206,8 +206,76 @@ extern int cpumod;
#define CPU_SUN4M 2
/*
* Values for cpumod (cpu model) variable. XXX currently valid only for sun4
+ * or Sun4M
*/
#define SUN4_100 0x22
#define SUN4_200 0x21
#define SUN4_300 0x23
#define SUN4_400 0x24
+#define SUN4M_MS 0x04 /* MicroSPARC-II */
+#define SUN4M_SS 0x40 /* Generic SuperSPARC */
+#define SUN4M_HS 0x10 /* Generic ROSS sparc product (HyperSPARC) */
+#define SUN4M_RT620 0x1f /* Ross HyperSPARC RT620 */
+#define SUN4M_STP1020N 0x41 /* TI SuperSPARC STP1020N */
+#define SUN4M_STP1020P 0x40 /* TI SuperSPARC STP1020P */
+#define SUN4M_STP1020A 0x40 /* TI SuperSPARC STP1020A */
+
+/* Values for mmumod (mmu model) variable. Valid only for Sun4M */
+#define SUN4M_MMU_HS 0x1 /* ROSS HyperSparc */
+#define SUN4M_MMU_SS 0x0 /* TI SuperSPARC */
+#define SUN4M_MMU_MS1 0x4 /* MicroSPARC-I (??? XXX) */
+#define SUN4M_MMU_MS 0x0 /* MicroSPARC-II (ugh, conflicts w/SS) */
+
+/*
+ * Shorthand CPU-type macros. Enumerate all eight cases.
+ * Let compiler optimize away code conditional on constants.
+ */
+#if defined(SUN4M) && defined(SUN4C) && defined(SUN4)
+# define CPU_ISSUN4M (cputyp == CPU_SUN4M)
+# define CPU_ISSUN4C (cputyp == CPU_SUN4C)
+# define CPU_ISSUN4 (cputyp == CPU_SUN4)
+# define CPU_ISSUN4OR4C (cputyp == CPU_SUN4 || cputyp == CPU_SUN4C)
+# define CPU_ISSUN4COR4M (cputyp == CPU_SUN4C || cputyp == CPU_SUN4M)
+#elif defined(SUN4M) && defined(SUN4C) && !defined(SUN4)
+# define CPU_ISSUN4M (cputyp == CPU_SUN4M)
+# define CPU_ISSUN4C (cputyp == CPU_SUN4C)
+# define CPU_ISSUN4 (0)
+# define CPU_ISSUN4OR4C (cputyp == CPU_SUN4C)
+# define CPU_ISSUN4COR4M (cputyp == CPU_SUN4C || cputyp == CPU_SUN4M)
+#elif defined(SUN4M) && !defined(SUN4C) && defined(SUN4)
+# define CPU_ISSUN4M (cputyp == CPU_SUN4M)
+# define CPU_ISSUN4C (0)
+# define CPU_ISSUN4 (cputyp == CPU_SUN4)
+# define CPU_ISSUN4OR4C (cputyp == CPU_SUN4)
+# define CPU_ISSUN4COR4M (cputyp == CPU_SUN4M)
+#elif defined(SUN4M) && !defined(SUN4C) && !defined(SUN4)
+# define CPU_ISSUN4M (1)
+# define CPU_ISSUN4C (0)
+# define CPU_ISSUN4 (0)
+# define CPU_ISSUN4OR4C (0)
+# define CPU_ISSUN4COR4M (1)
+#elif !defined(SUN4M) && defined(SUN4C) && defined(SUN4)
+# define CPU_ISSUN4M (0)
+# define CPU_ISSUN4C (cputyp == CPU_SUN4C)
+# define CPU_ISSUN4 (cputyp == CPU_SUN4)
+# define CPU_ISSUN4OR4C (1)
+# define CPU_ISSUN4COR4M (cputyp == CPU_SUN4C)
+#elif !defined(SUN4M) && defined(SUN4C) && !defined(SUN4)
+# define CPU_ISSUN4M (0)
+# define CPU_ISSUN4C (1)
+# define CPU_ISSUN4 (0)
+# define CPU_ISSUN4OR4C (1)
+# define CPU_ISSUN4COR4M (1)
+#elif !defined(SUN4M) && !defined(SUN4C) && defined(SUN4)
+# define CPU_ISSUN4M (0)
+# define CPU_ISSUN4C (0)
+# define CPU_ISSUN4 (1)
+# define CPU_ISSUN4OR4C (1)
+# define CPU_ISSUN4COR4M (0)
+#elif !defined(SUN4M) && !defined(SUN4C) && !defined(SUN4)
+# define CPU_ISSUN4M (0)
+# define CPU_ISSUN4C (0)
+# define CPU_ISSUN4 (0)
+# define CPU_ISSUN4OR4C (0)
+# define CPU_ISSUN4COR4M (0)
+#endif
diff --git a/sys/arch/sparc/include/pmap.h b/sys/arch/sparc/include/pmap.h
index 6106e4df3aa..e67de44782d 100644
--- a/sys/arch/sparc/include/pmap.h
+++ b/sys/arch/sparc/include/pmap.h
@@ -1,6 +1,8 @@
-/* $NetBSD: pmap.h,v 1.16 1995/04/13 16:24:40 pk Exp $ */
+/* $NetBSD: pmap.h,v 1.22.4.1 1996/06/12 20:29:01 pk Exp $ */
/*
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
*
@@ -10,19 +12,16 @@
*
* All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
+ * This product includes software developed by Aaron Brown and
+ * Harvard University.
* This product includes software developed by the University of
* California, Lawrence Berkeley Laboratory.
*
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
+ * @InsertRedistribution@
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
+ * This product includes software developed by Aaron Brown and
+ * Harvard University.
* This product includes software developed by the University of
* California, Berkeley and its contributors.
* 4. Neither the name of the University nor the names of its contributors
@@ -75,6 +74,9 @@
* requires 3584 segments, while the kernel (including DVMA) requires
* only 512 segments.
*
+ *
+ ** FOR THE SUN4/SUN4C
+ *
* The segment map entry for virtual segment vseg is offset in
* pmap->pm_rsegmap by 0 if pmap is not the kernel pmap, or by
* NUSEG if it is. We keep a pointer called pmap->pm_segmap
@@ -102,13 +104,37 @@
* `stolen' out of the the MMU, we just keep all its PTEs there, and
* have no software copies. Its mmu entries are nonetheless kept on lists
* so that the code that fiddles with mmu lists has something to fiddle.
+ *
+ ** FOR THE SUN4M
+ *
+ * On this architecture, the virtual-to-physical translation (page) tables
+ * are *not* stored within the MMU as they are in the earlier Sun architect-
+ * ures; instead, they are maintained entirely within physical memory (there
+ * is a TLB cache to prevent the high performance hit from keeping all page
+ * tables in core). Thus there is no need to dynamically allocate PMEGs or
+ * SMEGs; only contexts must be shared.
+ *
+ * We maintain two parallel sets of tables: one is the actual MMU-edible
+ * hierarchy of page tables in allocated kernel memory; these tables refer
+ * to each other by physical address pointers in SRMMU format (thus they
+ * are not very useful to the kernel's management routines). The other set
+ * of tables is similar to those used for the Sun4/100's 3-level MMU; it
+ * is a hierarchy of regmap and segmap structures which contain kernel virtual
+ * pointers to each other. These must (unfortunately) be kept in sync.
+ *
*/
#define NKREG ((int)((-(unsigned)KERNBASE) / NBPRG)) /* i.e., 8 */
#define NUREG (256 - NKREG) /* i.e., 248 */
TAILQ_HEAD(mmuhd,mmuentry);
-/* data appearing in both user and kernel pmaps */
+/*
+ * data appearing in both user and kernel pmaps
+ *
+ * note: if we want the same binaries to work on the 4/4c and 4m, we have to
+ * include the fields for both to make sure that the struct kproc
+ * is the same size.
+ */
struct pmap {
union ctxinfo *pm_ctx; /* current context, if any */
int pm_ctxnum; /* current context's number */
@@ -117,10 +143,15 @@ struct pmap {
#endif
int pm_refcount; /* just what it says */
- struct mmuhd pm_reglist; /* MMU regions on this pmap */
- struct mmuhd pm_seglist; /* MMU segments on this pmap */
+ struct mmuhd pm_reglist; /* MMU regions on this pmap (4/4c) */
+ struct mmuhd pm_seglist; /* MMU segments on this pmap (4/4c) */
+
void *pm_regstore;
struct regmap *pm_regmap;
+
+ int *pm_reg_ptps; /* SRMMU-edible region table for 4m */
+ int pm_reg_ptps_pa; /* _Physical_ address of pm_reg_ptps */
+
int pm_gap_start; /* Starting with this vreg there's */
int pm_gap_end; /* no valid mapping until here */
@@ -129,18 +160,29 @@ struct pmap {
struct regmap {
struct segmap *rg_segmap; /* point to NSGPRG PMEGs */
- smeg_t rg_smeg; /* the MMU region number */
+ int *rg_seg_ptps; /* SRMMU-edible segment tables (NULL
+ * indicates invalid region (4m) */
+ smeg_t rg_smeg; /* the MMU region number (4c) */
u_char rg_nsegmap; /* number of valid PMEGS */
};
struct segmap {
int *sg_pte; /* points to NPTESG PTEs */
- pmeg_t sg_pmeg; /* the MMU segment number */
+ pmeg_t sg_pmeg; /* the MMU segment number (4c) */
u_char sg_npte; /* number of valid PTEs per seg */
};
typedef struct pmap *pmap_t;
+#if 0
+struct kvm_cpustate {
+ int kvm_npmemarr;
+ struct memarr kvm_pmemarr[MA_SIZE];
+ int kvm_seginval; /* [4,4c] */
+ struct segmap kvm_segmap_store[NKREG*NSEGRG]; /* [4,4c] */
+}/*not yet used*/;
+#endif
+
#ifdef _KERNEL
#define PMAP_NULL ((pmap_t)0)
@@ -161,17 +203,24 @@ extern vm_offset_t vm_first_phys, vm_num_phys;
#define PMAP_VME16 2 /* etc */
#define PMAP_VME32 3 /* etc */
#define PMAP_NC 4 /* tells pmap_enter to set PG_NC */
-#define PMAP_TNC 7 /* mask to get PG_TYPE & PG_NC */
-/*#define PMAP_IOC 0x00800000 -* IO cacheable, NOT shifted */
+#define PMAP_TYPE4M 0x78 /* mask to get 4m page type */
+#define PMAP_PTESHFT4M 25 /* right shift to put type in pte */
+#define PMAP_SHFT4M 0x3 /* left shift to extract type */
+#define PMAP_TNC \
+ (CPU_ISSUN4M?127:7) /* mask to get PG_TYPE & PG_NC */
+/*#define PMAP_IOC 0x00800000 -* IO cacheable, NOT shifted */
+
+
+#if xxx
void pmap_bootstrap __P((int nmmu, int nctx, int nregion));
int pmap_count_ptes __P((struct pmap *));
-vm_offset_t pmap_prefer __P((vm_offset_t, vm_offset_t));
+void pmap_prefer __P((vm_offset_t, vm_offset_t *));
int pmap_pa_exists __P((vm_offset_t));
-int pmap_dumpsize __P((void));
-int pmap_dumpmmu __P((int (*)__P((dev_t, daddr_t, caddr_t, size_t)),
- daddr_t));
-int mmu_pagein __P((struct pmap *, int va, vm_prot_t prot));
+#endif
+int pmap_dumpsize __P((void));
+int pmap_dumpmmu __P((int (*)__P((dev_t, daddr_t, caddr_t, size_t)),
+ daddr_t));
#define pmap_kernel() (&kernel_pmap_store)
#define pmap_resident_count(pmap) pmap_count_ptes(pmap)
@@ -179,7 +228,144 @@ int mmu_pagein __P((struct pmap *, int va, vm_prot_t prot));
#define PMAP_ACTIVATE(pmap, pcb, iscurproc)
#define PMAP_DEACTIVATE(pmap, pcb)
-#define PMAP_PREFER(pa, va) pmap_prefer((pa), (va))
+#define PMAP_PREFER(fo, ap) pmap_prefer((fo), (ap))
+
+#define PMAP_EXCLUDE_DECLS /* tells MI pmap.h *not* to include decls */
+
+/* FUNCTION DECLARATIONS FOR COMMON PMAP MODULE */
+
+void pmap_bootstrap __P((int nmmu, int nctx, int nregion));
+int pmap_count_ptes __P((struct pmap *));
+void pmap_prefer __P((vm_offset_t, vm_offset_t *));
+int pmap_pa_exists __P((vm_offset_t));
+void *pmap_bootstrap_alloc __P((int));
+void pmap_change_wiring __P((pmap_t, vm_offset_t, boolean_t));
+void pmap_collect __P((pmap_t));
+void pmap_copy __P((pmap_t,
+ pmap_t, vm_offset_t, vm_size_t, vm_offset_t));
+pmap_t pmap_create __P((vm_size_t));
+void pmap_destroy __P((pmap_t));
+void pmap_init __P((void));
+vm_offset_t pmap_map __P((vm_offset_t, vm_offset_t, vm_offset_t, int));
+void pmap_pageable __P((pmap_t,
+ vm_offset_t, vm_offset_t, boolean_t));
+vm_offset_t pmap_phys_address __P((int));
+void pmap_pinit __P((pmap_t));
+void pmap_reference __P((pmap_t));
+void pmap_release __P((pmap_t));
+void pmap_remove __P((pmap_t, vm_offset_t, vm_offset_t));
+void pmap_update __P((void));
+u_int pmap_free_pages __P((void));
+void pmap_init __P((void));
+boolean_t pmap_next_page __P((vm_offset_t *));
+int pmap_page_index __P((vm_offset_t));
+void pmap_virtual_space __P((vm_offset_t *, vm_offset_t *));
+void pmap_redzone __P((void));
+void kvm_uncache __P((caddr_t, int));
+struct user;
+void switchexit __P((vm_map_t, struct user *, int));
+int mmu_pagein __P((struct pmap *pm, int, int));
+#ifdef DEBUG
+int mmu_pagein4m __P((struct pmap *pm, int, int));
+#endif
+
+
+/* SUN4/SUN4C SPECIFIC DECLARATIONS */
+
+#if defined(SUN4) || defined(SUN4C)
+void pmap_clear_modify4_4c __P((vm_offset_t pa));
+void pmap_clear_reference4_4c __P((vm_offset_t pa));
+void pmap_copy_page4_4c __P((vm_offset_t, vm_offset_t));
+void pmap_enter4_4c __P((pmap_t,
+ vm_offset_t, vm_offset_t, vm_prot_t, boolean_t));
+vm_offset_t pmap_extract4_4c __P((pmap_t, vm_offset_t));
+boolean_t pmap_is_modified4_4c __P((vm_offset_t pa));
+boolean_t pmap_is_referenced4_4c __P((vm_offset_t pa));
+void pmap_page_protect4_4c __P((vm_offset_t, vm_prot_t));
+void pmap_protect4_4c __P((pmap_t,
+ vm_offset_t, vm_offset_t, vm_prot_t));
+void pmap_zero_page4_4c __P((vm_offset_t));
+void pmap_changeprot4_4c __P((pmap_t, vm_offset_t, vm_prot_t, int));
+
+#endif
+
+/* SIMILAR DECLARATIONS FOR SUN4M MODULE */
+
+#if defined(SUN4M)
+void pmap_clear_modify4m __P((vm_offset_t pa));
+void pmap_clear_reference4m __P((vm_offset_t pa));
+void pmap_copy_page4m __P((vm_offset_t, vm_offset_t));
+void pmap_enter4m __P((pmap_t,
+ vm_offset_t, vm_offset_t, vm_prot_t, boolean_t));
+vm_offset_t pmap_extract4m __P((pmap_t, vm_offset_t));
+boolean_t pmap_is_modified4m __P((vm_offset_t pa));
+boolean_t pmap_is_referenced4m __P((vm_offset_t pa));
+void pmap_page_protect4m __P((vm_offset_t, vm_prot_t));
+void pmap_protect4m __P((pmap_t,
+ vm_offset_t, vm_offset_t, vm_prot_t));
+void pmap_zero_page4m __P((vm_offset_t));
+void pmap_changeprot4m __P((pmap_t, vm_offset_t, vm_prot_t, int));
+
+#endif /* defined SUN4M */
+
+#if !defined(SUN4M) && (defined(SUN4) || defined(SUN4C))
+
+#define pmap_clear_modify pmap_clear_modify4_4c
+#define pmap_clear_reference pmap_clear_reference4_4c
+#define pmap_copy_page pmap_copy_page4_4c
+#define pmap_enter pmap_enter4_4c
+#define pmap_extract pmap_extract4_4c
+#define pmap_is_modified pmap_is_modified4_4c
+#define pmap_is_referenced pmap_is_referenced4_4c
+#define pmap_page_protect pmap_page_protect4_4c
+#define pmap_protect pmap_protect4_4c
+#define pmap_zero_page pmap_zero_page4_4c
+#define pmap_changeprot pmap_changeprot4_4c
+
+#elif defined(SUN4M) && !(defined(SUN4) || defined(SUN4C))
+
+#define pmap_clear_modify pmap_clear_modify4m
+#define pmap_clear_reference pmap_clear_reference4m
+#define pmap_copy_page pmap_copy_page4m
+#define pmap_enter pmap_enter4m
+#define pmap_extract pmap_extract4m
+#define pmap_is_modified pmap_is_modified4m
+#define pmap_is_referenced pmap_is_referenced4m
+#define pmap_page_protect pmap_page_protect4m
+#define pmap_protect pmap_protect4m
+#define pmap_zero_page pmap_zero_page4m
+#define pmap_changeprot pmap_changeprot4m
+
+#else /* must use function pointers */
+
+extern void (*pmap_clear_modify_p) __P((vm_offset_t pa));
+extern void (*pmap_clear_reference_p) __P((vm_offset_t pa));
+extern void (*pmap_copy_page_p) __P((vm_offset_t, vm_offset_t));
+extern void (*pmap_enter_p) __P((pmap_t,
+ vm_offset_t, vm_offset_t, vm_prot_t, boolean_t));
+extern vm_offset_t (*pmap_extract_p) __P((pmap_t, vm_offset_t));
+extern boolean_t (*pmap_is_modified_p) __P((vm_offset_t pa));
+extern boolean_t (*pmap_is_referenced_p) __P((vm_offset_t pa));
+extern void (*pmap_page_protect_p) __P((vm_offset_t, vm_prot_t));
+extern void (*pmap_protect_p) __P((pmap_t,
+ vm_offset_t, vm_offset_t, vm_prot_t));
+extern void (*pmap_zero_page_p) __P((vm_offset_t));
+extern void (*pmap_changeprot_p) __P((pmap_t, vm_offset_t,
+ vm_prot_t, int));
+
+#define pmap_clear_modify (*pmap_clear_modify_p)
+#define pmap_clear_reference (*pmap_clear_reference_p)
+#define pmap_copy_page (*pmap_copy_page_p)
+#define pmap_enter (*pmap_enter_p)
+#define pmap_extract (*pmap_extract_p)
+#define pmap_is_modified (*pmap_is_modified_p)
+#define pmap_is_referenced (*pmap_is_referenced_p)
+#define pmap_page_protect (*pmap_page_protect_p)
+#define pmap_protect (*pmap_protect_p)
+#define pmap_zero_page (*pmap_zero_page_p)
+#define pmap_changeprot (*pmap_changeprot_p)
+
+#endif
#endif /* _KERNEL */
diff --git a/sys/arch/sparc/include/profile.h b/sys/arch/sparc/include/profile.h
index c34287882bb..0a27b368163 100644
--- a/sys/arch/sparc/include/profile.h
+++ b/sys/arch/sparc/include/profile.h
@@ -1,4 +1,4 @@
-/* $NetBSD: profile.h,v 1.4 1995/08/14 15:44:36 pk Exp $ */
+/* $NetBSD: profile.h,v 1.6 1996/04/08 20:55:36 pk Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -47,25 +47,26 @@
#ifdef PIC
/* Inline expansion of PICCY_SET() (see <machine/asm.h>). */
#define MCOUNT \
- __asm(".global mcount");\
- __asm("mcount:");\
- __asm("add %o7, 8, %o1");\
- __asm("1: call 2f; nop; 2:");\
- __asm("add %o7,__mcount-1b, %o2");\
- __asm("ld [%o2], %o2");\
- __asm("jmpl %o2, %g0");\
- __asm("add %i7, 8, %o0");
+ asm(".global mcount");\
+ asm("mcount:");\
+ asm("add %o7, 8, %o1");\
+ asm("1: call 2f; nop; 2:");\
+ asm("add %o7,__mcount-1b, %o2");\
+ asm("ld [%o2], %o2");\
+ asm("jmpl %o2, %g0");\
+ asm("add %i7, 8, %o0");
#else
#define MCOUNT \
- __asm(".global mcount");\
- __asm("mcount:");\
- __asm("add %i7, 8, %o0");\
- __asm("sethi %hi(__mcount), %o2");\
- __asm("jmpl %o2 + %lo(__mcount), %g0");\
- __asm("add %o7, 8, %o1");
+ asm(".global mcount");\
+ asm("mcount:");\
+ asm("add %i7, 8, %o0");\
+ asm("sethi %hi(__mcount), %o2");\
+ asm("jmpl %o2 + %lo(__mcount), %g0");\
+ asm("add %o7, 8, %o1");
#endif
#define _MCOUNT_DECL static void _mcount
+_MCOUNT_DECL __P((unsigned long, unsigned long));
#ifdef _KERNEL
/*
diff --git a/sys/arch/sparc/include/psl.h b/sys/arch/sparc/include/psl.h
index 9dc8b1baa26..a93b2724348 100644
--- a/sys/arch/sparc/include/psl.h
+++ b/sys/arch/sparc/include/psl.h
@@ -1,4 +1,4 @@
-/* $NetBSD: psl.h,v 1.7 1995/08/13 00:29:56 mycroft Exp $ */
+/* $NetBSD: psl.h,v 1.11 1996/03/31 22:20:14 pk Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -77,25 +77,36 @@
#define PIL_CLOCK 10
-#if defined(_KERNEL) && !defined(LOCORE)
+#if defined(_KERNEL) && !defined(_LOCORE)
+
+static __inline int getpsr __P((void));
+static __inline void setpsr __P((int));
+static __inline int spl0 __P((void));
+static __inline int splhigh __P((void));
+static __inline void splx __P((int));
+
/*
* GCC pseudo-functions for manipulating PSR (primarily PIL field).
*/
-static __inline int getpsr() {
+static __inline int getpsr()
+{
int psr;
__asm __volatile("rd %%psr,%0" : "=r" (psr));
return (psr);
}
-static __inline void setpsr(int newpsr) {
+static __inline void setpsr(newpsr)
+ int newpsr;
+{
__asm __volatile("wr %0,0,%%psr" : : "r" (newpsr));
__asm __volatile("nop");
__asm __volatile("nop");
__asm __volatile("nop");
}
-static __inline int spl0() {
+static __inline int spl0()
+{
int psr, oldipl;
/*
@@ -121,10 +132,28 @@ static __inline int spl0() {
* into the ipl field.)
*/
#define SPL(name, newipl) \
-static __inline int name() { \
+static __inline int name __P((void)); \
+static __inline int name() \
+{ \
+ int psr, oldipl; \
+ __asm __volatile("rd %%psr,%0" : "=r" (psr)); \
+ oldipl = psr & PSR_PIL; \
+ psr &= ~oldipl; \
+ __asm __volatile("wr %0,%1,%%psr" : : \
+ "r" (psr), "n" ((newipl) << 8)); \
+ __asm __volatile("nop; nop; nop"); \
+ return (oldipl); \
+}
+/* A non-priority-decreasing version of SPL */
+#define SPLHOLD(name, newipl) \
+static __inline int name __P((void)); \
+static __inline int name() \
+{ \
int psr, oldipl; \
__asm __volatile("rd %%psr,%0" : "=r" (psr)); \
oldipl = psr & PSR_PIL; \
+ if ((newipl << 8) <= oldipl) \
+ return oldipl; \
psr &= ~oldipl; \
__asm __volatile("wr %0,%1,%%psr" : : \
"r" (psr), "n" ((newipl) << 8)); \
@@ -136,6 +165,17 @@ SPL(splsoftint, 1)
#define splsoftclock splsoftint
#define splsoftnet splsoftint
+/* audio software interrupts are at software level 4 */
+#define PIL_AUSOFT 4
+SPL(splausoft, PIL_AUSOFT)
+
+/* floppy software interrupts are at software level 4 too */
+#define PIL_FDSOFT 4
+SPL(splfdsoft, PIL_FDSOFT)
+
+/* Block devices */
+SPL(splbio, 5)
+
/* network hardware interrupts are at level 6 */
#define PIL_NET 6
SPL(splnet, PIL_NET)
@@ -144,18 +184,11 @@ SPL(splnet, PIL_NET)
#define PIL_TTY 6
SPL(spltty, PIL_TTY)
-/* Memory allocation (must be as high as highest network or tty device) */
+/*
+ * Memory allocation (must be as high as highest network, tty, or disk device)
+ */
SPL(splimp, 7)
-
-/* audio software interrupts are at software level 4 */
-#define PIL_AUSOFT 4
-SPL(splausoft, PIL_AUSOFT)
-
-/* floppy software interrupts are at software level 4 too */
-#define PIL_FDSOFT 4
-SPL(splfdsoft, PIL_FDSOFT)
-
-SPL(splbio, 9)
+SPLHOLD(splpmap, 7)
SPL(splclock, PIL_CLOCK)
@@ -171,7 +204,8 @@ SPL(splaudio, 13)
/* second sparc timer interrupts at level 14 */
SPL(splstatclock, 14)
-static __inline int splhigh() {
+static __inline int splhigh()
+{
int psr, oldipl;
__asm __volatile("rd %%psr,%0" : "=r" (psr));
@@ -182,7 +216,9 @@ static __inline int splhigh() {
}
/* splx does not have a return value */
-static __inline void splx(int newipl) {
+static __inline void splx(newipl)
+ int newipl;
+{
int psr;
__asm __volatile("rd %%psr,%0" : "=r" (psr));
@@ -190,6 +226,6 @@ static __inline void splx(int newipl) {
"r" (psr & ~PSR_PIL), "rn" (newipl));
__asm __volatile("nop; nop; nop");
}
-#endif /* KERNEL && !LOCORE */
+#endif /* KERNEL && !_LOCORE */
#endif /* PSR_IMPL */
diff --git a/sys/arch/sparc/include/pte.h b/sys/arch/sparc/include/pte.h
index 978e6a84d3a..7418188bb85 100644
--- a/sys/arch/sparc/include/pte.h
+++ b/sys/arch/sparc/include/pte.h
@@ -1,6 +1,8 @@
-/* $NetBSD: pte.h,v 1.12 1995/07/05 17:53:41 pk Exp $ */
+/* $NetBSD: pte.h,v 1.17 1996/05/16 15:57:03 abrown Exp $ */
/*
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
*
@@ -9,7 +11,8 @@
* contributed to Berkeley.
*
* All advertising materials mentioning features or use of this software
- * must display the following acknowledgement:
+ * must display the following acknowledgements:
+ * This product includes software developed by Harvard University.
* This product includes software developed by the University of
* California, Lawrence Berkeley Laboratory.
*
@@ -22,7 +25,8 @@
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
- * must display the following acknowledgement:
+ * must display the following acknowledgements:
+ * This product includes software developed by Harvard University.
* This product includes software developed by the University of
* California, Berkeley and its contributors.
* 4. Neither the name of the University nor the names of its contributors
@@ -45,11 +49,11 @@
*/
/*
- * Sun-4 (sort of) and 4c (SparcStation) Page Table Entries
- * (Sun call them `Page Map Entries').
+ * Sun-4 (sort of), 4c (SparcStation), and 4m Page Table Entries
+ * (Sun calls them `Page Map Entries').
*/
-#ifndef LOCORE
+#ifndef _LOCORE
/*
* Segment maps contain `pmeg' (Page Map Entry Group) numbers.
* A PMEG is simply an index that names a group of 32 (sun4) or
@@ -83,6 +87,17 @@ typedef u_char smeg_t; /* 8 bits needed per Sun-4 regmap entry */
* (for sun4)
* 3. take the value from (2 or 3) above and concatenate va<17:13> to
* get a `page map entry' index. This gives a 32-bit PTE.
+ **
+ * For sun4m:
+ * 1. Use context_reg<3:0> to index the context table (located at
+ * (context_reg << 2) | ((ctx_tbl_ptr_reg >> 2) << 6) ). This
+ * gives a 32-bit page-table-descriptor (PTP).
+ * 2. Use va<31:24> to index the region table located by the PTP from (1):
+ * PTP<31:6> << 10. This gives another PTP for the segment tables
+ * 3. Use va<23:18> to index the segment table located by the PTP from (2)
+ * as follows: PTP<31:4> << 8. This gives another PTP for the page tbl.
+ * 4. Use va<17:12> to index the page table given by (3)'s PTP:
+ * PTP<31:4> << 8. This gives a 32-bit PTE.
*
* In other words:
*
@@ -105,14 +120,46 @@ typedef u_char smeg_t; /* 8 bits needed per Sun-4 regmap entry */
* va_off:12; (offset within page)
* };
*
+ * struct sun4m_virtual_addr {
+ * u_int va_reg:8, (virtual region)
+ * va_seg:6, (virtual segment within region)
+ * va_pg:6, (virtual page within segment)
+ * va_off:12; (offset within page)
+ * };
+ *
* Then, given any `va':
*
* extern smeg_t regmap[16][1<<8]; (3-level MMU only)
* extern pmeg_t segmap[8][1<<12]; ([16][1<<12] for sun4)
* extern int ptetable[128][1<<6]; ([512][1<<5] for sun4)
*
- * (the above being in the hardware, accessed as Alternate Address Spaces)
+ * extern u_int s4m_ctxmap[16]; (sun4m SRMMU only)
+ * extern u_int s4m_regmap[16][1<<8]; (sun4m SRMMU only)
+ * extern u_int s4m_segmap[1<<8][1<<6]; (sun4m SRMMU only)
+ * extern u_int s4m_pagmap[1<<14][1<<6]; (sun4m SRMMU only)
+ *
+ * (the above being in the hardware, accessed as Alternate Address Spaces on
+ * all machines but the Sun4m SRMMU, in which case the tables are in physical
+ * kernel memory. In the 4m architecture, the tables are not layed out as
+ * 2-dim arrays, but are sparsely allocated as needed, and point to each
+ * other.)
*
+ * if (cputyp==CPU_SUN4M) // SPARC Reference MMU
+ * regptp = s4m_ctxmap[curr_ctx];
+ * if (!(regptp & SRMMU_TEPTD)) TRAP();
+ * segptp = *(u_int *)(((regptp & ~0x3) << 4) | va.va_reg);
+ * if (!(segptp & SRMMU_TEPTD)) TRAP();
+ * pagptp = *(u_int *)(((segptp & ~0x3) << 4) | va.va_seg);
+ * if (!(pagptp & SRMMU_TEPTD)) TRAP();
+ * pte = *(u_int *)(((pagptp & ~0x3) << 4) | va.va_pg);
+ * if (!(pte & SRMMU_TEPTE)) TRAP(); // like PG_V
+ * if (usermode && PTE_PROT_LEVEL(pte) > 0x5) TRAP();
+ * if (writing && !PTE_PROT_LEVEL_ALLOWS_WRITING(pte)) TRAP();
+ * if (!(pte & SRMMU_PG_C)) DO_NOT_USE_CACHE_FOR_THIS_ACCESS();
+ * pte |= SRMMU_PG_U;
+ * if (writing) pte |= PG_M;
+ * physaddr = ((pte & SRMMU_PG_PFNUM) << SRMMU_PGSHIFT)|va.va_off;
+ * return;
* if (mmu_3l)
* physreg = regmap[curr_ctx][va.va_reg];
* physseg = segmap[physreg][va.va_seg];
@@ -147,7 +194,7 @@ extern int mmu_3l;
#define SGOFSET (NBPSG - 1) /* mask for segment offset */
/* number of PTEs that map one segment (not number that fit in one segment!) */
-#if defined(SUN4) && defined(SUN4C)
+#if defined(SUN4) && (defined(SUN4C) || defined(SUN4M))
extern int nptesg;
#define NPTESG nptesg /* (which someone will have to initialize) */
#else
@@ -163,6 +210,12 @@ extern int nptesg;
/* virtual address to virtual page number, for Sun-4 and Sun-4c */
#define VA_SUN4_VPG(va) (((int)(va) >> 13) & 31)
#define VA_SUN4C_VPG(va) (((int)(va) >> 12) & 63)
+#define VA_SUN4M_VPG(va) (((int)(va) >> 12) & 63)
+
+/* virtual address to offset within page */
+#define VA_SUN4_OFF(va) (((int)(va)) & 0x1FFF)
+#define VA_SUN4C_OFF(va) (((int)(va)) & 0xFFF)
+#define VA_SUN4M_OFF(va) (((int)(va)) & 0xFFF)
/* truncate virtual address to region base */
#define VA_ROUNDDOWNTOREG(va) ((int)(va) & ~RGOFSET)
@@ -171,16 +224,29 @@ extern int nptesg;
#define VA_ROUNDDOWNTOSEG(va) ((int)(va) & ~SGOFSET)
/* virtual segment to virtual address (must sign extend on holy MMUs!) */
-#if defined(MMU_3L)
-#define VRTOVA(vr) (mmu_3l \
+#if defined(SUN4M) && !(defined(SUN4C) || defined(SUN4))
+#define VRTOVA(vr) ((int)(vr) << RGSHIFT)
+#define VSTOVA(vr,vs) (((int)(vr) << RGSHIFT) + ((int)(vs) << SGSHIFT))
+#else
+#if defined(MMU_3L) || defined(SUN4M) /* hairy.. */
+#if !defined(MMU_3L)
+#define _PTE_HAIRY_3L_TEST (cputyp==CPU_SUN4M)
+#elif !defined(SUN4M)
+#define _PTE_HAIRY_3L_TEST (mmu_3l)
+#else
+#define _PTE_HAIRY_3L_TEST (mmu_3l || cputyp==CPU_SUN4M)
+#endif
+#define VRTOVA(vr) (_PTE_HAIRY_3L_TEST \
? ((int)(vr) << RGSHIFT) \
: (((int)(vr) << (RGSHIFT+2)) >> 2))
-#define VSTOVA(vr,vs) (mmu_3l \
- ? (((int)vr << RGSHIFT) + ((int)vs << SGSHIFT)) \
- : ((((int)vr << (RGSHIFT+2)) >> 2) + ((int)vs << SGSHIFT)))
+#define VSTOVA(vr,vs) (_PTE_HAIRY_3L_TEST \
+ ? (((int)(vr) << RGSHIFT) + ((int)(vs) << SGSHIFT)) \
+ : ((((int)(vr) << (RGSHIFT+2)) >> 2) + ((int)(vs) << SGSHIFT)))
#else
-#define VRTOVA(vr) (((int)vr << (RGSHIFT+2)) >> 2)
-#define VSTOVA(vr,vs) ((((int)vr << (RGSHIFT+2)) >> 2) + ((int)vs << SGSHIFT))
+#define VRTOVA(vr) (((int)(vr) << (RGSHIFT+2)) >> 2)
+#define VSTOVA(vr,vs) ((((int)(vr) << (RGSHIFT+2)) >> 2) + \
+ ((int)(vs) << SGSHIFT))
+#endif
#endif
extern int mmu_has_hole;
@@ -192,17 +258,30 @@ extern int mmu_has_hole;
#define MMU_HOLE_START 0x20000000
#define MMU_HOLE_END 0xe0000000
+#if defined(SUN4M) /* Optimization: sun4m, sun4c have same page */
+#if defined(SUN4) /* size, so they're used interchangeably */
+#define VA_VPG(va) (cputyp==CPU_SUN4 ? VA_SUN4_VPG(va) : VA_SUN4C_VPG(va))
+#define VA_OFF(VA) (cputyp==CPU_SUN4 ? VA_SUN4_OFF(va) : VA_SUN4C_OFF(va))
+#else
+#define VA_VPG(va) VA_SUN4M_VPG(va)
+#define VA_OFF(va) VA_SUN4M_OFF(va)
+#endif /* defined SUN4 */
+#else /* 4m not defined */
#if defined(SUN4) && defined(SUN4C)
#define VA_VPG(va) (cputyp==CPU_SUN4C ? VA_SUN4C_VPG(va) : VA_SUN4_VPG(va))
+#define VA_OFF(va) (cputyp==CPU_SUN4C ? VA_SUN4C_OFF(va) : VA_SUN4_OFF(va))
#endif
#if defined(SUN4C) && !defined(SUN4)
#define VA_VPG(va) VA_SUN4C_VPG(va)
+#define VA_OFF(va) VA_SUN4C_OFF(va)
#endif
#if !defined(SUN4C) && defined(SUN4)
#define VA_VPG(va) VA_SUN4_VPG(va)
+#define VA_OFF(va) VA_SUN4_OFF(va)
#endif
+#endif /* defined 4m */
-/* there is no `struct pte'; we just use `int' */
+/* there is no `struct pte'; we just use `int'; this is for non-4M only */
#define PG_V 0x80000000
#define PG_PROT 0x60000000 /* both protection bits */
#define PG_W 0x40000000 /* allowed to write */
@@ -212,20 +291,23 @@ extern int mmu_has_hole;
#define PG_OBMEM 0x00000000 /* on board memory */
#define PG_OBIO 0x04000000 /* on board I/O (incl. Sbus on 4c) */
-#ifdef SUN4
#define PG_VME16 0x08000000 /* 16-bit-data VME space */
#define PG_VME32 0x0c000000 /* 32-bit-data VME space */
+#if defined(SUN4M)
+#define PG_SUN4M_OBMEM 0x0 /* No type bits=>obmem on 4m */
+#define PG_SUN4M_OBIO 0xf /* obio maps to 0xf on 4M */
+#define SRMMU_PGTYPE 0xf0000000 /* Top 4 bits of pte PPN give type */
#endif
#define PG_U 0x02000000
#define PG_M 0x01000000
-#define PG_IOC 0x00800000 /* IO-cacheable */
+#define PG_IOC 0x00800000
#define PG_MBZ 0x00780000 /* unused; must be zero (oh really?) */
#define PG_PFNUM 0x0007ffff /* n.b.: only 16 bits on sun4c */
#define PG_TNC_SHIFT 26 /* shift to get PG_TYPE + PG_NC */
#define PG_M_SHIFT 24 /* shift to get PG_M, PG_U */
-
+#define PG_M_SHIFT4M 5 /* shift to get SRMMU_PG_M,R on 4m */
/*efine PG_NOACC 0 ** XXX */
#define PG_KR 0x20000000
#define PG_KW 0x60000000
@@ -245,6 +327,18 @@ struct pte {
pg_mbz:5,
pg_pfnum:19;
};
+#if defined(SUN4M)
+struct srmmu_pte {
+ u_int pg_pfnum:20,
+ pg_c:1,
+ pg_m:1,
+ pg_u:1;
+ enum pgprot { pprot_r_r, pprot_rw_rw, pprot_rx_rx, pprot_rwx_rwx,
+ pprot_x_x, pprot_r_rw, pprot_n_rx, pprot_n_rwx }
+ pg_prot:3; /* prot. bits: pprot_<user>_<supervisor> */
+ u_int pg_must_be_2:2;
+};
+#endif
#endif
/*
@@ -258,48 +352,61 @@ struct pte {
#define PG_PROTUWRITE 6 /* PG_V,PG_W,!PG_S */
#define PG_PROTUREAD 4 /* PG_V,!PG_W,!PG_S */
+/* %%%: Fix above and below for 4m? */
+
/* static __inline int PG_VALID(void *va) {
register int t = va; t >>= PG_VSHIFT; return (t == 0 || t == -1);
} */
-#if defined(SUN4M)
-
-/*
- * Reference MMU PTE bits.
- */
-#define SRPTE_PPN_MASK 0x07ffff00
-#define SRPTE_PPN_SHIFT 8
-#define SRPTE_CACHEABLE 0x00000080 /* Page is cacheable */
-#define SRPTE_MOD 0x00000040 /* Page is modified */
-#define SRPTE_REF 0x00000020 /* Page is referenced */
-#define SRPTE_ACCMASK 0x0000001c /* Access rights mask */
-#define SRPTE_ACCSHIFT 2 /* Access rights shift */
-#define SRPTE_TYPEMASK 0x00000003 /* PTE Type */
-#define SRPTE_PTE 0x00000002 /* A PTE (Page Table Entry) */
-#define SRPTE_PTP 0x00000001 /* A PTP (Page Table Pointer) */
/*
- * Reference MMU access permission bits.
- * format: SRACC_sssuuu,
- * where <sss> denote the supervisor rights
- * and <uuu> denote the user rights
+ * Here are the bit definitions for 4M/SRMMU pte's
*/
-#define SRACC_R__R__ 0
-#define SRACC_RW_RW_ 1
-#define SRACC_R_XR_X 2
-#define SRACC_RWXRWX 3
-#define SRACC___X__X 4
-#define SRACC_RW_R__ 5
-#define SRACC_R_X___ 6
-#define SRACC_RWX___ 7
+ /* MMU TABLE ENTRIES */
+#define SRMMU_TEINVALID 0x0 /* invalid (serves as !valid bit) */
+#define SRMMU_TEPTD 0x1 /* Page Table Descriptor */
+#define SRMMU_TEPTE 0x2 /* Page Table Entry */
+#define SRMMU_TERES 0x3 /* reserved */
+#define SRMMU_TETYPE 0x3 /* mask for table entry type */
+ /* PTE FIELDS */
+#define SRMMU_PPNMASK 0xFFFFFF00
+#define SRMMU_PPNSHIFT 0x8
+#define SRMMU_PPNPASHIFT 0x4 /* shift to put ppn into PAddr */
+#define SRMMU_L1PPNSHFT 0x14
+#define SRMMU_L1PPNMASK 0xFFF00000
+#define SRMMU_L2PPNSHFT 0xE
+#define SRMMU_L2PPNMASK 0xFC000
+#define SRMMU_L3PPNSHFT 0x8
+#define SRMMU_L3PPNMASK 0x3F00
+ /* PTE BITS */
+#define SRMMU_PG_C 0x80 /* cacheable */
+#define SRMMU_PG_M 0x40 /* modified (dirty) */
+#define SRMMU_PG_R 0x20 /* referenced */
+#define SRMMU_PGBITSMSK 0xE0
+ /* PTE PROTECTION */
+#define SRMMU_PROT_MASK 0x1C /* Mask protection bits out of pte */
+#define SRMMU_PROT_SHFT 0x2
+#define PPROT_R_R 0x0 /* These are in the form: */
+#define PPROT_RW_RW 0x4 /* PPROT_<u>_<s> */
+#define PPROT_RX_RX 0x8 /* where <u> is the user-mode */
+#define PPROT_RWX_RWX 0xC /* permission, and <s> is the */
+#define PPROT_X_X 0x10 /* supervisor mode permission. */
+#define PPROT_R_RW 0x14 /* R=read, W=write, X=execute */
+#define PPROT_N_RX 0x18 /* N=none. */
+#define PPROT_N_RWX 0x1C
+#define PPROT_WRITE 0x4 /* set iff write priv. allowed */
+#define PPROT_S 0x18 /* effective S bit */
+#define PPROT_U2S_OMASK 0x18 /* OR with prot. to revoke user priv */
+ /* TABLE SIZES */
+#define SRMMU_L1SIZE 0x100
+#define SRMMU_L2SIZE 0x40
+#define SRMMU_L3SIZE 0x40
/*
* IOMMU PTE bits.
*/
-#define IOPTE_PPN_MASK 0x07ffff00
-#define IOPTE_PPN_SHIFT 8
-#define IOPTE_RSVD 0x000000f1
-#define IOPTE_WRITE 0x00000004
-#define IOPTE_VALID 0x00000002
-
-#endif /* SUN4M */
+#define IOPTE_PPN_MASK 0x07ffff00
+#define IOPTE_PPN_SHIFT 8
+#define IOPTE_RSVD 0x000000f1
+#define IOPTE_WRITE 0x00000004
+#define IOPTE_VALID 0x00000002
diff --git a/sys/arch/sparc/include/signal.h b/sys/arch/sparc/include/signal.h
index 1dad46faf47..e8339c38520 100644
--- a/sys/arch/sparc/include/signal.h
+++ b/sys/arch/sparc/include/signal.h
@@ -1,4 +1,4 @@
-/* $NetBSD: signal.h,v 1.3 1995/01/10 19:01:46 jtc Exp $ */
+/* $NetBSD: signal.h,v 1.4 1996/02/01 22:32:35 mycroft Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -47,12 +47,12 @@
#ifndef _SPARC_SIGNAL_H_
#define _SPARC_SIGNAL_H_
-#ifndef LOCORE
+#ifndef _LOCORE
typedef int sig_atomic_t;
#endif
#ifndef _ANSI_SOURCE
-#ifndef LOCORE
+#ifndef _LOCORE
/*
* Information pushed on stack when a signal is delivered.
@@ -74,14 +74,14 @@ struct sigcontext {
int sc_g1; /* %g1 to restore */
int sc_o0; /* %o0 to restore */
};
-#else /* LOCORE */
+#else /* _LOCORE */
#define SC_SP_OFFSET 8
#define SC_PC_OFFSET 12
#define SC_NPC_OFFSET 16
#define SC_PSR_OFFSET 20
#define SC_G1_OFFSET 24
#define SC_O0_OFFSET 28
-#endif /* LOCORE */
+#endif /* _LOCORE */
/*
* `Code' arguments to signal handlers. The names, and the funny numbering.
diff --git a/sys/arch/sparc/include/sun_disklabel.h b/sys/arch/sparc/include/sun_disklabel.h
index e1070bcf273..8f01b84115d 100644
--- a/sys/arch/sparc/include/sun_disklabel.h
+++ b/sys/arch/sparc/include/sun_disklabel.h
@@ -51,9 +51,9 @@
#define SUN_DKMAGIC 55998
/* These are the guys that Sun's dkinfo needs... */
-#define SUNOS_DKIOCGGEOM _IOR('d', 2, struct sun_dkgeom) /* geometry info */
-#define SUNOS_DKIOCINFO _IOR('d', 8, struct sun_dkctlr) /* controller info */
-#define SUNOS_DKIOCGPART _IOR('d', 4, struct sun_dkpart) /* partition info */
+#define DKIOCGGEOM _IOR('d', 2, struct sun_dkgeom) /* geometry info */
+#define DKIOCINFO _IOR('d', 8, struct sun_dkctlr) /* controller info */
+#define DKIOCGPART _IOR('d', 4, struct sun_dkpart) /* partition info */
/* geometry info */
struct sun_dkgeom {
@@ -108,8 +108,8 @@ struct sun_disklabel { /* total size = 512 bytes */
#ifdef _KERNEL
/* reads sun label in sector at [cp..cp+511] and sets *lp to BSD label */
-int sunos_disklabel __P((caddr_t, struct disklabel *)); /* true on success */
+int sun_disklabel __P((caddr_t, struct disklabel *)); /* true on success */
/* compatability dk ioctl's */
-int sunos_dkioctl __P((struct disk *, u_long, caddr_t, int));
+int sun_dkioctl __P((struct disk *, u_long, caddr_t, int));
#endif
diff --git a/sys/arch/sparc/include/svr4_machdep.h b/sys/arch/sparc/include/svr4_machdep.h
index f8c52bab9bd..d2687a7a896 100644
--- a/sys/arch/sparc/include/svr4_machdep.h
+++ b/sys/arch/sparc/include/svr4_machdep.h
@@ -1,4 +1,4 @@
-/* $NetBSD: svr4_machdep.h,v 1.1 1995/03/31 02:55:59 christos Exp $ */
+/* $NetBSD: svr4_machdep.h,v 1.4 1996/03/31 22:21:45 pk Exp $ */
/*
* Copyright (c) 1994 Christos Zoulas
@@ -64,7 +64,7 @@
typedef int svr4_greg_t;
-typedef struct {
+typedef struct {
svr4_greg_t rwin_lo[8];
svr4_greg_t rwin_in[8];
} svr4_rwindow_t;
@@ -78,7 +78,10 @@ typedef struct {
typedef svr4_greg_t svr4_gregset_t[SVR4_SPARC_MAXREG];
typedef struct {
- double fpu_regs[32];
+ union {
+ u_int fp_ri[32];
+ double fp_rd[16];
+ } fpu_regs;
void *fp_q;
unsigned fp_fsr;
u_char fp_nqel;
@@ -99,5 +102,6 @@ void svr4_getcontext __P((struct proc *, struct svr4_ucontext *,
int, int));
int svr4_setcontext __P((struct proc *p, struct svr4_ucontext *));
void svr4_sendsig __P((sig_t, int, int, u_long));
+int svr4_trap __P((int, struct proc *));
#endif /* !_SPARC_SVR4_MACHDEP_H_ */
diff --git a/sys/arch/sparc/include/trap.h b/sys/arch/sparc/include/trap.h
index 9c1f1529592..86c45589f80 100644
--- a/sys/arch/sparc/include/trap.h
+++ b/sys/arch/sparc/include/trap.h
@@ -1,4 +1,4 @@
-/* $NetBSD: trap.h,v 1.6 1995/07/04 22:58:51 christos Exp $ */
+/* $NetBSD: trap.h,v 1.9 1996/05/16 15:57:04 abrown Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -43,6 +43,11 @@
*
* @(#)trap.h 8.1 (Berkeley) 6/11/93
*/
+/*
+ * Sun4m support by Aaron Brown, Harvard University.
+ * Changes Copyright (c) 1995 The President and Fellows of Harvard College.
+ * All rights reserved.
+ */
#ifndef _MACHINE_TRAP_H
#define _MACHINE_TRAP_H
@@ -87,6 +92,9 @@
/* through 0x27 unused */
#define T_CPEXCEPTION 0x28 /* (9) coprocessor exception */
/* 0x29 unused */
+/* through 0x2a unused */
+#define T_STOREBUFFAULT 0x2b /* SuperSPARC: Store buffer copy-back fault */
+/* 0x2c unused */
/* through 0x7f unused */
/* beginning of `user' vectors (from trap instructions) - all priority 12 */
diff --git a/sys/arch/sparc/include/types.h b/sys/arch/sparc/include/types.h
index 4ac77b33791..b9d0a0a6914 100644
--- a/sys/arch/sparc/include/types.h
+++ b/sys/arch/sparc/include/types.h
@@ -1,4 +1,4 @@
-/* $NetBSD: types.h,v 1.8 1995/12/09 04:41:51 mycroft Exp $ */
+/* $NetBSD: types.h,v 1.9 1996/03/14 00:48:30 pk Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -73,13 +73,9 @@ typedef short int16_t;
typedef unsigned short u_int16_t;
typedef int int32_t;
typedef unsigned int u_int32_t;
-/* LONGLONG */
typedef long long int64_t;
-/* LONGLONG */
typedef unsigned long long u_int64_t;
typedef int32_t register_t;
-#define __FORK_BRAINDAMAGE
-
#endif /* _MACHTYPES_H_ */
diff --git a/sys/arch/sparc/include/vmparam.h b/sys/arch/sparc/include/vmparam.h
index ef899abd76a..78ce9c97cee 100644
--- a/sys/arch/sparc/include/vmparam.h
+++ b/sys/arch/sparc/include/vmparam.h
@@ -1,4 +1,4 @@
-/* $NetBSD: vmparam.h,v 1.6 1995/07/05 18:04:48 pk Exp $ */
+/* $NetBSD: vmparam.h,v 1.10 1996/03/14 19:49:20 christos Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -140,8 +140,8 @@
#define MACHINE_NONCONTIG /* VM <=> pmap interface modifier */
-#if defined (_KERNEL) && !defined(LOCORE)
+#if defined (_KERNEL) && !defined(_LOCORE)
struct vm_map;
vm_offset_t dvma_mapin __P((struct vm_map *, vm_offset_t, int, int));
-int dvma_mapout __P((vm_offset_t, vm_offset_t, int));
+void dvma_mapout __P((vm_offset_t, vm_offset_t, int));
#endif
diff --git a/sys/arch/sparc/sparc/amd7930intr.s b/sys/arch/sparc/sparc/amd7930intr.s
index eaaf1656c2d..001695ab6cf 100644
--- a/sys/arch/sparc/sparc/amd7930intr.s
+++ b/sys/arch/sparc/sparc/amd7930intr.s
@@ -44,10 +44,7 @@
*/
#ifndef AUDIO_C_HANDLER
-#ifndef LOCORE
-#define LOCORE
-#endif
-#include "assym.s"
+#include "assym.h"
#include <sparc/sparc/intreg.h>
/* XXX this goes in a header file -- currently, it's hidden in locore.s */
#define INTREG_ADDR 0xf8002000
@@ -95,14 +92,14 @@ _amd7930_trap:
cmp R_data, R_end
bgu 1f
nop
-
+
ldub [R_amd + AMD_BBRB], %l6 ! *d = amd->bbrb
stb %l6, [ R_data ]
cmp R_data, R_end
inc R_data ! au->au_rdata++
bne 1f ! if (d == e)
st R_data, [%l7 + AU_RDATA]
-
+
sethi %hi(INTREG_ADDR), %l5
ldub [%l5 + %lo(INTREG_ADDR)], %l6
or %l6, IE_L4, %l6
@@ -118,7 +115,7 @@ _amd7930_trap:
cmp R_data, R_end
bgu 2f
nop
-
+
ldub [ R_data ], %l6 ! amd->bbtb = *d
stb %l6, [ R_amd + AMD_BBTB ]
@@ -126,7 +123,7 @@ _amd7930_trap:
inc R_data ! au->au_pdata++
bne 2f ! if (d == e)
st R_data, [%l7 + AU_PDATA]
-
+
sethi %hi(INTREG_ADDR), %l5
ldub [%l5 + %lo(INTREG_ADDR)], %l6
or %l6, IE_L4, %l6
diff --git a/sys/arch/sparc/sparc/autoconf.c b/sys/arch/sparc/sparc/autoconf.c
index fde266bbeae..d5fc9eaf742 100644
--- a/sys/arch/sparc/sparc/autoconf.c
+++ b/sys/arch/sparc/sparc/autoconf.c
@@ -1,35 +1,8 @@
-/* $NetBSD: autoconf.c,v 1.38 1995/10/09 22:34:03 pk Exp $ */
+/* $NetBSD: autoconf.c,v 1.58.2.2 1996/07/02 23:49:54 jtc Exp $ */
/*
- * Copyright (c) 1995 Theo de Raadt
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- * 3. All advertising materials mentioning features or use of this software
- * must display the following acknowledgement:
- * This product includes software developed under OpenBSD by
- * Theo de Raadt for Willowglen Singapore.
- * 4. The name of the author may not be used to endorse or promote products
- * derived from this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
- * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
- * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
- * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
- * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
- * SUCH DAMAGE.
- *
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
*
@@ -39,6 +12,7 @@
*
* All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
+ * This product includes software developed by Harvard University.
* This product includes software developed by the University of
* California, Lawrence Berkeley Laboratory.
*
@@ -74,6 +48,7 @@
*/
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/map.h>
#include <sys/buf.h>
#include <sys/disklabel.h>
@@ -84,16 +59,15 @@
#include <sys/dmap.h>
#include <sys/reboot.h>
#include <sys/socket.h>
-#include <sys/systm.h>
-
-#include "fd.h"
-#if NFD > 0
-#include <sys/ioctl.h>
-#include <sys/mtio.h>
-#endif /* NFD */
+#include <sys/malloc.h>
+#include <sys/queue.h>
#include <net/if.h>
+#include <dev/cons.h>
+
+#include <vm/vm.h>
+
#include <machine/autoconf.h>
#include <machine/bsd_openprom.h>
#ifdef SUN4
@@ -102,6 +76,17 @@
#include <sparc/sparc/memreg.h>
#endif
#include <machine/cpu.h>
+#include <machine/ctlreg.h>
+#include <machine/pmap.h>
+#include <sparc/sparc/asm.h>
+#include <sparc/sparc/timerreg.h>
+
+#ifdef DDB
+#include <machine/db_machdep.h>
+#include <ddb/db_sym.h>
+#include <ddb/db_extern.h>
+#endif
+
/*
* The following several variables are related to
@@ -111,26 +96,44 @@
int cold; /* if 1, still working on cold-start */
int fbnode; /* node ID of ROM's console frame buffer */
int optionsnode; /* node ID of ROM's options */
-int cpumod; /* CPU model, valid only if cputyp==CPU_SUN4 */
int mmu_3l; /* SUN4_400 models have a 3-level MMU */
-extern struct promvec *promvec;
-
#ifdef KGDB
-extern int kgdb_debug_panic;
+extern int kgdb_debug_panic;
#endif
static int rootnode;
-int findroot __P((void));
void setroot __P((void));
+static char *str2hex __P((char *, int *));
static int getstr __P((char *, int));
static int findblkmajor __P((struct device *));
static struct device *getdisk __P((char *, int, int, dev_t *));
+static int mbprint __P((void *, char *));
+static void crazymap __P((char *, int *));
+int st_crazymap __P((int));
+void swapconf __P((void));
+void sync_crash __P((void));
+int mainbus_match __P((struct device *, void *, void *));
+static void mainbus_attach __P((struct device *, struct device *, void *));
struct bootpath bootpath[8];
+int nbootpath;
static void bootpath_build __P((void));
static void bootpath_fake __P((struct bootpath *, char *));
static void bootpath_print __P((struct bootpath *));
+int search_prom __P((int, char *));
+
+/*
+ * The mountroot_hook is provided as a mechanism for devices to perform
+ * a special function if they're the root device, such as the floppy
+ * drive ejecting the current disk and prompting for a filesystem floppy.
+ */
+struct mountroot_hook {
+ LIST_ENTRY(mountroot_hook) mr_link;
+ struct device *mr_device;
+ void (*mr_func) __P((struct device *));
+};
+LIST_HEAD(, mountroot_hook) mrh_list;
/*
* Most configuration on the SPARC is done by matching OPENPROM Forth
@@ -144,13 +147,11 @@ matchbyname(parent, vcf, aux)
struct cfdata *cf = vcf;
struct confargs *ca = aux;
-#if defined(SUN4)
- if (cputyp == CPU_SUN4) {
+ if (CPU_ISSUN4) {
printf("WARNING: matchbyname not valid on sun4!");
printf("%s\n", cf->cf_driver->cd_name);
return (0);
}
-#endif
return (strcmp(cf->cf_driver->cd_name, ca->ca_ra.ra_name) == 0);
}
@@ -199,14 +200,13 @@ void getidprom __P((struct idprom *, int size));
void
bootstrap()
{
- int nregion, nsegment, ncontext, node;
+ int nregion = 0, nsegment = 0, ncontext = 0;
extern int msgbufmapped;
#if defined(SUN4)
- extern void oldmon_w_cmd();
- extern struct msgbuf *msgbufp;
-
- if (cputyp == CPU_SUN4) {
+ if (CPU_ISSUN4) {
+ extern void oldmon_w_cmd __P((u_long, char *));
+ extern struct msgbuf *msgbufp;
/*
* XXX
* Some boot programs mess up physical page 0, which
@@ -269,50 +269,186 @@ bootstrap()
}
}
#endif /* SUN4 */
-#if defined(SUN4C) || defined(SUN4M)
- if (cputyp == CPU_SUN4C || cputyp == CPU_SUN4M) {
- node = findroot();
+
+#if defined(SUN4C)
+ if (CPU_ISSUN4C) {
+ register int node = findroot();
nsegment = getpropint(node, "mmu-npmg", 128);
ncontext = getpropint(node, "mmu-nctx", 8);
}
-#endif /* SUN4C || SUN4M */
+#endif /* SUN4C */
+
+#if defined (SUN4M)
+ if (CPU_ISSUN4M) {
+ nsegment = 0;
+ cpumod = (u_int) getpsr() >> 24;
+ mmumod = (u_int) lda(SRMMU_PCR, ASI_SRMMU) >> 28;
+ /*
+ * We use the max. number of contexts on the micro and
+ * hyper SPARCs. The SuperSPARC would let us use up to 65536
+ * contexts (by powers of 2), but we keep it at 4096 since
+ * the table must be aligned to #context*4. With 4K contexts,
+ * we waste at most 16K of memory. Note that the context
+ * table is *always* page-aligned, so there can always be
+ * 1024 contexts without sacrificing memory space (given
+ * that the chip supports 1024 contexts).
+ *
+ * Currently known limits: MS2=256, HS=4096, SS=65536
+ * some old SS's=4096
+ *
+ * XXX Should this be a tuneable parameter?
+ */
+ switch (mmumod) {
+ case SUN4M_MMU_MS1:
+ ncontext = 64;
+ break;
+ case SUN4M_MMU_MS:
+ ncontext = 256;
+ break;
+ default:
+ ncontext = 4096;
+ break;
+ }
+ }
+#endif /* SUN4M */
+
pmap_bootstrap(ncontext, nregion, nsegment);
- msgbufmapped = 1;
+ msgbufmapped = 1; /* enable message buffer */
#ifdef KGDB
- zs_kgdb_init(); /* XXX */
+ zs_kgdb_init(); /* XXX */
#endif
#ifdef DDB
db_machine_init();
ddb_init();
#endif
-}
+ /*
+ * On sun4ms we have to do some nasty stuff here. We need to map
+ * in the interrupt registers (since we need to find out where
+ * they are from the PROM, since they aren't in a fixed place), and
+ * disable all interrupts. We can't do this easily from locore
+ * since the PROM is ugly to use from assembly. We also need to map
+ * in the counter registers because we can't disable the level 14
+ * (statclock) interrupt, so we need a handler early on (ugh).
+ *
+ * NOTE: We *demand* the psl to stay at splhigh() at least until
+ * we get here. The system _cannot_ take interrupts until we map
+ * the interrupt registers.
+ */
+
+#if defined(SUN4M)
+#define getpte4m(va) lda(((va) & 0xFFFFF000) | ASI_SRMMUFP_L3, ASI_SRMMUFP)
+
+ /* First we'll do the interrupt registers */
+ if (CPU_ISSUN4M) {
+ register int node;
+ struct romaux ra;
+ register u_int pte;
+ register int i;
+ extern void setpte4m __P((u_int, u_int));
+ extern struct timer_4m *timerreg_4m;
+ extern struct counter_4m *counterreg_4m;
+
+ if ((node = opennode("/obio/interrupt")) == 0)
+ if ((node=search_prom(findroot(),"interrupt"))==0)
+ panic("bootstrap: could not get interrupt "
+ "node from prom");
+
+ if (!romprop(&ra, "interrupt", node))
+ panic("bootstrap: could not get interrupt properties");
+ if (ra.ra_nvaddrs < 2)
+ panic("bootstrap: less than 2 interrupt regs. available");
+ if (ra.ra_nvaddrs > 5)
+ panic("bootstrap: cannot support capability of > 4 CPUs");
+
+ for (i = 0; i < ra.ra_nvaddrs - 1; i++) {
+
+ pte = getpte4m((u_int)ra.ra_vaddrs[i]);
+ if ((pte & SRMMU_TETYPE) != SRMMU_TEPTE)
+ panic("bootstrap: PROM has invalid mapping for "
+ "processor interrupt register %d",i);
+ pte |= PPROT_S;
+
+ /* Duplicate existing mapping */
+
+ setpte4m(PI_INTR_VA + (_MAXNBPG * i), pte);
+ }
+
+ /*
+ * That was the processor register...now get system register;
+ * it is the last returned by the PROM
+ */
+ pte = getpte4m((u_int)ra.ra_vaddrs[i]);
+ if ((pte & SRMMU_TETYPE) != SRMMU_TEPTE)
+ panic("bootstrap: PROM has invalid mapping for system "
+ "interrupt register");
+ pte |= PPROT_S;
+
+ setpte4m(SI_INTR_VA, pte);
+
+ /* Now disable interrupts */
+ ienab_bis(SINTR_MA);
+
+ /* Send all interrupts to primary processor */
+ *((u_int *)ICR_ITR) = 0;
+
+#ifdef DEBUG
+/* printf("SINTR: mask: 0x%x, pend: 0x%x\n", *(int*)ICR_SI_MASK,
+ *(int*)ICR_SI_PEND);
+*/
+#endif
+
+ /*
+ * Now map in the counters
+ * (XXX: fix for multiple CPUs! We assume 1)
+ * The processor register is the first; the system is the last.
+ * See also timerattach() in clock.c.
+ * This shouldn't be necessary; we ought to keep interrupts off
+ * and/or disable the (level 14) counter...
+ */
+
+ if ((node = opennode("/obio/counter")) == 0)
+ if ((node=search_prom(findroot(),"counter"))==0)
+ panic("bootstrap: could not find counter in OPENPROM");
+
+ if (!romprop(&ra, "counter", node))
+ panic("bootstrap: could not find counter properties");
+
+ counterreg_4m = (struct counter_4m *)ra.ra_vaddrs[0];
+ timerreg_4m = (struct timer_4m *)ra.ra_vaddrs[ra.ra_nvaddrs-1];
+ }
+#endif /* SUN4M */
+}
/*
* bootpath_build: build a bootpath. Used when booting a generic
* kernel to find our root device. Newer proms give us a bootpath,
* for older proms we have to create one. An element in a bootpath
- * has 3 fields: name (device name), val[0], and val[1]. Note that:
+ * has 4 fields: name (device name), val[0], val[1], and val[2]. Note that:
+ * Interpretation of val[] is device-dependent. Some examples:
*
- * if (val[0] == -1) {
+ * if (val[0] == -1) {
* val[1] is a unit number (happens most often with old proms)
* } else {
- * val[0] is a sbus slot, and val[1] is an sbus offset [if sbus]
+ * [sbus device] val[0] is a sbus slot, and val[1] is an sbus offset
+ * [scsi disk] val[0] is target, val[1] is lun, val[2] is partition
+ * [scsi tape] val[0] is target, val[1] is lun, val[2] is file #
* }
+ *
*/
-static void
+
+static void
bootpath_build()
{
register char *cp, *pp;
register struct bootpath *bp;
- int v0val[3];
/*
* On SS1s, promvec->pv_v0bootargs->ba_argv[1] contains the flags
* that were given after the boot command. On SS2s, pv_v0bootargs
* is NULL but *promvec->pv_v2bootargs.v2_bootargs points to
* "vmunix -s" or whatever.
- * ### DO THIS BEFORE pmap_boostrap?
+ * XXX DO THIS BEFORE pmap_boostrap?
*/
bzero(bootpath, sizeof(bootpath));
bp = bootpath;
@@ -332,7 +468,7 @@ bootpath_build()
if (cp == NULL || *cp != '-')
return;
} else {
- /*
+ /*
* Grab boot path from PROM
*/
cp = *promvec->pv_v2bootargs.v2_bootpath;
@@ -348,15 +484,15 @@ bootpath_build()
cp = str2hex(++cp, &bp->val[0]);
if (*cp == ',')
cp = str2hex(++cp, &bp->val[1]);
+ if (*cp == ':')
+ /* XXX - we handle just one char */
+ bp->val[2] = *++cp - 'a', ++cp;
} else {
bp->val[0] = -1; /* no #'s: assume unit 0, no
sbus offset/adddress */
}
-#ifdef notyet
- if (strcmp(bp->name, "iommu") == 0)
- continue;
-#endif
++bp;
+ ++nbootpath;
}
bp->name[0] = 0;
@@ -403,8 +539,9 @@ bootpath_build()
/*
* Fake a ROM generated bootpath.
- * The argument `cp' points to a string such as "xd(0,0,0)bsd"
+ * The argument `cp' points to a string such as "xd(0,0,0)netbsd"
*/
+
static void
bootpath_fake(bp, cp)
struct bootpath *bp;
@@ -412,39 +549,48 @@ bootpath_fake(bp, cp)
{
register char *pp;
int v0val[3];
- char tmpname[8];
-#define BP_APPEND(BP,N,V0,V1) { \
+#define BP_APPEND(BP,N,V0,V1,V2) { \
strcpy((BP)->name, N); \
(BP)->val[0] = (V0); \
(BP)->val[1] = (V1); \
- (BP)++; }
+ (BP)->val[2] = (V2); \
+ (BP)++; \
+ nbootpath++; \
+}
+#if defined(SUN4)
+ if (CPU_ISSUN4M) {
+ printf("twas brillig..\n");
+ return;
+ }
+#endif
pp = cp + 2;
v0val[0] = v0val[1] = v0val[2] = 0;
- if (*pp == '(' /* for vi: ) */
- && *(pp = str2hex(++pp, &v0val[0])) == ','
+ if (*pp == '(' /* for vi: ) */
+ && *(pp = str2hex(++pp, &v0val[0])) == ','
&& *(pp = str2hex(++pp, &v0val[1])) == ',')
(void)str2hex(++pp, &v0val[2]);
-#ifdef SUN4
- if (cputyp == CPU_SUN4) {
+#if defined(SUN4)
+ if (CPU_ISSUN4) {
+ char tmpname[8];
/*
- * xylogics VME dev: xd, xy, xt
+ * xylogics VME dev: xd, xy, xt
* fake looks like: /vmel0/xdc0/xd@1,0
*/
- if (cp[0] == 'x') {
+ if (cp[0] == 'x') {
if (cp[1] == 'd') {/* xd? */
- BP_APPEND(bp,"vmel",-1,0);
+ BP_APPEND(bp, "vmel", -1, 0, 0);
} else {
- BP_APPEND(bp,"vmes",-1,0);
+ BP_APPEND(bp, "vmes", -1, 0, 0);
}
sprintf(tmpname,"x%cc", cp[1]); /* e.g. xdc */
- BP_APPEND(bp,tmpname,-1,v0val[0]);
+ BP_APPEND(bp, tmpname,-1, v0val[0], 0);
sprintf(tmpname,"%c%c", cp[0], cp[1]);
- BP_APPEND(bp,tmpname,v0val[1], v0val[2]); /* e.g. xd */
+ BP_APPEND(bp, tmpname,v0val[1], v0val[2], 0); /* e.g. xd */
return;
}
@@ -453,92 +599,108 @@ bootpath_fake(bp, cp)
* fake looks like: /obio0/le0
*/
if ((cp[0] == 'i' || cp[0] == 'l') && cp[1] == 'e') {
- BP_APPEND(bp,"obio",-1,0);
+ BP_APPEND(bp, "obio", -1, 0, 0);
sprintf(tmpname,"%c%c", cp[0], cp[1]);
- BP_APPEND(bp,tmpname,-1,0);
+ BP_APPEND(bp, tmpname, -1, 0, 0);
return;
}
/*
* scsi: sd, st, sr
- * assume: 4/100 = sw: /obio0/sw0/sd@0,0
- * 4/200 & 4/400 = si/sc: /vmes0/si0/sd@0,0
- * 4/300 = esp: /obio0/esp0/sd@0,0
+ * assume: 4/100 = sw: /obio0/sw0/sd@0,0:a
+ * 4/200 & 4/400 = si/sc: /vmes0/si0/sd@0,0:a
+ * 4/300 = esp: /obio0/esp0/sd@0,0:a
* (note we expect sc to mimic an si...)
*/
- if (cp[0] == 's' &&
+ if (cp[0] == 's' &&
(cp[1] == 'd' || cp[1] == 't' || cp[1] == 'r')) {
-
+
+ int target, lun;
+
switch (cpumod) {
case SUN4_200:
case SUN4_400:
- BP_APPEND(bp,"vmes",-1,0);
- BP_APPEND(bp,"si",-1,v0val[0]);
- sprintf(tmpname,"%c%c", cp[0], cp[1]);
- BP_APPEND(bp,tmpname,v0val[1],v0val[2]);
- return;
+ BP_APPEND(bp, "vmes", -1, 0, 0);
+ BP_APPEND(bp, "si", -1, v0val[0], 0);
+ break;
case SUN4_100:
- if (v0val[0] == 0) {
- BP_APPEND(bp,"obio",-1,0);
- BP_APPEND(bp,"sw",-1,v0val[0]);
- } else {
- BP_APPEND(bp,"obio",-1,0);
- BP_APPEND(bp,"si",-1,v0val[0]-1);
- }
- sprintf(tmpname,"%c%c", cp[0], cp[1]);
- BP_APPEND(bp,tmpname,v0val[1],v0val[2]);
- return;
+ BP_APPEND(bp, "obio", -1, 0, 0);
+ BP_APPEND(bp, "sw", -1, v0val[0], 0);
+ break;
case SUN4_300:
- BP_APPEND(bp,"obio",-1,0);
- BP_APPEND(bp,"esp",-1,v0val[0]);
- sprintf(tmpname,"%c%c", cp[0], cp[1]);
- BP_APPEND(bp,tmpname,v0val[1],v0val[2]);
- return;
+ BP_APPEND(bp, "obio", -1, 0, 0);
+ BP_APPEND(bp, "esp", -1, v0val[0], 0);
+ break;
+ default:
+ panic("bootpath_fake: unknown cpumod %d",
+ cpumod);
+ }
+ /*
+ * Deal with target/lun encodings.
+ * Note: more special casing in dk_establish().
+ */
+ if (oldpvec->monId[0] > '1') {
+ target = v0val[1] >> 3; /* new format */
+ lun = v0val[1] & 0x7;
+ } else {
+ target = v0val[1] >> 2; /* old format */
+ lun = v0val[1] & 0x3;
}
- panic("bootpath_fake: unknown cpumod?");
+ sprintf(tmpname, "%c%c", cp[0], cp[1]);
+ BP_APPEND(bp, tmpname, target, lun, v0val[2]);
+ return;
}
return; /* didn't grok bootpath, no change */
}
#endif /* SUN4 */
-
-#ifdef SUN4C
+
+#if defined(SUN4C)
/*
* sun4c stuff
*/
/*
* floppy: fd
- * fake looks like: /fd@0,0
+ * fake looks like: /fd@0,0:a
*/
if (cp[0] == 'f' && cp[1] == 'd') {
- BP_APPEND(bp,"fd",v0val[1],v0val[2]);
+ /*
+ * Assume `fd(c,u,p)' means:
+ * partition `p' on floppy drive `u' on controller `c'
+ */
+ BP_APPEND(bp, "fd", v0val[0], v0val[1], v0val[2]);
return;
}
/*
- * ethenet: le
+ * ethernet: le
* fake looks like: /sbus0/le0
*/
if (cp[0] == 'l' && cp[1] == 'e') {
- BP_APPEND(bp,"sbus",-1,0);
- BP_APPEND(bp,"le",-1,v0val[0]);
+ BP_APPEND(bp, "sbus", -1, 0, 0);
+ BP_APPEND(bp, "le", -1, v0val[0], 0);
return;
}
/*
* scsi: sd, st, sr
- * fake looks like: /sbus0/esp0/sd@3,0
- */
- if (cp[0] == 's' &&
- (cp[1] == 'd' || cp[1] == 't' || cp[1] == 'r')) {
- BP_APPEND(bp,"sbus",-1,0);
- BP_APPEND(bp,"esp",-1,v0val[0]);
- if (cp[1] == 'r')
- sprintf(tmpname, "cd"); /* use 'cd', not 'sr'*/
+ * fake looks like: /sbus0/esp0/sd@3,0:a
+ */
+ if (cp[0] == 's' && (cp[1] == 'd' || cp[1] == 't' || cp[1] == 'r')) {
+ char tmpname[8];
+ int target, lun;
+
+ BP_APPEND(bp, "sbus", -1, 0, 0);
+ BP_APPEND(bp, "esp", -1, v0val[0], 0);
+ if (cp[1] == 'r')
+ sprintf(tmpname, "cd"); /* netbsd uses 'cd', not 'sr'*/
else
sprintf(tmpname,"%c%c", cp[0], cp[1]);
- BP_APPEND(bp,tmpname,v0val[1], v0val[2]);
+ /* XXX - is TARGET/LUN encoded in v0val[1]? */
+ target = v0val[1];
+ lun = 0;
+ BP_APPEND(bp, tmpname, target, lun, v0val[2]);
return;
}
#endif /* SUN4C */
@@ -547,14 +709,15 @@ bootpath_fake(bp, cp)
/*
* unknown; return
*/
-
+
#undef BP_APPEND
}
/*
* print out the bootpath
*/
-static void
+
+static void
bootpath_print(bp)
struct bootpath *bp;
{
@@ -564,18 +727,22 @@ bootpath_print(bp)
printf("/%s%x", bp->name, bp->val[1]);
else
printf("/%s@%x,%x", bp->name, bp->val[0], bp->val[1]);
+ if (bp->val[2] != 0)
+ printf(":%c", bp->val[2] + 'a');
bp++;
}
printf("\n");
}
+
/*
* save or read a bootpath pointer from the boothpath store.
*
* XXX. required because of SCSI... we don't have control over the "sd"
- * device, so we can't set boot device there. we patch in with
+ * device, so we can't set boot device there. we patch in with
* dk_establish(), and use this to recover the bootpath.
*/
+
struct bootpath *
bootpath_store(storep, bp)
int storep;
@@ -591,7 +758,7 @@ bootpath_store(storep, bp)
return (retval);
}
-/*
+/*
* Set up the sd target mappings for non SUN4 PROMs.
* Find out about the real SCSI target, given the PROM's idea of the
* target of the (boot) device (i.e., the value in bp->v0val[0]).
@@ -603,9 +770,8 @@ crazymap(prop, map)
{
int i;
char *propval;
- struct nodeops *no;
- if (cputyp != CPU_SUN4 && promvec->pv_romvec_vers < 2) {
+ if (!CPU_ISSUN4 && promvec->pv_romvec_vers < 2) {
/*
* Machines with real v0 proms have an `s[dt]-targets' property
* which contains the mapping for us to use. v2 proms donot
@@ -613,7 +779,7 @@ crazymap(prop, map)
*/
propval = getpropstring(optionsnode, prop);
if (propval == NULL || strlen(propval) != 8) {
-build_default_map:
+ build_default_map:
printf("WARNING: %s map is bogus, using default\n",
prop);
for (i = 0; i < 8; ++i)
@@ -640,7 +806,7 @@ build_default_map:
}
}
-int
+int
sd_crazymap(n)
int n;
{
@@ -654,7 +820,7 @@ sd_crazymap(n)
return prom_sd_crazymap[n];
}
-int
+int
st_crazymap(n)
int n;
{
@@ -674,63 +840,76 @@ st_crazymap(n)
* attach it as `mainbus0'. We also set up to handle the PROM `sync'
* command.
*/
+void
configure()
{
struct confargs oca;
register int node = 0;
register char *cp;
- void sync_crash();
+
+ /* Initialize the mountroot_hook list. */
+ LIST_INIT(&mrh_list);
/* build the bootpath */
bootpath_build();
#if defined(SUN4)
- if (cputyp == CPU_SUN4) {
+ if (CPU_ISSUN4) {
extern struct cfdata cfdata[];
- extern struct cfdriver memregcd, obiocd;
+ extern struct cfdriver memreg_cd, obio_cd;
struct cfdata *cf, *memregcf = NULL;
register short *p;
+ struct rom_reg rr;
- /*
- * On the 4/100 obio addresses must be mapped at
- * 0x0YYYYYYY, but alias higher up (we avoid the
- * alias condition because it causes pmap difficulties)
- * XXX: We also assume that 4/[23]00 obio addresses
- * must be 0xZYYYYYYY, where (Z != 0)
- * make sure we get the correct memreg cfdriver!
- */
for (cf = cfdata; memregcf==NULL && cf->cf_driver; cf++) {
- if (cf->cf_driver != &memregcd)
+ if (cf->cf_driver != &memreg_cd ||
+ cf->cf_loc[0] == -1) /* avoid sun4m memreg0 */
continue;
+ /*
+ * On the 4/100 obio addresses must be mapped at
+ * 0x0YYYYYYY, but alias higher up (we avoid the
+ * alias condition because it causes pmap difficulties)
+ * XXX: We also assume that 4/[23]00 obio addresses
+ * must be 0xZYYYYYYY, where (Z != 0)
+ * make sure we get the correct memreg cfdriver!
+ */
if (cpumod==SUN4_100 && (cf->cf_loc[0] & 0xf0000000))
continue;
if (cpumod!=SUN4_100 && !(cf->cf_loc[0] & 0xf0000000))
continue;
for (p = cf->cf_parents; memregcf==NULL && *p >= 0; p++)
- if (cfdata[*p].cf_driver == &obiocd)
+ if (cfdata[*p].cf_driver == &obio_cd)
memregcf = cf;
}
if (memregcf==NULL)
panic("configure: no memreg found!");
- par_err_reg = (int *)bus_map((void *)memregcf->cf_loc[0],
- NBPG, BUS_OBIO);
+
+ rr.rr_iospace = BUS_OBIO;
+ rr.rr_paddr = (void *)memregcf->cf_loc[0];
+ rr.rr_len = NBPG;
+ par_err_reg = (u_int *)bus_map(&rr, NBPG, BUS_OBIO);
if (par_err_reg == NULL)
panic("configure: ROM hasn't mapped memreg!");
}
#endif
-#if defined(SUN4C) || defined(SUN4M)
- if (cputyp == CPU_SUN4C || cputyp == CPU_SUN4M) {
+#if defined(SUN4C)
+ if (CPU_ISSUN4C) {
node = findroot();
cp = getpropstring(node, "device_type");
if (strcmp(cp, "cpu") != 0)
panic("PROM root device type = %s (need CPU)\n", cp);
- *promvec->pv_synchook = sync_crash;
}
#endif
+#if defined(SUN4M)
+ if (CPU_ISSUN4M)
+ node = findroot();
+#endif
+
+ *promvec->pv_synchook = sync_crash;
oca.ca_ra.ra_node = node;
oca.ca_ra.ra_name = cp = "mainbus";
- if (!config_rootfound(cp, (void *)&oca))
+ if (config_rootfound(cp, (void *)&oca) == NULL)
panic("mainbus not configured");
(void)spl0();
@@ -807,7 +986,7 @@ findroot()
int
findnode(first, name)
int first;
- register char *name;
+ register const char *name;
{
register int node;
@@ -828,11 +1007,15 @@ romprop(rp, cp, node)
register int node;
{
register int len;
- union { char regbuf[128]; struct rom_reg rr[RA_MAXREG]; } u;
+ union { char regbuf[256]; struct rom_reg rr[RA_MAXREG]; } u;
static const char pl[] = "property length";
bzero(u.regbuf, sizeof u);
- len = getprop(node, "reg", (void *)u.regbuf, sizeof u.regbuf);
+ len = getprop(node, "reg", (void *)u.regbuf, sizeof(u.regbuf));
+ if (len == -1 &&
+ node_has_property(node, "device_type") &&
+ strcmp(getpropstring(node, "device_type"), "hierarchical") == 0)
+ len = 0;
if (len % sizeof(struct rom_reg)) {
printf("%s \"reg\" %s = %d (need multiple of %d)\n",
cp, pl, len, sizeof(struct rom_reg));
@@ -846,11 +1029,12 @@ romprop(rp, cp, node)
rp->ra_nreg = len / sizeof(struct rom_reg);
bcopy(u.rr, rp->ra_reg, len);
- rp->ra_vaddr = (caddr_t)getpropint(node, "address", 0);
len = getprop(node, "address", (void *)rp->ra_vaddrs,
sizeof(rp->ra_vaddrs));
- if (len == -1)
+ if (len == -1) {
+ rp->ra_vaddr = 0; /* XXX - driver compat */
len = 0;
+ }
if (len & 3) {
printf("%s \"address\" %s = %d (need multiple of 4)\n",
cp, pl, len);
@@ -874,22 +1058,44 @@ romprop(rp, cp, node)
cp, len);
break;
}
+#if defined(SUN4M)
+ if (CPU_ISSUN4M) {
+ /* What's in these high bits anyway? */
+ rp->ra_intr[len].int_pri &= 0xf;
+ /* Look at "interrupts" property too? */
+ }
+#endif
+
}
+#if defined(SUN4M)
+ if (CPU_ISSUN4M) {
+ len = getprop(node, "ranges", (void *)&rp->ra_range,
+ sizeof rp->ra_range);
+ if (len == -1)
+ len = 0;
+ rp->ra_nrange = len / sizeof(struct rom_range);
+ } else
+#endif
+ rp->ra_nrange = 0;
+
return (1);
}
int
-mainbus_match(parent, cf, aux)
+mainbus_match(parent, self, aux)
struct device *parent;
- struct cfdata *cf;
+ void *self;
void *aux;
{
+ struct cfdata *cf = self;
register struct confargs *ca = aux;
register struct romaux *ra = &ca->ca_ra;
return (strcmp(cf->cf_driver->cd_name, ra->ra_name) == 0);
}
+int autoconf_nzs = 0; /* must be global so obio.c can see it */
+
/*
* Attach the mainbus.
*
@@ -902,12 +1108,16 @@ mainbus_attach(parent, dev, aux)
struct device *parent, *dev;
void *aux;
{
- struct confargs oca, *ca = aux;
+ struct confargs oca;
+ register const char *const *ssp, *sp = NULL;
+#if defined(SUN4C) || defined(SUN4M)
+ struct confargs *ca = aux;
register int node0, node;
- register const char *cp, *const *ssp, *sp;
+ const char *const *openboot_special;
#define L1A_HACK /* XXX hack to allow L1-A during autoconf */
#ifdef L1A_HACK
- int nzs = 0, audio = 0;
+ int audio = 0;
+#endif
#endif
#if defined(SUN4)
static const char *const oldmon_special[] = {
@@ -917,8 +1127,8 @@ mainbus_attach(parent, dev, aux)
};
#endif /* SUN4 */
-#if defined(SUN4C) || defined(SUN4M)
- static const char *const openboot_special[] = {
+#if defined(SUN4C)
+ static const char *const openboot_special4c[] = {
/* find these first (end with empty string) */
"memory-error", /* as early as convenient, in case of error */
"eeprom",
@@ -936,17 +1146,40 @@ mainbus_attach(parent, dev, aux)
"virtual-memory",
NULL
};
-#endif /* SUN4C || SUN4M */
+#else
+#define openboot_special4c ((void *)0)
+#endif
+#if defined(SUN4M)
+ static const char *const openboot_special4m[] = {
+ /* find these first */
+ "obio", /* smart enough to get eeprom/etc mapped */
+ "",
- printf("\n");
+ /* ignore these (end with NULL) */
+ /*
+ * These are _root_ devices to ignore. Others must be handled
+ * elsewhere.
+ */
+ "SUNW,sx", /* XXX: no driver for SX yet */
+ "eccmemctl",
+ "virtual-memory",
+ "aliases",
+ "memory",
+ "openprom",
+ "options",
+ "packages",
+ /* we also skip any nodes with device_type == "cpu" */
+ NULL
+ };
+#else
+#define openboot_special4m ((void *)0)
+#endif
- /* configure the cpu */
- node = ca->ca_ra.ra_node;
- oca.ca_ra.ra_node = node;
- oca.ca_ra.ra_name = cp = "cpu";
- oca.ca_ra.ra_paddr = 0;
- oca.ca_ra.ra_nreg = 0;
- config_found(dev, (void *)&oca, mbprint);
+#if defined(SUN4M)
+ if (CPU_ISSUN4M)
+ printf(": %s", getpropstring(ca->ca_ra.ra_node, "name"));
+#endif
+ printf("\n");
/*
* Locate and configure the ``early'' devices. These must be
@@ -954,80 +1187,138 @@ mainbus_attach(parent, dev, aux)
* EEPROM contains the Ethernet address for the LANCE chip.
* If the device cannot be located or configured, panic.
*/
+
#if defined(SUN4)
- if (cputyp == CPU_SUN4) {
+ if (CPU_ISSUN4) {
+ /* Configure the CPU. */
+ bzero(&oca, sizeof(oca));
+ oca.ca_ra.ra_name = "cpu";
+ (void)config_found(dev, (void *)&oca, mbprint);
/* Start at the beginning of the bootpath */
+ bzero(&oca, sizeof(oca));
oca.ca_ra.ra_bp = bootpath;
oca.ca_bustype = BUS_MAIN;
oca.ca_ra.ra_name = "obio";
- if (!config_found(dev, (void *)&oca, mbprint))
- panic(sp);
+ if (config_found(dev, (void *)&oca, mbprint) == NULL)
+ panic("obio missing");
for (ssp = oldmon_special; (sp = *ssp) != NULL; ssp++) {
oca.ca_bustype = BUS_MAIN;
oca.ca_ra.ra_name = sp;
(void)config_found(dev, (void *)&oca, mbprint);
}
+ return;
}
#endif
+
+/*
+ * The rest of this routine is for OBP machines exclusively.
+ */
#if defined(SUN4C) || defined(SUN4M)
- if (cputyp == CPU_SUN4C || cputyp == CPU_SUN4M) {
+ openboot_special = CPU_ISSUN4M
+ ? openboot_special4m
+ : openboot_special4c;
+
+ node = ca->ca_ra.ra_node; /* i.e., the root node */
+
+ /* the first early device to be configured is the cpu */
+#if defined(SUN4M)
+ if (CPU_ISSUN4M) {
+ /* XXX - what to do on multiprocessor machines? */
+ register const char *cp;
+
+ for (node = firstchild(node); node; node = nextsibling(node)) {
+ cp = getpropstring(node, "device_type");
+ if (strcmp(cp, "cpu") == 0)
+ break;
+ }
+ if (node == 0)
+ panic("None of the CPUs found\n");
+ }
+#endif
+
+ oca.ca_ra.ra_node = node;
+ oca.ca_ra.ra_name = "cpu";
+ oca.ca_ra.ra_paddr = 0;
+ oca.ca_ra.ra_nreg = 0;
+ config_found(dev, (void *)&oca, mbprint);
+
+ node = ca->ca_ra.ra_node; /* re-init root node */
+
+ if (promvec->pv_romvec_vers <= 2)
/* remember which frame buffer, if any, is to be /dev/fb */
fbnode = getpropint(node, "fb", 0);
- /* Find the "options" node */
- node0 = firstchild(node);
- optionsnode = findnode(node0, "options");
- if (optionsnode == 0)
- panic("no options in OPENPROM");
+ /* Find the "options" node */
+ node0 = firstchild(node);
+ optionsnode = findnode(node0, "options");
+ if (optionsnode == 0)
+ panic("no options in OPENPROM");
- /* Start at the beginning of the bootpath */
- oca.ca_ra.ra_bp = bootpath;
+ /* Start at the beginning of the bootpath */
+ oca.ca_ra.ra_bp = bootpath;
- for (ssp = openboot_special; *(sp = *ssp) != 0; ssp++) {
- if ((node = findnode(node0, sp)) == 0) {
- printf("could not find %s in OPENPROM\n", sp);
- panic(sp);
- }
- oca.ca_bustype = BUS_MAIN;
- if (!romprop(&oca.ca_ra, sp, node) ||
- !config_found(dev, (void *)&oca, mbprint))
- panic(sp);
+ for (ssp = openboot_special; *(sp = *ssp) != 0; ssp++) {
+ if ((node = findnode(node0, sp)) == 0) {
+ printf("could not find %s in OPENPROM\n", sp);
+ panic(sp);
}
+ oca.ca_bustype = BUS_MAIN;
+ if (!romprop(&oca.ca_ra, sp, node) ||
+ (config_found(dev, (void *)&oca, mbprint) == NULL))
+ panic(sp);
+ }
- /*
- * Configure the rest of the devices, in PROM order. Skip
- * PROM entries that are not for devices, or which must be
- * done before we get here.
- */
- for (node = node0; node; node = nextsibling(node)) {
- cp = getpropstring(node, "name");
- for (ssp = openboot_special; (sp = *ssp) != NULL; ssp++)
- if (strcmp(cp, sp) == 0)
- break;
- if (sp == NULL && romprop(&oca.ca_ra, cp, node)) {
+ /*
+ * Configure the rest of the devices, in PROM order. Skip
+ * PROM entries that are not for devices, or which must be
+ * done before we get here.
+ */
+ for (node = node0; node; node = nextsibling(node)) {
+ register const char *cp;
+
+#if defined(SUN4M)
+ if (CPU_ISSUN4M) /* skip the CPUs */
+ if (node_has_property(node, "device_type") &&
+ !strcmp(getpropstring(node, "device_type"), "cpu"))
+ continue;
+#endif
+ cp = getpropstring(node, "name");
+ for (ssp = openboot_special; (sp = *ssp) != NULL; ssp++)
+ if (strcmp(cp, sp) == 0)
+ break;
+ if (sp == NULL && romprop(&oca.ca_ra, cp, node)) {
#ifdef L1A_HACK
- if (strcmp(cp, "audio") == 0)
- audio = 1;
- if (strcmp(cp, "zs") == 0)
- nzs++;
- if (audio && nzs >= 2)
- (void) splx(11 << 8); /* XXX */
+ if (strcmp(cp, "audio") == 0)
+ audio = 1;
+ if (strcmp(cp, "zs") == 0)
+ autoconf_nzs++;
+ if (/*audio &&*/ autoconf_nzs >= 2) /*XXX*/
+ (void) splx(11 << 8); /*XXX*/
#endif
- oca.ca_bustype = BUS_MAIN;
- (void) config_found(dev, (void *)&oca, mbprint);
- }
+ oca.ca_bustype = BUS_MAIN;
+ (void) config_found(dev, (void *)&oca, mbprint);
}
}
+#if defined(SUN4M)
+ if (CPU_ISSUN4M) {
+ /* Enable device interrupts */
+ ienab_bic(SINTR_MA);
+ }
+#endif
#endif /* SUN4C || SUN4M */
}
-struct cfdriver mainbuscd =
- { NULL, "mainbus", mainbus_match, mainbus_attach,
- DV_DULL, sizeof(struct device) };
+struct cfattach mainbus_ca = {
+ sizeof(struct device), mainbus_match, mainbus_attach
+};
+
+struct cfdriver mainbus_cd = {
+ NULL, "mainbus", DV_DULL
+};
/*
* findzs() is called from the zs driver (which is, at least in theory,
@@ -1040,38 +1331,49 @@ void *
findzs(zs)
int zs;
{
- register int node, addr;
-#ifdef SUN4
+#if defined(SUN4)
#define ZS0_PHYS 0xf1000000
#define ZS1_PHYS 0xf0000000
#define ZS2_PHYS 0xe0000000
- if (cputyp == CPU_SUN4) {
- void *paddr;
+ if (CPU_ISSUN4) {
+ struct rom_reg rr;
+ register void *vaddr;
switch (zs) {
case 0:
- paddr = (void *)ZS0_PHYS;
+ rr.rr_paddr = (void *)ZS0_PHYS;
break;
case 1:
- paddr = (void *)ZS1_PHYS;
+ rr.rr_paddr = (void *)ZS1_PHYS;
break;
case 2:
- paddr = (void *)ZS2_PHYS;
+ rr.rr_paddr = (void *)ZS2_PHYS;
break;
default:
panic("findzs: unknown zs device %d", zs);
}
- addr = (int)bus_map(paddr, NBPG, BUS_OBIO);
- if (addr)
- return ((void *)addr);
+ rr.rr_iospace = BUS_OBIO;
+ rr.rr_len = NBPG;
+ vaddr = bus_map(&rr, NBPG, BUS_OBIO);
+ if (vaddr)
+ return (vaddr);
}
#endif
+
#if defined(SUN4C) || defined(SUN4M)
- if (cputyp == CPU_SUN4C || cputyp == CPU_SUN4M) {
+ if (CPU_ISSUN4COR4M) {
+ register int node, addr;
+
node = firstchild(findroot());
+ if (CPU_ISSUN4M) { /* zs is in "obio" tree on Sun4M */
+ node = findnode(node, "obio");
+ if (!node)
+ panic("findzs: no obio node");
+ node = firstchild(node);
+ }
while ((node = findnode(node, "zs")) != 0) {
if (getpropint(node, "slave", -1) == zs) {
if ((addr = getpropint(node, "address", 0)) == 0)
@@ -1082,7 +1384,6 @@ findzs(zs)
}
}
#endif
-bail:
panic("findzs: cannot find zs%d", zs);
/* NOTREACHED */
}
@@ -1105,7 +1406,7 @@ makememarr(ap, max, which)
#endif
#if defined(SUN4)
- if (cputyp == CPU_SUN4) {
+ if (CPU_ISSUN4) {
switch (which) {
case MEMARR_AVAILPHYS:
ap[0].addr = 0;
@@ -1157,6 +1458,7 @@ makememarr(ap, max, which)
i);
/* FALLTHROUGH */
+ case 3:
case 2:
/*
* Version 2 PROMs use a property array to describe them.
@@ -1219,15 +1521,18 @@ getprop(node, name, buf, bufsiz)
void *buf;
register int bufsiz;
{
+#if defined(SUN4C) || defined(SUN4M)
register struct nodeops *no;
register int len;
+#endif
#if defined(SUN4)
- if (cputyp==CPU_SUN4) {
+ if (CPU_ISSUN4) {
printf("WARNING: getprop not valid on sun4! %s\n", name);
return (0);
}
#endif
+
#if defined(SUN4C) || defined(SUN4M)
no = promvec->pv_nodeops;
len = no->no_proplen(node, name);
@@ -1259,6 +1564,8 @@ getpropstring(node, name)
static char stringbuf[32];
len = getprop(node, name, (void *)stringbuf, sizeof stringbuf - 1);
+ if (len == -1)
+ len = 0;
stringbuf[len] = '\0'; /* usually unnecessary */
return (stringbuf);
}
@@ -1302,6 +1609,67 @@ nextsibling(node)
return (promvec->pv_nodeops->no_nextnode(node));
}
+char *strchr __P((const char *, int));
+u_int hexatoi __P((const char *));
+
+/* The following recursively searches a PROM tree for a given node */
+int
+search_prom(rootnode, name)
+ register int rootnode;
+ register char *name;
+{
+ register int rtnnode;
+ register int node = rootnode;
+
+ if (node == findroot() || !strcmp("hierarchical",
+ getpropstring(node, "device_type")))
+ node = firstchild(node);
+
+ if (!node)
+ panic("search_prom: null node");
+
+ do {
+ if (strcmp(getpropstring(node, "name"),name) == 0)
+ return node;
+
+ if (node_has_property(node,"device_type") &&
+ (!strcmp(getpropstring(node, "device_type"),"hierarchical")
+ || !strcmp(getpropstring(node, "name"),"iommu"))
+ && (rtnnode = search_prom(node, name)) != 0)
+ return rtnnode;
+
+ } while ((node = nextsibling(node)));
+
+ return 0;
+}
+
+/* The following are used primarily in consinit() */
+
+int
+opennode(path) /* translate phys. device path to node */
+ register char *path;
+{
+ register int fd;
+
+ if (promvec->pv_romvec_vers < 2) {
+ printf("WARNING: opennode not valid on sun4! %s\n", path);
+ return (0);
+ }
+ fd = promvec->pv_v2devops.v2_open(path);
+ if (fd == 0)
+ return 0;
+ return promvec->pv_v2devops.v2_fd_phandle(fd);
+}
+
+int
+node_has_property(node, prop) /* returns 1 if node has given property */
+ register int node;
+ register const char *prop;
+{
+
+ return ((*promvec->pv_nodeops->no_proplen)(node, (caddr_t)prop) != -1);
+}
+
#ifdef RASTERCONSOLE
/* Pass a string to the FORTH PROM to be interpreted */
void
@@ -1332,40 +1700,45 @@ romgetcursoraddr(rowp, colp)
*/
if (promvec->pv_romvec_vers < 2 || promvec->pv_printrev < 0x00020009)
sprintf(buf,
- "' line# >body >user %x ! ' column# >body >user %x !",
- rowp, colp);
+ "' line# >body >user %lx ! ' column# >body >user %lx !",
+ (u_long)rowp, (u_long)colp);
else
sprintf(buf,
- "stdout @ is my-self addr line# %x ! addr column# %x !",
- rowp, colp);
+ "stdout @ is my-self addr line# %lx ! addr column# %lx !",
+ (u_long)rowp, (u_long)colp);
*rowp = *colp = NULL;
rominterpret(buf);
return (*rowp == NULL || *colp == NULL);
}
#endif
-volatile void
+void
romhalt()
{
+ if (CPU_ISSUN4COR4M)
+ *promvec->pv_synchook = NULL;
promvec->pv_halt();
panic("PROM exit failed");
}
-volatile void
+void
romboot(str)
char *str;
{
+ if (CPU_ISSUN4COR4M)
+ *promvec->pv_synchook = NULL;
promvec->pv_reboot(str);
panic("PROM boot failed");
}
+void
callrom()
{
#if 0 /* sun4c FORTH PROMs do this for us */
- if (cputyp == CPU_SUN4)
+ if (CPU_ISSUN4)
fb_unblank();
#endif
promvec->pv_abort();
@@ -1374,6 +1747,7 @@ callrom()
/*
* Configure swap space and related parameters.
*/
+void
swapconf()
{
register struct swdevt *swp;
@@ -1433,7 +1807,8 @@ getdisk(str, len, defpart, devp)
if ((dv = parsedisk(str, len, defpart, devp)) == NULL) {
printf("use one of:");
- for (dv = alldevs; dv != NULL; dv = dv->dv_next) {
+ for (dv = alldevs.tqh_first; dv != NULL;
+ dv = dv->dv_list.tqe_next) {
if (dv->dv_class == DV_DISK)
printf(" %s[a-h]", dv->dv_xname);
#ifdef NFSCLIENT
@@ -1441,10 +1816,6 @@ getdisk(str, len, defpart, devp)
printf(" %s", dv->dv_xname);
#endif
}
-#if NFD > 0
- if (defpart == 0)
- printf(" fdeject");
-#endif /* NFD */
printf("\n");
}
return (dv);
@@ -1470,7 +1841,7 @@ parsedisk(str, len, defpart, devp)
} else
part = defpart;
- for (dv = alldevs; dv != NULL; dv = dv->dv_next) {
+ for (dv = alldevs.tqh_first; dv != NULL; dv = dv->dv_list.tqe_next) {
if (dv->dv_class == DV_DISK &&
strcmp(str, dv->dv_xname) == 0) {
majdev = findblkmajor(dv);
@@ -1493,11 +1864,29 @@ parsedisk(str, len, defpart, devp)
return (dv);
}
+void
+mountroot_hook_establish(func, dev)
+ void (*func) __P((struct device *));
+ struct device *dev;
+{
+ struct mountroot_hook *mrhp;
+
+ mrhp = (struct mountroot_hook *)malloc(sizeof(struct mountroot_hook),
+ M_DEVBUF, M_NOWAIT);
+ if (mrhp == NULL)
+ panic("no memory for mountroot_hook");
+
+ bzero(mrhp, sizeof(struct mountroot_hook));
+ mrhp->mr_device = dev;
+ mrhp->mr_func = func;
+ LIST_INSERT_HEAD(&mrh_list, mrhp, mr_link);
+}
+
/*
* Attempt to find the device from which we were booted.
* If we can do so, and not instructed not to do so,
* change rootdev to correspond to the load device.
- *
+ *
* XXX Actually, swap and root must be on the same type of device,
* (ie. DV_DISK or DV_IFNET) because of how (*mountroot) is written.
* That should be fixed.
@@ -1510,37 +1899,39 @@ setroot()
register int len, majdev, mindev;
dev_t nrootdev, nswapdev = NODEV;
char buf[128];
- extern int (*mountroot)();
+ extern int (*mountroot) __P((void *));
dev_t temp;
+ struct mountroot_hook *mrhp;
+ struct device *bootdv;
+ struct bootpath *bp;
#if defined(NFSCLIENT)
extern char *nfsbootdevname;
- extern int nfs_mountroot();
+ extern int nfs_mountroot __P((void *));
#endif
#if defined(FFS)
- extern int ffs_mountroot();
+ extern int ffs_mountroot __P((void *));
#endif
+ bp = nbootpath == 0 ? NULL : &bootpath[nbootpath-1];
+ bootdv = bp == NULL ? NULL : bp->dev;
+
+ /*
+ * If `swap generic' and we couldn't determine boot device,
+ * ask the user.
+ */
+ if (mountroot == NULL && bootdv == NULL)
+ boothowto |= RB_ASKNAME;
+
if (boothowto & RB_ASKNAME) {
for (;;) {
printf("root device ");
if (bootdv != NULL)
- printf("(default %s%s)", bootdv->dv_xname,
- bootdv->dv_class == DV_DISK?"a":"");
+ printf("(default %s%c)",
+ bootdv->dv_xname,
+ bootdv->dv_class == DV_DISK
+ ? bp->val[2]+'a' : ' ');
printf(": ");
len = getstr(buf, sizeof(buf));
-#if NFD > 0
- /*
- * I will go punish myself now.
- */
- if (len > 0 && strcmp(buf, "fdeject")==0) {
- struct mtop mtop;
-
- bzero(&mtop, sizeof mtop);
- mtop.mt_op = MTOFFL;
- (void) fdioctl(0, MTIOCTOP, &mtop, 0);
- continue;
- }
-#endif /* NFD */
if (len == 0 && bootdv != NULL) {
strcpy(buf, bootdv->dv_xname);
len = strlen(buf);
@@ -1554,7 +1945,7 @@ setroot()
goto gotswap;
}
}
- dv = getdisk(buf, len, 0, &nrootdev);
+ dv = getdisk(buf, len, bp?bp->val[2]:0, &nrootdev);
if (dv != NULL) {
bootdv = dv;
break;
@@ -1564,7 +1955,6 @@ setroot()
/*
* because swap must be on same device as root, for
* network devices this is easy.
- * XXX: IS THIS STILL TRUE?
*/
if (bootdv->dv_class == DV_IFNET) {
goto gotswap;
@@ -1572,8 +1962,9 @@ setroot()
for (;;) {
printf("swap device ");
if (bootdv != NULL)
- printf("(default %s%s)", bootdv->dv_xname,
- bootdv->dv_class == DV_DISK?"b":"");
+ printf("(default %s%c)",
+ bootdv->dv_xname,
+ bootdv->dv_class == DV_DISK?'b':' ');
printf(": ");
len = getstr(buf, sizeof(buf));
if (len == 0 && bootdv != NULL) {
@@ -1585,6 +1976,11 @@ setroot()
nswapdev = makedev(major(nrootdev),
(minor(nrootdev) & ~ PARTITIONMASK) | 1);
break;
+ case DV_TAPE:
+ case DV_TTY:
+ case DV_DULL:
+ case DV_CPU:
+ break;
}
break;
}
@@ -1600,7 +1996,9 @@ gotswap:
dumpdev = nswapdev;
swdevt[0].sw_dev = nswapdev;
swdevt[1].sw_dev = NODEV;
+
} else if (mountroot == NULL) {
+
/*
* `swap generic': Use the device the ROM told us to use.
*/
@@ -1608,10 +2006,11 @@ gotswap:
if (majdev >= 0) {
/*
* Root and swap are on a disk.
- * Assume that we are supposed to put root on
- * partition a, and swap on partition b.
+ * val[2] of the boot device is the partition number.
+ * Assume swap is on partition b.
*/
- mindev = (bootdv->dv_unit << PARTITIONSHIFT) + 0;
+ int part = bp->val[2];
+ mindev = (bootdv->dv_unit << PARTITIONSHIFT) + part;
rootdev = makedev(majdev, mindev);
nswapdev = dumpdev = makedev(major(rootdev),
(minor(rootdev) & ~ PARTITIONMASK) | 1);
@@ -1630,10 +2029,11 @@ gotswap:
* `root DEV swap DEV': honour rootdev/swdevt.
* rootdev/swdevt/mountroot already properly set.
*/
- return;
+ majdev = major(rootdev);
+ mindev = minor(rootdev);
+ goto gotroot;
}
- if (bootdv == NULL)
- panic("boot device not known");
+
switch (bootdv->dv_class) {
#if defined(NFSCLIENT)
case DV_IFNET:
@@ -1654,6 +2054,41 @@ gotswap:
printf("can't figure root, hope your kernel is right\n");
return;
}
+
+ /*
+ * Make the swap partition on the root drive the primary swap.
+ */
+ mindev &= ~PARTITIONMASK;
+ temp = NODEV;
+ for (swp = swdevt; swp->sw_dev != NODEV; swp++) {
+ if (majdev == major(swp->sw_dev) &&
+ mindev == (minor(swp->sw_dev) & ~PARTITIONMASK)) {
+ temp = swdevt[0].sw_dev;
+ swdevt[0].sw_dev = swp->sw_dev;
+ swp->sw_dev = temp;
+ break;
+ }
+ }
+ if (swp->sw_dev != NODEV) {
+ /*
+ * If dumpdev was the same as the old primary swap device,
+ * move it to the new primary swap device.
+ */
+ if (temp == dumpdev)
+ dumpdev = swdevt[0].sw_dev;
+ }
+
+gotroot:
+ /*
+ * Find mountroot hook and execute.
+ */
+ for (mrhp = mrh_list.lh_first; mrhp != NULL;
+ mrhp = mrhp->mr_link.le_next)
+ if (mrhp->mr_device == bootdv) {
+ (*mrhp->mr_func)(bootdv);
+ break;
+ }
+
}
static int
@@ -1703,7 +2138,7 @@ getstr(cp, size)
}
-/*
+/*
* find a device matching "name" and unit number
*/
struct device *
@@ -1711,7 +2146,7 @@ getdevunit(name, unit)
char *name;
int unit;
{
- struct device *dev = alldevs;
+ struct device *dev = alldevs.tqh_first;
char num[10], fullname[16];
int lunit;
@@ -1725,8 +2160,37 @@ getdevunit(name, unit)
strcat(fullname, num);
while (strcmp(dev->dv_xname, fullname) != 0) {
- if ((dev = dev->dv_next) == NULL)
+ if ((dev = dev->dv_list.tqe_next) == NULL)
return NULL;
}
return dev;
}
+
+
+/*
+ * The $#!@$&%# kernel library doesn't have strchr or atoi. Ugh. We duplicate
+ * here.
+ */
+
+char *
+strchr(p, ch) /* cribbed from libc */
+ register const char *p, ch;
+{
+ for (;; ++p) {
+ if (*p == ch)
+ return((char *)p);
+ if (!*p)
+ return((char *)NULL);
+ }
+ /* NOTREACHED */
+}
+
+u_int
+hexatoi(nptr) /* atoi assuming hex, no 0x */
+ const char *nptr;
+{
+ u_int retval;
+ str2hex((char *)nptr, &retval);
+ return retval;
+}
+
diff --git a/sys/arch/sparc/sparc/auxreg.c b/sys/arch/sparc/sparc/auxreg.c
index 0b40f13f5ea..1b44aee9bbf 100644
--- a/sys/arch/sparc/sparc/auxreg.c
+++ b/sys/arch/sparc/sparc/auxreg.c
@@ -1,4 +1,4 @@
-/* $NetBSD: auxreg.c,v 1.9 1995/12/11 12:45:16 pk Exp $ */
+/* $NetBSD: auxreg.c,v 1.14 1996/04/13 17:40:03 abrown Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -45,6 +45,7 @@
*/
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/device.h>
#include <sys/kernel.h>
@@ -55,16 +56,23 @@
static int auxregmatch __P((struct device *, void *, void *));
static void auxregattach __P((struct device *, struct device *, void *));
-struct cfdriver auxregcd =
- { 0, "auxreg", auxregmatch, auxregattach, DV_DULL, sizeof(struct device) };
+
+struct cfattach auxreg_ca = {
+ sizeof(struct device), auxregmatch, auxregattach
+};
+
+struct cfdriver auxreg_cd = {
+ 0, "auxreg", DV_DULL
+};
#ifdef BLINK
-static int
+static void blink __P((void *zero));
+
+static void
blink(zero)
void *zero;
{
register int s;
- register fixpt_t lav;
s = splhigh();
LED_FLIP;
@@ -76,7 +84,7 @@ blink(zero)
* full cycle every 3 seconds if loadav = 2
* etc.
*/
- s = (((averunnable[0] + FSCALE) * hz) >> (FSHIFT + 1));
+ s = (((averunnable.ldavg[0] + FSCALE) * hz) >> (FSHIFT + 1));
timeout(blink, (caddr_t)0, s);
}
#endif
@@ -90,11 +98,17 @@ auxregmatch(parent, vcf, aux)
void *aux, *vcf;
{
register struct confargs *ca = aux;
- struct cfdata *cf = vcf;
- if (cputyp==CPU_SUN4)
+ switch (cputyp) {
+ case CPU_SUN4:
return (0);
- return (strcmp("auxiliary-io", ca->ca_ra.ra_name) == 0);
+ case CPU_SUN4C:
+ return (strcmp("auxiliary-io", ca->ca_ra.ra_name) == 0);
+ case CPU_SUN4M:
+ return (strcmp("auxio", ca->ca_ra.ra_name) == 0);
+ default:
+ panic("auxregmatch");
+ }
}
/* ARGSUSED */
@@ -116,6 +130,7 @@ auxregattach(parent, self, aux)
unsigned int
auxregbisc(bis, bic)
+ int bis, bic;
{
register int v, s = splhigh();
diff --git a/sys/arch/sparc/sparc/auxreg.h b/sys/arch/sparc/sparc/auxreg.h
index 751e944a7d0..321c9a38a83 100644
--- a/sys/arch/sparc/sparc/auxreg.h
+++ b/sys/arch/sparc/sparc/auxreg.h
@@ -1,4 +1,4 @@
-/* $NetBSD: auxreg.h,v 1.3 1995/02/22 21:13:02 pk Exp $ */
+/* $NetBSD: auxreg.h,v 1.5 1996/03/14 21:08:54 christos Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -69,10 +69,12 @@
#define AUXIO_BITS "\20\6FHD\5FDC\4FDS\3FTC\2FEJ\1LED"
-#ifndef LOCORE
+#ifndef _LOCORE
/*
* Copy of AUXIO_REG for the benefit of assembler modules (eg. trap handlers)
* as AUXREG_VA depends on NBPG which is not a constant.
*/
volatile u_char *auxio_reg;
+unsigned int auxregbisc __P((int, int));
#endif
+
diff --git a/sys/arch/sparc/sparc/bsd_fdintr.s b/sys/arch/sparc/sparc/bsd_fdintr.s
index 000cf5eef8f..e0e52a690e8 100644
--- a/sys/arch/sparc/sparc/bsd_fdintr.s
+++ b/sys/arch/sparc/sparc/bsd_fdintr.s
@@ -1,4 +1,4 @@
-/* $NetBSD: bsd_fdintr.s,v 1.4 1995/04/25 20:01:23 pk Exp $ */
+/* $NetBSD: bsd_fdintr.s,v 1.8 1996/03/31 23:45:00 pk Exp $ */
/*
* Copyright (c) 1995 Paul Kranenburg
@@ -32,10 +32,7 @@
*/
#ifndef FDC_C_HANDLER
-#ifndef LOCORE
-#define LOCORE
-#endif
-#include "assym.s"
+#include "assym.h"
#include <sparc/sparc/intreg.h>
#include <sparc/sparc/auxreg.h>
#include <sparc/sparc/vaddrs.h>
@@ -103,7 +100,7 @@ _fdchwintr:
!!ld [R_fdc + FDC_REG_DOR], R_dor ! get chip DOR reg addr
! find out what we are supposed to do
- ld [R_fdc + FDC_ISTATE], %l7 ! examine flags
+ ld [R_fdc + FDC_ISTATE], %l7 ! examine flags
cmp %l7, ISTATE_SENSEI
be sensei
nop
diff --git a/sys/arch/sparc/sparc/cache.c b/sys/arch/sparc/sparc/cache.c
index abc187995f1..35c00087300 100644
--- a/sys/arch/sparc/sparc/cache.c
+++ b/sys/arch/sparc/sparc/cache.c
@@ -1,6 +1,8 @@
-/* $NetBSD: cache.c,v 1.5 1995/04/13 14:32:44 pk Exp $ */
+/* $NetBSD: cache.c,v 1.8.4.1 1996/06/12 20:40:35 pk Exp $ */
/*
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
*
@@ -10,12 +12,14 @@
*
* All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
+ * This product includes software developed by Harvard University.
* This product includes software developed by the University of
* California, Lawrence Berkeley Laboratory.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
+ *
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
@@ -23,6 +27,8 @@
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
+ * This product includes software developed by Aaron Brown and
+ * Harvard University.
* This product includes software developed by the University of
* California, Berkeley and its contributors.
* 4. Neither the name of the University nor the names of its contributors
@@ -42,6 +48,7 @@
* SUCH DAMAGE.
*
* @(#)cache.c 8.2 (Berkeley) 10/30/93
+ *
*/
/*
@@ -52,6 +59,7 @@
*/
#include <sys/param.h>
+#include <sys/systm.h>
#include <machine/ctlreg.h>
#include <machine/pte.h>
@@ -62,6 +70,46 @@
enum vactype vactype;
struct cachestats cachestats;
+#if defined(SUN4M)
+int cache_alias_dist; /* Cache anti-aliasing constants */
+int cache_alias_bits;
+#endif
+
+/*
+ * A few quick macros to allow for different ASI's between 4M/4/4C machines
+ */
+#define flushls_ctx(p) do { \
+ CPU_ISSUN4M \
+ ? sta(p, ASI_IDCACHELFC, 0) \
+ : sta(p, ASI_FLUSHCTX, 0); \
+} while (0)
+#define flushls_reg(p) do { \
+ CPU_ISSUN4M \
+ ? sta(p, ASI_IDCACHELFR, 0) \
+ : sta(p, ASI_FLUSHREG, 0); \
+} while (0)
+#define flushls_seg(p) do { \
+ CPU_ISSUN4M \
+ ? sta(p, ASI_IDCACHELFS, 0) \
+ : sta(p, ASI_FLUSHSEG, 0); \
+} while (0)
+#define flushls_pag(p) do { \
+ CPU_ISSUN4M \
+ ? sta(p, ASI_IDCACHELFP, 0) \
+ : sta(p, ASI_FLUSHPG, 0); \
+} while (0)
+#define flushls_usr(p) do { \
+ if (CPU_ISSUN4M) \
+ sta(p, ASI_IDCACHELFU, 0); \
+while(0)
+
+/* XXX: (ABB) need more generic Sun4m cache support */
+#if !defined(SUN4) && !defined(SUN4C)
+#define ASI_HWFLUSHSEG 0x05 /* hardware assisted version of FLUSHSEG */
+#define ASI_HWFLUSHPG 0x06 /* hardware assisted version of FLUSHPG */
+#define ASI_HWFLUSHCTX 0x07 /* hardware assisted version of FLUSHCTX */
+#endif
+
/*
* Enable the cache.
* We need to clear out the valid bits first.
@@ -73,25 +121,116 @@ cache_enable()
ls = cacheinfo.c_linesize;
ts = cacheinfo.c_totalsize;
- for (i = AC_CACHETAGS, lim = i + ts; i < lim; i += ls)
- sta(i, ASI_CONTROL, 0);
- stba(AC_SYSENABLE, ASI_CONTROL,
- lduba(AC_SYSENABLE, ASI_CONTROL) | SYSEN_CACHE);
- cacheinfo.c_enabled = 1;
+ if (CPU_ISSUN4M) {
+ i = lda(SRMMU_PCR, ASI_SRMMU);
+ switch (mmumod) {
+ case SUN4M_MMU_HS: /* HyperSPARC */
+ /*
+ * First we determine what type of cache we have, and
+ * setup the anti-aliasing constants appropriately.
+ */
+ if (i & SRMMU_PCR_CS) {
+ cache_alias_bits = CACHE_ALIAS_BITS_HS256k;
+ cache_alias_dist = CACHE_ALIAS_DIST_HS256k;
+ } else {
+ cache_alias_bits = CACHE_ALIAS_BITS_HS128k;
+ cache_alias_dist = CACHE_ALIAS_DIST_HS128k;
+ }
+ /* Now reset cache tag memory */
+ for (i = 0, lim = ts; i < lim; i += ls)
+ sta(i, ASI_DCACHETAG, 0);
- printf("cache enabled\n");
+ sta(SRMMU_PCR, ASI_SRMMU, /* Enable write-back cache */
+ lda(SRMMU_PCR, ASI_SRMMU) | SRMMU_PCR_CE |
+ SRMMU_PCR_CM);
+ cacheinfo.c_enabled = 1;
+ vactype = VAC_NONE;
+ /* HyperSPARC uses phys. tagged cache */
+
+ /* XXX: should add support */
+ if (cacheinfo.c_hwflush)
+ panic("cache_enable: can't handle 4M with hw-flush cache");
+
+ printf("cache enabled\n");
+ break;
+
+ case SUN4M_MMU_SS: /* SuperSPARC */
+ if ((cpumod & 0xf0) != (SUN4M_SS & 0xf0)) {
+ printf(
+ "cache NOT enabled for %x/%x cpu/mmu combination\n",
+ cpumod, mmumod);
+ break; /* ugh, SS and MS have same MMU # */
+ }
+ cache_alias_bits = CACHE_ALIAS_BITS_SS;
+ cache_alias_dist = CACHE_ALIAS_DIST_SS;
+
+ /* We "flash-clear" the I/D caches. */
+ sta(0x80000000, ASI_ICACHECLR, 0); /* Unlock */
+ sta(0, ASI_ICACHECLR, 0); /* clear */
+ sta(0x80000000, ASI_DCACHECLR, 0);
+ sta(0, ASI_DCACHECLR, 0);
+
+ /* Turn on caches via MMU */
+ sta(SRMMU_PCR, ASI_SRMMU,
+ lda(SRMMU_PCR,ASI_SRMMU) | SRMMU_PCR_DCE |
+ SRMMU_PCR_ICE);
+ cacheinfo.c_enabled = cacheinfo.dc_enabled = 1;
+
+ /* Now try to turn on MultiCache if it exists */
+ /* XXX (ABB) THIS IS BROKEN MUST FIX */
+ if (0&&(lda(SRMMU_PCR, ASI_SRMMU) & SRMMU_PCR_MB) == 0
+ && cacheinfo.ec_totalsize > 0) {
+ /* Multicache controller */
+ sta(MXCC_ENABLE_ADDR, ASI_CONTROL,
+ lda(MXCC_ENABLE_ADDR, ASI_CONTROL) |
+ MXCC_ENABLE_BIT);
+ cacheinfo.ec_enabled = 1;
+ }
+ printf("cache enabled\n");
+ break;
+ case SUN4M_MMU_MS1: /* MicroSPARC */
+ /* We "flash-clear" the I/D caches. */
+ sta(0, ASI_ICACHECLR, 0); /* clear */
+ sta(0, ASI_DCACHECLR, 0);
+
+ /* Turn on caches via MMU */
+ sta(SRMMU_PCR, ASI_SRMMU,
+ lda(SRMMU_PCR,ASI_SRMMU) | SRMMU_PCR_DCE |
+ SRMMU_PCR_ICE);
+ cacheinfo.c_enabled = cacheinfo.dc_enabled = 1;
+
+ printf("cache enabled\n");
+ break;
+
+ default:
+ printf("Unknown MMU architecture...cache disabled.\n");
+ /* XXX: not HyperSPARC -- guess for now */
+ cache_alias_bits = GUESS_CACHE_ALIAS_BITS;
+ cache_alias_dist = GUESS_CACHE_ALIAS_DIST;
+ }
+
+ } else {
+
+ for (i = AC_CACHETAGS, lim = i + ts; i < lim; i += ls)
+ sta(i, ASI_CONTROL, 0);
-#ifdef notyet
- if (cpumod == SUN4_400) {
stba(AC_SYSENABLE, ASI_CONTROL,
- lduba(AC_SYSENABLE, ASI_CONTROL) | SYSEN_IOCACHE);
- printf("iocache enabled\n");
- }
+ lduba(AC_SYSENABLE, ASI_CONTROL) | SYSEN_CACHE);
+ cacheinfo.c_enabled = 1;
+
+ printf("cache enabled\n");
+
+#ifdef notyet
+ if (cpumod == SUN4_400) {
+ stba(AC_SYSENABLE, ASI_CONTROL,
+ lduba(AC_SYSENABLE, ASI_CONTROL) | SYSEN_IOCACHE);
+ printf("iocache enabled\n");
+ }
#endif
+ }
}
-
/*
* Flush the current context from the cache.
*
@@ -116,11 +255,11 @@ cache_flush_context()
ls = cacheinfo.c_linesize;
i = cacheinfo.c_totalsize >> cacheinfo.c_l2linesize;
for (; --i >= 0; p += ls)
- sta(p, ASI_FLUSHCTX, 0);
+ flushls_ctx(p);
}
}
-#ifdef MMU_3L
+#if defined(MMU_3L) || defined(SUN4M)
/*
* Flush the given virtual region from the cache.
*
@@ -128,7 +267,7 @@ cache_flush_context()
* now the addresses must include the virtual region number, and
* we use the `flush region' space.
*
- * This function is only called on sun4's with 3-level MMUs; there's
+ * This function is only called on sun4m's or sun4's with 3-level MMUs; there's
* no hw-flush space.
*/
void
@@ -143,7 +282,7 @@ cache_flush_region(vreg)
ls = cacheinfo.c_linesize;
i = cacheinfo.c_totalsize >> cacheinfo.c_l2linesize;
for (; --i >= 0; p += ls)
- sta(p, ASI_FLUSHREG, 0);
+ flushls_reg(p);
}
#endif
@@ -174,7 +313,7 @@ cache_flush_segment(vreg, vseg)
ls = cacheinfo.c_linesize;
i = cacheinfo.c_totalsize >> cacheinfo.c_l2linesize;
for (; --i >= 0; p += ls)
- sta(p, ASI_FLUSHSEG, 0);
+ flushls_seg(p);
}
}
@@ -190,6 +329,11 @@ cache_flush_page(va)
register int i, ls;
register char *p;
+#ifdef DEBUG
+ if (va & PGOFSET)
+ panic("cache_flush_page: asked to flush misaligned va %x",va);
+#endif
+
cachestats.cs_npgflush++;
p = (char *)va;
if (cacheinfo.c_hwflush)
@@ -198,7 +342,7 @@ cache_flush_page(va)
ls = cacheinfo.c_linesize;
i = NBPG >> cacheinfo.c_l2linesize;
for (; --i >= 0; p += ls)
- sta(p, ASI_FLUSHPG, 0);
+ flushls_pag(p);
}
}
@@ -209,6 +353,9 @@ cache_flush_page(va)
*
* We choose the best of (context,segment,page) here.
*/
+
+#define CACHE_FLUSH_MAGIC (cacheinfo.c_totalsize / NBPG)
+
void
cache_flush(base, len)
caddr_t base;
@@ -216,6 +363,25 @@ cache_flush(base, len)
{
register int i, ls, baseoff;
register char *p;
+#if defined(MMU_3L)
+ extern int mmu_3l;
+#endif
+
+#if defined(SUN4M)
+ /*
+ * Although physically tagged, we still need to flush the
+ * data cache after (if we have a write-through cache) or before
+ * (in case of write-back caches) DMA operations.
+ *
+ * SS10s and 20s (supersparcs) use cached DVMA, no need to flush.
+ */
+ if (CPU_ISSUN4M && mmumod != SUN4M_MMU_SS) {
+ /* XXX (ABB) - Need more generic cache interface */
+ /* XXX above test conflicts on MicroSPARC (SUN4M_MMU_MS=_SS) */
+ sta(0, ASI_DCACHECLR, 0);
+ return;
+ }
+#endif
if (vactype == VAC_NONE)
return;
@@ -223,11 +389,12 @@ cache_flush(base, len)
/*
* Figure out how much must be flushed.
*
- * If we need to do 16 pages, we can do a segment in the same
- * number of loop iterations. We can also do the context. If
- * we would need to do two segments, do the whole context.
- * This might not be ideal (e.g., fsck likes to do 65536-byte
- * reads, which might not necessarily be aligned).
+ * If we need to do CACHE_FLUSH_MAGIC pages, we can do a segment
+ * in the same number of loop iterations. We can also do the whole
+ * region. If we need to do between 2 and NSEGRG, do the region.
+ * If we need to do two or more regions, just go ahead and do the
+ * whole context. This might not be ideal (e.g., fsck likes to do
+ * 65536-byte reads, which might not necessarily be aligned).
*
* We could try to be sneaky here and use the direct mapping
* to avoid flushing things `below' the start and `above' the
@@ -245,7 +412,7 @@ cache_flush(base, len)
cachestats.cs_ra[min(i, MAXCACHERANGE)]++;
#endif
- if (i <= 15) {
+ if (i < CACHE_FLUSH_MAGIC) {
/* cache_flush_page, for i pages */
p = (char *)((int)base & ~baseoff);
if (cacheinfo.c_hwflush) {
@@ -255,7 +422,7 @@ cache_flush(base, len)
ls = cacheinfo.c_linesize;
i <<= PGSHIFT - cacheinfo.c_l2linesize;
for (; --i >= 0; p += ls)
- sta(p, ASI_FLUSHPG, 0);
+ flushls_pag(p);
}
return;
}
@@ -263,6 +430,22 @@ cache_flush(base, len)
i = (baseoff + len + SGOFSET) >> SGSHIFT;
if (i == 1)
cache_flush_segment(VA_VREG(base), VA_VSEG(base));
- else
- cache_flush_context();
+ else {
+#if defined(MMU_3L) || defined(SUN4M)
+ baseoff = (u_int)base & RGOFSET;
+ i = (baseoff + len + RGOFSET) >> RGSHIFT;
+ if (i == 1
+#if !defined(MMU_3L)
+ && CPU_ISSUN4M
+#elif !defined(SUN4M)
+ && mmu_3l
+#else
+ && (CPU_ISSUN4M || mmu_3l)
+#endif
+ )
+ cache_flush_region(VA_VREG(base));
+ else
+#endif
+ cache_flush_context();
+ }
}
diff --git a/sys/arch/sparc/sparc/cache.h b/sys/arch/sparc/sparc/cache.h
index 9624e9ae775..6076ee314b6 100644
--- a/sys/arch/sparc/sparc/cache.h
+++ b/sys/arch/sparc/sparc/cache.h
@@ -1,6 +1,8 @@
-/* $NetBSD: cache.h,v 1.5 1995/04/13 14:48:51 pk Exp $ */
+/* $NetBSD: cache.h,v 1.7.4.1 1996/06/12 20:41:22 pk Exp $ */
/*
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
*
@@ -8,11 +10,6 @@
* at Lawrence Berkeley Laboratory under DARPA contract BG 91-66 and
* contributed to Berkeley.
*
- * All advertising materials mentioning features or use of this software
- * must display the following acknowledgement:
- * This product includes software developed by the University of
- * California, Lawrence Berkeley Laboratory.
- *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
@@ -23,6 +20,8 @@
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
+ * This product includes software developed by Aaron Brown and
+ * Harvard University.
* This product includes software developed by the University of
* California, Berkeley and its contributors.
* 4. Neither the name of the University nor the names of its contributors
@@ -108,27 +107,48 @@ extern enum vactype vactype; /* XXX move into cacheinfo struct */
* bits that are one position higher.
*/
+/* Some more well-known values: */
+
#define CACHE_ALIAS_DIST_SUN4 0x20000
#define CACHE_ALIAS_DIST_SUN4C 0x10000
#define CACHE_ALIAS_BITS_SUN4 0x1e000
#define CACHE_ALIAS_BITS_SUN4C 0xf000
-#if defined(SUN4) && defined(SUN4C)
-#define CACHE_ALIAS_DIST ((cputyp == CPU_SUN4) ? CACHE_ALIAS_DIST_SUN4 : \
- CACHE_ALIAS_DIST_SUN4C)
-#define CACHE_ALIAS_BITS ((cputyp == CPU_SUN4) ? CACHE_ALIAS_BITS_SUN4 : \
- CACHE_ALIAS_BITS_SUN4C)
-#else
-#if defined(SUN4)
-#define CACHE_ALIAS_DIST CACHE_ALIAS_DIST_SUN4
-#define CACHE_ALIAS_BITS CACHE_ALIAS_BITS_SUN4
-#endif
-#if defined(SUN4C)
-#define CACHE_ALIAS_DIST CACHE_ALIAS_DIST_SUN4C
-#define CACHE_ALIAS_BITS CACHE_ALIAS_BITS_SUN4C
-#endif
-#endif
+#define CACHE_ALIAS_DIST_HS128k 0x20000
+#define CACHE_ALIAS_BITS_HS128k 0x1f000
+#define CACHE_ALIAS_DIST_HS256k 0x40000
+#define CACHE_ALIAS_BITS_HS256k 0x3f000
+
+#define CACHE_ALIAS_DIST_SS 0x0
+#define CACHE_ALIAS_BITS_SS 0x0
+
+#define CACHE_ALIAS_DIST_MS GUESS_CACHE_ALIAS_DIST /* fix! */
+#define CACHE_ALIAS_BITS_MS GUESS_CACHE_ALIAS_BITS /* %%% */
+
+/*
+ * Assuming a tag format where the least significant bits are the byte offset
+ * into the cache line, and the next-most significant bits are the line id,
+ * we can calculate the appropriate aliasing constants. We also assume that
+ * the linesize and total cache size are powers of 2.
+ */
+#define GUESS_CACHE_ALIAS_BITS ((cacheinfo.c_totalsize - 1) & ~PGOFSET)
+#define GUESS_CACHE_ALIAS_DIST (cacheinfo.c_totalsize)
+
+extern int cache_alias_dist; /* If `guessed' */
+extern int cache_alias_bits;
+
+#define CACHE_ALIAS_DIST (CPU_ISSUN4M \
+ ? cache_alias_dist \
+ : (CPU_ISSUN4C \
+ ? CACHE_ALIAS_DIST_SUN4C \
+ : CACHE_ALIAS_DIST_SUN4))
+
+#define CACHE_ALIAS_BITS (CPU_ISSUN4M \
+ ? cache_alias_bits \
+ : (CPU_ISSUN4C \
+ ? CACHE_ALIAS_BITS_SUN4C \
+ : CACHE_ALIAS_BITS_SUN4))
/*
* True iff a1 and a2 are `bad' aliases (will cause cache duplication).
@@ -140,19 +160,35 @@ extern enum vactype vactype; /* XXX move into cacheinfo struct */
*/
void cache_enable __P((void)); /* turn it on */
void cache_flush_context __P((void)); /* flush current context */
-void cache_flush_segment __P((int vreg, int vseg)); /* flush seg in cur ctx */
+void cache_flush_region __P((int)); /* flush region in cur ctx */
+void cache_flush_segment __P((int, int)); /* flush seg in cur ctx */
void cache_flush_page __P((int va)); /* flush page in cur ctx */
-void cache_flush __P((caddr_t base, u_int len));/* flush region */
+void cache_flush __P((caddr_t, u_int)); /* flush region */
/*
* Cache control information.
*/
struct cacheinfo {
int c_totalsize; /* total size, in bytes */
+ /* if split, MAX(icache,dcache) */
int c_enabled; /* true => cache is enabled */
int c_hwflush; /* true => have hardware flush */
int c_linesize; /* line size, in bytes */
int c_l2linesize; /* log2(linesize) */
+ int c_physical; /* true => cache is physical */
+ int c_split; /* true => cache is split */
+ int ic_totalsize; /* instruction cache */
+ int ic_enabled;
+ int ic_linesize;
+ int ic_l2linesize;
+ int dc_totalsize; /* data cache */
+ int dc_enabled;
+ int dc_linesize;
+ int dc_l2linesize;
+ int ec_totalsize; /* external cache info */
+ int ec_enabled;
+ int ec_linesize;
+ int ec_l2linesize;
};
extern struct cacheinfo cacheinfo;
diff --git a/sys/arch/sparc/sparc/clock.c b/sys/arch/sparc/sparc/clock.c
index 1f30672b5b2..d8606c7b21d 100644
--- a/sys/arch/sparc/sparc/clock.c
+++ b/sys/arch/sparc/sparc/clock.c
@@ -1,10 +1,13 @@
-/* $NetBSD: clock.c,v 1.26 1995/12/11 12:45:18 pk Exp $ */
+/* $NetBSD: clock.c,v 1.44 1996/05/16 15:57:12 abrown Exp $ */
/*
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
* Copyright (c) 1994 Gordon W. Ross
* Copyright (c) 1993 Adam Glass
+ * Copyright (c) 1996 Paul Kranenburg
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
*
* This software was developed by the Computer Systems Engineering group
* at Lawrence Berkeley Laboratory under DARPA contract BG 91-66 and
@@ -12,12 +15,14 @@
*
* All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
+ * This product includes software developed by Harvard University.
* This product includes software developed by the University of
* California, Lawrence Berkeley Laboratory.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
+ *
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
@@ -27,6 +32,8 @@
* must display the following acknowledgement:
* This product includes software developed by the University of
* California, Berkeley and its contributors.
+ * This product includes software developed by Paul Kranenburg.
+ * This product includes software developed by Harvard University.
* 4. Neither the name of the University nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
@@ -44,6 +51,7 @@
* SUCH DAMAGE.
*
* @(#)clock.c 8.1 (Berkeley) 6/11/93
+ *
*/
/*
@@ -57,6 +65,7 @@
#include <sys/proc.h>
#include <sys/resourcevar.h>
#include <sys/malloc.h>
+#include <sys/systm.h>
#ifdef GPROF
#include <sys/gmon.h>
#endif
@@ -65,7 +74,9 @@
#include <machine/autoconf.h>
#include <machine/eeprom.h>
+#include <machine/cpu.h>
+#include <sparc/sparc/vaddrs.h>
#include <sparc/sparc/clockreg.h>
#include <sparc/sparc/intreg.h>
#include <sparc/sparc/timerreg.h>
@@ -84,13 +95,9 @@ int statvar = 8192;
int statmin; /* statclock interval - 1/2*variance */
int timerok;
-#if defined(SUN4)
#include <sparc/sparc/intersil7170.h>
extern struct idprom idprom;
-static int oldclk = 0;
-volatile struct intersil7170 *i7;
-
#define intersil_command(run, interrupt) \
(run | interrupt | INTERSIL_CMD_FREQ_32K | INTERSIL_CMD_24HR_MODE | \
INTERSIL_CMD_NORMAL_MODE)
@@ -105,23 +112,31 @@ volatile struct intersil7170 *i7;
#define intersil_clear(CLOCK) CLOCK->clk_intr_reg
+#if defined(SUN4)
/*
- * OCLOCK support: 4/100's and 4/200's have the old clock.
+ * OCLOCK support: 4/100's and 4/200's have the old clock.
*/
+static int oldclk = 0;
+struct intersil7170 *i7;
+
static long oclk_get_secs __P((void));
static void oclk_get_dt __P((struct date_time *));
static void dt_to_gmt __P((struct date_time *, long *));
static void oclk_set_dt __P((struct date_time *));
static void oclk_set_secs __P((long));
static void gmt_to_dt __P((long *, struct date_time *));
+#endif
static int oclockmatch __P((struct device *, void *, void *));
static void oclockattach __P((struct device *, struct device *, void *));
-struct cfdriver oclockcd =
- { NULL, "oclock", oclockmatch, oclockattach, DV_DULL,
- sizeof(struct device) };
-#endif /* SUN4 */
+struct cfattach oclock_ca = {
+ sizeof(struct device), oclockmatch, oclockattach
+};
+
+struct cfdriver oclock_cd = {
+ NULL, "oclock", DV_DULL
+};
/*
* Sun 4 machines use the old-style (a'la Sun 3) EEPROM. On the
@@ -133,61 +148,84 @@ struct cfdriver oclockcd =
* attached.
*/
char *eeprom_va = NULL;
+#if defined(SUN4)
static int eeprom_busy = 0;
static int eeprom_wanted = 0;
static int eeprom_nvram = 0; /* non-zero if eeprom is on Mostek */
+static int eeprom_take __P((void));
+static void eeprom_give __P((void));
+static int eeprom_update __P((char *, int, int));
+#endif
static int eeprom_match __P((struct device *, void *, void *));
static void eeprom_attach __P((struct device *, struct device *, void *));
-struct cfdriver eepromcd = {
- NULL, "eeprom", eeprom_match, eeprom_attach,
- DV_DULL, sizeof(struct device)
+
+struct cfattach eeprom_ca = {
+ sizeof(struct device), eeprom_match, eeprom_attach
};
-#if defined(SUN4)
-static int eeprom_update __P((char *, int, int));
-static int eeprom_take __P((void));
-static void eeprom_give __P((void));
-#endif
+struct cfdriver eeprom_cd = {
+ NULL, "eeprom", DV_DULL
+};
static int clockmatch __P((struct device *, void *, void *));
static void clockattach __P((struct device *, struct device *, void *));
-struct cfdriver clockcd = {
- NULL, "clock", clockmatch, clockattach,
- DV_DULL, sizeof(struct device)
+
+struct cfattach clock_ca = {
+ sizeof(struct device), clockmatch, clockattach
+};
+
+struct cfdriver clock_cd = {
+ NULL, "clock", DV_DULL
};
static int timermatch __P((struct device *, void *, void *));
static void timerattach __P((struct device *, struct device *, void *));
-struct cfdriver timercd = {
- NULL, "timer", timermatch, timerattach,
- DV_DULL, sizeof(struct device)
+
+struct timer_4m *timerreg_4m; /* XXX - need more cleanup */
+struct counter_4m *counterreg_4m;
+#define timerreg4 ((struct timerreg_4 *)TIMERREG_VA)
+
+struct cfattach timer_ca = {
+ sizeof(struct device), timermatch, timerattach
+};
+
+struct cfdriver timer_cd = {
+ NULL, "timer", DV_DULL
};
+struct chiptime;
+void clk_wenable __P((int));
+void myetheraddr __P((u_char *));
+int chiptotime __P((int, int, int, int, int, int));
+void timetochip __P((struct chiptime *));
+
+int timerblurb = 10; /* Guess a value; used before clock is attached */
/*
* old clock match routine
*/
-
static int
oclockmatch(parent, vcf, aux)
struct device *parent;
void *aux, *vcf;
{
- struct cfdata *cf = vcf;
register struct confargs *ca = aux;
-#if defined(SUN4)
- if (cputyp==CPU_SUN4) {
- if (cpumod == SUN4_100 || cpumod == SUN4_200)
- return (strcmp(oclockcd.cd_name, ca->ca_ra.ra_name) == 0);
+ /* Only these sun4s have oclock */
+ if (!CPU_ISSUN4 || (cpumod != SUN4_100 && cpumod != SUN4_200))
return (0);
- }
-#endif /* SUN4 */
- return (0); /* only sun4 has oclock */
-}
-#if defined(SUN4)
+ /* Check configuration name */
+ if (strcmp(oclock_cd.cd_name, ca->ca_ra.ra_name) != 0)
+ return (0);
+
+ /* Make sure there is something there */
+ if (probeget(ca->ca_ra.ra_vaddr, 1) == -1)
+ return (0);
+
+ return (1);
+}
/* ARGSUSED */
static void
@@ -195,16 +233,16 @@ oclockattach(parent, self, aux)
struct device *parent, *self;
void *aux;
{
+#if defined(SUN4)
struct confargs *ca = aux;
struct romaux *ra = &ca->ca_ra;
struct idprom *idp;
register int h;
oldclk = 1; /* we've got an oldie! */
- printf("\n");
- i7 = (volatile struct intersil7170 *) mapiodev(ra->ra_reg, 0,
- sizeof(*i7), ca->ca_bustype);
+ i7 = (struct intersil7170 *)
+ mapiodev(ra->ra_reg, 0, sizeof(*i7), ca->ca_bustype);
idp = &idprom;
h = idp->id_machine << 24;
@@ -212,8 +250,36 @@ oclockattach(parent, self, aux)
h |= idp->id_hostid[1] << 8;
h |= idp->id_hostid[2];
hostid = h;
-}
+
+ /*
+ * calibrate delay()
+ */
+ ienab_bic(IE_L14 | IE_L10); /* disable all clock intrs */
+ for (timerblurb = 1; ; timerblurb++) {
+ volatile register char *ireg = &i7->clk_intr_reg;
+ register int ival;
+ *ireg = INTERSIL_INTER_CSECONDS; /* 1/100 sec */
+ intersil_enable(i7); /* enable clock */
+ while ((*ireg & INTERSIL_INTER_PENDING) == 0)
+ /* sync with interrupt */;
+ while ((*ireg & INTERSIL_INTER_PENDING) == 0)
+ /* XXX: do it again, seems to need it */;
+ delay(10000); /* Probe 1/100 sec delay */
+ ival = *ireg; /* clear, save value */
+ intersil_disable(i7); /* disable clock */
+ if (ival & INTERSIL_INTER_PENDING) {
+ printf(" delay constant %d%s\n", timerblurb,
+ (timerblurb == 1) ? " [TOO SMALL?]" : "");
+ break;
+ }
+ if (timerblurb > 10) {
+ printf("\noclock: calibration failing; clamped at %d\n",
+ timerblurb);
+ break;
+ }
+ }
#endif /* SUN4 */
+}
/*
* Sun 4/100, 4/200 EEPROM match routine.
@@ -223,28 +289,31 @@ eeprom_match(parent, vcf, aux)
struct device *parent;
void *aux, *vcf;
{
-#if defined(SUN4)
struct cfdata *cf = vcf;
struct confargs *ca = aux;
- if (cputyp == CPU_SUN4) {
- if (cf->cf_unit != 0)
- return (0);
+ if (!CPU_ISSUN4)
+ return (0);
- if (cpumod == SUN4_100 || cpumod == SUN4_200) {
- if (strcmp(eepromcd.cd_name, ca->ca_ra.ra_name))
- return (0);
- /*
- * Make sure there's something there...
- * This is especially important if we want to
- * use the same kernel on a 4/100 as a 4/200.
- */
- if (probeget(ca->ca_ra.ra_vaddr, 1) != -1)
- return (1);
- }
- }
-#endif /* SUN4 */
- return (0);
+ if (cf->cf_unit != 0)
+ return (0);
+
+ if (cpumod != SUN4_100 && cpumod != SUN4_200)
+ return (0);
+
+ if (strcmp(eeprom_cd.cd_name, ca->ca_ra.ra_name) != 0)
+ return (0);
+
+ /*
+ * Make sure there's something there...
+ * This is especially important if we want to
+ * use the same kernel on a 4/100 as a 4/200.
+ */
+ if (probeget(ca->ca_ra.ra_vaddr, 1) == -1)
+ return (0);
+
+ /* Passed all tests */
+ return (1);
}
static void
@@ -258,8 +327,7 @@ eeprom_attach(parent, self, aux)
printf("\n");
- eeprom_va = (char *)mapiodev(ra->ra_reg, 0,
- sizeof(struct eeprom), ca->ca_bustype);
+ eeprom_va = (char *)mapiodev(ra->ra_reg, 0, EEPROM_SIZE, ca->ca_bustype);
eeprom_nvram = 0;
#endif /* SUN4 */
@@ -274,17 +342,23 @@ clockmatch(parent, vcf, aux)
struct device *parent;
void *aux, *vcf;
{
- struct cfdata *cf = vcf;
register struct confargs *ca = aux;
-#if defined(SUN4)
- if (cputyp==CPU_SUN4) {
- if (cpumod == SUN4_300 || cpumod == SUN4_400)
- return (strcmp(clockcd.cd_name,
- ca->ca_ra.ra_name) == 0);
- return (0);
+ if (CPU_ISSUN4) {
+ /* Only these sun4s have "clock" (others have "oclock") */
+ if (cpumod != SUN4_300 && cpumod != SUN4_400)
+ return (0);
+
+ if (strcmp(clock_cd.cd_name, ca->ca_ra.ra_name) != 0)
+ return (0);
+
+ /* Make sure there is something there */
+ if (probeget(ca->ca_ra.ra_vaddr, 1) == -1)
+ return (0);
+
+ return (1);
}
-#endif /* SUN4 */
+
return (strcmp("eeprom", ca->ca_ra.ra_name) == 0);
}
@@ -299,17 +373,20 @@ clockattach(parent, self, aux)
register struct idprom *idp;
struct confargs *ca = aux;
struct romaux *ra = &ca->ca_ra;
- char *prop;
+ char *prop = NULL;
-#if defined(SUN4)
- if (cputyp == CPU_SUN4)
+ if (CPU_ISSUN4)
prop = "mk48t02";
-#endif
-#if defined(SUN4C) || defined(SUN4M)
- if (cputyp == CPU_SUN4C || cputyp == CPU_SUN4M)
+
+ if (CPU_ISSUN4COR4M)
prop = getpropstring(ra->ra_node, "model");
+
+#ifdef DIAGNOSTIC
+ if (prop == NULL)
+ panic("no prop");
#endif
printf(": %s (eeprom)\n", prop);
+
/*
* We ignore any existing virtual address as we need to map
* this read-only and make it read-write only temporarily,
@@ -322,25 +399,25 @@ clockattach(parent, self, aux)
/*
* the MK48T08 is 8K
*/
- cl = (struct clockreg *)mapiodev(ra->ra_reg, 0, 2 * NBPG,
- ca->ca_bustype);
- pmap_changeprot(pmap_kernel(), (vm_offset_t)cl,
- VM_PROT_READ, 1);
- pmap_changeprot(pmap_kernel(), (vm_offset_t)cl + NBPG,
- VM_PROT_READ, 1);
+ cl = (struct clockreg *)mapiodev(ra->ra_reg, 0, 8192,
+ ca->ca_bustype);
+ pmap_changeprot(pmap_kernel(), (vm_offset_t)cl, VM_PROT_READ, 1);
+ pmap_changeprot(pmap_kernel(), (vm_offset_t)cl + 4096,
+ VM_PROT_READ, 1);
cl = (struct clockreg *)((int)cl + CLK_MK48T08_OFF);
} else {
/*
* the MK48T02 is 2K
*/
- cl = (struct clockreg *)mapiodev(ra->ra_reg, 0, sizeof *clockreg,
- ca->ca_bustype);
+ cl = (struct clockreg *)mapiodev(ra->ra_reg, 0,
+ sizeof *clockreg,
+ ca->ca_bustype);
pmap_changeprot(pmap_kernel(), (vm_offset_t)cl, VM_PROT_READ, 1);
- idp = &cl->cl_idprom;
}
+ idp = &cl->cl_idprom;
#if defined(SUN4)
- if (cputyp == CPU_SUN4) {
+ if (CPU_ISSUN4) {
idp = &idprom;
if (cpumod == SUN4_300 || cpumod == SUN4_400) {
@@ -349,6 +426,7 @@ clockattach(parent, self, aux)
}
}
#endif
+
h = idp->id_machine << 24;
h |= idp->id_hostid[0] << 16;
h |= idp->id_hostid[1] << 8;
@@ -365,17 +443,31 @@ timermatch(parent, vcf, aux)
struct device *parent;
void *aux, *vcf;
{
- struct cfdata *cf = vcf;
register struct confargs *ca = aux;
-#if defined(SUN4)
- if (cputyp==CPU_SUN4) {
- if (cpumod == SUN4_300 || cpumod == SUN4_400)
- return (strcmp("timer", ca->ca_ra.ra_name) == 0);
- return (0);
+ if (CPU_ISSUN4) {
+ if (cpumod != SUN4_300 && cpumod != SUN4_400)
+ return (0);
+
+ if (strcmp("timer", ca->ca_ra.ra_name) != 0)
+ return (0);
+
+ /* Make sure there is something there */
+ if (probeget(ca->ca_ra.ra_vaddr, 4) == -1)
+ return (0);
+
+ return (1);
}
-#endif /* SUN4 */
- return (strcmp("counter-timer", ca->ca_ra.ra_name) == 0);
+
+ if (CPU_ISSUN4C) {
+ return (strcmp("counter-timer", ca->ca_ra.ra_name) == 0);
+ }
+
+ if (CPU_ISSUN4M) {
+ return (strcmp("counter", ca->ca_ra.ra_name) == 0);
+ }
+
+ return (0);
}
/* ARGSUSED */
@@ -386,16 +478,70 @@ timerattach(parent, self, aux)
{
struct confargs *ca = aux;
register struct romaux *ra = &ca->ca_ra;
+ volatile int *cnt = NULL, *lim = NULL;
+ /* XXX: must init to NULL to avoid stupid gcc -Wall warning */
+
+ if (CPU_ISSUN4M) {
+ (void)mapdev(&ra->ra_reg[ra->ra_nreg-1], TIMERREG_VA, 0,
+ sizeof(struct timer_4m), ca->ca_bustype);
+ (void)mapdev(&ra->ra_reg[0], COUNTERREG_VA, 0,
+ sizeof(struct counter_4m), ca->ca_bustype);
+ timerreg_4m = (struct timer_4m *)TIMERREG_VA;
+ counterreg_4m = (struct counter_4m *)COUNTERREG_VA;
+
+ /* Put processor counter in "timer" mode */
+ timerreg_4m->t_cfg = 0;
+
+ cnt = &counterreg_4m->t_counter;
+ lim = &counterreg_4m->t_limit;
+ }
+
+ if (CPU_ISSUN4OR4C) {
+ /*
+ * This time, we ignore any existing virtual address because
+ * we have a fixed virtual address for the timer, to make
+ * microtime() faster (in SUN4/SUN4C kernel only).
+ */
+ (void)mapdev(ra->ra_reg, TIMERREG_VA, 0,
+ sizeof(struct timerreg_4), ca->ca_bustype);
+
+ cnt = &timerreg4->t_c14.t_counter;
+ lim = &timerreg4->t_c14.t_limit;
+ }
+
+ timerok = 1;
- printf("\n");
/*
- * This time, we ignore any existing virtual address because
- * we have a fixed virtual address for the timer, to make
- * microtime() faster.
+ * Calibrate delay() by tweaking the magic constant
+ * until a delay(100) actually reads (at least) 100 us on the clock.
+ * Note: sun4m clocks tick with 500ns periods.
*/
- (void)mapdev(ra->ra_reg, TIMERREG_VA, 0, sizeof(struct timerreg),
- ca->ca_bustype);
- timerok = 1;
+
+ for (timerblurb = 1; ; timerblurb++) {
+ volatile int discard;
+ register int t0, t1;
+
+ /* Reset counter register by writing some large limit value */
+ discard = *lim;
+ *lim = tmr_ustolim(TMR_MASK-1);
+
+ t0 = *cnt;
+ delay(100);
+ t1 = *cnt;
+
+ if (t1 & TMR_LIMIT)
+ panic("delay calibration");
+
+ t0 = (t0 >> TMR_SHIFT) & TMR_MASK;
+ t1 = (t1 >> TMR_SHIFT) & TMR_MASK;
+
+ if (t1 >= t0 + 100)
+ break;
+
+ }
+
+ printf(" delay constant %d\n", timerblurb);
+
/* should link interrupt handlers here, rather than compiled-in? */
}
@@ -418,7 +564,9 @@ clk_wenable(onoff)
prot = --writers == 0 ? VM_PROT_READ : 0;
splx(s);
if (prot)
- pmap_changeprot(pmap_kernel(), (vm_offset_t)clockreg, prot, 1);
+ pmap_changeprot(pmap_kernel(),
+ (vm_offset_t)clockreg & ~(NBPG-1),
+ prot, 1);
}
/*
@@ -432,9 +580,10 @@ myetheraddr(cp)
register struct idprom *idp = &cl->cl_idprom;
#if defined(SUN4)
- if (cputyp == CPU_SUN4)
+ if (CPU_ISSUN4)
idp = &idprom;
#endif
+
cp[0] = idp->id_ether[0];
cp[1] = idp->id_ether[1];
cp[2] = idp->id_ether[2];
@@ -444,56 +593,12 @@ myetheraddr(cp)
}
/*
- * This is easy to do on the machine with new-style clock chips
- * since we have freerunning microsecond timers -- no need to
- * guess at cpu speed factors. We just wait for it to change
- * n times (if we calculated a limit, we might overshoot, and
- * precision is irrelevant here---we want less object code).
- * On machines with the old-style clocks, we've determined good
- * estimates.
- */
-delay(n)
- register int n;
-{
- register int c, t;
-
-#if defined(SUN4)
- if (oldclk) {
- volatile register int lcv;
-
- /*
- * Two cases: 4/100 and 4/200.
- */
- if (cpumod == SUN4_200)
- n = cacheinfo.c_enabled ? (n << 3) : (n << 1);
- else
- n = n << 1;
- while (--n >= 0)
- lcv = n;
- return (0);
- }
-#endif /* SUN4 */
-
- if (timerok==0)
- return (0);
-
- if (timercd.cd_ndevs == 0)
- panic("delay");
- c = TIMERREG->t_c10.t_counter;
- while (--n >= 0) {
- while ((t = TIMERREG->t_c10.t_counter) == c)
- continue;
- c = t;
- }
- return (0);
-}
-
-/*
* Set up the real-time and statistics clocks. Leave stathz 0 only if
* no alternative timer is available.
*
* The frequencies of these clocks must be an even number of microseconds.
*/
+void
cpu_initclocks()
{
register int statint, minint;
@@ -501,22 +606,19 @@ cpu_initclocks()
#if defined(SUN4)
if (oldclk) {
int dummy;
+
profhz = hz = 100;
tick = 1000000 / hz;
i7->clk_intr_reg = INTERSIL_INTER_CSECONDS; /* 1/100 sec */
- ienab_bic(IE_L14 | IE_L10); /* disable all clock intrs */
-
- intersil_disable(i7); /* disable clock */
-
- dummy = intersil_clear(i7); /* clear interrupts */
-
- ienab_bis(IE_L10); /* enable l10 interrupt */
+ ienab_bic(IE_L14 | IE_L10); /* disable all clock intrs */
+ intersil_disable(i7); /* disable clock */
+ dummy = intersil_clear(i7); /* clear interrupts */
+ ienab_bis(IE_L10); /* enable l10 interrupt */
+ intersil_enable(i7); /* enable clock */
- intersil_enable(i7); /* enable clock */
-
- return (0);
+ return;
}
#endif /* SUN4 */
@@ -537,10 +639,27 @@ cpu_initclocks()
minint = statint / 2 + 100;
while (statvar > minint)
statvar >>= 1;
- TIMERREG->t_c10.t_limit = tmr_ustolim(tick);
- TIMERREG->t_c14.t_limit = tmr_ustolim(statint);
+
+ if (CPU_ISSUN4M) {
+ timerreg_4m->t_limit = tmr_ustolim(tick);
+ counterreg_4m->t_limit = tmr_ustolim(statint);
+ }
+
+ if (CPU_ISSUN4OR4C) {
+ timerreg4->t_c10.t_limit = tmr_ustolim(tick);
+ timerreg4->t_c14.t_limit = tmr_ustolim(statint);
+ }
+
statmin = statint - (statvar >> 1);
- ienab_bis(IE_L14 | IE_L10);
+
+#if defined(SUN4M)
+ if (CPU_ISSUN4M)
+ ienab_bic(SINTR_T);
+#endif
+
+ if (CPU_ISSUN4OR4C)
+ ienab_bis(IE_L14 | IE_L10);
+
}
/*
@@ -563,9 +682,18 @@ int
clockintr(cap)
void *cap;
{
- volatile register int discard;
+ register volatile int discard;
+ int s;
extern int rom_console_input;
+ /*
+ * Protect the clearing of the clock interrupt. If we don't
+ * do this, and we're interrupted (by the zs, for example),
+ * the clock stops!
+ * XXX WHY DOES THIS HAPPEN?
+ */
+ s = splhigh();
+
#if defined(SUN4)
if (oldclk) {
discard = intersil_clear(i7);
@@ -575,8 +703,18 @@ clockintr(cap)
}
#endif
/* read the limit register to clear the interrupt */
- discard = TIMERREG->t_c10.t_limit;
+ if (CPU_ISSUN4M) {
+ discard = timerreg_4m->t_limit;
+ }
+
+ if (CPU_ISSUN4OR4C) {
+ discard = timerreg4->t_c10.t_limit;
+ }
+#if defined(SUN4)
forward:
+#endif
+ splx(s);
+
hardclock((struct clockframe *)cap);
if (rom_console_input && cnrom())
setsoftint();
@@ -591,7 +729,7 @@ int
statintr(cap)
void *cap;
{
- volatile register int discard;
+ register volatile int discard;
register u_long newint, r, var;
#if defined(SUN4)
@@ -602,7 +740,20 @@ statintr(cap)
#endif
/* read the limit register to clear the interrupt */
- discard = TIMERREG->t_c14.t_limit;
+ if (CPU_ISSUN4M) {
+ discard = counterreg_4m->t_limit;
+ if (timerok == 0) {
+ /* Stop the clock */
+ counterreg_4m->t_limit = 0;
+ counterreg_4m->t_ss = 0;
+ timerreg_4m->t_cfg = TMR_CFG_USER;
+ return 1;
+ }
+ }
+
+ if (CPU_ISSUN4OR4C) {
+ discard = timerreg4->t_c14.t_limit;
+ }
statclock((struct clockframe *)cap);
/*
@@ -616,7 +767,13 @@ statintr(cap)
} while (r == 0);
newint = statmin + r;
- TIMERREG->t_c14.t_limit = tmr_ustolim(newint);
+ if (CPU_ISSUN4M) {
+ counterreg_4m->t_limit = tmr_ustolim(newint);
+ }
+
+ if (CPU_ISSUN4OR4C) {
+ timerreg4->t_c14.t_limit = tmr_ustolim(newint);
+ }
return (1);
}
@@ -628,6 +785,11 @@ statintr(cap)
#define SECDAY (24 * 60 * 60)
#define SECYR (SECDAY * 365)
+/*
+ * should use something like
+ * #define LEAPYEAR(y) ((((y) % 4) == 0 && ((y) % 100) != 0) || ((y) % 400) == 0)
+ * but it's unlikely that we'll still be around in 2100.
+ */
#define LEAPYEAR(y) (((y) & 3) == 0)
/*
@@ -637,6 +799,7 @@ statintr(cap)
const short dayyr[12] =
{ 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334 };
+int
chiptotime(sec, min, hour, day, mon, year)
register int sec, min, hour, day, mon, year;
{
@@ -672,6 +835,7 @@ struct chiptime {
int year;
};
+void
timetochip(c)
register struct chiptime *c;
{
@@ -720,6 +884,7 @@ timetochip(c)
/*
* Set up the system's time, given a `reasonable' time value.
*/
+void
inittodr(base)
time_t base;
{
@@ -756,7 +921,9 @@ inittodr(base)
clk_wenable(0);
time.tv_sec = chiptotime(sec, min, hour, day, mon, year);
+#if defined(SUN4)
forward:
+#endif
if (time.tv_sec == 0) {
printf("WARNING: bad date in battery clock");
/*
@@ -785,6 +952,7 @@ forward:
* and when rebooting. Do nothing if the time is not yet known, e.g.,
* when crashing during autoconfig.
*/
+void
resettodr()
{
register struct clockreg *cl;
@@ -822,24 +990,24 @@ resettodr()
static long
oclk_get_secs()
{
- struct date_time dt;
- long gmt;
+ struct date_time dt;
+ long gmt;
- oclk_get_dt(&dt);
- dt_to_gmt(&dt, &gmt);
- return (gmt);
+ oclk_get_dt(&dt);
+ dt_to_gmt(&dt, &gmt);
+ return (gmt);
}
static void
oclk_set_secs(secs)
long secs;
{
- struct date_time dt;
- long gmt;
+ struct date_time dt;
+ long gmt;
- gmt = secs;
- gmt_to_dt(&gmt, &dt);
- oclk_set_dt(&dt);
+ gmt = secs;
+ gmt_to_dt(&gmt, &dt);
+ oclk_set_dt(&dt);
}
/*
@@ -851,48 +1019,48 @@ static void
oclk_get_dt(dt)
struct date_time *dt;
{
- int s;
- register volatile char *src, *dst;
+ int s;
+ register volatile char *src, *dst;
- src = (char *) &i7->counters;
+ src = (char *) &i7->counters;
- s = splhigh();
- i7->clk_cmd_reg =
- intersil_command(INTERSIL_CMD_STOP, INTERSIL_CMD_IENABLE);
+ s = splhigh();
+ i7->clk_cmd_reg =
+ intersil_command(INTERSIL_CMD_STOP, INTERSIL_CMD_IENABLE);
- dst = (char *) dt;
- dt++; /* end marker */
- do {
- *dst++ = *src++;
- } while (dst < (char*)dt);
+ dst = (char *) dt;
+ dt++; /* end marker */
+ do {
+ *dst++ = *src++;
+ } while (dst < (char*)dt);
- i7->clk_cmd_reg =
- intersil_command(INTERSIL_CMD_RUN, INTERSIL_CMD_IENABLE);
- splx(s);
+ i7->clk_cmd_reg =
+ intersil_command(INTERSIL_CMD_RUN, INTERSIL_CMD_IENABLE);
+ splx(s);
}
static void
oclk_set_dt(dt)
struct date_time *dt;
{
- int s;
- register volatile char *src, *dst;
+ int s;
+ register volatile char *src, *dst;
- dst = (char *) &i7->counters;
+ dst = (char *) &i7->counters;
- s = splhigh();
- i7->clk_cmd_reg =
- intersil_command(INTERSIL_CMD_STOP, INTERSIL_CMD_IENABLE);
+ s = splhigh();
+ i7->clk_cmd_reg =
+ intersil_command(INTERSIL_CMD_STOP, INTERSIL_CMD_IENABLE);
- src = (char *) dt;
- dt++; /* end marker */
- do {
- *dst++ = *src++;
- } while (src < (char *)dt);
+ src = (char *) dt;
+ dt++; /* end marker */
+ do {
+ *dst++ = *src++;
+ } while (src < (char *)dt);
- i7->clk_cmd_reg =
- intersil_command(INTERSIL_CMD_RUN, INTERSIL_CMD_IENABLE);
- splx(s);
+ i7->clk_cmd_reg =
+ intersil_command(INTERSIL_CMD_RUN, INTERSIL_CMD_IENABLE);
+ splx(s);
}
@@ -904,14 +1072,14 @@ oclk_set_dt(dt)
/* Traditional UNIX base year */
#define POSIX_BASE_YEAR 1970
-#define FEBRUARY 2
+#define FEBRUARY 2
-#define leapyear(year) ((year) % 4 == 0)
-#define days_in_year(a) (leapyear(a) ? 366 : 365)
-#define days_in_month(a) (month_days[(a) - 1])
+#define leapyear(year) ((year) % 4 == 0)
+#define days_in_year(a) (leapyear(a) ? 366 : 365)
+#define days_in_month(a) (month_days[(a) - 1])
static int month_days[12] = {
- 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31
+ 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31
};
static void
@@ -919,90 +1087,90 @@ gmt_to_dt(tp, dt)
long *tp;
struct date_time *dt;
{
- register int i;
- register long days, secs;
-
- days = *tp / SECDAY;
- secs = *tp % SECDAY;
-
- /* Hours, minutes, seconds are easy */
- dt->dt_hour = secs / 3600;
- secs = secs % 3600;
- dt->dt_min = secs / 60;
- secs = secs % 60;
- dt->dt_sec = secs;
-
- /* Day of week (Note: 1/1/1970 was a Thursday) */
- dt->dt_dow = (days + 4) % 7;
-
- /* Number of years in days */
- i = POSIX_BASE_YEAR;
- while (days >= days_in_year(i)) {
- days -= days_in_year(i);
- i++;
- }
- dt->dt_year = i - CLOCK_BASE_YEAR;
-
- /* Number of months in days left */
- if (leapyear(i))
- days_in_month(FEBRUARY) = 29;
- for (i = 1; days >= days_in_month(i); i++)
- days -= days_in_month(i);
- days_in_month(FEBRUARY) = 28;
- dt->dt_month = i;
-
- /* Days are what is left over (+1) from all that. */
- dt->dt_day = days + 1;
+ register int i;
+ register long days, secs;
+
+ days = *tp / SECDAY;
+ secs = *tp % SECDAY;
+
+ /* Hours, minutes, seconds are easy */
+ dt->dt_hour = secs / 3600;
+ secs = secs % 3600;
+ dt->dt_min = secs / 60;
+ secs = secs % 60;
+ dt->dt_sec = secs;
+
+ /* Day of week (Note: 1/1/1970 was a Thursday) */
+ dt->dt_dow = (days + 4) % 7;
+
+ /* Number of years in days */
+ i = POSIX_BASE_YEAR;
+ while (days >= days_in_year(i)) {
+ days -= days_in_year(i);
+ i++;
+ }
+ dt->dt_year = i - CLOCK_BASE_YEAR;
+
+ /* Number of months in days left */
+ if (leapyear(i))
+ days_in_month(FEBRUARY) = 29;
+ for (i = 1; days >= days_in_month(i); i++)
+ days -= days_in_month(i);
+ days_in_month(FEBRUARY) = 28;
+ dt->dt_month = i;
+
+ /* Days are what is left over (+1) from all that. */
+ dt->dt_day = days + 1;
}
+
static void
dt_to_gmt(dt, tp)
struct date_time *dt;
long *tp;
{
- register int i;
- register long tmp;
- int year;
+ register int i;
+ register long tmp;
+ int year;
- /*
- * Hours are different for some reason. Makes no sense really.
- */
- tmp = 0;
+ /*
+ * Hours are different for some reason. Makes no sense really.
+ */
- if (dt->dt_hour >= 24)
- goto out;
- if (dt->dt_day > 31)
- goto out;
- if (dt->dt_month > 12)
- goto out;
+ tmp = 0;
- year = dt->dt_year + CLOCK_BASE_YEAR;
+ if (dt->dt_hour >= 24) goto out;
+ if (dt->dt_day > 31) goto out;
+ if (dt->dt_month > 12) goto out;
+
+ year = dt->dt_year + CLOCK_BASE_YEAR;
- /*
- * Compute days since start of time
- * First from years, then from months.
- */
- for (i = POSIX_BASE_YEAR; i < year; i++)
- tmp += days_in_year(i);
- if (leapyear(year) && dt->dt_month > FEBRUARY)
- tmp++;
- /* Months */
- for (i = 1; i < dt->dt_month; i++)
- tmp += days_in_month(i);
- tmp += (dt->dt_day - 1);
+ /*
+ * Compute days since start of time
+ * First from years, then from months.
+ */
+ for (i = POSIX_BASE_YEAR; i < year; i++)
+ tmp += days_in_year(i);
+ if (leapyear(year) && dt->dt_month > FEBRUARY)
+ tmp++;
- /* Now do hours */
- tmp = tmp * 24 + dt->dt_hour;
+ /* Months */
+ for (i = 1; i < dt->dt_month; i++)
+ tmp += days_in_month(i);
+ tmp += (dt->dt_day - 1);
- /* Now do minutes */
- tmp = tmp * 60 + dt->dt_min;
+ /* Now do hours */
+ tmp = tmp * 24 + dt->dt_hour;
- /* Now do seconds */
- tmp = tmp * 60 + dt->dt_sec;
+ /* Now do minutes */
+ tmp = tmp * 60 + dt->dt_min;
+
+ /* Now do seconds */
+ tmp = tmp * 60 + dt->dt_sec;
out:
- *tp = tmp;
+ *tp = tmp;
}
#endif /* SUN4 */
@@ -1023,26 +1191,21 @@ microtime(tvp)
{
int s;
static struct timeval lasttime;
+ static struct timeval oneusec = {0, 1};
if (!oldclk) {
lo_microtime(tvp);
return;
}
+
s = splhigh();
*tvp = time;
- tvp->tv_usec;
- while (tvp->tv_usec > 1000000) {
- tvp->tv_sec++;
- tvp->tv_usec -= 1000000;
- }
- if (tvp->tv_sec == lasttime.tv_sec &&
- tvp->tv_usec <= lasttime.tv_usec &&
- (tvp->tv_usec = lasttime.tv_usec + 1) > 1000000) {
- tvp->tv_sec++;
- tvp->tv_usec -= 1000000;
- }
- lasttime = *tvp;
splx(s);
+
+ if (timercmp(tvp, &lasttime, <=))
+ timeradd(&lasttime, &oneusec, tvp);
+
+ lasttime = *tvp;
}
#endif /* SUN4 */
@@ -1061,19 +1224,18 @@ eeprom_uio(uio)
u_int cnt, bcnt;
caddr_t buf = NULL;
- if (cputyp != CPU_SUN4)
+ if (!CPU_ISSUN4)
return (ENODEV);
off = uio->uio_offset;
- if (off > sizeof(struct eeprom))
+ if (off > EEPROM_SIZE)
return (EFAULT);
cnt = uio->uio_resid;
- if (cnt > sizeof(struct eeprom) - off)
- cnt = sizeof(struct eeprom) - off;
+ if (cnt > (EEPROM_SIZE - off))
+ cnt = (EEPROM_SIZE - off);
- error = eeprom_take();
- if (error != 0)
+ if ((error = eeprom_take()) != 0)
return (error);
if (eeprom_va == NULL) {
@@ -1087,14 +1249,14 @@ eeprom_uio(uio)
* this, we byte-by-byte copy the eeprom contents into a
* temporary buffer.
*/
- buf = malloc(sizeof(struct eeprom), M_DEVBUF, M_WAITOK);
+ buf = malloc(EEPROM_SIZE, M_DEVBUF, M_WAITOK);
if (buf == NULL) {
error = EAGAIN;
goto out;
}
if (uio->uio_rw == UIO_READ)
- for (bcnt = 0; bcnt < sizeof(struct eeprom); ++bcnt)
+ for (bcnt = 0; bcnt < EEPROM_SIZE; ++bcnt)
*(char *)(buf + bcnt) = *(char *)(eeprom_va + bcnt);
if ((error = uiomove(buf + off, (int)cnt, uio)) != 0)
@@ -1103,7 +1265,7 @@ eeprom_uio(uio)
if (uio->uio_rw != UIO_READ)
error = eeprom_update(buf, off, cnt);
-out:
+ out:
if (buf)
free(buf, M_DEVBUF);
eeprom_give();
@@ -1132,10 +1294,6 @@ eeprom_update(buf, off, cnt)
ep = eeprom_va + off;
bp = buf + off;
- /*
- * XXX: I'm not totally sure if this is necessary, and I don't
- * know if there are any harmful side effects, either.
- */
if (eeprom_nvram)
clk_wenable(1);
@@ -1164,8 +1322,7 @@ eeprom_update(buf, off, cnt)
++bp;
--cnt;
}
-out:
- /* XXX: see above. */
+ out:
if (eeprom_nvram)
clk_wenable(0);
@@ -1186,7 +1343,7 @@ eeprom_take()
goto out;
}
eeprom_busy = 1;
-out:
+ out:
return (error);
}
@@ -1194,6 +1351,7 @@ out:
static void
eeprom_give()
{
+
eeprom_busy = 0;
if (eeprom_wanted) {
eeprom_wanted = 0;
diff --git a/sys/arch/sparc/sparc/conf.c b/sys/arch/sparc/sparc/conf.c
index a409f8c77c5..17f23d77fda 100644
--- a/sys/arch/sparc/sparc/conf.c
+++ b/sys/arch/sparc/sparc/conf.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: conf.c,v 1.6 1996/07/15 14:57:07 mickey Exp $ */
+/* $NetBSD: conf.c,v 1.40 1996/04/11 19:20:03 thorpej Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -52,49 +52,49 @@
#include <sys/tty.h>
#include <sys/conf.h>
-int ttselect __P((dev_t, int, struct proc *));
+#include <machine/conf.h>
-bdev_decl(sw);
+#include "pty.h"
+#include "bpfilter.h"
+#include "tun.h"
+#include "audio.h"
+#include "vnd.h"
+#include "ccd.h"
+#include "ch.h"
+#include "ss.h"
#include "sd.h"
-bdev_decl(sd);
-#include "xd.h"
-bdev_decl(xd);
-#include "xy.h"
-bdev_decl(xy);
#include "st.h"
-bdev_decl(st);
#include "cd.h"
-bdev_decl(cd);
-#include "vnd.h"
-bdev_decl(vnd);
-#define fdopen Fdopen /* conflicts with fdopen() in kern_descrip.c */
-#include "fd.h"
-bdev_decl(fd);
-#undef fdopen
-#include "ccd.h"
-bdev_decl(ccd);
+
+#include "zs.h"
+#include "fdc.h" /* has NFDC and NFD; see files.sparc */
+#include "bwtwo.h"
+#include "cgthree.h"
+#include "cgfour.h"
+#include "cgsix.h"
+#include "cgeight.h"
+#include "xd.h"
+#include "xy.h"
struct bdevsw bdevsw[] =
{
bdev_notdef(), /* 0 */
bdev_notdef(), /* 1 */
bdev_notdef(), /* 2 */
- bdev_disk_init(NXY,xy), /* 3: XY SMD disk */
+ bdev_disk_init(NXY,xy), /* 3: SMD disk */
bdev_swap_init(1,sw), /* 4 */
bdev_notdef(), /* 5 */
bdev_notdef(), /* 6 */
bdev_disk_init(NSD,sd), /* 7: SCSI disk */
bdev_disk_init(NVND,vnd), /* 8: vnode disk driver */
bdev_disk_init(NCCD,ccd), /* 9: concatenated disk driver */
- bdev_disk_init(NXD,xd), /* 10: XD SMD disk */
+ bdev_disk_init(NXD,xd), /* 10: SMD disk */
bdev_tape_init(NST,st), /* 11: SCSI tape */
bdev_notdef(), /* 12 */
bdev_notdef(), /* 13 */
bdev_notdef(), /* 14 */
bdev_notdef(), /* 15 */
-#define fdopen Fdopen /* conflicts with fdopen() in kern_descrip.c */
bdev_disk_init(NFD,fd), /* 16: floppy disk */
-#undef fdopen
bdev_notdef(), /* 17 */
bdev_disk_init(NCD,cd), /* 18: SCSI CD-ROM */
bdev_lkm_dummy(), /* 19 */
@@ -106,86 +106,6 @@ struct bdevsw bdevsw[] =
};
int nblkdev = sizeof(bdevsw) / sizeof(bdevsw[0]);
-/* open, close, read, write, ioctl, select */
-#define cdev_gen_init(c,n) { \
- dev_init(c,n,open), dev_init(c,n,close), dev_init(c,n,read), \
- dev_init(c,n,write), dev_init(c,n,ioctl), (dev_type_stop((*))) nullop, \
- 0, dev_init(c,n,select), (dev_type_mmap((*))) enodev }
-
-/* open, close, ioctl */
-#define cdev_openprom_init(c,n) { \
- dev_init(c,n,open), dev_init(c,n,close), (dev_type_read((*))) enodev, \
- (dev_type_write((*))) enodev, dev_init(c,n,ioctl), \
- (dev_type_stop((*))) nullop, 0, (dev_type_select((*))) enodev, \
- (dev_type_mmap((*))) enodev }
-
-cdev_decl(cn);
-cdev_decl(ctty);
-#define mmread mmrw
-#define mmwrite mmrw
-cdev_decl(mm);
-cdev_decl(sw);
-#include "zs.h"
-cdev_decl(zs);
-cdev_decl(ms);
-cdev_decl(log);
-cdev_decl(sd);
-cdev_decl(st);
-cdev_decl(xd);
-cdev_decl(xy);
-#include "ch.h"
-cdev_decl(ch);
-#include "pty.h"
-#define ptstty ptytty
-#define ptsioctl ptyioctl
-cdev_decl(pts);
-#define ptctty ptytty
-#define ptcioctl ptyioctl
-cdev_decl(ptc);
-cdev_decl(fb);
-dev_decl(fd,open);
-#include "bwtwo.h"
-cdev_decl(bwtwo);
-cdev_decl(kbd);
-#define fdopen Fdopen
-cdev_decl(fd);
-#undef fdopen
-#include "cgthree.h"
-cdev_decl(cgthree);
-#include "cgfour.h"
-cdev_decl(cgfour);
-cdev_decl(cd);
-#include "cgsix.h"
-cdev_decl(cgsix);
-#include "cgeight.h"
-cdev_decl(cgeight);
-#include "audio.h"
-cdev_decl(audio);
-cdev_decl(openprom);
-#include "bpfilter.h"
-cdev_decl(bpf);
-cdev_decl(vnd);
-cdev_decl(ccd);
-#include "tun.h"
-cdev_decl(tun);
-cdev_decl(svr4_net);
-#ifdef LKM
-#define NLKM 1
-#else
-#define NLKM 0
-#endif
-cdev_decl(lkm);
-#include "rnd.h"
-cdev_decl(rnd);
-
-/* open, close, read, ioctl */
-cdev_decl(ipl);
-#ifdef IPFILTER
-#define NIPF 1
-#else
-#define NIPF 0
-#endif
-
struct cdevsw cdevsw[] =
{
cdev_cn_init(1,cn), /* 0: virtual console */
@@ -197,7 +117,7 @@ struct cdevsw cdevsw[] =
cdev_notdef(), /* 6 */
cdev_swap_init(1,sw), /* 7: /dev/drum (swap pseudo-device) */
cdev_notdef(), /* 8 */
- cdev_disk_init(NXY,xy), /* 9: XY SMD disk */
+ cdev_disk_init(NXY,xy), /* 9: SMD disk */
cdev_notdef(), /* 10 */
cdev_notdef(), /* 11 */
cdev_tty_init(NZS,zs), /* 12: zs serial */
@@ -212,7 +132,7 @@ struct cdevsw cdevsw[] =
cdev_ptc_init(NPTY,ptc), /* 21: pseudo-tty master */
cdev_fb_init(1,fb), /* 22: /dev/fb indirect driver */
cdev_disk_init(NCCD,ccd), /* 23: concatenated disk driver */
- cdev_fd_init(1,fd), /* 24: file descriptor pseudo-device */
+ cdev_fd_init(1,filedesc), /* 24: file descriptor pseudo-device */
cdev_notdef(), /* 25 */
cdev_notdef(), /* 26 */
cdev_fb_init(NBWTWO,bwtwo), /* 27: /dev/bwtwo */
@@ -230,12 +150,8 @@ struct cdevsw cdevsw[] =
cdev_fb_init(NCGFOUR,cgfour), /* 39: /dev/cgfour */
cdev_notdef(), /* 40 */
cdev_notdef(), /* 41 */
- cdev_disk_init(NXD,xd), /* 42: XD SMD disk */
-#ifdef COMPAT_SVR4
- cdev_svr4_net_init(1,svr4_net), /* 43: svr4 net pseudo-device */
-#else
- cdev_notdef(), /* 43 */
-#endif
+ cdev_disk_init(NXD,xd), /* 42: SMD disk */
+ cdev_svr4_net_init(NSVR4_NET,svr4_net), /* 43: svr4 net pseudo-device */
cdev_notdef(), /* 44 */
cdev_notdef(), /* 45 */
cdev_notdef(), /* 46 */
@@ -246,14 +162,12 @@ struct cdevsw cdevsw[] =
cdev_notdef(), /* 51 */
cdev_notdef(), /* 52 */
cdev_notdef(), /* 53 */
-#define fdopen Fdopen /* conflicts with fdopen() in kern_descrip.c */
cdev_disk_init(NFD,fd), /* 54: floppy disk */
-#undef fdopen
cdev_fb_init(NCGTHREE,cgthree), /* 55: /dev/cgthree */
cdev_notdef(), /* 56 */
cdev_notdef(), /* 57 */
cdev_disk_init(NCD,cd), /* 58 SCSI CD-ROM */
- cdev_gen_ipf(NIPF,ipl), /* 59 */
+ cdev_notdef(), /* 59 */
cdev_notdef(), /* 60 */
cdev_notdef(), /* 61 */
cdev_notdef(), /* 62 */
@@ -313,7 +227,6 @@ struct cdevsw cdevsw[] =
cdev_lkm_dummy(), /* 116 */
cdev_lkm_dummy(), /* 117 */
cdev_lkm_dummy(), /* 118 */
- cdev_rnd_init(NRND,rnd), /* 119: random data source */
};
int nchrdev = sizeof(cdevsw) / sizeof(cdevsw[0]);
@@ -335,6 +248,7 @@ dev_t swapdev = makedev(4, 0);
*
* A minimal stub routine can always return 0.
*/
+int
iskmemdev(dev)
dev_t dev;
{
@@ -342,6 +256,7 @@ iskmemdev(dev)
return (major(dev) == mem_no && minor(dev) < 2);
}
+int
iszerodev(dev)
dev_t dev;
{
@@ -475,6 +390,7 @@ static int chrtoblktbl[] = {
/*
* Routine to convert from character to block device number.
*/
+int
chrtoblk(dev)
dev_t dev;
{
diff --git a/sys/arch/sparc/sparc/cpu.c b/sys/arch/sparc/sparc/cpu.c
index 43437db234b..f4d7d493a50 100644
--- a/sys/arch/sparc/sparc/cpu.c
+++ b/sys/arch/sparc/sparc/cpu.c
@@ -1,6 +1,8 @@
-/* $NetBSD: cpu.c,v 1.14 1995/02/01 12:37:52 pk Exp $ */
+/* $NetBSD: cpu.c,v 1.22.4.1 1996/06/12 20:39:45 pk Exp $ */
/*
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
*
@@ -10,12 +12,14 @@
*
* All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
+ * This product includes software developed by Harvard University.
* This product includes software developed by the University of
* California, Lawrence Berkeley Laboratory.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
+ *
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
@@ -23,6 +27,8 @@
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
+ * This product includes software developed by Aaron Brown and
+ * Harvard University.
* This product includes software developed by the University of
* California, Berkeley and its contributors.
* 4. Neither the name of the University nor the names of its contributors
@@ -42,17 +48,24 @@
* SUCH DAMAGE.
*
* @(#)cpu.c 8.5 (Berkeley) 11/23/93
+ *
*/
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/device.h>
+#include <vm/vm.h>
+
#include <machine/autoconf.h>
#include <machine/cpu.h>
#include <machine/reg.h>
+#include <machine/ctlreg.h>
#include <machine/trap.h>
+#include <machine/pmap.h>
#include <sparc/sparc/cache.h>
+#include <sparc/sparc/asm.h>
/* This is declared here so that you must include a CPU for the cache code. */
struct cacheinfo cacheinfo;
@@ -65,14 +78,25 @@ char cpu_model[100];
static void cpu_attach __P((struct device *, struct device *, void *));
int cpu_match __P((struct device *, void *, void *));
-struct cfdriver cpucd =
- { NULL, "cpu", cpu_match, cpu_attach, DV_CPU, sizeof(struct device) };
+struct cfattach cpu_ca = {
+ sizeof(struct device), cpu_match, cpu_attach
+};
+
+struct cfdriver cpu_cd = {
+ NULL, "cpu", DV_CPU
+};
+#if defined(SUN4C) || defined(SUN4M)
static char *psrtoname __P((int, int, int, char *));
+#endif
static char *fsrtoname __P((int, int, int, char *));
#define IU_IMPL(psr) ((u_int)(psr) >> 28)
#define IU_VERS(psr) (((psr) >> 24) & 0xf)
+#if defined(SUN4M)
+#define SRMMU_IMPL(mmusr) ((u_int)(mmusr) >> 28)
+#define SRMMU_VERS(mmusr) (((mmusr) >> 24) & 0xf)
+#endif
#ifdef notdef
/*
@@ -101,7 +125,7 @@ static char *iu_vendor[16] = {
/*
* 4/110 comment: the 4/110 chops off the top 4 bits of an OBIO address.
- * this confuses autoconf. for example, if you try and map
+ * this confuses autoconf. for example, if you try and map
* 0xfe000000 in obio space on a 4/110 it actually maps 0x0e000000.
* this is easy to verify with the PROM. this causes problems
* with devices like "esp0 at obio0 addr 0xfa000000" because the
@@ -135,12 +159,17 @@ cpu_attach(parent, dev, aux)
struct device *dev;
void *aux;
{
- register int node, clk, bug = 0, i, l;
+ register int node, clk, bug = 0, i;
register int impl, vers, fver;
register char *fpuname;
struct confargs *ca = aux;
struct fpstate fpstate;
- char iubuf[40], fpbuf[40];
+ char *sep;
+ char fpbuf[40];
+#if defined(SUN4C) || defined(SUN4M)
+ register int l;
+ char iubuf[40];
+#endif
/*
* Get the FSR and clear any exceptions. If we do not unload
@@ -165,8 +194,9 @@ cpu_attach(parent, dev, aux)
/* tell them what we have */
node = ca->ca_ra.ra_node;
+
#ifdef SUN4
- if (cputyp == CPU_SUN4) {
+ if (CPU_ISSUN4) {
clk = 0;
vactype = VAC_WRITEBACK;
switch (cpumod) {
@@ -188,7 +218,6 @@ cpu_attach(parent, dev, aux)
case SUN4_300:
sprintf(cpu_model, "SUN-4/300 series (%s FPU)", fpuname);
bug = 1;
- vactype = VAC_WRITETHROUGH;
cacheinfo.c_totalsize = 128*1024;
cacheinfo.c_hwflush = 0;
cacheinfo.c_linesize = 16;
@@ -204,29 +233,136 @@ cpu_attach(parent, dev, aux)
}
printf(": %s\n", cpu_model);
}
-#endif
+#endif /* SUN4 */
+
#if defined(SUN4C) || defined(SUN4M)
- if (cputyp == CPU_SUN4C || cputyp == CPU_SUN4M) {
+ if (CPU_ISSUN4C || CPU_ISSUN4M) {
clk = getpropint(node, "clock-frequency", 0);
- sprintf(cpu_model, "%s (%s @ %s MHz, %s FPU)",
- getpropstring(node, "name"),
- psrtoname(impl, vers, fver, iubuf), clockfreq(clk), fpuname);
+ if (clk == 0) {
+ /*
+ * Try to find it in the OpenPROM root...
+ */
+ clk = getpropint(findroot(), "clock-frequency", 0);
+ }
+ if (CPU_ISSUN4C)
+ sprintf(cpu_model, "%s (%s @ %s MHz, %s FPU)",
+ getpropstring(node, "name"),
+ psrtoname(impl, vers, fver, iubuf),
+ clockfreq(clk), fpuname);
+ else
+ /* On sun4m, the "name" property identifies CPU */
+ sprintf(cpu_model, "%s @ %s MHz, %s FPU",
+ getpropstring(node, "name"),
+ clockfreq(clk), fpuname);
printf(": %s\n", cpu_model);
/*
* Fill in the cache info. Note, vac-hwflush is spelled
* with an underscore on 4/75s.
*/
- cacheinfo.c_totalsize = getpropint(node, "vac-size", 65536);
- cacheinfo.c_hwflush = getpropint(node, "vac_hwflush", 0) |
- getpropint(node, "vac-hwflush", 0);
- cacheinfo.c_linesize = l = getpropint(node, "vac-linesize", 16);
- for (i = 0; (1 << i) < l; i++)
- /* void */;
- if ((1 << i) != l)
- panic("bad cache line size %d", l);
- cacheinfo.c_l2linesize = i;
- vactype = VAC_WRITETHROUGH;
+ /*bzero(&cacheinfo, sizeof(cacheinfo));*/
+#if defined(SUN4M)
+ if (CPU_ISSUN4M) {
+ cacheinfo.c_physical = 1;
+ vactype = node_has_property(node, "cache-physical?")
+ ? VAC_NONE
+ : VAC_WRITETHROUGH; /* ??? */
+ /*
+ * Sun4m physical caches are nice since we never
+ * have to flush them. Unfortunately it is a pain
+ * to determine what size they are, since they may
+ * be split...
+ */
+ switch (mmumod) {
+ case SUN4M_MMU_SS:
+ case SUN4M_MMU_MS1:
+ cacheinfo.c_split = 1;
+ cacheinfo.ic_linesize = l =
+ getpropint(node, "icache-line-size", 0);
+ for (i = 0; (1 << i) < l && l; i++)
+ /* void */;
+ if ((1 << i) != l && l)
+ panic("bad icache line size %d", l);
+ cacheinfo.ic_l2linesize = i;
+ cacheinfo.ic_totalsize = l *
+ getpropint(node, "icache-nlines", 64) *
+ getpropint(node, "icache-associativity", 1);
+
+ cacheinfo.dc_linesize = l =
+ getpropint(node, "dcache-line-size",0);
+ for (i = 0; (1 << i) < l && l; i++)
+ /* void */;
+ if ((1 << i) != l && l)
+ panic("bad dcache line size %d", l);
+ cacheinfo.dc_l2linesize = i;
+ cacheinfo.dc_totalsize = l *
+ getpropint(node, "dcache-nlines", 128) *
+ getpropint(node, "dcache-associativity", 1);
+
+ cacheinfo.ec_linesize = l =
+ getpropint(node, "ecache-line-size", 0);
+ for (i = 0; (1 << i) < l && l; i++)
+ /* void */;
+ if ((1 << i) != l && l)
+ panic("bad ecache line size %d", l);
+ cacheinfo.ec_l2linesize = i;
+ cacheinfo.ec_totalsize = l *
+ getpropint(node, "ecache-nlines", 32768) *
+ getpropint(node, "ecache-associativity", 1);
+
+ /*
+ * XXX - The following will have to do until
+ * we have per-cpu cache handling.
+ */
+ cacheinfo.c_l2linesize =
+ min(cacheinfo.ic_l2linesize,
+ cacheinfo.dc_l2linesize);
+ cacheinfo.c_linesize =
+ min(cacheinfo.ic_linesize,
+ cacheinfo.dc_linesize);
+ cacheinfo.c_totalsize =
+ cacheinfo.ic_totalsize +
+ cacheinfo.dc_totalsize;
+ break;
+ case SUN4M_MMU_HS:
+ printf("Warning, guessing on HyperSPARC cache...\n");
+ cacheinfo.c_split = 0;
+ i = lda(SRMMU_PCR, ASI_SRMMU);
+ if (i & SRMMU_PCR_CS)
+ cacheinfo.c_totalsize = 256 * 1024;
+ else
+ cacheinfo.c_totalsize = 128 * 1024;
+ /* manual says it has 4096 lines */
+ cacheinfo.c_linesize = l =
+ cacheinfo.c_totalsize / 4096;
+ for (i = 0; (1 << i) < l; i++)
+ /* void */;
+ if ((1 << i) != l)
+ panic("bad cache line size %d", l);
+ cacheinfo.c_l2linesize = i;
+ break;
+ default:
+ printf("warning: couldn't identify cache\n");
+ cacheinfo.c_totalsize = 0;
+ }
+ } else
+#endif /* SUN4M */
+ {
+ cacheinfo.c_physical = 0;
+ cacheinfo.c_totalsize =
+ getpropint(node, "vac-size", 65536);
+ cacheinfo.c_hwflush =
+ getpropint(node, "vac_hwflush", 0) |
+ getpropint(node, "vac-hwflush", 0);
+ cacheinfo.c_linesize = l =
+ getpropint(node, "vac-linesize", 16);
+ for (i = 0; (1 << i) < l; i++)
+ /* void */;
+ if ((1 << i) != l)
+ panic("bad cache line size %d", l);
+ cacheinfo.c_l2linesize = i;
+ vactype = VAC_WRITETHROUGH;
+ }
/*
* Machines with "buserr-type" 1 have a bug in the cache
@@ -236,19 +372,51 @@ cpu_attach(parent, dev, aux)
bug = (getpropint(node, "buserr-type", 0) == 1);
}
#endif /* SUN4C || SUN4M */
+
if (bug) {
kvm_uncache((caddr_t)trapbase, 1);
printf("%s: cache chip bug; trap page uncached\n",
dev->dv_xname);
}
- if (cacheinfo.c_totalsize) {
+ if (cacheinfo.c_totalsize == 0)
+ return;
+
+ if (!cacheinfo.c_physical) {
printf("%s: %d byte write-%s, %d bytes/line, %cw flush ",
dev->dv_xname, cacheinfo.c_totalsize,
(vactype == VAC_WRITETHROUGH) ? "through" : "back",
cacheinfo.c_linesize,
cacheinfo.c_hwflush ? 'h' : 's');
cache_enable();
+ } else {
+ sep = " ";
+ if (cacheinfo.c_split) {
+ printf("%s: physical", dev->dv_xname);
+ if (cacheinfo.ic_totalsize > 0) {
+ printf("%s%dK instruction (%d b/l)", sep,
+ cacheinfo.ic_totalsize/1024,
+ cacheinfo.ic_linesize);
+ sep = ", ";
+ }
+ if (cacheinfo.dc_totalsize > 0) {
+ printf("%s%dK data (%d b/l)", sep,
+ cacheinfo.dc_totalsize/1024,
+ cacheinfo.dc_linesize);
+ sep = ", ";
+ }
+ if (cacheinfo.ec_totalsize > 0) {
+ printf("%s%dK external (%d b/l)", sep,
+ cacheinfo.ec_totalsize/1024,
+ cacheinfo.ec_linesize);
+ }
+ printf(" ");
+ } else
+ printf("%s: physical %dK combined cache (%d bytes/"
+ "line) ", dev->dv_xname,
+ cacheinfo.c_totalsize/1024,
+ cacheinfo.c_linesize);
+ cache_enable();
}
}
@@ -260,8 +428,6 @@ cpu_attach(parent, dev, aux)
*
* The table contents (and much of the structure here) are from Guy Harris.
*
- * NOTE: we have Sun-4m cpu types here, even though this only runs on the
- * Sun-4c (yet)...
*/
struct info {
u_char valid;
@@ -273,14 +439,16 @@ struct info {
#define ANY 0xff /* match any FPU version (or, later, IU version) */
+#if defined(SUN4C) || defined(SUN4M)
static struct info iu_types[] = {
+ { 1, 0x0, 0x4, 4, "MB86904" },
{ 1, 0x0, 0x0, ANY, "MB86900/1A or L64801" },
{ 1, 0x1, 0x0, ANY, "RT601 or L64811 v1" },
{ 1, 0x1, 0x1, ANY, "RT601 or L64811 v2" },
{ 1, 0x1, 0x3, ANY, "RT611" },
{ 1, 0x1, 0xf, ANY, "RT620" },
{ 1, 0x2, 0x0, ANY, "B5010" },
- { 1, 0x4, 0x0, 0, "TMS390Z50 v0" },
+ { 1, 0x4, 0x0, 0, "TMS390Z50 v0 or TMS390Z55" },
{ 1, 0x4, 0x1, 0, "TMS390Z50 v1" },
{ 1, 0x4, 0x1, 4, "TMS390S10" },
{ 1, 0x5, 0x0, ANY, "MN10501" },
@@ -304,6 +472,7 @@ psrtoname(impl, vers, fver, buf)
sprintf(buf, "IU impl 0x%x vers 0x%x", impl, vers);
return (buf);
}
+#endif /* SUN4C || SUN4M */
/* NB: table order matters here; specific numbers must appear before ANY. */
static struct info fpu_types[] = {
@@ -314,6 +483,7 @@ static struct info fpu_types[] = {
{ 1, 0x0, ANY, 1, "MB86911 or WTL1164/5" },
{ 1, 0x0, ANY, 2, "L64802 or ACT8847" },
{ 1, 0x0, ANY, 3, "WTL3170/2" },
+ { 1, 0x0, 4, 4, "on-chip" }, /* Swift */
{ 1, 0x0, ANY, 4, "L64804" },
/*
@@ -358,8 +528,8 @@ fsrtoname(impl, vers, fver, buf)
for (p = fpu_types; p->valid; p++)
if (p->iu_impl == impl &&
- (p->iu_vers == vers || p->iu_vers == ANY) &
- p->fpu_vers == fver)
+ (p->iu_vers == vers || p->iu_vers == ANY) &&
+ (p->fpu_vers == fver))
return (p->name);
sprintf(buf, "version %x", fver);
return (buf);
diff --git a/sys/arch/sparc/sparc/db_disasm.c b/sys/arch/sparc/sparc/db_disasm.c
index a76ce891a18..901597194d3 100644
--- a/sys/arch/sparc/sparc/db_disasm.c
+++ b/sys/arch/sparc/sparc/db_disasm.c
@@ -1,4 +1,4 @@
-/* $NetBSD: db_disasm.c,v 1.6 1995/04/19 21:24:29 pk Exp $ */
+/* $NetBSD: db_disasm.c,v 1.9 1996/03/31 23:45:07 pk Exp $ */
/*
* Copyright (c) 1994 David S. Miller, davem@nadzieja.rutgers.edu
@@ -33,8 +33,12 @@
#include <sys/param.h>
#include <machine/db_machdep.h>
-#include <ddb/db_sym.h>
#include <machine/instr.h>
+#include <ddb/db_sym.h>
+#include <ddb/db_interface.h>
+#include <ddb/db_extern.h>
+#include <ddb/db_output.h>
+#include <ddb/db_access.h>
/*
* All Sparc instructions are 32-bits, with the one exception being
@@ -42,8 +46,8 @@
* two instructions...
*
* There are 5 different fields that can be used to identify which
- * operation is encoded into a particular 32-bit insn. There are 3
- * formats for instuctions, which one being used is determined by
+ * operation is encoded into a particular 32-bit insn. There are 3
+ * formats for instuctions, which one being used is determined by
* bits 30-31 of the insn. Here are the bit fields and their names:
*
* 1100 0000 0000 0000 0000 0000 0000 0000 op field, determines format
@@ -52,35 +56,34 @@
* 0000 0000 0000 0000 0010 0000 0000 0000 f3i bit, format 3 only
*/
-#define OP(x) ((x & 0x3) << 30)
-#define OP2(x) ((x & 0x7) << 22)
-#define OP3(x) ((x & 0x3f) << 19)
-#define OPF(x) ((x & 0x1ff) << 5)
-#define F3I(x) ((x & 0x1) << 13)
+#define OP(x) (((x) & 0x3) << 30)
+#define OP2(x) (((x) & 0x7) << 22)
+#define OP3(x) (((x) & 0x3f) << 19)
+#define OPF(x) (((x) & 0x1ff) << 5)
+#define F3I(x) (((x) & 0x1) << 13)
/* various other fields */
-#define A(x) ((x & 0x1) << 29)
-#define P(x) ((x & 0x1) << 19)
-#define X(x) ((x & 0x1) << 12)
-#define FCN(x) ((x & 0x1f) << 25)
-#define OPF(x) ((x & 0x1ff) << 5)
-#define RCOND2(x) ((x & 0x7) << 25)
-#define RCOND34(x) ((x & 0x7) << 10)
-#define COND(x) ((x & 0xf) << 25)
-#define SW_TRAP(x) (x & 0x7f)
-#define SHCNT32(x) (x & 0x1f)
-#define SHCNT64(x) (x & 0x3f)
-#define IMM11(x) (x & 0x7ff)
-#define IMM22(x) (x & 0x3fffff)
-#define DISP19(x) (x & 0x7ffff)
-#define DISP22(x) (x & 0x3fffff)
-#define DISP30(x) (x & 0x3fffffff)
+#define A(x) (((x) & 0x1) << 29)
+#define P(x) (((x) & 0x1) << 19)
+#define X(x) (((x) & 0x1) << 12)
+#define FCN(x) (((x) & 0x1f) << 25)
+#define RCOND2(x) (((x) & 0x7) << 25)
+#define RCOND34(x) (((x) & 0x7) << 10)
+#define COND(x) (((x) & 0xf) << 25)
+#define SW_TRAP(x) ((x) & 0x7f)
+#define SHCNT32(x) ((x) & 0x1f)
+#define SHCNT64(x) ((x) & 0x3f)
+#define IMM11(x) ((x) & 0x7ff)
+#define IMM22(x) ((x) & 0x3fffff)
+#define DISP19(x) ((x) & 0x7ffff)
+#define DISP22(x) ((x) & 0x3fffff)
+#define DISP30(x) ((x) & 0x3fffffff)
/* Register Operand Fields */
-#define RS1(x) ((x & 0x1f) << 14)
-#define RS2(x) (x & 0x1f)
-#define RD(x) ((x & 0x1f) << 25)
+#define RS1(x) (((x) & 0x1f) << 14)
+#define RS2(x) ((x) & 0x1f)
+#define RD(x) (((x) & 0x1f) << 25)
/* FORMAT macros used in sparc_i table to decode each opcode */
#define FORMAT1(a) (OP(a))
@@ -93,7 +96,7 @@
#define OPF_X(x,y) ((((x) & 0x1f) << 4) | ((y) & 0xf))
/* COND condition codes field... */
-#define COND2(x) ((x & 0xf) << 14)
+#define COND2(x) (((x) & 0xf) << 14)
struct sparc_insn {
unsigned long int match;
@@ -790,7 +793,7 @@ struct sparc_insn sparc_i[] = {
};
-vm_offset_t
+db_addr_t
db_disasm(loc, altfmt)
vm_offset_t loc;
boolean_t altfmt;
@@ -883,13 +886,13 @@ db_disasm(loc, altfmt)
db_printf("%%%s", regs[((insn >> 25) & 0x1f)]);
break;
case '3':
- db_printf("%%f%d", ((insn >> 14) & 0x1f));
+ db_printf("%%f%ld", ((insn >> 14) & 0x1f));
break;
case '4':
- db_printf("%%f%d", (insn & 0x1f));
+ db_printf("%%f%ld", (insn & 0x1f));
break;
case 'e':
- db_printf("%%f%d", ((insn >> 25) & 0x1f));
+ db_printf("%%f%ld", ((insn >> 25) & 0x1f));
break;
case 'i':
db_printf("0x%lx", (insn & 0x1fff));
@@ -923,7 +926,7 @@ db_disasm(loc, altfmt)
db_printf("0x%-2.2lx", ((insn >> 5) & 0xff));
break;
case 'o':
- db_printf("%%fcc%d", ((insn >> 25) & 0x3));
+ db_printf("%%fcc%ld", ((insn >> 25) & 0x3));
break;
case 'p':
case '7':
@@ -931,7 +934,7 @@ db_disasm(loc, altfmt)
regs[((insn >> 14) & 0x1f)],
regs[(insn & 0x1f)]);
if (*f_ptr == '7')
- db_printf(" %d", ((insn >> 5) & 0xff));
+ db_printf(" %ld", ((insn >> 5) & 0xff));
break;
case 'q':
case '8':
@@ -948,7 +951,7 @@ db_disasm(loc, altfmt)
db_printf("%%fsr");
break;
case '9':
- db_printf("0x%xl",
+ db_printf("0x%lxl",
((insn & 0xf) | ((insn >> 4) & 0x7)));
break;
case '0':
diff --git a/sys/arch/sparc/sparc/db_interface.c b/sys/arch/sparc/sparc/db_interface.c
index ebee65ddae2..4548b04f27e 100644
--- a/sys/arch/sparc/sparc/db_interface.c
+++ b/sys/arch/sparc/sparc/db_interface.c
@@ -1,27 +1,27 @@
-/* $NetBSD: db_interface.c,v 1.6 1995/04/09 11:51:10 pk Exp $ */
+/* $NetBSD: db_interface.c,v 1.12 1996/05/18 12:27:45 mrg Exp $ */
-/*
+/*
* Mach Operating System
* Copyright (c) 1991,1990 Carnegie Mellon University
* All Rights Reserved.
- *
+ *
* Permission to use, copy, modify and distribute this software and its
* documentation is hereby granted, provided that both the copyright
* notice and this permission notice appear in all copies of the
* software, derivative works or modified versions, and any portions
* thereof, and that both notices appear in supporting documentation.
- *
- * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS
+ *
+ * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS
* CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
* ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
- *
+ *
* Carnegie Mellon requests users of this software to return to
- *
+ *
* Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
* School of Computer Science
* Carnegie Mellon University
* Pittsburgh PA 15213-3890
- *
+ *
* any improvements or extensions that they make and grant Carnegie the
* rights to redistribute these changes.
*
@@ -34,14 +34,19 @@
#include <sys/param.h>
#include <sys/proc.h>
#include <sys/reboot.h>
-#include <sys/systm.h> /* just for boothowto --eichin */
-#include <setjmp.h>
+#include <sys/systm.h>
#include <vm/vm.h>
+#include <dev/cons.h>
+
#include <machine/db_machdep.h>
#include <ddb/db_command.h>
+#include <ddb/db_sym.h>
#include <ddb/db_variables.h>
+#include <ddb/db_extern.h>
+#include <ddb/db_access.h>
+#include <ddb/db_output.h>
#include <machine/bsd_openprom.h>
#include <machine/ctlreg.h>
#include <sparc/sparc/asm.h>
@@ -89,27 +94,33 @@ struct db_variable db_regs[] = {
};
struct db_variable *db_eregs = db_regs + sizeof(db_regs)/sizeof(db_regs[0]);
-extern jmp_buf *db_recover;
+extern label_t *db_recover;
int db_active = 0;
extern char *trap_type[];
+void kdb_kbd_trap __P((struct trapframe *));
+static void db_write_text __P((unsigned char *, int));
+void db_prom_cmd __P((db_expr_t, int, db_expr_t, char *));
+
/*
* Received keyboard interrupt sequence.
*/
+void
kdb_kbd_trap(tf)
struct trapframe *tf;
{
if (db_active == 0 && (boothowto & RB_KDB)) {
printf("\n\nkernel: keyboard interrupt\n");
- kdb_trap(-1, 0, tf);
+ kdb_trap(-1, tf);
}
}
/*
* kdb_trap - field a TRACE or BPT trap
*/
+int
kdb_trap(type, tf)
int type;
register struct trapframe *tf;
@@ -152,40 +163,88 @@ kdb_trap(type, tf)
void
db_read_bytes(addr, size, data)
vm_offset_t addr;
- register int size;
+ register size_t size;
register char *data;
{
register char *src;
src = (char *)addr;
- while (--size >= 0)
+ while (size-- > 0)
*data++ = *src++;
}
+/*
+ * XXX - stolen from pmap.c
+ */
+#if defined(SUN4M)
+#define getpte4m(va) lda((va & 0xFFFFF000) | ASI_SRMMUFP_L3, \
+ ASI_SRMMUFP)
+void setpte4m __P((vm_offset_t va, int pte));
+
+#endif
+#define getpte4(va) lda(va, ASI_PTE)
+#define setpte4(va, pte) sta(va, ASI_PTE, pte)
+#if defined(SUN4M) && !(defined(SUN4C) || defined(SUN4))
+#define getpte getpte4m
+#define setpte setpte4m
+#elif defined(SUN4M)
+#define getpte(va) (cputyp==CPU_SUN4M ? getpte4m(va) : getpte4(va))
+#define setpte(va, pte) (cputyp==CPU_SUN4M ? setpte4m(va, pte) \
+ : setpte4(va,pte))
+#else
+#define getpte getpte4
+#define setpte setpte4
+#endif
+
#define splpmap() splimp()
static void
db_write_text(dst, ch)
unsigned char *dst;
int ch;
-{
+{
+ int s, pte0, pte;
vm_offset_t va;
- int s;
s = splpmap();
va = (unsigned long)dst & (~PGOFSET);
+ pte0 = getpte(va);
+
+#if defined(SUN4M)
+#if defined(SUN4) || defined(SUN4C)
+ if (cputyp == CPU_SUN4M) {
+#endif
+ if ((pte0 & SRMMU_TETYPE) != SRMMU_TEPTE) {
+ db_printf(" address %p not a valid page\n", dst);
+ splx(s);
+ return;
+ }
- if (pmap_extract(pmap_kernel(), va) == 0) {
- db_printf(" address 0x%x not a valid page\n", dst);
+ pte = pte0 | PPROT_WRITE;
+ setpte(va, pte);
+
+#if defined(SUN4) || defined(SUN4C)
+ } else {
+#endif
+#endif /* 4m */
+#if defined(SUN4) || defined(SUN4C)
+ if ((pte0 & PG_V) == 0) {
+ db_printf(" address %p not a valid page\n", dst);
splx(s);
return;
}
- pmap_changeprot(pmap_kernel(), va, VM_PROT_ALL, 0);
+ pte = pte0 | PG_W;
+ setpte(va, pte);
+#if defined(SUN4M)
+ }
+#endif
+#endif /* 4/4c */
+
*dst = (unsigned char)ch;
- pmap_protect(pmap_kernel(), va, va, VM_PROT_READ|VM_PROT_EXECUTE);
+ setpte(va, pte0);
splx(s);
}
@@ -195,14 +254,14 @@ db_write_text(dst, ch)
void
db_write_bytes(addr, size, data)
vm_offset_t addr;
- register int size;
+ register size_t size;
register char *data;
{
extern char etext[];
register char *dst;
dst = (char *)addr;
- while (--size >= 0) {
+ while (size-- > 0) {
if ((dst >= (char *)VM_MIN_KERNEL_ADDRESS) && (dst < etext))
db_write_text(dst, *data);
else
@@ -212,16 +271,19 @@ db_write_bytes(addr, size, data)
}
-int
+void
Debugger()
{
asm("ta 0x81");
}
void
-db_prom_cmd()
+db_prom_cmd(addr, have_addr, count, modif)
+ db_expr_t addr;
+ int have_addr;
+ db_expr_t count;
+ char *modif;
{
- extern struct promvec *promvec;
promvec->pv_abort();
}
diff --git a/sys/arch/sparc/sparc/db_trace.c b/sys/arch/sparc/sparc/db_trace.c
index b97cc3d7cb6..8da02ce3082 100644
--- a/sys/arch/sparc/sparc/db_trace.c
+++ b/sys/arch/sparc/sparc/db_trace.c
@@ -1,38 +1,40 @@
-/* $NetBSD: db_trace.c,v 1.4 1995/07/05 18:51:41 pk Exp $ */
+/* $NetBSD: db_trace.c,v 1.8 1996/04/04 23:25:35 pk Exp $ */
-/*
+/*
* Mach Operating System
* Copyright (c) 1991,1990 Carnegie Mellon University
* All Rights Reserved.
- *
+ *
* Permission to use, copy, modify and distribute this software and its
* documentation is hereby granted, provided that both the copyright
* notice and this permission notice appear in all copies of the
* software, derivative works or modified versions, and any portions
* thereof, and that both notices appear in supporting documentation.
- *
- * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS
+ *
+ * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS
* CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
* ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
- *
+ *
* Carnegie Mellon requests users of this software to return to
- *
+ *
* Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
* School of Computer Science
* Carnegie Mellon University
* Pittsburgh PA 15213-3890
- *
+ *
* any improvements or extensions that they make and grant Carnegie the
* rights to redistribute these changes.
*/
-#include "param.h"
-#include "proc.h"
+#include <sys/param.h>
+#include <sys/proc.h>
#include <machine/db_machdep.h>
#include <ddb/db_access.h>
#include <ddb/db_sym.h>
-
+#include <ddb/db_interface.h>
+#include <ddb/db_output.h>
+
#define INKERNEL(va) (((vm_offset_t)(va)) >= USRSTACK)
void
@@ -63,7 +65,6 @@ db_stack_trace_cmd(addr, have_addr, count, modif)
while (count--) {
int i;
db_expr_t offset;
- db_sym_t sym;
char *name;
db_addr_t pc;
@@ -71,6 +72,13 @@ db_stack_trace_cmd(addr, have_addr, count, modif)
if (!INKERNEL(pc))
break;
+ /*
+ * Switch to frame that contains arguments
+ */
+ frame = frame->fr_fp;
+ if (!INKERNEL(frame))
+ break;
+
db_find_sym_and_offset(pc, &name, &offset);
if (name == NULL)
name = "?";
@@ -78,11 +86,6 @@ db_stack_trace_cmd(addr, have_addr, count, modif)
db_printf("%s(", name);
/*
- * Switch to frame that contains arguments
- */
- frame = frame->fr_fp;
-
- /*
* Print %i0..%i5, hope these still reflect the
* actual arguments somewhat...
*/
diff --git a/sys/arch/sparc/sparc/disksubr.c b/sys/arch/sparc/sparc/disksubr.c
index 95b97464ec8..4446fd1927f 100644
--- a/sys/arch/sparc/sparc/disksubr.c
+++ b/sys/arch/sparc/sparc/disksubr.c
@@ -1,4 +1,4 @@
-/* $NetBSD: disksubr.c,v 1.11 1996/01/07 22:03:12 thorpej Exp $ */
+/* $NetBSD: disksubr.c,v 1.16 1996/04/28 20:25:59 thorpej Exp $ */
/*
* Copyright (c) 1994, 1995 Gordon W. Ross
@@ -32,6 +32,7 @@
*/
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/buf.h>
#include <sys/ioccom.h>
#include <sys/device.h>
@@ -58,7 +59,7 @@ static int disklabel_bsd_to_sun __P((struct disklabel *, char *));
extern struct device *bootdv;
-/*
+/*
* find the boot device (if it was a disk). we must check to see if
* unit info in saved bootpath structure matches unit info in our softc.
* note that knowing the device name (e.g. "xd0") is not useful... we
@@ -66,68 +67,56 @@ extern struct device *bootdv;
* (XXX is it worth ifdef'ing this?)
*/
-int
+void
dk_establish(dk, dev)
struct disk *dk;
struct device *dev;
{
struct bootpath *bp = bootpath_store(0, NULL); /* restore bootpath! */
- char name[10];
struct scsibus_softc *sbsc;
- int targ, lun;
+ int target, lun;
if (bp == NULL)
- return -1;
+ return;
/*
* scsi: sd,cd
*/
- if (strncmp("sd", dev->dv_xname, 2) == 0 ||
+ if (strncmp("sd", dev->dv_xname, 2) == 0 ||
strncmp("cd", dev->dv_xname, 2) == 0) {
+
sbsc = (struct scsibus_softc *)dev->dv_parent;
- targ = lun = 0;
-#if defined(SUN4)
- if (cputyp == CPU_SUN4) {
- if (dev->dv_xname[0] == 's' && bp->val[0] == 0) {
- /* disk unit 0 is magic */
- if (sbsc->sc_link[0][0] == NULL) {
- targ = 3; /* remap to 3 */
- lun = 0;
- }
- } else {
- extern struct om_vector *oldpvec;
- if (oldpvec->monId[0] > '1') {
- targ = bp->val[0] >> 3; /* new format */
- lun = bp->val[0] & 0x7;
- } else {
- targ = bp->val[0] >> 2; /* old format */
- lun = bp->val[0] & 0x3;
- }
- }
- } else
-#endif
- {
- lun = bp->val[1];
- targ = (dev->dv_xname[0] == 's')
- ? sd_crazymap(bp->val[0])
- : bp->val[0];
+ target = bp->val[0];
+ lun = bp->val[1];
+
+ if (CPU_ISSUN4 && dev->dv_xname[0] == 's' &&
+ target == 0 && sbsc->sc_link[0][0] == NULL) {
+ /*
+ * disk unit 0 is magic: if there is actually no
+ * target 0 scsi device, the PROM will call
+ * target 3 `sd0'.
+ * XXX - what if someone puts a tape at target 0?
+ */
+ target = 3; /* remap to 3 */
+ lun = 0;
}
- if (sbsc->sc_link[targ][lun] != NULL &&
- sbsc->sc_link[targ][lun]->device_softc == (void *)dev) {
- bootdv = dev; /* got it! */
- return 1;
+ if (CPU_ISSUN4C && dev->dv_xname[0] == 's')
+ target = sd_crazymap(target);
+
+ if (sbsc->sc_link[target][lun] != NULL &&
+ sbsc->sc_link[target][lun]->device_softc == (void *)dev) {
+ bp->dev = dev; /* got it! */
+ return;
}
}
-
- return -1;
}
-/*
+/*
* Attempt to read a disk label from a device
* using the indicated stategy routine.
- * The label must be partly set up before this:
+ * The label must be partly set up before this:
* secpercyl, secsize and anything required for a block i/o read
* operation in the driver's strategy/start routines
* must be filled in before calling us.
@@ -138,8 +127,8 @@ dk_establish(dk, dev)
*/
char *
readdisklabel(dev, strat, lp, clp)
- dev_t dev;
- void (*strat)();
+ dev_t dev;
+ void (*strat) __P((struct buf *));
struct disklabel *lp;
struct cpu_disklabel *clp;
{
@@ -165,7 +154,7 @@ readdisklabel(dev, strat, lp, clp)
bp->b_cylin = 0;
bp->b_bcount = lp->d_secsize;
bp->b_flags = B_BUSY | B_READ;
- (*strat)(bp);
+ (*strat)(bp);
/* if successful, locate disk label within block and validate */
error = biowait(bp);
@@ -215,13 +204,13 @@ setdisklabel(olp, nlp, openmask, clp)
return(EINVAL);
/* special case to allow disklabel to be invalidated */
- if (nlp->d_magic == 0xffffffff) {
+ if (nlp->d_magic == 0xffffffff) {
*olp = *nlp;
return (0);
}
if (nlp->d_magic != DISKMAGIC || nlp->d_magic2 != DISKMAGIC ||
- dkcksum(nlp) != 0)
+ dkcksum(nlp) != 0)
return (EINVAL);
while ((i = ffs((long)openmask)) != 0) {
@@ -233,19 +222,20 @@ setdisklabel(olp, nlp, openmask, clp)
npp = &nlp->d_partitions[i];
if (npp->p_offset != opp->p_offset || npp->p_size < opp->p_size)
return (EBUSY);
- }
+ }
- *olp = *nlp;
- return (0);
+ *olp = *nlp;
+ return (0);
}
/*
* Write disk label back to device after modification.
* Current label is already in clp->cd_block[]
*/
+int
writedisklabel(dev, strat, lp, clp)
dev_t dev;
- void (*strat)();
+ void (*strat) __P((struct buf *));
register struct disklabel *lp;
struct cpu_disklabel *clp;
{
@@ -281,7 +271,7 @@ writedisklabel(dev, strat, lp, clp)
return (error);
}
-/*
+/*
* Determine the size of the transfer, and make sure it is
* within the boundaries of the partition. Adjust transfer
* if needed, and signal errors or early completion.
@@ -300,14 +290,14 @@ bounds_check_with_label(bp, lp, wlabel)
int sz = (bp->b_bcount + DEV_BSIZE - 1) >> DEV_BSHIFT;
/* overwriting disk label ? */
- /* XXX should also protect bootstrap in first 8K */
+ /* XXX should also protect bootstrap in first 8K */
if (bp->b_blkno + p->p_offset <= LABELSECTOR + labelsect &&
(bp->b_flags & B_READ) == 0 && wlabel == 0) {
bp->b_error = EROFS;
goto bad;
}
- /* beyond partition? */
+ /* beyond partition? */
if (bp->b_blkno < 0 || bp->b_blkno + sz > maxsz) {
/* if exactly at end of disk, return an EOF */
if (bp->b_blkno == maxsz) {
@@ -321,7 +311,7 @@ bounds_check_with_label(bp, lp, wlabel)
goto bad;
}
bp->b_bcount = sz << DEV_BSHIFT;
- }
+ }
/* calculate cylinder for disksort to order transfers with */
bp->b_resid = (bp->b_blkno + p->p_offset) / lp->d_secpercyl;
@@ -487,7 +477,7 @@ disklabel_bsd_to_sun(lp, cp)
/* move this to compat/sunos */
int
-sunos_dkioctl(dk, cmd, data, partition)
+sun_dkioctl(dk, cmd, data, partition)
struct disk *dk;
u_long cmd;
caddr_t data;
@@ -496,7 +486,7 @@ sunos_dkioctl(dk, cmd, data, partition)
register struct partition *p;
switch (cmd) {
- case SUNOS_DKIOCGGEOM:
+ case DKIOCGGEOM:
#define geom ((struct sun_dkgeom *)data)
bzero(data, sizeof(*geom));
geom->sdkc_ncylinders = dk->dk_label->d_ncylinders;
@@ -510,11 +500,11 @@ sunos_dkioctl(dk, cmd, data, partition)
dk->dk_label->d_ncylinders + dk->dk_label->d_acylinders;
#undef geom
break;
- case SUNOS_DKIOCINFO:
+ case DKIOCINFO:
/* Homey don't do DKIOCINFO */
bzero(data, sizeof(struct sun_dkctlr));
break;
- case SUNOS_DKIOCGPART:
+ case DKIOCGPART:
if (dk->dk_label->d_secpercyl == 0)
return (ERANGE); /* XXX */
p = &dk->dk_label->d_partitions[partition];
@@ -539,6 +529,7 @@ sunos_dkioctl(dk, cmd, data, partition)
int
isbad(bt, cyl, trk, sec)
register struct dkbad *bt;
+ int cyl, trk, sec;
{
register int i;
register long blk, bblk;
diff --git a/sys/arch/sparc/sparc/genassym.c b/sys/arch/sparc/sparc/genassym.c
index f7bc26e3a9a..ac0b0b8e36b 100644
--- a/sys/arch/sparc/sparc/genassym.c
+++ b/sys/arch/sparc/sparc/genassym.c
@@ -1,4 +1,4 @@
-/* $NetBSD: genassym.c,v 1.14 1995/05/04 19:42:36 pk Exp $ */
+/* $NetBSD: genassym.c,v 1.17.4.1 1996/06/12 20:31:21 pk Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -78,6 +78,10 @@
#define off(what, str, mem) def(what, (int)offsetof(str, mem))
+void def __P((char *, int));
+void flush __P((void));
+int main __P((void));
+
void
def(what, where)
char *what;
@@ -100,23 +104,16 @@ flush()
}
}
+int
main()
{
/* general constants */
def("BSD", BSD);
- def("USPACE", USPACE);
def("SUN4_PGSHIFT", SUN4_PGSHIFT);
def("SUN4CM_PGSHIFT", SUN4CM_PGSHIFT);
- def("KERNBASE", KERNBASE);
def("USRSTACK", USRSTACK);
- def("CPU_SUN4", CPU_SUN4);
- def("CPU_SUN4C", CPU_SUN4C);
- def("CPU_SUN4M", CPU_SUN4M);
-
- def("SUN4_400", SUN4_400);
-
/* proc fields and values */
off("P_ADDR", struct proc, p_addr);
off("P_STAT", struct proc, p_stat);
@@ -140,6 +137,15 @@ main()
def("PG_PROTSHIFT", PG_PROTSHIFT);
def("PG_PROTUREAD", PG_PROTUREAD);
def("PG_PROTUWRITE", PG_PROTUWRITE);
+#if defined(SUN4M)
+ def("SRMMU_TETYPE", SRMMU_TETYPE);
+ def("SRMMU_TEPTE", SRMMU_TEPTE);
+ def("SRMMU_PROT_MASK", SRMMU_PROT_MASK);
+ def("PPROT_R_RW", PPROT_R_RW);
+ def("PPROT_RX_RX", PPROT_RX_RX);
+ def("PPROT_RWX_RWX", PPROT_RWX_RWX);
+ def("PPROT_WRITE", PPROT_WRITE);
+#endif
/* FPU state */
off("FS_REGS", struct fpstate, fs_regs);
@@ -218,5 +224,5 @@ main()
flush();
- exit(0);
+ return(0);
}
diff --git a/sys/arch/sparc/sparc/in_cksum.c b/sys/arch/sparc/sparc/in_cksum.c
index 9f2f34c122b..38554d39569 100644
--- a/sys/arch/sparc/sparc/in_cksum.c
+++ b/sys/arch/sparc/sparc/in_cksum.c
@@ -1,8 +1,6 @@
-/* $NetBSD: in_cksum.c,v 1.3 1995/04/26 13:30:03 pk Exp $ */
+/* $NetBSD: in_cksum.c,v 1.5 1996/03/31 23:45:24 pk Exp $ */
/*
- * Copyright (c) 1995 Zubin Dittia.
- * Copyright (c) 1995 Theo de Raadt.
* Copyright (c) 1995 Matthew Green.
* Copyright (c) 1994 Charles Hannum.
* Copyright (c) 1992, 1993
@@ -46,6 +44,7 @@
#include <sys/param.h>
#include <sys/mbuf.h>
+#include <netinet/in.h>
/*
* Checksum routine for Internet Protocol family headers.
@@ -57,81 +56,50 @@
*/
/*
- * The checksum computation code here is significantly faster than its
- * vanilla C counterpart (by significantly, I mean 2-3 times faster if
- * the data is in cache, and 1.5-2 times faster if the data is not in
- * cache).
- * We optimize on three fronts:
- * 1. By using the add-with-carry (addxcc) instruction, we can use
- * 32-bit operations instead of 16-bit operations.
- * 2. By unrolling the main loop to reduce branch overheads.
- * 3. By doing a sequence of load,load,add,add,load,load,add,add,
- * we can avoid the extra stall cycle which is incurred if the
- * instruction immediately following a load tries to use the
- * target register of the load.
- * Another possible optimization is to replace a pair of 32-bit loads
- * with a single 64-bit load (ldd) instruction, but I found that although
- * this improves performance somewhat on Sun4c machines, it actually
- * reduces performance considerably on Sun4m machines (because of their
- * superscaler architecture). So I chose to leave it out.
+ * This idea here is that we do as many 32 bit operations as possible
+ * for maximum efficiency. We also unroll all loops in to assembly.
+ * This gains about 20% extra efficiency over the non-pipelined method.
*
- * Zubin Dittia (zubin@dworkin.wustl.edu)
+ * XXX - this code really needs further performance analysis. At the
+ * moment it has only been run on a SPARC ELC.
*/
-#define Asm __asm __volatile
-#define ADD64 Asm(" ld [%2],%3; ld [%2+4],%4; \
- addcc %0,%3,%0; addxcc %0,%4,%0; \
- ld [%2+8],%3; ld [%2+12],%4; \
- addxcc %0,%3,%0; addxcc %0,%4,%0; \
- ld [%2+16],%3; ld [%2+20],%4; \
- addxcc %0,%3,%0; addxcc %0,%4,%0; \
- ld [%2+24],%3; ld [%2+28],%4; \
- addxcc %0,%3,%0; addxcc %0,%4,%0; \
- ld [%2+32],%3; ld [%2+36],%4; \
- addxcc %0,%3,%0; addxcc %0,%4,%0; \
- ld [%2+40],%3; ld [%2+44],%4; \
- addxcc %0,%3,%0; addxcc %0,%4,%0; \
- ld [%2+48],%3; ld [%2+52],%4; \
- addxcc %0,%3,%0; addxcc %0,%4,%0; \
- ld [%2+56],%3; ld [%2+60],%4; \
- addxcc %0,%3,%0; addxcc %0,%4,%0; \
+#define Asm __asm __volatile
+#define ADD32 Asm(" ld [%2+28],%%i0; ld [%2+24],%%i1; \
+ ld [%2+20],%%i2; ld [%2+16],%%i3; \
+ ld [%2+12],%%i4; ld [%2+8],%%i5; \
+ ld [%2+4],%%g3; ld [%2],%%g4; \
+ addcc %0,%%i0,%0; addxcc %0,%%i1,%0; \
+ addxcc %0,%%i2,%0; addxcc %0,%%i3,%0; \
+ addxcc %0,%%i4,%0; addxcc %0,%%i5,%0; \
+ addxcc %0,%%g3,%0; addxcc %0,%%g4,%0; \
addxcc %0,0,%0" \
- : "=r" (sum) \
- : "0" (sum), "r" (w), "r" (tmp1), "r" (tmp2))
-#define ADD32 Asm(" ld [%2],%3; ld [%2+4],%4; \
- addcc %0,%3,%0; addxcc %0,%4,%0; \
- ld [%2+8],%3; ld [%2+12],%4; \
- addxcc %0,%3,%0; addxcc %0,%4,%0; \
- ld [%2+16],%3; ld [%2+20],%4; \
- addxcc %0,%3,%0; addxcc %0,%4,%0; \
- ld [%2+24],%3; ld [%2+28],%4; \
- addxcc %0,%3,%0; addxcc %0,%4,%0; \
+ : "=r" (sum) : "0" (sum), "r" (w) \
+ : "%i0", "%i1", "%i2", "%i3", \
+ "%i4", "%i5", "%g3", "%g4")
+#define ADD16 Asm(" ld [%2+12],%%i0; ld [%2+8],%%i1; \
+ ld [%2+4],%%i2; ld [%2],%%i3; \
+ addcc %0,%%i0,%0; addxcc %0,%%i1,%0; \
+ addxcc %0,%%i2,%0; addxcc %0,%%i3,%0; \
addxcc %0,0,%0" \
- : "=r" (sum) \
- : "0" (sum), "r" (w), "r" (tmp1), "r" (tmp2))
-#define ADD16 Asm(" ld [%2],%3; ld [%2+4],%4; \
- addcc %0,%3,%0; addxcc %0,%4,%0; \
- ld [%2+8],%3; ld [%2+12],%4; \
- addxcc %0,%3,%0; addxcc %0,%4,%0; \
+ : "=r" (sum) : "0" (sum), "r" (w) \
+ : "%i0", "%i1", "%i2", "%i3")
+#define ADD8 Asm(" ld [%2+4],%%i0; ld [%2],%%i1; \
+ addcc %0,%%i0,%0; addxcc %0,%%i1,%0; \
addxcc %0,0,%0" \
- : "=r" (sum) \
- : "0" (sum), "r" (w), "r" (tmp1), "r" (tmp2))
-#define ADD8 Asm(" ld [%2],%3; ld [%2+4],%4; \
- addcc %0,%3,%0; addxcc %0,%4,%0; \
+ : "=r" (sum) : "0" (sum), "r" (w) \
+ : "%i0", "%i1")
+#define ADD4 Asm(" ld [%2],%%i0; addcc %0,%%i0,%0; \
addxcc %0,0,%0" \
- : "=r" (sum) \
- : "0" (sum), "r" (w), "r" (tmp1), "r" (tmp2))
-#define ADD4 Asm(" ld [%2],%3; addcc %0,%3,%0; \
- addxcc %0,0,%0" \
- : "=r" (sum) \
- : "0" (sum), "r" (w), "r" (tmp1))
+ : "=r" (sum) : "0" (sum), "r" (w) \
+ : "%i0")
#define REDUCE {sum = (sum & 0xffff) + (sum >> 16);}
#define ADDCARRY {if (sum > 0xffff) sum -= 0xffff;}
#define ROL {sum = sum << 8;} /* depends on recent REDUCE */
-#define ADDBYTE {ROL; sum += *w; byte_swapped ^= 1;}
-#define ADDSHORT {sum += *(u_short *)w;}
-#define ADVANCE(n) {w += n; mlen -= n;}
+#define ADDB {ROL; sum += *w; byte_swapped ^= 1;}
+#define ADDS {sum += *(u_short *)w;}
+#define SHIFT(n) {w += n; mlen -= n;}
int
in_cksum(m, len)
@@ -143,13 +111,6 @@ in_cksum(m, len)
register int mlen = 0;
int byte_swapped = 0;
- /*
- * Declare two temporary registers for use by the asm code. We
- * allow the compiler to pick which specific machine registers to
- * use, instead of hard-coding this in the asm code above.
- */
- register u_int tmp1, tmp2;
-
for (; m && len; m = m->m_next) {
if (m->m_len == 0)
continue;
@@ -158,7 +119,7 @@ in_cksum(m, len)
if (len < mlen)
mlen = len;
len -= mlen;
-
+
/*
* Ensure that we're aligned on a word boundary here so
* that we can do 32 bit operations below.
@@ -166,50 +127,45 @@ in_cksum(m, len)
if ((3 & (long)w) != 0) {
REDUCE;
if ((1 & (long)w) != 0 && mlen >= 1) {
- ADDBYTE;
- ADVANCE(1);
+ ADDB;
+ SHIFT(1);
}
if ((2 & (long)w) != 0 && mlen >= 2) {
- ADDSHORT;
- ADVANCE(2);
+ ADDS;
+ SHIFT(2);
}
}
-
/*
* Do as many 32 bit operattions as possible using the
- * 64/32/16/8/4 macro's above, using as many as possible of
+ * 32/16/8/4 macro's above, using as many as possible of
* these.
*/
- while (mlen >= 64) {
- ADD64;
- ADVANCE(64);
- }
- if (mlen >= 32) {
+ while (mlen >= 32) {
ADD32;
- ADVANCE(32);
+ SHIFT(32);
}
if (mlen >= 16) {
ADD16;
- ADVANCE(16);
+ SHIFT(16);
}
if (mlen >= 8) {
ADD8;
- ADVANCE(8);
+ SHIFT(8);
}
if (mlen >= 4) {
ADD4;
- ADVANCE(4)
+ SHIFT(4)
}
if (mlen == 0)
continue;
REDUCE;
if (mlen >= 2) {
- ADDSHORT;
- ADVANCE(2);
+ ADDS;
+ SHIFT(2);
}
if (mlen == 1) {
- ADDBYTE;
+ ADDB;
}
}
if (byte_swapped) {
@@ -218,6 +174,5 @@ in_cksum(m, len)
}
REDUCE;
ADDCARRY;
-
return (0xffff ^ sum);
}
diff --git a/sys/arch/sparc/sparc/intersil7170.h b/sys/arch/sparc/sparc/intersil7170.h
index 2ca4e4035ab..b91f526e62e 100644
--- a/sys/arch/sparc/sparc/intersil7170.h
+++ b/sys/arch/sparc/sparc/intersil7170.h
@@ -1,4 +1,4 @@
-/* $NetBSD: intersil7170.h,v 1.1 1994/12/16 22:17:00 deraadt Exp $ */
+/* $NetBSD: intersil7170.h,v 1.3 1996/05/02 18:17:40 pk Exp $ */
/*
* Copyright (c) 1993 Adam Glass
@@ -34,57 +34,57 @@
/*
* Driver support for the intersil7170 used in sun[34]s to provide
* real time clock and time-of-day support.
- *
+ *
* Derived from: datasheet "ICM7170 a uP-Compatible Real-Time Clock"
* document #301680-005, Dec 85
*/
struct date_time { /* from p. 7 of 10 */
- u_char dt_csec;
- u_char dt_hour;
- u_char dt_min;
- u_char dt_sec;
- u_char dt_month;
- u_char dt_day;
- u_char dt_year;
- u_char dt_dow;
+ volatile unsigned char dt_csec;
+ volatile unsigned char dt_hour;
+ volatile unsigned char dt_min;
+ volatile unsigned char dt_sec;
+ volatile unsigned char dt_month;
+ volatile unsigned char dt_day;
+ volatile unsigned char dt_year;
+ volatile unsigned char dt_dow;
};
struct intersil7170 {
struct date_time counters;
- struct date_time clk_ram; /* should be ok as both are word aligned */
- u_char clk_intr_reg;
- u_char clk_cmd_reg;
+ struct date_time clk_ram; /* should be ok as both are word aligned */
+ volatile unsigned char clk_intr_reg;
+ volatile unsigned char clk_cmd_reg;
};
/* bit assignments for command register, p. 6 of 10, write-only */
-#define INTERSIL_CMD_FREQ_32K 0x00
-#define INTERSIL_CMD_FREQ_1M 0x01
-#define INTERSIL_CMD_FREQ_2M 0x02
-#define INTERSIL_CMD_FREQ_4M 0x03
+#define INTERSIL_CMD_FREQ_32K 0x0
+#define INTERSIL_CMD_FREQ_1M 0x1
+#define INTERSIL_CMD_FREQ_2M 0x2
+#define INTERSIL_CMD_FREQ_4M 0x3
-#define INTERSIL_CMD_12HR_MODE 0x00
-#define INTERSIL_CMD_24HR_MODE 0x04
+#define INTERSIL_CMD_12HR_MODE 0x0
+#define INTERSIL_CMD_24HR_MODE 0x4
-#define INTERSIL_CMD_STOP 0x00
-#define INTERSIL_CMD_RUN 0x08
+#define INTERSIL_CMD_STOP 0x0
+#define INTERSIL_CMD_RUN 0x8
-#define INTERSIL_CMD_IDISABLE 0x00
-#define INTERSIL_CMD_IENABLE 0x10
+#define INTERSIL_CMD_IDISABLE 0x0
+#define INTERSIL_CMD_IENABLE 0x10
-#define INTERSIL_CMD_TEST_MODE 0x20
-#define INTERSIL_CMD_NORMAL_MODE 0x00
+#define INTERSIL_CMD_TEST_MODE 0x20
+#define INTERSIL_CMD_NORMAL_MODE 0x0
/* bit assignments for interrupt register r/w, p 7 of 10*/
-#define INTERSIL_INTER_ALARM 0x01 /* r/w */
-#define INTERSIL_INTER_CSECONDS 0x02 /* r/w */
-#define INTERSIL_INTER_DSECONDS 0x04 /* r/w */
-#define INTERSIL_INTER_SECONDS 0x08 /* r/w */
-#define INTERSIL_INTER_MINUTES 0x10 /* r/w */
-#define INTERSIL_INTER_HOURS 0x20 /* r/w */
-#define INTERSIL_INTER_DAYS 0x40 /* r/w */
-#define INTERSIL_INTER_PENDING 0x80 /* read-only */
+#define INTERSIL_INTER_ALARM 0x1 /* r/w */
+#define INTERSIL_INTER_CSECONDS 0x2 /* r/w */
+#define INTERSIL_INTER_DSECONDS 0x4 /* r/w */
+#define INTERSIL_INTER_SECONDS 0x8 /* r/w */
+#define INTERSIL_INTER_MINUTES 0x10 /* r/w */
+#define INTERSIL_INTER_HOURS 0x20 /* r/w */
+#define INTERSIL_INTER_DAYS 0x40 /* r/w */
+#define INTERSIL_INTER_PENDING 0x80 /* read-only */
#define INTERSIL_INTER_BITS "\20\10PENDING\7DAYS\6HRS\5MIN\4SCDS\3DSEC\2CSEC\1ALARM"
diff --git a/sys/arch/sparc/sparc/intr.c b/sys/arch/sparc/sparc/intr.c
index 8d360371173..5173b00616f 100644
--- a/sys/arch/sparc/sparc/intr.c
+++ b/sys/arch/sparc/sparc/intr.c
@@ -1,4 +1,4 @@
-/* $NetBSD: intr.c,v 1.9 1995/07/04 12:34:37 paulus Exp $ */
+/* $NetBSD: intr.c,v 1.13 1996/03/31 23:35:20 pk Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -44,18 +44,45 @@
* @(#)intr.c 8.3 (Berkeley) 11/11/93
*/
+#include "ppp.h"
+
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/kernel.h>
+#include <sys/socket.h>
#include <vm/vm.h>
+#include <dev/cons.h>
+
#include <net/netisr.h>
+#include <net/if.h>
#include <machine/cpu.h>
#include <machine/ctlreg.h>
#include <machine/instr.h>
#include <machine/trap.h>
+#ifdef INET
+#include <netinet/in.h>
+#include <netinet/if_ether.h>
+#include <netinet/ip_var.h>
+#endif
+#ifdef NS
+#include <netns/ns_var.h>
+#endif
+#ifdef ISO
+#include <netiso/iso.h>
+#include <netiso/clnp.h>
+#endif
+#include "ppp.h"
+#if NPPP > 0
+#include <net/ppp_defs.h>
+#include <net/if_ppp.h>
+#endif
+
+void strayintr __P((struct clockframe *));
+int soft01intr __P((void *));
/*
* Stray interrupt handler. Clear it if possible.
@@ -69,7 +96,7 @@ strayintr(fp)
int timesince;
printf("stray interrupt ipl %x pc=%x npc=%x psr=%b\n",
- fp->ipl, fp->pc, fp->npc, fp->psr, PSR_BITS);
+ fp->ipl, fp->pc, fp->npc, fp->psr, PSR_BITS);
timesince = time.tv_sec - straytime;
if (timesince <= 10) {
if (++nstray > 9)
@@ -80,10 +107,7 @@ strayintr(fp)
}
}
-extern int clockintr(); /* level 10 (clock) interrupt code */
static struct intrhand level10 = { clockintr };
-
-extern int statintr(); /* level 14 (statclock) interrupt code */
static struct intrhand level14 = { statintr };
/*
@@ -129,7 +153,6 @@ soft01intr(fp)
if (n & (1 << NETISR_ISO))
clnlintr();
#endif
-#include "ppp.h"
#if NPPP > 0
if (n & (1 << NETISR_PPP))
pppintr();
@@ -137,7 +160,7 @@ soft01intr(fp)
}
if (sir.sir_which[SIR_CLOCK]) {
sir.sir_which[SIR_CLOCK] = 0;
- softclock(fp);
+ softclock();
}
}
return (1);
@@ -153,11 +176,11 @@ static struct intrhand level01 = { soft01intr };
struct intrhand *intrhand[15] = {
NULL, /* 0 = error */
&level01, /* 1 = software level 1 + Sbus */
- NULL, /* 2 = Sbus level 2 */
- NULL, /* 3 = SCSI + DMA + Sbus level 3 */
- NULL, /* 4 = software level 4 (tty softint) */
- NULL, /* 5 = Ethernet + Sbus level 4 */
- NULL, /* 6 = software level 6 (not used) */
+ NULL, /* 2 = Sbus level 2 (4m: Sbus L1) */
+ NULL, /* 3 = SCSI + DMA + Sbus level 3 (4m: L2,lpt)*/
+ NULL, /* 4 = software level 4 (tty softint) (scsi) */
+ NULL, /* 5 = Ethernet + Sbus level 4 (4m: Sbus L3) */
+ NULL, /* 6 = software level 6 (not used) (4m: enet)*/
NULL, /* 7 = video + Sbus level 5 */
NULL, /* 8 = Sbus level 6 */
NULL, /* 9 = Sbus level 7 */
@@ -170,7 +193,8 @@ struct intrhand *intrhand[15] = {
static int fastvec; /* marks fast vectors (see below) */
#ifdef DIAGNOSTIC
-extern int sparc_interrupt[];
+extern int sparc_interrupt4m[];
+extern int sparc_interrupt44c[];
#endif
/*
@@ -195,14 +219,17 @@ intr_establish(level, ih)
level);
#ifdef DIAGNOSTIC
/* double check for legal hardware interrupt */
- if (level != 1 && level != 4 && level != 6) {
+ if ((level != 1 && level != 4 && level != 6) || CPU_ISSUN4M ) {
tv = &trapbase[T_L1INT - 1 + level];
- displ = &sparc_interrupt[0] - &tv->tv_instr[1];
+ displ = (CPU_ISSUN4M)
+ ? &sparc_interrupt4m[0] - &tv->tv_instr[1]
+ : &sparc_interrupt44c[0] - &tv->tv_instr[1];
+
/* has to be `mov level,%l3; ba _sparc_interrupt; rdpsr %l0' */
if (tv->tv_instr[0] != I_MOVi(I_L3, level) ||
tv->tv_instr[1] != I_BA(0, displ) ||
tv->tv_instr[2] != I_RDPSR(I_L0))
- panic("intr_establish(%d, %x)\n%x %x %x != %x %x %x",
+ panic("intr_establish(%d, %p)\n%x %x %x != %x %x %x",
level, ih,
tv->tv_instr[0], tv->tv_instr[1], tv->tv_instr[2],
I_MOVi(I_L3, level), I_BA(0, displ), I_RDPSR(I_L0));
@@ -243,12 +270,15 @@ intr_fasttrap(level, vec)
panic("intr_fasttrap: already handling level %d interrupts",
level);
#ifdef DIAGNOSTIC
- displ = &sparc_interrupt[0] - &tv->tv_instr[1];
+ displ = (CPU_ISSUN4M)
+ ? &sparc_interrupt4m[0] - &tv->tv_instr[1]
+ : &sparc_interrupt44c[0] - &tv->tv_instr[1];
+
/* has to be `mov level,%l3; ba _sparc_interrupt; rdpsr %l0' */
if (tv->tv_instr[0] != I_MOVi(I_L3, level) ||
tv->tv_instr[1] != I_BA(0, displ) ||
tv->tv_instr[2] != I_RDPSR(I_L0))
- panic("intr_fasttrap(%d, %x)\n%x %x %x != %x %x %x",
+ panic("intr_fasttrap(%d, %p)\n%x %x %x != %x %x %x",
level, vec,
tv->tv_instr[0], tv->tv_instr[1], tv->tv_instr[2],
I_MOVi(I_L3, level), I_BA(0, displ), I_RDPSR(I_L0));
diff --git a/sys/arch/sparc/sparc/intreg.h b/sys/arch/sparc/sparc/intreg.h
index 899103353dd..5abbb8342f1 100644
--- a/sys/arch/sparc/sparc/intreg.h
+++ b/sys/arch/sparc/sparc/intreg.h
@@ -1,4 +1,4 @@
-/* $NetBSD: intreg.h,v 1.3 1995/06/25 21:34:28 pk Exp $ */
+/* $NetBSD: intreg.h,v 1.5 1996/03/31 23:03:39 pk Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -44,6 +44,8 @@
* @(#)intreg.h 8.1 (Berkeley) 6/11/93
*/
+#include <sparc/sparc/vaddrs.h>
+
/*
* sun4c interrupt enable register.
*
@@ -69,13 +71,22 @@
#define IE_L1 0x02 /* request software level 1 interrupt */
#define IE_ALLIE 0x01 /* enable interrupts */
-#ifndef LOCORE
+#ifndef _LOCORE
void ienab_bis __P((int bis)); /* set given bits */
void ienab_bic __P((int bic)); /* clear given bits */
#endif
#if defined(SUN4M)
-#define ICR_REG_PHYSADR 0x71e00000 /* XXX - phys addr in IOspace */
+#ifdef notyet
+#define IENAB_SYS ((_MAXNBPG * _MAXNCPU) + 0xc)
+#define IENAB_P0 0x0008
+#define IENAB_P1 0x1008
+#define IENAB_P2 0x2008
+#define IENAB_P3 0x3008
+#endif /* notyet */
+#endif
+
+#if defined(SUN4M)
/*
* Interrupt Control Registers, located in IO space.
* (mapped to `locore' for now..)
@@ -83,21 +94,27 @@ void ienab_bic __P((int bic)); /* clear given bits */
* and `System Interrupts'. The `Processor' set corresponds to the 15
* interrupt levels as seen by the CPU. The `System' set corresponds to
* a set of devices supported by the implementing chip-set.
+ *
+ * Briefly, the ICR_PI_* are per-processor interrupts; the ICR_SI_* are
+ * system-wide interrupts, and the ICR_ITR selects the processor to get
+ * the system's interrupts.
*/
-#define ICR_PI_PEND (IE_reg_addr + 0x0)
-#define ICR_PI_CLR (IE_reg_addr + 0x4)
-#define ICR_PI_SET (IE_reg_addr + 0x8)
-#define ICR_SI_PEND (IE_reg_addr + 0x1000)
-#define ICR_SI_MASK (IE_reg_addr + 0x1004)
-#define ICR_SI_CLR (IE_reg_addr + 0x1008)
-#define ICR_SI_SET (IE_reg_addr + 0x100c)
+#define ICR_PI_PEND (PI_INTR_VA + 0x0)
+#define ICR_PI_CLR (PI_INTR_VA + 0x4)
+#define ICR_PI_SET (PI_INTR_VA + 0x8)
+#define ICR_SI_PEND (SI_INTR_VA)
+#define ICR_SI_MASK (SI_INTR_VA + 0x4)
+#define ICR_SI_CLR (SI_INTR_VA + 0x8)
+#define ICR_SI_SET (SI_INTR_VA + 0xc)
+#define ICR_ITR (SI_INTR_VA + 0x10)
+
/*
* Bits in interrupt registers. Software interrupt requests must
* be cleared in software. This is done in locore.s.
* There are separate registers for reading pending interrupts and
* setting/clearing (software) interrupts.
*/
-#define PINTR_SOFTINTR(n) ((n)) << 16)
+#define PINTR_SINTRLEV(n) (1 << (16 + (n)))
#define PINTR_IC 0x8000 /* Level 15 clear */
#define SINTR_MA 0x80000000 /* Mask All interrupts */
diff --git a/sys/arch/sparc/sparc/iommu.c b/sys/arch/sparc/sparc/iommu.c
new file mode 100644
index 00000000000..7a0457f6b65
--- /dev/null
+++ b/sys/arch/sparc/sparc/iommu.c
@@ -0,0 +1,352 @@
+/* $NetBSD: iommu.c,v 1.4 1996/05/21 07:25:07 pk Exp $ */
+
+/*
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
+ * Copyright (c) 1995 Paul Kranenburg
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by Aaron Brown and
+ * Harvard University.
+ * This product includes software developed by Paul Kranenburg.
+ * 4. Neither the name of the University nor the names of its contributors
+ * may be used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ *
+ */
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/device.h>
+#include <vm/vm.h>
+
+#include <machine/autoconf.h>
+#include <machine/ctlreg.h>
+#include <sparc/sparc/asm.h>
+#include <sparc/sparc/vaddrs.h>
+#include <sparc/sparc/iommureg.h>
+
+struct iommu_softc {
+ struct device sc_dev; /* base device */
+ struct iommureg *sc_reg;
+ u_int sc_pagesize;
+ u_int sc_range;
+ u_int sc_dvmabase;
+ iopte_t *sc_ptes;
+ int sc_hasiocache;
+};
+struct iommu_softc *iommu_sc;/*XXX*/
+int has_iocache;
+
+
+/* autoconfiguration driver */
+int iommu_print __P((void *, char *));
+void iommu_attach __P((struct device *, struct device *, void *));
+int iommu_match __P((struct device *, void *, void *));
+
+struct cfattach iommu_ca = {
+ sizeof(struct iommu_softc), iommu_match, iommu_attach
+};
+
+struct cfdriver iommu_cd = {
+ NULL, "iommu", DV_DULL
+};
+
+/*
+ * Print the location of some iommu-attached device (called just
+ * before attaching that device). If `iommu' is not NULL, the
+ * device was found but not configured; print the iommu as well.
+ * Return UNCONF (config_find ignores this if the device was configured).
+ */
+int
+iommu_print(args, iommu)
+ void *args;
+ char *iommu;
+{
+ register struct confargs *ca = args;
+
+ if (iommu)
+ printf("%s at %s", ca->ca_ra.ra_name, iommu);
+ return (UNCONF);
+}
+
+int
+iommu_match(parent, vcf, aux)
+ struct device *parent;
+ void *vcf, *aux;
+{
+ struct cfdata *cf = vcf;
+ register struct confargs *ca = aux;
+ register struct romaux *ra = &ca->ca_ra;
+
+ if (CPU_ISSUN4OR4C)
+ return (0);
+ return (strcmp(cf->cf_driver->cd_name, ra->ra_name) == 0);
+}
+
+/*
+ * Attach the iommu.
+ */
+void
+iommu_attach(parent, self, aux)
+ struct device *parent;
+ struct device *self;
+ void *aux;
+{
+#if defined(SUN4M)
+ register struct iommu_softc *sc = (struct iommu_softc *)self;
+ struct confargs oca, *ca = aux;
+ register struct romaux *ra = &ca->ca_ra;
+ register int node;
+ register char *name;
+ register u_int pbase, pa;
+ register int i, mmupcrsav, s, wierdviking = 0;
+ register iopte_t *tpte_p;
+ extern u_int *kernel_iopte_table;
+ extern u_int kernel_iopte_table_pa;
+
+/*XXX-GCC!*/mmupcrsav=0;
+ iommu_sc = sc;
+ /*
+ * XXX there is only one iommu, for now -- do not know how to
+ * address children on others
+ */
+ if (sc->sc_dev.dv_unit > 0) {
+ printf(" unsupported\n");
+ return;
+ }
+ node = ra->ra_node;
+
+#if 0
+ if (ra->ra_vaddr)
+ sc->sc_reg = (struct iommureg *)ca->ca_ra.ra_vaddr;
+#else
+ /*
+ * Map registers into our space. The PROM may have done this
+ * already, but I feel better if we have our own copy. Plus, the
+ * prom doesn't map the entire register set
+ *
+ * XXX struct iommureg is bigger than ra->ra_len; what are the
+ * other fields for?
+ */
+ sc->sc_reg = (struct iommureg *)
+ mapdev(ra->ra_reg, 0, 0, ra->ra_len, ra->ra_iospace);
+#endif
+
+ sc->sc_hasiocache = node_has_property(node, "cache-coherence?");
+ has_iocache = sc->sc_hasiocache; /* Set global flag */
+
+ sc->sc_pagesize = getpropint(node, "page-size", NBPG),
+ sc->sc_range = (1 << 24) <<
+ ((sc->sc_reg->io_cr & IOMMU_CTL_RANGE) >> IOMMU_CTL_RANGESHFT);
+#if 0
+ sc->sc_dvmabase = (0 - sc->sc_range);
+#endif
+ pbase = (sc->sc_reg->io_bar & IOMMU_BAR_IBA) <<
+ (14 - IOMMU_BAR_IBASHFT);
+
+ /*
+ * Now we build our own copy of the IOMMU page tables. We need to
+ * do this since we're going to change the range to give us 64M of
+ * mappings, and thus we can move DVMA space down to 0xfd000000 to
+ * give us lots of space and to avoid bumping into the PROM, etc.
+ *
+ * XXX Note that this is rather messy.
+ */
+ sc->sc_ptes = (iopte_t *) kernel_iopte_table;
+
+ /*
+ * Now discache the page tables so that the IOMMU sees our
+ * changes.
+ */
+ kvm_uncache((caddr_t)sc->sc_ptes,
+ (((0 - DVMA4M_BASE)/sc->sc_pagesize) * sizeof(iopte_t)) / NBPG);
+
+ /*
+ * Ok. We've got to read in the original table using MMU bypass,
+ * and copy all of its entries to the appropriate place in our
+ * new table, even if the sizes are different.
+ * This is pretty easy since we know DVMA ends at 0xffffffff.
+ *
+ * XXX: PGOFSET, NBPG assume same page size as SRMMU
+ */
+ if ((getpsr() & 0x40000000) && (!(lda(SRMMU_PCR,ASI_SRMMU) & 0x800))) {
+ wierdviking = 1;
+ sta(SRMMU_PCR, ASI_SRMMU, /* set MMU AC bit */
+ ((mmupcrsav = lda(SRMMU_PCR,ASI_SRMMU)) | SRMMU_PCR_AC));
+ }
+
+ for (tpte_p = &sc->sc_ptes[((0 - DVMA4M_BASE)/NBPG) - 1],
+ pa = (u_int)pbase - sizeof(iopte_t) +
+ ((u_int)sc->sc_range/NBPG)*sizeof(iopte_t);
+ tpte_p >= &sc->sc_ptes[0] && pa >= (u_int)pbase;
+ tpte_p--, pa -= sizeof(iopte_t)) {
+
+ IOMMU_FLUSHPAGE(sc,
+ (tpte_p - &sc->sc_ptes[0])*NBPG + DVMA4M_BASE);
+ *tpte_p = lda(pa, ASI_BYPASS);
+ }
+ if (wierdviking) { /* restore mmu after bug-avoidance */
+ sta(SRMMU_PCR, ASI_SRMMU, mmupcrsav);
+ }
+
+ /*
+ * Now we can install our new pagetable into the IOMMU
+ */
+ sc->sc_range = 0 - DVMA4M_BASE;
+ sc->sc_dvmabase = DVMA4M_BASE;
+
+ /* calculate log2(sc->sc_range/16MB) */
+ i = ffs(sc->sc_range/(1 << 24)) - 1;
+ if ((1 << i) != (sc->sc_range/(1 << 24)))
+ panic("bad iommu range: %d\n",i);
+
+ s = splhigh();
+ IOMMU_FLUSHALL(sc);
+
+ sc->sc_reg->io_cr = (sc->sc_reg->io_cr & ~IOMMU_CTL_RANGE) |
+ (i << IOMMU_CTL_RANGESHFT) | IOMMU_CTL_ME;
+ sc->sc_reg->io_bar = (kernel_iopte_table_pa >> 4) & IOMMU_BAR_IBA;
+
+ IOMMU_FLUSHALL(sc);
+ splx(s);
+
+ printf(": version %x/%x, page-size %d, range %dMB\n",
+ (sc->sc_reg->io_cr & IOMMU_CTL_VER) >> 24,
+ (sc->sc_reg->io_cr & IOMMU_CTL_IMPL) >> 28,
+ sc->sc_pagesize,
+ sc->sc_range >> 20);
+
+ /* Propagate bootpath */
+ if (ra->ra_bp != NULL && strcmp(ra->ra_bp->name, "iommu") == 0)
+ oca.ca_ra.ra_bp = ra->ra_bp + 1;
+ else
+ oca.ca_ra.ra_bp = NULL;
+
+ /*
+ * Loop through ROM children (expect Sbus among them).
+ */
+ for (node = firstchild(node); node; node = nextsibling(node)) {
+ name = getpropstring(node, "name");
+ if (!romprop(&oca.ca_ra, name, node))
+ continue;
+ oca.ca_bustype = BUS_MAIN; /* ??? */
+ (void) config_found(&sc->sc_dev, (void *)&oca, iommu_print);
+ }
+#endif
+}
+
+void
+iommu_enter(va, pa)
+ u_int va, pa;
+{
+ struct iommu_softc *sc = iommu_sc;
+ int pte;
+
+#ifdef DEBUG
+ if (va < sc->sc_dvmabase)
+ panic("iommu_enter: va 0x%x not in DVMA space",va);
+#endif
+
+ pte = atop(pa) << IOPTE_PPNSHFT;
+ pte &= IOPTE_PPN;
+ pte |= IOPTE_V | IOPTE_W | (has_iocache ? IOPTE_C : 0);
+ sc->sc_ptes[atop(va - sc->sc_dvmabase)] = pte;
+ IOMMU_FLUSHPAGE(sc, va);
+}
+
+/*
+ * iommu_clear: clears mappings created by iommu_enter
+ */
+void
+iommu_remove(va, len)
+ register u_int va, len;
+{
+ register struct iommu_softc *sc = iommu_sc;
+
+#ifdef DEBUG
+ if (va < sc->sc_dvmabase)
+ panic("iommu_enter: va 0x%x not in DVMA space", va);
+#endif
+
+ while (len > 0) {
+#ifdef notyet
+#ifdef DEBUG
+ if ((sc->sc_ptes[atop(va - sc->sc_dvmabase)] & IOPTE_V) == 0)
+ panic("iommu_clear: clearing invalid pte at va 0x%x",
+ va);
+#endif
+#endif
+ sc->sc_ptes[atop(va - sc->sc_dvmabase)] = 0;
+ sta(sc->sc_ptes + atop(va - sc->sc_dvmabase), ASI_BYPASS, 0);
+ IOMMU_FLUSHPAGE(sc, va);
+ len -= sc->sc_pagesize;
+ va += sc->sc_pagesize;
+ }
+}
+
+#if 0 /* These registers aren't there??? */
+void
+iommu_error()
+{
+ struct iommu_softc *sc = X;
+ struct iommureg *iop = sc->sc_reg;
+
+ printf("iommu: afsr %x, afar %x\n", iop->io_afsr, iop->io_afar);
+ printf("iommu: mfsr %x, mfar %x\n", iop->io_mfsr, iop->io_mfar);
+}
+int
+iommu_alloc(va, len)
+ u_int va, len;
+{
+ struct iommu_softc *sc = X;
+ int off, tva, pa, iovaddr, pte;
+
+ off = (int)va & PGOFSET;
+ len = round_page(len + off);
+ va -= off;
+
+if ((int)sc->sc_dvmacur + len > 0)
+ sc->sc_dvmacur = sc->sc_dvmabase;
+
+ iovaddr = tva = sc->sc_dvmacur;
+ sc->sc_dvmacur += len;
+ while (len) {
+ pa = pmap_extract(pmap_kernel(), va);
+
+#define IOMMU_PPNSHIFT 8
+#define IOMMU_V 0x00000002
+#define IOMMU_W 0x00000004
+
+ pte = atop(pa) << IOMMU_PPNSHIFT;
+ pte |= IOMMU_V | IOMMU_W;
+ sta(sc->sc_ptes + atop(tva - sc->sc_dvmabase), ASI_BYPASS, pte);
+ sc->sc_reg->io_flushpage = tva;
+ len -= NBPG;
+ va += NBPG;
+ tva += NBPG;
+ }
+ return iovaddr + off;
+}
+#endif
diff --git a/sys/arch/sparc/sparc/iommureg.h b/sys/arch/sparc/sparc/iommureg.h
new file mode 100644
index 00000000000..24c5ac9f8cf
--- /dev/null
+++ b/sys/arch/sparc/sparc/iommureg.h
@@ -0,0 +1,93 @@
+/* $NetBSD: iommureg.h,v 1.2 1996/05/16 15:57:18 abrown Exp $ */
+
+/*
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
+ * Copyright (c) 1995 Paul Kranenburg
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by Aaron Brown and
+ * Harvard University.
+ * This product includes software developed by Paul Kranenburg.
+ * 4. Neither the name of the University nor the names of its contributors
+ * may be used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ *
+ */
+
+struct iommureg {
+ u_int32_t io_cr; /* IOMMU control register */
+ u_int32_t io_bar; /* IOMMU PTE base register */
+ u_int32_t io_fill1[3];
+ u_int32_t io_flashclear; /* Flush all TLB entries */
+ u_int32_t io_flushpage; /* Flush page from TLB */
+ u_int32_t io_fill2[0x1000-28];
+ u_int32_t io_afsr; /* Asynchronous Fault Status */
+ u_int32_t io_afar; /* Asynchronous Fault Address */
+ u_int32_t io_sbuscfg0; /* Sbus configuration control */
+ u_int32_t io_sbuscfg1;
+ u_int32_t io_sbuscfg2;
+ u_int32_t io_sbuscfg3;
+ u_int32_t io_mfsr; /* Memory Fault Status */
+ u_int32_t io_mfar; /* Memory Fault Address */
+ u_int32_t io_fill3[0x1000-32];
+ u_int32_t io_mid; /* it might be mud */
+};
+
+#define IOMMU_CTL_IMPL 0xf0000000
+#define IOMMU_CTL_VER 0x0f000000
+#define IOMMU_CTL_RSVD1 0x00ffffe0
+#define IOMMU_CTL_RANGE 0x0000001c
+#define IOMMU_CTL_RANGESHFT 2
+#define IOMMU_CTL_RSVD2 0x00000002
+#define IOMMU_CTL_ME 0x00000001
+
+#define IOMMU_BAR_IBA 0xfffffc00
+#define IOMMU_BAR_IBASHFT 10
+
+/* Flushpage fields */
+#define IOMMU_FLPG_VADDR 0xfffff000
+#define IOMMU_FLUSH_MASK 0xfffff000
+
+#define IOMMU_FLUSHPAGE(sc, va) do { \
+ (sc)->sc_reg->io_flushpage = (va) & IOMMU_FLUSH_MASK; \
+} while (0);
+#define IOMMU_FLUSHALL(sc) do { \
+ (sc)->sc_reg->io_flashclear = 0; \
+} while (0)
+
+/* to pte.h ? */
+typedef u_int32_t iopte_t;
+
+#define IOPTE_PPN 0xffffff00 /* PA<35:12> */
+#define IOPTE_C 0x00000080 /* cacheable */
+#define IOPTE_W 0x00000004 /* writeable */
+#define IOPTE_V 0x00000002 /* valid */
+#define IOPTE_WAZ 0x00000001 /* must write as zero */
+
+#define IOPTE_PPNSHFT 8 /* shift to get ppn from IOPTE */
+#define IOPTE_PPNPASHFT 4 /* shift to get pa from ioppn */
+
+#define IOPTE_BITS "\20\10C\3W\2V\1WAZ"
+
diff --git a/sys/arch/sparc/sparc/kgdb_proto.h b/sys/arch/sparc/sparc/kgdb_proto.h
index edd86775d7f..8d1869a3a47 100644
--- a/sys/arch/sparc/sparc/kgdb_proto.h
+++ b/sys/arch/sparc/sparc/kgdb_proto.h
@@ -1,18 +1,22 @@
-/* $NetBSD: kgdb_proto.h,v 1.2 1994/11/20 20:54:21 deraadt Exp $ */
+/* $NetBSD: kgdb_proto.h,v 1.4 1996/05/16 15:57:19 abrown Exp $ */
/*
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
+ * Copyright (c) 1995
+ * The President and Fellows of Harvard College. All rights reserved.
*
* This software was developed by the Computer Systems Engineering group
* at Lawrence Berkeley Laboratory under DARPA contract BG 91-66 and
* contributed to Berkeley.
*
* All advertising materials mentioning features or use of this software
- * must display the following acknowledgement:
+ * must display the following acknowledgements:
* This product includes software developed by the University of
* California, Lawrence Berkeley Laboratory.
*
+ * This product includes software developed by Harvard University.
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
@@ -56,6 +60,8 @@
#define KGDB_KILL 0x07
#define KGDB_SIGNAL 0x08
#define KGDB_EXEC 0x09
+#define KGDB_HALT 0x0a
+#define KGDB_BOOT 0x0b
#define KGDB_CMD(x) ((x) & 0x0f)
diff --git a/sys/arch/sparc/sparc/kgdb_stub.c b/sys/arch/sparc/sparc/kgdb_stub.c
index 0739215c3a1..3ba0c5e9975 100644
--- a/sys/arch/sparc/sparc/kgdb_stub.c
+++ b/sys/arch/sparc/sparc/kgdb_stub.c
@@ -1,18 +1,22 @@
-/* $NetBSD: kgdb_stub.c,v 1.2 1994/11/20 20:54:23 deraadt Exp $ */
+/* $NetBSD: kgdb_stub.c,v 1.8 1996/05/16 15:57:20 abrown Exp $ */
/*
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
+ * Copyright (c) 1995
+ * The President and Fellows of Harvard College. All rights reserved.
*
* This software was developed by the Computer Systems Engineering group
* at Lawrence Berkeley Laboratory under DARPA contract BG 91-66 and
* contributed to Berkeley.
*
* All advertising materials mentioning features or use of this software
- * must display the following acknowledgement:
+ * must display the following acknowledgements:
* This product includes software developed by the University of
* California, Lawrence Berkeley Laboratory.
*
+ * This product includes software developed by Harvard University.
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
@@ -53,12 +57,16 @@
#include <sys/systm.h>
#include <sys/buf.h>
+#include <vm/vm.h>
+
#include <machine/ctlreg.h>
#include <machine/psl.h>
-#include <machine/pte.h>
#include <machine/reg.h>
#include <machine/remote-sl.h>
#include <machine/trap.h>
+#include <machine/cpu.h>
+#include <machine/autoconf.h>
+#include <machine/bsd_openprom.h>
#include <sparc/sparc/asm.h>
#include <sparc/sparc/kgdb_proto.h>
@@ -75,12 +83,30 @@ int kgdb_rate = KGDBRATE; /* remote debugging baud rate */
int kgdb_active = 0; /* remote debugging active if != 0 */
int kgdb_debug_panic = 0; /* != 0 waits for remote on panic */
-#define getpte(va) lda(va, ASI_PTE)
-#define setpte(va, pte) sta(va, ASI_PTE, pte)
+#if defined(SUN4M)
+#define getpte4m(va) lda(((vm_offset_t)va & 0xFFFFF000) | \
+ ASI_SRMMUFP_L3, ASI_SRMMUFP)
+void setpte4m __P((vm_offset_t, int));
+#endif
+
+#define getpte4(va) lda(va, ASI_PTE)
+#define setpte4(va, pte) sta(va, ASI_PTE, pte)
+
+void kgdb_copy __P((char *, char *, int));
+void kgdb_zero __P((char *, int));
+static void kgdb_send __P((u_int, u_char *, int));
+static int kgdb_recv __P((u_char *, int *));
+static int computeSignal __P((int));
+void regs_to_gdb __P((struct trapframe *, u_long *));
+void gdb_to_regs __P((struct trapframe *, u_long *));
+int kgdb_trap __P((int, struct trapframe *));
+int kgdb_acc __P((caddr_t, int, int, int));
+void kdb_mkwrite __P((caddr_t, int));
/*
* This little routine exists simply so that bcopy() can be debugged.
*/
+void
kgdb_copy(src, dst, len)
register char *src, *dst;
register int len;
@@ -90,6 +116,16 @@ kgdb_copy(src, dst, len)
*dst++ = *src++;
}
+/* ditto for bzero */
+void
+kgdb_zero(ptr, len)
+ register char *ptr;
+ register int len;
+{
+ while (--len >= 0)
+ *ptr++ = (char) 0;
+}
+
static int (*kgdb_getc) __P((void *));
static void (*kgdb_putc) __P((void *, int));
static void *kgdb_ioarg;
@@ -110,6 +146,7 @@ static void *kgdb_ioarg;
PUTC(c); \
}
+void
kgdb_attach(getfn, putfn, ioarg)
int (*getfn) __P((void *));
void (*putfn) __P((void *, int));
@@ -126,7 +163,7 @@ kgdb_attach(getfn, putfn, ioarg)
*/
static void
kgdb_send(type, bp, len)
- register u_char type;
+ register u_int type;
register u_char *bp;
register int len;
{
@@ -181,7 +218,7 @@ restart:
if (escape)
c = FRAME_START;
break;
-
+
case FRAME_START:
goto restart;
@@ -215,9 +252,9 @@ restart:
/*
* Translate a trap number into a unix compatible signal value.
* (gdb only understands unix signal numbers).
- * ### should this be done at the other end?
+ * XXX should this be done at the other end?
*/
-static int
+static int
computeSignal(type)
int type;
{
@@ -264,9 +301,10 @@ computeSignal(type)
}
/*
- * Trap into kgdb to wait for debugger to connect,
+ * Trap into kgdb to wait for debugger to connect,
* noting on the console why nothing else is going on.
*/
+void
kgdb_connect(verbose)
int verbose;
{
@@ -282,6 +320,7 @@ kgdb_connect(verbose)
/*
* Decide what to do on panic.
*/
+void
kgdb_panic()
{
@@ -330,7 +369,7 @@ regs_to_gdb(tf, gdb_regs)
kgdb_copy((caddr_t)tf->tf_out[6], (caddr_t)&gdb_regs[GDB_L0], 16 * 4);
/* %f0..%f31 -- fake, kernel does not use FP */
- bzero((caddr_t)&gdb_regs[GDB_FP0], 32 * 4);
+ kgdb_zero((caddr_t)&gdb_regs[GDB_FP0], 32 * 4);
/* %y, %psr, %wim, %tbr, %pc, %npc, %fsr, %csr */
gdb_regs[GDB_Y] = tf->tf_y;
@@ -353,7 +392,7 @@ gdb_to_regs(tf, gdb_regs)
{
kgdb_copy((caddr_t)&gdb_regs[1], (caddr_t)&tf->tf_global[1], 15 * 4);
- kgdb_copy((caddr_t)&gdb_regs[GDB_L0], (caddr_t)tf->tf_out[6]);
+ kgdb_copy((caddr_t)&gdb_regs[GDB_L0], (caddr_t)tf->tf_out[6], 16 * 4);
tf->tf_y = gdb_regs[GDB_Y];
tf->tf_psr = gdb_regs[GDB_PSR];
tf->tf_pc = gdb_regs[GDB_PC];
@@ -365,10 +404,10 @@ static u_char inbuffer[SL_RPCSIZE];
static u_char outbuffer[SL_RPCSIZE];
/*
- * This function does all command procesing for interfacing to
+ * This function does all command procesing for interfacing to
* a remote gdb.
*/
-int
+int
kgdb_trap(type, tf)
int type;
register struct trapframe *tf;
@@ -394,7 +433,7 @@ kgdb_trap(type, tf)
/*
* If the packet that woke us up isn't an exec packet,
* ignore it since there is no active debugger. Also,
- * we check that it's not an ack to be sure that the
+ * we check that it's not an ack to be sure that the
* remote side doesn't send back a response after the
* local gdb has exited. Otherwise, the local host
* could trap into gdb if it's running a gdb kernel too.
@@ -415,7 +454,7 @@ kgdb_trap(type, tf)
continue;
/*
* Do the printf *before* we ack the message. This way
- * we won't drop any inbound characters while we're
+ * we won't drop any inbound characters while we're
* doing the polling printf.
*/
printf("kgdb started from device %x\n", kgdb_dev);
@@ -487,7 +526,7 @@ kgdb_trap(type, tf)
}
}
break;
-
+
case KGDB_REG_W:
case KGDB_REG_W | KGDB_DELTA:
cp = inbuffer;
@@ -501,7 +540,7 @@ kgdb_trap(type, tf)
gdb_to_regs(tf, gdb_regs);
outlen = 0;
break;
-
+
case KGDB_MEM_R:
len = inbuffer[0];
kgdb_copy((caddr_t)&inbuffer[1], (caddr_t)&addr, 4);
@@ -541,6 +580,16 @@ kgdb_trap(type, tf)
kgdb_send(out, 0, 0);
return (1);
+ case KGDB_HALT:
+ kgdb_send(out, 0, 0);
+ callrom();
+ /* NOTREACHED */
+
+ case KGDB_BOOT:
+ kgdb_send(out, 0, 0);
+ romboot("");
+ /* NOTREACHED */
+
case KGDB_EXEC:
default:
/* Unknown command. Ack with a null message. */
@@ -552,21 +601,31 @@ kgdb_trap(type, tf)
}
}
-extern int kernacc();
-extern void chgkprot();
extern char *kernel_map; /* XXX! */
extern char *curproc; /* XXX! */
/*
* XXX do kernacc and useracc calls if safe, otherwise use PTE protections.
+ *
+ * Note: kernacc fails currently on the Sun4M port--we use pte bits instead.
+ * Plus this lets us debug kernacc. (%%% XXX)
*/
+int
kgdb_acc(addr, len, rw, usertoo)
caddr_t addr;
int len, rw, usertoo;
{
- extern char end[];
int pte;
+#if defined(SUN4M) /* we just use ptes here...its easier */
+ if (CPU_ISSUN4M) {
+ pte = getpte4m(addr);
+ if ((pte & SRMMU_TETYPE) != SRMMU_TEPTE ||
+ (rw == B_WRITE && (pte & PPROT_WRITE) == 0))
+ return (0);
+ }
+#endif
+
/* XXX icky: valid address but causes timeout */
if (addr >= (caddr_t)0xfffff000)
return (0);
@@ -581,24 +640,42 @@ kgdb_acc(addr, len, rw, usertoo)
if (((int)addr >> PG_VSHIFT) != 0 &&
((int)addr >> PG_VSHIFT) != -1)
return (0);
- pte = getpte(addr);
- if ((pte & PG_V) == 0 || rw == B_WRITE && (pte & PG_W) == 0)
- return (0);
+#if defined(SUN4M)
+ if (CPU_ISSUN4M) {
+ pte = getpte4m(addr);
+ if ((pte & SRMMU_TETYPE) != SRMMU_TEPTE ||
+ (rw == B_WRITE && (pte & PPROT_WRITE) == 0))
+ return (0);
+ } else
+#endif
+ {
+ pte = getpte4(addr);
+ if ((pte & PG_V) == 0 ||
+ (rw == B_WRITE && (pte & PG_W) == 0))
+ return (0);
+ }
}
return (1);
}
+void
kdb_mkwrite(addr, len)
register caddr_t addr;
register int len;
{
-
- if (kernel_map != NULL) {
+ if (CPU_ISSUN4OR4C && kernel_map != NULL) {
chgkprot(addr, len, B_WRITE);
return;
}
+
addr = (caddr_t)((int)addr & ~PGOFSET);
for (; len > 0; len -= NBPG, addr += NBPG)
- setpte(addr, getpte(addr) | PG_W);
+#if defined(SUN4M)
+ if (CPU_ISSUN4M)
+ setpte4m((vm_offset_t)addr,
+ getpte4m(addr) | PPROT_WRITE);
+ else
+#endif
+ setpte4(addr, getpte4(addr) | PG_W);
}
#endif
diff --git a/sys/arch/sparc/sparc/locore.s b/sys/arch/sparc/sparc/locore.s
index 33f2e97417d..31721c9ed81 100644
--- a/sys/arch/sparc/sparc/locore.s
+++ b/sys/arch/sparc/sparc/locore.s
@@ -1,40 +1,10 @@
/*
- * Copyright (c) 1995 Theo de Raadt
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- * 3. All advertising materials mentioning features or use of this software
- * must display the following acknowledgement:
- * This product includes software developed under OpenBSD by
- * Theo de Raadt for Willowglen Singapore.
- * 4. The name of the author may not be used to endorse or promote products
- * derived from this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
- * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
- * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
- * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
- * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
- * SUCH DAMAGE.
- *
+ * Copyright (c) 1996 Paul Kranenburg
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
*
- * All advertising materials mentioning features or use of this software
- * must display the following acknowledgement:
- * This product includes software developed by Theo de Raadt.
- *
* This software was developed by the Computer Systems Engineering group
* at Lawrence Berkeley Laboratory under DARPA contract BG 91-66 and
* contributed to Berkeley.
@@ -43,6 +13,7 @@
* must display the following acknowledgement:
* This product includes software developed by the University of
* California, Lawrence Berkeley Laboratory.
+ * This product includes software developed by Harvard University.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
@@ -56,6 +27,8 @@
* must display the following acknowledgement:
* This product includes software developed by the University of
* California, Berkeley and its contributors.
+ * This product includes software developed by Harvard University.
+ * This product includes software developed by Paul Kranenburg.
* 4. Neither the name of the University nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
@@ -75,12 +48,12 @@
* @(#)locore.s 8.4 (Berkeley) 12/10/93
*/
-#define LOCORE
-#include "assym.s"
+#include "assym.h"
+#include <machine/param.h>
#include <sparc/sparc/intreg.h>
#include <sparc/sparc/timerreg.h>
-#ifdef notyet
#include <sparc/sparc/vaddrs.h>
+#ifdef notyet
#include <sparc/dev/zsreg.h>
#endif
#include <machine/ctlreg.h>
@@ -105,7 +78,7 @@
#endif
/* use as needed to align things on longword boundaries */
-#define ALIGN .align 4
+#define _ALIGN .align 4
/*
* CCFSZ (C Compiler Frame SiZe) is the size of a stack frame required if
@@ -206,7 +179,7 @@ _kgdb_stack:
.globl _cpcb
_cpcb: .word _u0
-/*
+/*
* _cputyp is the current cpu type, used to distinguish between
* the many variations of different sun4* machines. It contains
* the value CPU_SUN4, CPU_SUN4C, or CPU_SUN4M.
@@ -214,6 +187,23 @@ _cpcb: .word _u0
.globl _cputyp
_cputyp:
.word 1
+/*
+ * _cpumod is the current cpu model, used to distinguish between variants
+ * in the Sun4 and Sun4M families. See /sys/arch/sparc/include/param.h for
+ * possible values.
+ */
+ .globl _cpumod
+_cpumod:
+ .word 1
+/*
+ * _mmumod is the current mmu model, used to distinguish between the
+ * various implementations of the SRMMU in the sun4m family of machines.
+ * See /sys/arch/sparc/include/param.h for possible values.
+ */
+ .globl _mmumod
+_mmumod:
+ .word 0
+
#if defined(SUN4C) || defined(SUN4M)
_cputypval:
.asciz "sun4c"
@@ -221,7 +211,7 @@ _cputypval:
_cputypvar:
.asciz "compatible"
_cputypvallen = _cputypvar - _cputypval
- ALIGN
+ _ALIGN
#endif
/*
@@ -230,11 +220,15 @@ _cputypvallen = _cputypvar - _cputypval
*/
.globl _pgshift, _nbpg, _pgofset
_pgshift:
- .word 1
+ .word 0
_nbpg:
- .word 1
+ .word 0
_pgofset:
- .word 1
+ .word 0
+
+ .globl _trapbase
+_trapbase:
+ .word 0
#if defined(SUN4M)
_mapme:
@@ -255,7 +249,7 @@ sun4_notsup:
! before we halt. Sick, eh?
.asciz "NetBSD/sparc: this kernel does not support the sun4\n\r \b"
#endif
- ALIGN
+ _ALIGN
.text
@@ -280,6 +274,10 @@ _msgbuf = KERNBASE
* boot process, so that we can handle NMIs (parity errors) halfway
* sensibly during boot. We use virtual address f8002000 (`page 2')
* for this, wasting a page of physical memory.
+ *
+ * This doesn't work for the Sun4M, which can have 5 or more pages of
+ * registers. Thus we use a reserved piece of the virtual address space, set
+ * up in bootstrap().
*/
IE_reg_addr = KERNBASE + 8192 ! this page not used; points to IEreg
@@ -319,13 +317,19 @@ IE_reg_addr = KERNBASE + 8192 ! this page not used; points to IEreg
mov (type), %l3; b label; mov %psr, %l0; nop
/* hardware interrupts (can be linked or made `fast') */
-#define HARDINT(lev) \
- mov (lev), %l3; b _sparc_interrupt; mov %psr, %l0; nop
+#define HARDINT44C(lev) \
+ mov (lev), %l3; b _sparc_interrupt44c; mov %psr, %l0; nop
+
+ /* hardware interrupts (can be linked or made `fast') */
+#define HARDINT4M(lev) \
+ mov (lev), %l3; b _sparc_interrupt4m; mov %psr, %l0; nop
/* software interrupts (may not be made direct, sorry---but you
should not be using them trivially anyway) */
-#define SOFTINT(lev, bit) \
- mov (lev), %l3; mov (bit), %l4; b softintr; mov %psr, %l0
+#define SOFTINT44C(lev, bit) \
+ mov (lev), %l3; mov (bit), %l4; b softintr_sun44c; mov %psr, %l0
+
+ /* There's no SOFTINT4M(): both hard and soft vector the same way */
/* traps that just call trap() */
#define TRAP(type) VTRAP(type, slowtrap)
@@ -352,16 +356,282 @@ IE_reg_addr = KERNBASE + 8192 ! this page not used; points to IEreg
#ifdef notyet
#define ZS_INTERRUPT b zshard; mov %psr, %l0; nop; nop
#else
-#define ZS_INTERRUPT HARDINT(12)
+#define ZS_INTERRUPT44C HARDINT44C(12)
+#define ZS_INTERRUPT4M HARDINT4M(12)
#endif
- .globl start
- .globl _trapbase
+ .globl start, _kernel_text
+ _kernel_text = start ! for kvm_mkdb(8)
start:
-_trapbase:
+/*
+ * Put sun4 traptable first, since it needs the most stringent aligment (8192)
+ */
+#if defined(SUN4)
+trapbase_sun4:
+ /* trap 0 is special since we cannot receive it */
+ b dostart; nop; nop; nop ! 00 = reset (fake)
+ VTRAP(T_TEXTFAULT, memfault_sun4) ! 01 = instr. fetch fault
+ TRAP(T_ILLINST) ! 02 = illegal instruction
+ TRAP(T_PRIVINST) ! 03 = privileged instruction
+ TRAP(T_FPDISABLED) ! 04 = fp instr, but EF bit off in psr
+ WINDOW_OF ! 05 = window overflow
+ WINDOW_UF ! 06 = window underflow
+ TRAP(T_ALIGN) ! 07 = address alignment error
+ VTRAP(T_FPE, fp_exception) ! 08 = fp exception
+ VTRAP(T_DATAFAULT, memfault_sun4) ! 09 = data fetch fault
+ TRAP(T_TAGOF) ! 0a = tag overflow
+ UTRAP(0x0b)
+ UTRAP(0x0c)
+ UTRAP(0x0d)
+ UTRAP(0x0e)
+ UTRAP(0x0f)
+ UTRAP(0x10)
+ SOFTINT44C(1, IE_L1) ! 11 = level 1 interrupt
+ HARDINT44C(2) ! 12 = level 2 interrupt
+ HARDINT44C(3) ! 13 = level 3 interrupt
+ SOFTINT44C(4, IE_L4) ! 14 = level 4 interrupt
+ HARDINT44C(5) ! 15 = level 5 interrupt
+ SOFTINT44C(6, IE_L6) ! 16 = level 6 interrupt
+ HARDINT44C(7) ! 17 = level 7 interrupt
+ HARDINT44C(8) ! 18 = level 8 interrupt
+ HARDINT44C(9) ! 19 = level 9 interrupt
+ HARDINT44C(10) ! 1a = level 10 interrupt
+ HARDINT44C(11) ! 1b = level 11 interrupt
+ ZS_INTERRUPT44C ! 1c = level 12 (zs) interrupt
+ HARDINT44C(13) ! 1d = level 13 interrupt
+ HARDINT44C(14) ! 1e = level 14 interrupt
+ VTRAP(15, nmi_sun4) ! 1f = nonmaskable interrupt
+ UTRAP(0x20)
+ UTRAP(0x21)
+ UTRAP(0x22)
+ UTRAP(0x23)
+ TRAP(T_CPDISABLED) ! 24 = coprocessor instr, EC bit off in psr
+ UTRAP(0x25)
+ UTRAP(0x26)
+ UTRAP(0x27)
+ TRAP(T_CPEXCEPTION) ! 28 = coprocessor exception
+ UTRAP(0x29)
+ UTRAP(0x2a)
+ UTRAP(0x2b)
+ UTRAP(0x2c)
+ UTRAP(0x2d)
+ UTRAP(0x2e)
+ UTRAP(0x2f)
+ UTRAP(0x30)
+ UTRAP(0x31)
+ UTRAP(0x32)
+ UTRAP(0x33)
+ UTRAP(0x34)
+ UTRAP(0x35)
+ UTRAP(0x36)
+ UTRAP(0x37)
+ UTRAP(0x38)
+ UTRAP(0x39)
+ UTRAP(0x3a)
+ UTRAP(0x3b)
+ UTRAP(0x3c)
+ UTRAP(0x3d)
+ UTRAP(0x3e)
+ UTRAP(0x3f)
+ UTRAP(0x40)
+ UTRAP(0x41)
+ UTRAP(0x42)
+ UTRAP(0x43)
+ UTRAP(0x44)
+ UTRAP(0x45)
+ UTRAP(0x46)
+ UTRAP(0x47)
+ UTRAP(0x48)
+ UTRAP(0x49)
+ UTRAP(0x4a)
+ UTRAP(0x4b)
+ UTRAP(0x4c)
+ UTRAP(0x4d)
+ UTRAP(0x4e)
+ UTRAP(0x4f)
+ UTRAP(0x50)
+ UTRAP(0x51)
+ UTRAP(0x52)
+ UTRAP(0x53)
+ UTRAP(0x54)
+ UTRAP(0x55)
+ UTRAP(0x56)
+ UTRAP(0x57)
+ UTRAP(0x58)
+ UTRAP(0x59)
+ UTRAP(0x5a)
+ UTRAP(0x5b)
+ UTRAP(0x5c)
+ UTRAP(0x5d)
+ UTRAP(0x5e)
+ UTRAP(0x5f)
+ UTRAP(0x60)
+ UTRAP(0x61)
+ UTRAP(0x62)
+ UTRAP(0x63)
+ UTRAP(0x64)
+ UTRAP(0x65)
+ UTRAP(0x66)
+ UTRAP(0x67)
+ UTRAP(0x68)
+ UTRAP(0x69)
+ UTRAP(0x6a)
+ UTRAP(0x6b)
+ UTRAP(0x6c)
+ UTRAP(0x6d)
+ UTRAP(0x6e)
+ UTRAP(0x6f)
+ UTRAP(0x70)
+ UTRAP(0x71)
+ UTRAP(0x72)
+ UTRAP(0x73)
+ UTRAP(0x74)
+ UTRAP(0x75)
+ UTRAP(0x76)
+ UTRAP(0x77)
+ UTRAP(0x78)
+ UTRAP(0x79)
+ UTRAP(0x7a)
+ UTRAP(0x7b)
+ UTRAP(0x7c)
+ UTRAP(0x7d)
+ UTRAP(0x7e)
+ UTRAP(0x7f)
+ SYSCALL ! 80 = sun syscall
+ BPT ! 81 = pseudo breakpoint instruction
+ TRAP(T_DIV0) ! 82 = divide by zero
+ TRAP(T_FLUSHWIN) ! 83 = flush windows
+ TRAP(T_CLEANWIN) ! 84 = provide clean windows
+ TRAP(T_RANGECHECK) ! 85 = ???
+ TRAP(T_FIXALIGN) ! 86 = fix up unaligned accesses
+ TRAP(T_INTOF) ! 87 = integer overflow
+ SYSCALL ! 88 = svr4 syscall
+ SYSCALL ! 89 = bsd syscall
+ BPT_KGDB_EXEC ! 8a = enter kernel gdb on kernel startup
+ STRAP(0x8b)
+ STRAP(0x8c)
+ STRAP(0x8d)
+ STRAP(0x8e)
+ STRAP(0x8f)
+ STRAP(0x90)
+ STRAP(0x91)
+ STRAP(0x92)
+ STRAP(0x93)
+ STRAP(0x94)
+ STRAP(0x95)
+ STRAP(0x96)
+ STRAP(0x97)
+ STRAP(0x98)
+ STRAP(0x99)
+ STRAP(0x9a)
+ STRAP(0x9b)
+ STRAP(0x9c)
+ STRAP(0x9d)
+ STRAP(0x9e)
+ STRAP(0x9f)
+ STRAP(0xa0)
+ STRAP(0xa1)
+ STRAP(0xa2)
+ STRAP(0xa3)
+ STRAP(0xa4)
+ STRAP(0xa5)
+ STRAP(0xa6)
+ STRAP(0xa7)
+ STRAP(0xa8)
+ STRAP(0xa9)
+ STRAP(0xaa)
+ STRAP(0xab)
+ STRAP(0xac)
+ STRAP(0xad)
+ STRAP(0xae)
+ STRAP(0xaf)
+ STRAP(0xb0)
+ STRAP(0xb1)
+ STRAP(0xb2)
+ STRAP(0xb3)
+ STRAP(0xb4)
+ STRAP(0xb5)
+ STRAP(0xb6)
+ STRAP(0xb7)
+ STRAP(0xb8)
+ STRAP(0xb9)
+ STRAP(0xba)
+ STRAP(0xbb)
+ STRAP(0xbc)
+ STRAP(0xbd)
+ STRAP(0xbe)
+ STRAP(0xbf)
+ STRAP(0xc0)
+ STRAP(0xc1)
+ STRAP(0xc2)
+ STRAP(0xc3)
+ STRAP(0xc4)
+ STRAP(0xc5)
+ STRAP(0xc6)
+ STRAP(0xc7)
+ STRAP(0xc8)
+ STRAP(0xc9)
+ STRAP(0xca)
+ STRAP(0xcb)
+ STRAP(0xcc)
+ STRAP(0xcd)
+ STRAP(0xce)
+ STRAP(0xcf)
+ STRAP(0xd0)
+ STRAP(0xd1)
+ STRAP(0xd2)
+ STRAP(0xd3)
+ STRAP(0xd4)
+ STRAP(0xd5)
+ STRAP(0xd6)
+ STRAP(0xd7)
+ STRAP(0xd8)
+ STRAP(0xd9)
+ STRAP(0xda)
+ STRAP(0xdb)
+ STRAP(0xdc)
+ STRAP(0xdd)
+ STRAP(0xde)
+ STRAP(0xdf)
+ STRAP(0xe0)
+ STRAP(0xe1)
+ STRAP(0xe2)
+ STRAP(0xe3)
+ STRAP(0xe4)
+ STRAP(0xe5)
+ STRAP(0xe6)
+ STRAP(0xe7)
+ STRAP(0xe8)
+ STRAP(0xe9)
+ STRAP(0xea)
+ STRAP(0xeb)
+ STRAP(0xec)
+ STRAP(0xed)
+ STRAP(0xee)
+ STRAP(0xef)
+ STRAP(0xf0)
+ STRAP(0xf1)
+ STRAP(0xf2)
+ STRAP(0xf3)
+ STRAP(0xf4)
+ STRAP(0xf5)
+ STRAP(0xf6)
+ STRAP(0xf7)
+ STRAP(0xf8)
+ STRAP(0xf9)
+ STRAP(0xfa)
+ STRAP(0xfb)
+ STRAP(0xfc)
+ STRAP(0xfd)
+ STRAP(0xfe)
+ STRAP(0xff)
+#endif
+
+#if defined(SUN4C)
+trapbase_sun4c:
/* trap 0 is special since we cannot receive it */
b dostart; nop; nop; nop ! 00 = reset (fake)
- VTRAP(T_TEXTFAULT, memfault_bootup) ! 01 = instr. fetch fault (backpatch to memfault)
+ VTRAP(T_TEXTFAULT, memfault_sun4c) ! 01 = instr. fetch fault
TRAP(T_ILLINST) ! 02 = illegal instruction
TRAP(T_PRIVINST) ! 03 = privileged instruction
TRAP(T_FPDISABLED) ! 04 = fp instr, but EF bit off in psr
@@ -369,7 +639,7 @@ _trapbase:
WINDOW_UF ! 06 = window underflow
TRAP(T_ALIGN) ! 07 = address alignment error
VTRAP(T_FPE, fp_exception) ! 08 = fp exception
- VTRAP(T_DATAFAULT, memfault_bootup) ! 09 = data fetch fault (backpatch to memfault)
+ VTRAP(T_DATAFAULT, memfault_sun4c) ! 09 = data fetch fault
TRAP(T_TAGOF) ! 0a = tag overflow
UTRAP(0x0b)
UTRAP(0x0c)
@@ -377,21 +647,21 @@ _trapbase:
UTRAP(0x0e)
UTRAP(0x0f)
UTRAP(0x10)
- SOFTINT(1, IE_L1) ! 11 = level 1 interrupt
- HARDINT(2) ! 12 = level 2 interrupt
- HARDINT(3) ! 13 = level 3 interrupt
- SOFTINT(4, IE_L4) ! 14 = level 4 interrupt
- HARDINT(5) ! 15 = level 5 interrupt
- SOFTINT(6, IE_L6) ! 16 = level 6 interrupt
- HARDINT(7) ! 17 = level 7 interrupt
- HARDINT(8) ! 18 = level 8 interrupt
- HARDINT(9) ! 19 = level 9 interrupt
- HARDINT(10) ! 1a = level 10 interrupt
- HARDINT(11) ! 1b = level 11 interrupt
- ZS_INTERRUPT ! 1c = level 12 (zs) interrupt
- HARDINT(13) ! 1d = level 13 interrupt
- HARDINT(14) ! 1e = level 14 interrupt
- VTRAP(15, nmi) ! 1f = nonmaskable interrupt
+ SOFTINT44C(1, IE_L1) ! 11 = level 1 interrupt
+ HARDINT44C(2) ! 12 = level 2 interrupt
+ HARDINT44C(3) ! 13 = level 3 interrupt
+ SOFTINT44C(4, IE_L4) ! 14 = level 4 interrupt
+ HARDINT44C(5) ! 15 = level 5 interrupt
+ SOFTINT44C(6, IE_L6) ! 16 = level 6 interrupt
+ HARDINT44C(7) ! 17 = level 7 interrupt
+ HARDINT44C(8) ! 18 = level 8 interrupt
+ HARDINT44C(9) ! 19 = level 9 interrupt
+ HARDINT44C(10) ! 1a = level 10 interrupt
+ HARDINT44C(11) ! 1b = level 11 interrupt
+ ZS_INTERRUPT44C ! 1c = level 12 (zs) interrupt
+ HARDINT44C(13) ! 1d = level 13 interrupt
+ HARDINT44C(14) ! 1e = level 14 interrupt
+ VTRAP(15, nmi_sun4c) ! 1f = nonmaskable interrupt
UTRAP(0x20)
UTRAP(0x21)
UTRAP(0x22)
@@ -616,13 +886,278 @@ _trapbase:
STRAP(0xfd)
STRAP(0xfe)
STRAP(0xff)
+#endif
+
+#if defined(SUN4M)
+trapbase_sun4m:
+/* trap 0 is special since we cannot receive it */
+ b dostart; nop; nop; nop ! 00 = reset (fake)
+ VTRAP(T_TEXTFAULT, memfault_sun4m) ! 01 = instr. fetch fault
+ TRAP(T_ILLINST) ! 02 = illegal instruction
+ TRAP(T_PRIVINST) ! 03 = privileged instruction
+ TRAP(T_FPDISABLED) ! 04 = fp instr, but EF bit off in psr
+ WINDOW_OF ! 05 = window overflow
+ WINDOW_UF ! 06 = window underflow
+ TRAP(T_ALIGN) ! 07 = address alignment error
+ VTRAP(T_FPE, fp_exception) ! 08 = fp exception
+ VTRAP(T_DATAFAULT, memfault_sun4m) ! 09 = data fetch fault
+ TRAP(T_TAGOF) ! 0a = tag overflow
+ UTRAP(0x0b)
+ UTRAP(0x0c)
+ UTRAP(0x0d)
+ UTRAP(0x0e)
+ UTRAP(0x0f)
+ UTRAP(0x10)
+ HARDINT4M(1) ! 11 = level 1 interrupt
+ HARDINT4M(2) ! 12 = level 2 interrupt
+ HARDINT4M(3) ! 13 = level 3 interrupt
+ HARDINT4M(4) ! 14 = level 4 interrupt
+ HARDINT4M(5) ! 15 = level 5 interrupt
+ HARDINT4M(6) ! 16 = level 6 interrupt
+ HARDINT4M(7) ! 17 = level 7 interrupt
+ HARDINT4M(8) ! 18 = level 8 interrupt
+ HARDINT4M(9) ! 19 = level 9 interrupt
+ HARDINT4M(10) ! 1a = level 10 interrupt
+ HARDINT4M(11) ! 1b = level 11 interrupt
+ ZS_INTERRUPT4M ! 1c = level 12 (zs) interrupt
+ HARDINT4M(13) ! 1d = level 13 interrupt
+ HARDINT4M(14) ! 1e = level 14 interrupt
+ VTRAP(15, nmi_sun4m) ! 1f = nonmaskable interrupt
+ UTRAP(0x20)
+ UTRAP(0x21)
+ UTRAP(0x22)
+ UTRAP(0x23)
+ TRAP(T_CPDISABLED) ! 24 = coprocessor instr, EC bit off in psr
+ UTRAP(0x25)
+ UTRAP(0x26)
+ UTRAP(0x27)
+ TRAP(T_CPEXCEPTION) ! 28 = coprocessor exception
+ UTRAP(0x29)
+ UTRAP(0x2a)
+ VTRAP(T_STOREBUFFAULT, memfault_sun4m) ! 2b = SuperSPARC store buffer fault
+ UTRAP(0x2c)
+ UTRAP(0x2d)
+ UTRAP(0x2e)
+ UTRAP(0x2f)
+ UTRAP(0x30)
+ UTRAP(0x31)
+ UTRAP(0x32)
+ UTRAP(0x33)
+ UTRAP(0x34)
+ UTRAP(0x35)
+ UTRAP(0x36)
+ UTRAP(0x37)
+ UTRAP(0x38)
+ UTRAP(0x39)
+ UTRAP(0x3a)
+ UTRAP(0x3b)
+ UTRAP(0x3c)
+ UTRAP(0x3d)
+ UTRAP(0x3e)
+ UTRAP(0x3f)
+ UTRAP(0x40)
+ UTRAP(0x41)
+ UTRAP(0x42)
+ UTRAP(0x43)
+ UTRAP(0x44)
+ UTRAP(0x45)
+ UTRAP(0x46)
+ UTRAP(0x47)
+ UTRAP(0x48)
+ UTRAP(0x49)
+ UTRAP(0x4a)
+ UTRAP(0x4b)
+ UTRAP(0x4c)
+ UTRAP(0x4d)
+ UTRAP(0x4e)
+ UTRAP(0x4f)
+ UTRAP(0x50)
+ UTRAP(0x51)
+ UTRAP(0x52)
+ UTRAP(0x53)
+ UTRAP(0x54)
+ UTRAP(0x55)
+ UTRAP(0x56)
+ UTRAP(0x57)
+ UTRAP(0x58)
+ UTRAP(0x59)
+ UTRAP(0x5a)
+ UTRAP(0x5b)
+ UTRAP(0x5c)
+ UTRAP(0x5d)
+ UTRAP(0x5e)
+ UTRAP(0x5f)
+ UTRAP(0x60)
+ UTRAP(0x61)
+ UTRAP(0x62)
+ UTRAP(0x63)
+ UTRAP(0x64)
+ UTRAP(0x65)
+ UTRAP(0x66)
+ UTRAP(0x67)
+ UTRAP(0x68)
+ UTRAP(0x69)
+ UTRAP(0x6a)
+ UTRAP(0x6b)
+ UTRAP(0x6c)
+ UTRAP(0x6d)
+ UTRAP(0x6e)
+ UTRAP(0x6f)
+ UTRAP(0x70)
+ UTRAP(0x71)
+ UTRAP(0x72)
+ UTRAP(0x73)
+ UTRAP(0x74)
+ UTRAP(0x75)
+ UTRAP(0x76)
+ UTRAP(0x77)
+ UTRAP(0x78)
+ UTRAP(0x79)
+ UTRAP(0x7a)
+ UTRAP(0x7b)
+ UTRAP(0x7c)
+ UTRAP(0x7d)
+ UTRAP(0x7e)
+ UTRAP(0x7f)
+ SYSCALL ! 80 = sun syscall
+ BPT ! 81 = pseudo breakpoint instruction
+ TRAP(T_DIV0) ! 82 = divide by zero
+ TRAP(T_FLUSHWIN) ! 83 = flush windows
+ TRAP(T_CLEANWIN) ! 84 = provide clean windows
+ TRAP(T_RANGECHECK) ! 85 = ???
+ TRAP(T_FIXALIGN) ! 86 = fix up unaligned accesses
+ TRAP(T_INTOF) ! 87 = integer overflow
+ SYSCALL ! 88 = svr4 syscall
+ SYSCALL ! 89 = bsd syscall
+ BPT_KGDB_EXEC ! 8a = enter kernel gdb on kernel startup
+ STRAP(0x8b)
+ STRAP(0x8c)
+ STRAP(0x8d)
+ STRAP(0x8e)
+ STRAP(0x8f)
+ STRAP(0x90)
+ STRAP(0x91)
+ STRAP(0x92)
+ STRAP(0x93)
+ STRAP(0x94)
+ STRAP(0x95)
+ STRAP(0x96)
+ STRAP(0x97)
+ STRAP(0x98)
+ STRAP(0x99)
+ STRAP(0x9a)
+ STRAP(0x9b)
+ STRAP(0x9c)
+ STRAP(0x9d)
+ STRAP(0x9e)
+ STRAP(0x9f)
+ STRAP(0xa0)
+ STRAP(0xa1)
+ STRAP(0xa2)
+ STRAP(0xa3)
+ STRAP(0xa4)
+ STRAP(0xa5)
+ STRAP(0xa6)
+ STRAP(0xa7)
+ STRAP(0xa8)
+ STRAP(0xa9)
+ STRAP(0xaa)
+ STRAP(0xab)
+ STRAP(0xac)
+ STRAP(0xad)
+ STRAP(0xae)
+ STRAP(0xaf)
+ STRAP(0xb0)
+ STRAP(0xb1)
+ STRAP(0xb2)
+ STRAP(0xb3)
+ STRAP(0xb4)
+ STRAP(0xb5)
+ STRAP(0xb6)
+ STRAP(0xb7)
+ STRAP(0xb8)
+ STRAP(0xb9)
+ STRAP(0xba)
+ STRAP(0xbb)
+ STRAP(0xbc)
+ STRAP(0xbd)
+ STRAP(0xbe)
+ STRAP(0xbf)
+ STRAP(0xc0)
+ STRAP(0xc1)
+ STRAP(0xc2)
+ STRAP(0xc3)
+ STRAP(0xc4)
+ STRAP(0xc5)
+ STRAP(0xc6)
+ STRAP(0xc7)
+ STRAP(0xc8)
+ STRAP(0xc9)
+ STRAP(0xca)
+ STRAP(0xcb)
+ STRAP(0xcc)
+ STRAP(0xcd)
+ STRAP(0xce)
+ STRAP(0xcf)
+ STRAP(0xd0)
+ STRAP(0xd1)
+ STRAP(0xd2)
+ STRAP(0xd3)
+ STRAP(0xd4)
+ STRAP(0xd5)
+ STRAP(0xd6)
+ STRAP(0xd7)
+ STRAP(0xd8)
+ STRAP(0xd9)
+ STRAP(0xda)
+ STRAP(0xdb)
+ STRAP(0xdc)
+ STRAP(0xdd)
+ STRAP(0xde)
+ STRAP(0xdf)
+ STRAP(0xe0)
+ STRAP(0xe1)
+ STRAP(0xe2)
+ STRAP(0xe3)
+ STRAP(0xe4)
+ STRAP(0xe5)
+ STRAP(0xe6)
+ STRAP(0xe7)
+ STRAP(0xe8)
+ STRAP(0xe9)
+ STRAP(0xea)
+ STRAP(0xeb)
+ STRAP(0xec)
+ STRAP(0xed)
+ STRAP(0xee)
+ STRAP(0xef)
+ STRAP(0xf0)
+ STRAP(0xf1)
+ STRAP(0xf2)
+ STRAP(0xf3)
+ STRAP(0xf4)
+ STRAP(0xf5)
+ STRAP(0xf6)
+ STRAP(0xf7)
+ STRAP(0xf8)
+ STRAP(0xf9)
+ STRAP(0xfa)
+ STRAP(0xfb)
+ STRAP(0xfc)
+ STRAP(0xfd)
+ STRAP(0xfe)
+ STRAP(0xff)
+#endif
/*
- * pad the trap table to max page size
- * trap table size is 0x100 * 4instr * 4byte/instr = 4096 bytes
- * need to .skip 4096 to pad to page size
+ * Pad the trap table to max page size.
+ * Trap table size is 0x100 * 4instr * 4byte/instr = 4096 bytes;
+ * need to .skip 4096 to pad to page size iff. the number of trap tables
+ * defined above is odd.
*/
+#if defined(SUN4) + defined(SUN4C) + defined(SUN4M) - 2 == 0
.skip 4096
+#endif
#ifdef DEBUG
/*
@@ -640,7 +1175,7 @@ _redstack:
.text
Lpanic_red:
.asciz "stack overflow"
- ALIGN
+ _ALIGN
/* set stack pointer redzone to base+minstack; alters base */
#define SET_SP_REDZONE(base, tmp) \
@@ -684,7 +1219,6 @@ Lpanic_red:
#define CHECK_SP_REDZONE(t1, t2)
#endif
-#if defined(SUN4) || defined(SUN4C)
/*
* The window code must verify user stack addresses before using them.
* A user stack pointer is invalid if:
@@ -706,8 +1240,9 @@ Lpanic_red:
* The code below also assumes that PTE_OF_ADDR is safe in a delay
* slot; it is, at it merely sets its `pte' register to a temporary value.
*/
+#if defined(SUN4) || defined(SUN4C)
/* input: addr, output: pte; aux: bad address label */
-#define PTE_OF_ADDR(addr, pte, bad, page_offset) \
+#define PTE_OF_ADDR4_4C(addr, pte, bad, page_offset) \
sra addr, PG_VSHIFT, pte; \
cmp pte, -1; \
be,a 1f; andn addr, page_offset, pte; \
@@ -717,20 +1252,92 @@ Lpanic_red:
1:
/* input: pte; output: condition codes */
-#define CMP_PTE_USER_READ(pte) \
+#define CMP_PTE_USER_READ4_4C(pte) \
lda [pte] ASI_PTE, pte; \
srl pte, PG_PROTSHIFT, pte; \
andn pte, (PG_W >> PG_PROTSHIFT), pte; \
cmp pte, PG_PROTUREAD
/* input: pte; output: condition codes */
-#define CMP_PTE_USER_WRITE(pte) \
+#define CMP_PTE_USER_WRITE4_4C(pte) \
lda [pte] ASI_PTE, pte; \
srl pte, PG_PROTSHIFT, pte; \
cmp pte, PG_PROTUWRITE
#endif
/*
+ * The Sun4M does not have the memory hole that the 4C does. Thus all
+ * we need to do here is clear the page offset from addr.
+ */
+#if defined(SUN4M)
+#define PTE_OF_ADDR4M(addr, pte, bad, page_offset) \
+ andn addr, page_offset, pte
+
+/* note: pmap currently does not use the PPROT_R_R and PPROT_RW_RW cases */
+#define CMP_PTE_USER_READ4M(pte) \
+ or pte, ASI_SRMMUFP_L3, pte; \
+ lda [pte] ASI_SRMMUFP, pte; \
+ and pte, (SRMMU_TETYPE | SRMMU_PROT_MASK), pte; \
+ cmp pte, (SRMMU_TEPTE | PPROT_RWX_RWX); \
+ be 8f; nop; \
+ cmp pte, (SRMMU_TEPTE | PPROT_RX_RX); \
+8:
+
+
+/* note: PTE bit 4 set implies no user writes */
+#define CMP_PTE_USER_WRITE4M(pte) \
+ or pte, ASI_SRMMUFP_L3, pte; \
+ lda [pte] ASI_SRMMUFP, pte; \
+ and pte, (SRMMU_TETYPE | 0x14), pte; \
+ cmp pte, (SRMMU_TEPTE | PPROT_WRITE)
+#endif /* 4m */
+
+#if defined(SUN4M) && !(defined(SUN4C) || defined(SUN4))
+#define PTE_OF_ADDR PTE_OF_ADDR4M
+#define CMP_PTE_USER_WRITE(pte,tmp) CMP_PTE_USER_WRITE4M(pte)
+#define CMP_PTE_USER_READ(pte,tmp) CMP_PTE_USER_READ4M(pte)
+#elif (defined(SUN4C) || defined(SUN4)) && !defined(SUN4M)
+#define PTE_OF_ADDR PTE_OF_ADDR4_4C
+#define CMP_PTE_USER_WRITE(pte,tmp) CMP_PTE_USER_WRITE4_4C(pte)
+#define CMP_PTE_USER_READ(pte,tmp) CMP_PTE_USER_READ4_4C(pte)
+#else /* both defined, ugh */
+#define PTE_OF_ADDR(addr, pte, bad, page_offset) \
+ sethi %hi(_cputyp), pte; \
+ ld [pte + %lo(_cputyp)], pte; \
+ cmp pte, CPU_SUN4M; \
+ bne 2f; nop; \
+ PTE_OF_ADDR4M(addr, pte, bad, page_offset); \
+ b,a 3f; \
+2: \
+ PTE_OF_ADDR4_4C(addr, pte, bad, page_offset); \
+3:
+
+#define CMP_PTE_USER_READ(pte, tmp) \
+ sethi %hi(_cputyp), tmp; \
+ ld [tmp + %lo(_cputyp)], tmp; \
+ cmp tmp, CPU_SUN4M; \
+ bne 1f; nop; \
+ CMP_PTE_USER_READ4M(pte); \
+ b,a 2f; \
+1: \
+ CMP_PTE_USER_READ4_4C(pte); \
+2:
+
+#define CMP_PTE_USER_WRITE(pte, tmp) \
+ sethi %hi(_cputyp), tmp; \
+ ld [tmp + %lo(_cputyp)], tmp; \
+ cmp tmp, CPU_SUN4M; \
+ bne 1f; nop; \
+ CMP_PTE_USER_WRITE4M(pte); \
+ b,a 2f; \
+1: \
+ CMP_PTE_USER_WRITE4_4C(pte); \
+2:
+
+#endif
+
+
+/*
* The calculations in PTE_OF_ADDR and CMP_PTE_USER_* are rather slow:
* in particular, according to Gordon Irlam of the University of Adelaide
* in Australia, these consume at least 18 cycles on an SS1 and 37 on an
@@ -998,16 +1605,17 @@ ctw_user:
sethi %hi(_pgofset), %g6 ! trash %g6=curpcb
ld [%g6 + %lo(_pgofset)], %g6
PTE_OF_ADDR(%sp, %g7, ctw_invalid, %g6)
- CMP_PTE_USER_WRITE(%g7) ! likewise if not writable
+ CMP_PTE_USER_WRITE(%g7, %g5) ! likewise if not writable
bne ctw_invalid
EMPTY
+ /* Note side-effect of SLT_IF_1PAGE_RW: decrements %g6 by 62 */
SLT_IF_1PAGE_RW(%sp, %g7, %g6)
bl,a ctw_merge ! all ok if only 1
std %l0, [%sp]
add %sp, 7*8, %g5 ! check last addr too
- add %g6, 62, %g6
+ add %g6, 62, %g6 ! restore %g6 to `pgofset'
PTE_OF_ADDR(%g5, %g7, ctw_invalid, %g6)
- CMP_PTE_USER_WRITE(%g7)
+ CMP_PTE_USER_WRITE(%g7, %g6)
be,a ctw_merge ! all ok: store <l0,l1> and merge
std %l0, [%sp]
@@ -1098,13 +1706,6 @@ ctw_invalid:
* %l4 = %y, until we call mem_access_fault (then onto trapframe)
* %l5 = IE_reg_addr, if async mem error
*
- * We know about the layout of the error registers here.
- * addr reg
- * ---- ---
- * a AC_SYNC_ERR
- * a+4 AC_SYNC_VA
- * a+8 AC_ASYNC_ERR
- * a+12 AC_ASYNC_VA
*/
#if defined(SUN4)
@@ -1135,7 +1736,7 @@ memfault_sun4:
! as I got some parity errors and the
! correct bits were not on?
std %g6, [%sp + CCFSZ + 40]
- bz,a xnormal_mem_fault ! no, just a regular fault
+ bz,a 0f ! no, just a regular fault
wr %l0, PSR_ET, %psr ! (and reenable traps)
/* memory error = death for now XXX */
@@ -1146,11 +1747,11 @@ memfault_sun4:
call _callrom
nop
-xnormal_mem_fault:
+0:
/*
- * have to make SUN4 emulate SUN4C. SUN4C code expects SER in
- * %o1 and the offending VA in %o2, everything else remains the
- * same, but we must figure out if SER_WRITE should be set.
+ * have to make SUN4 emulate SUN4C. 4C code expects
+ * SER in %o1 and the offending VA in %o2, everything else is ok.
+ * (must figure out if SER_WRITE should be set)
*/
set AC_BUS_ERR, %o0 ! bus error register
cmp %l3, T_TEXTFAULT ! text fault always on PC
@@ -1166,18 +1767,30 @@ xnormal_mem_fault:
sethi %hi(SER_WRITE), %o5 ! damn SER_WRITE wont fit simm13
! or %lo(SER_WRITE), %o5, %o5! not necessary since %lo is zero
or %o5, %o1, %o1 ! set SER_WRITE
- ba normal_mem_fault
- nop ! XXX make efficient later
+#if defined(SUN4C) || defined(SUN4M)
+ ba,a normal_mem_fault
+ !!nop ! XXX make efficient later
+#endif /* SUN4C || SUN4M */
#endif /* SUN4 */
-#if defined(SUN4C)
memfault_sun4c:
+#if defined(SUN4C)
TRAP_SETUP(-CCFSZ-80)
INCR(_cnt+V_FAULTS) ! cnt.v_faults++ (clobbers %o0,%o1)
st %g1, [%sp + CCFSZ + 20] ! save g1
rd %y, %l4 ! save y
+ /*
+ * We know about the layout of the error registers here.
+ * addr reg
+ * ---- ---
+ * a AC_SYNC_ERR
+ * a+4 AC_SYNC_VA
+ * a+8 AC_ASYNC_ERR
+ * a+12 AC_ASYNC_VA
+ */
+
#if AC_SYNC_ERR + 4 != AC_SYNC_VA || \
AC_SYNC_ERR + 8 != AC_ASYNC_ERR || AC_SYNC_ERR + 12 != AC_ASYNC_VA
help help help ! I, I, I wanna be a lifeguard
@@ -1266,17 +1879,58 @@ memfault_sun4m:
st %g1, [%sp + CCFSZ + 20] ! save g1
rd %y, %l4 ! save y
- mov SRMMU_SFSTAT, %o0
+ set SRMMU_SFADDR, %o0
std %g2, [%sp + CCFSZ + 24] ! save g2, g3
- lda [%o0] ASI_SRMMU, %o1 ! srmmu fault status
- mov SRMMU_SFADDR, %o0
+ lda [%o0] ASI_SRMMU, %o2 ! sync virt addr; must be read first
+ set SRMMU_SFSTAT, %o0
+ lda [%o0] ASI_SRMMU, %o1 ! get sync fault status register
std %g4, [%sp + CCFSZ + 32] ! (sneak g4,g5 in here)
- lda [%o0] ASI_SRMMU, %o2 ! srmmu fault address
- ! XXX check for memory errors?
- std %g6, [%sp + CCFSZ + 40]
- b normal_mem_fault ! no, just a regular fault
- wr %l0, PSR_ET, %psr ! (and reenable traps)
- /* NOTREACHED */
+
+ /* Now test for a HyperSPARC. If we have one, get the async status */
+ sethi %hi(_mmumod), %o3 ! get MMU model
+ ld [%o3 + %lo(_mmumod)], %o3
+ cmp %o3, SUN4M_MMU_HS ! is it hypersparc?
+ std %g6, [%sp + CCFSZ + 40] ! sneak in g6, g7
+ be 1f ! yup, skip ahead
+
+ clr %o3 ! clear %o3 and %o4, not hypersparc
+ b 2f
+ clr %o4
+1:
+ set SRMMU_AFSTAT, %o3 ! must read status before fault on HS
+ lda [%o3] ASI_SRMMU, %o3 ! get async fault status
+ set SRMMU_AFADDR, %o4
+ lda [%o4] ASI_SRMMU, %o4 ! get async fault address
+2:
+ wr %l0, PSR_ET, %psr ! reenable traps
+
+ /* Finish stackframe, call C trap handler */
+ std %l0, [%sp + CCFSZ + 0] ! set tf.tf_psr, tf.tf_pc
+ mov %l3, %o0 ! (argument: type)
+ st %l2, [%sp + CCFSZ + 8] ! set tf.tf_npc
+ st %l4, [%sp + CCFSZ + 12] ! set tf.tf_y
+ std %i0, [%sp + CCFSZ + 48] ! tf.tf_out[0], etc
+ std %i2, [%sp + CCFSZ + 56]
+ std %i4, [%sp + CCFSZ + 64]
+ std %i6, [%sp + CCFSZ + 72]
+ call _mem_access_fault4m ! mem_access_fault(type, sfsr, sfva,
+ ! afsr, afva, &tf);
+ add %sp, CCFSZ, %o5 ! (argument: &tf)
+
+ ldd [%sp + CCFSZ + 0], %l0 ! load new values
+ ldd [%sp + CCFSZ + 8], %l2
+ wr %l3, 0, %y
+ ld [%sp + CCFSZ + 20], %g1
+ ldd [%sp + CCFSZ + 24], %g2
+ ldd [%sp + CCFSZ + 32], %g4
+ ldd [%sp + CCFSZ + 40], %g6
+ ldd [%sp + CCFSZ + 48], %i0
+ ldd [%sp + CCFSZ + 56], %i2
+ ldd [%sp + CCFSZ + 64], %i4
+ ldd [%sp + CCFSZ + 72], %i6
+
+ b return_from_trap ! go return
+ wr %l0, 0, %psr ! (but first disable traps again)
#endif /* SUN4M */
normal_mem_fault:
@@ -1315,14 +1969,6 @@ normal_mem_fault:
b return_from_trap ! go return
wr %l0, 0, %psr ! (but first disable traps again)
-/* At bootup, we have to point the data/text fault trap table entry
- * somewhere by default. Make the machine watchdog so that we can
- * see what happened by looking at %tbr and %pc under the PROM.
- */
-memfault_bootup:
- call 0
- nop
- /* NOTREACHED */
/*
* fp_exception has to check to see if we are trying to save
@@ -1340,8 +1986,6 @@ fp_exception:
EMPTY
sethi %hi(savefpcont), %l4 ! yes, "return" to the special code
or %lo(savefpcont), %l4, %l4
- wr %l0, 0x0, %psr
- nop; nop; nop;
jmp %l4
rett %l4 + 4
@@ -1621,8 +2265,8 @@ syscall:
ldd [%sp + CCFSZ + 0], %l0 ! new %psr, new pc
ldd [%sp + CCFSZ + 8], %l2 ! new npc, new %y
wr %l3, 0, %y
- /* see `dostart' for the reason for this label */
-init_syscall_ret:
+ /* see `proc_trampoline' for the reason for this label */
+return_from_syscall:
ld [%sp + CCFSZ + 20], %g1
ldd [%sp + CCFSZ + 24], %g2
ldd [%sp + CCFSZ + 32], %g4
@@ -1662,7 +2306,7 @@ init_syscall_ret:
* represents one or many requests)
* (3) we never announce a stray interrupt (because of (1), another
* interrupt request can come in while we're in the handler. If
- * the handler deal with everything for both the original & the
+ * the handler deals with everything for both the original & the
* new request, we'll erroneously report a stray interrupt when
* we take the software interrupt for the new request.
*
@@ -1683,11 +2327,13 @@ init_syscall_ret:
* this contains the psr, pc, npc, and interrupt level.
*/
.comm _intrhand, 15 * 8 ! intrhand[0..14]; 0 => error
-softintr:
+softintr_sun44c:
sethi %hi(IE_reg_addr), %l6
ldub [%l6 + %lo(IE_reg_addr)], %l5
andn %l5, %l4, %l5
stb %l5, [%l6 + %lo(IE_reg_addr)]
+
+softintr_common:
INTR_SETUP(-CCFSZ-80)
std %g2, [%sp + CCFSZ + 24] ! save registers
INCR(_cnt+V_INTR) ! cnt.v_intr++; (clobbers %o0,%o1)
@@ -1731,10 +2377,31 @@ softintr:
wr %l0, 0, %psr
/*
- * _sparc_interrupt is exported for paranoia checking (see intr.c).
+ * _sparc_interrupt{44c,4m} is exported for paranoia checking
+ * (see intr.c).
*/
- .globl _sparc_interrupt
-_sparc_interrupt:
+#if defined(SUN4M)
+ .globl _sparc_interrupt4m
+_sparc_interrupt4m:
+ mov 1, %l4
+ sethi %hi(ICR_PI_PEND), %l5
+ ld [%l5 + %lo(ICR_PI_PEND)], %l5
+ sll %l4, %l3, %l4
+ andcc %l5, %l4, %g0
+ bne _sparc_interrupt_common
+ nop
+
+ ! a soft interrupt; clear bit in interrupt-pending register
+ ! XXX - this is CPU0's register set.
+ sethi %hi(ICR_PI_CLR), %l6
+ sll %l4, 16, %l5
+ st %l5, [%l6 + %lo(ICR_PI_CLR)]
+ b,a softintr_common
+#endif
+
+ .globl _sparc_interrupt44c
+_sparc_interrupt44c:
+_sparc_interrupt_common:
INTR_SETUP(-CCFSZ-80)
std %g2, [%sp + CCFSZ + 24] ! save registers
INCR(_cnt+V_INTR) ! cnt.v_intr++; (clobbers %o0,%o1)
@@ -1821,7 +2488,9 @@ zshard:
* except that we already know the problem is not a `normal' fault,
* and that we must be extra-careful with interrupt enables.
*/
-nmi:
+
+#if defined(SUN4)
+nmi_sun4:
INTR_SETUP(-CCFSZ-80)
INCR(_cnt+V_INTR) ! cnt.v_intr++; (clobbers %o0,%o1)
/*
@@ -1830,33 +2499,40 @@ nmi:
*/
sethi %hi(IE_reg_addr), %o0
ldub [%o0 + %lo(IE_reg_addr)], %o1
- andn %o1, IE_ALLIE, %o1
+ andn %o0, IE_ALLIE, %o1
stb %o1, [%o0 + %lo(IE_reg_addr)]
wr %l0, PSR_ET, %psr ! okay, turn traps on again
std %g2, [%sp + CCFSZ + 0] ! save g2, g3
rd %y, %l4 ! save y
- ! must read the sync error register too.
-#if defined(SUN4) && (defined(SUN4C) || defined(SUN4M))
- sethi %hi(_cputyp), %o0 ! what cpu are we running on?
- ld [%o0 + %lo(_cputyp)], %o0
- cmp %o0, CPU_SUN4
- bne 1f
- nop
-#endif
-#if defined(SUN4)
std %g4, [%sp + CCFSZ + 8] ! save g4, g5
mov %g1, %l5 ! save g1, g6, g7
mov %g6, %l6
mov %g7, %l7
#if defined(SUN4C) || defined(SUN4M)
- b 2f
- nop
+ b,a nmi_common
#endif /* SUN4C || SUN4M */
-#endif /* SUN4 */
-1:
-#if defined(SUN4C) || defined(SUN4M)
+#endif
+
+#if defined(SUN4C)
+nmi_sun4c:
+ INTR_SETUP(-CCFSZ-80)
+ INCR(_cnt+V_INTR) ! cnt.v_intr++; (clobbers %o0,%o1)
+ /*
+ * Level 15 interrupts are nonmaskable, so with traps off,
+ * disable all interrupts to prevent recursion.
+ */
+ sethi %hi(IE_reg_addr), %o0
+ ldub [%o0 + %lo(IE_reg_addr)], %o1
+ andn %o0, IE_ALLIE, %o1
+ stb %o1, [%o0 + %lo(IE_reg_addr)]
+ wr %l0, PSR_ET, %psr ! okay, turn traps on again
+
+ std %g2, [%sp + CCFSZ + 0] ! save g2, g3
+ rd %y, %l4 ! save y
+
+ ! must read the sync error register too.
set AC_SYNC_ERR, %o0
lda [%o0] ASI_CONTROL, %o1 ! sync err reg
inc 4, %o0
@@ -1869,8 +2545,12 @@ nmi:
lda [%o0] ASI_CONTROL, %o3 ! async err reg
inc 4, %o0
lda [%o0] ASI_CONTROL, %o4 ! async virt addr
-#endif /* SUN4C || SUN4M */
-2:
+#if defined(SUN4M)
+ !!b,a nmi_common
+#endif /* SUN4M */
+#endif /* SUN4C */
+
+nmi_common:
! and call C code
call _memerr ! memerr(0, ser, sva, aer, ava)
clr %o0
@@ -1890,6 +2570,84 @@ nmi:
b return_from_trap
wr %l4, 0, %y ! restore y
+#if defined(SUN4M)
+nmi_sun4m:
+ INTR_SETUP(-CCFSZ-80)
+ INCR(_cnt+V_INTR) ! cnt.v_intr++; (clobbers %o0,%o1)
+ /*
+ * Level 15 interrupts are nonmaskable, so with traps off,
+ * disable all interrupts to prevent recursion.
+ */
+ sethi %hi(ICR_SI_SET), %o0
+ set SINTR_MA, %o1
+ st %o1, [%o0 + %lo(ICR_SI_SET)]
+
+ /* Now clear the NMI */
+
+ sethi %hi(ICR_PI_CLR), %o0
+ set PINTR_IC, %o1
+ st %o1, [%o0 + %lo(ICR_PI_CLR)]
+
+ wr %l0, PSR_ET, %psr ! okay, turn traps on again
+
+ std %g2, [%sp + CCFSZ + 0] ! save g2, g3
+ rd %y, %l4 ! save y
+
+ ! now read sync error registers
+ set SRMMU_SFADDR, %o0
+ lda [%o0] ASI_SRMMU, %o2 ! sync virt addr
+ set SRMMU_SFSTAT, %o0
+ lda [%o0] ASI_SRMMU, %o1 ! sync err reg
+ std %g4, [%sp + CCFSZ + 8] ! save g4,g5
+
+ /* Now test for a HyperSPARC. If we have one, get the async status */
+
+ sethi %hi(_mmumod), %o3 ! get MMU model
+ ld [%o3 + %lo(_mmumod)], %o3
+ cmp %o3, SUN4M_MMU_HS ! is it hypersparc?
+ be 1f ! yup, skip ahead
+
+ clr %o3 ! clear %o3 and %o4, not hypersparc
+ b 2f
+ clr %o4
+1:
+ set SRMMU_AFSTAT, %o3 ! read status first on hypersparc
+ lda [%o3] ASI_SRMMU, %o3 ! get async fault status
+ set SRMMU_AFADDR, %o4
+ lda [%o4] ASI_SRMMU, %o4 ! get async fault address
+2:
+ /* Finish stackframe, call C trap handler */
+ mov %g1, %l5 ! save g1,g6,g7
+ mov %g6, %l6
+ mov %g7, %l7
+ clr %o5
+
+ call _memerr4m ! memerr4m(0, sfsr, sfva, afsr, afva)
+ clr %o0
+
+ mov %l5, %g1 ! restore g1 through g7
+ ldd [%sp + CCFSZ + 0], %g2
+ ldd [%sp + CCFSZ + 8], %g4
+ wr %l0, 0, %psr ! re-disable traps
+ mov %l6, %g6
+ mov %l7, %g7
+
+ ! enable interrupts again (safe, we disabled traps again above)
+ sethi %hi(ICR_SI_CLR), %o0
+ set SINTR_MA, %o1
+ st %o1, [%o0 + %lo(ICR_SI_CLR)]
+
+ b return_from_trap
+ wr %l4, 0, %y ! restore y
+#endif /* SUN4M */
+
+#ifdef GPROF
+ .globl window_of, winof_user
+ .globl window_uf, winuf_user, winuf_ok, winuf_invalid
+ .globl return_from_trap, rft_kernel, rft_user, rft_invalid
+ .globl softtrap, slowtrap
+ .globl clean_trap_window, syscall
+#endif
/*
* Window overflow trap handler.
@@ -2064,7 +2822,7 @@ winuf_user:
sethi %hi(_pgofset), %l4
ld [%l4 + %lo(_pgofset)], %l4
PTE_OF_ADDR(%sp, %l7, winuf_invalid, %l4)
- CMP_PTE_USER_READ(%l7) ! if first page not readable,
+ CMP_PTE_USER_READ(%l7, %l5) ! if first page not readable,
bne winuf_invalid ! it is invalid
EMPTY
SLT_IF_1PAGE_RW(%sp, %l7, %l4) ! first page is readable
@@ -2073,7 +2831,7 @@ winuf_user:
add %sp, 7*8, %l5
add %l4, 62, %l4
PTE_OF_ADDR(%l5, %l7, winuf_invalid, %l4)
- CMP_PTE_USER_READ(%l7) ! check second page too
+ CMP_PTE_USER_READ(%l7, %l5) ! check second page too
be,a winuf_ok ! enter window X and goto ok
restore %g0, 1, %l1 ! (and then set %l1 to 1)
@@ -2095,12 +2853,7 @@ winuf_invalid:
* only window, and it is a user window).
*/
save %g0, %g0, %g0 ! back to R
-#ifdef SUN_AS /* this gives `as' mild heartburn */
save %g0, 1, %l4 ! back to T, then %l4 = 1
-#else
- save %g0, %g0, %g0 ! back to T
- mov 1, %l4 ! and set %l4 = 1
-#endif
sethi %hi(_cpcb), %l6
ld [%l6 + %lo(_cpcb)], %l6
st %l4, [%l6 + PCB_UW] ! pcb_uw = 1
@@ -2293,7 +3046,7 @@ rft_user:
sethi %hi(_pgofset), %l3
ld [%l3 + %lo(_pgofset)], %l3
PTE_OF_ADDR(%fp, %l7, rft_invalid, %l3)
- CMP_PTE_USER_READ(%l7) ! try first page
+ CMP_PTE_USER_READ(%l7, %l5) ! try first page
bne rft_invalid ! no good
EMPTY
SLT_IF_1PAGE_RW(%fp, %l7, %l3)
@@ -2302,7 +3055,7 @@ rft_user:
add %fp, 7*8, %l5
add %l3, 62, %l3
PTE_OF_ADDR(%l5, %l7, rft_invalid, %l3)
- CMP_PTE_USER_READ(%l7) ! check 2nd page too
+ CMP_PTE_USER_READ(%l7, %l5) ! check 2nd page too
be,a rft_user_ok
wr %g0, 0, %wim
@@ -2579,9 +3332,8 @@ dostart:
is_sun4m:
#if defined(SUN4M)
+ set trapbase_sun4m, %g6
mov SUN4CM_PGSHIFT, %g5
-
- set memfault_sun4m, %o1 ! ptr to our memfault routine
b start_havetype
mov CPU_SUN4M, %g4
#else
@@ -2596,12 +3348,12 @@ is_sun4m:
#endif
is_sun4c:
#if defined(SUN4C)
+ set trapbase_sun4c, %g6
mov SUN4CM_PGSHIFT, %g5
set AC_CONTEXT, %g1 ! paranoia: set context to kernel
stba %g0, [%g1] ASI_CONTROL
- set memfault_sun4c, %o1 ! ptr to our memfault routine
b start_havetype
mov CPU_SUN4C, %g4 ! XXX CPU_SUN4
#else
@@ -2630,12 +3382,12 @@ is_sun4c:
#endif
is_sun4:
#if defined(SUN4)
+ set trapbase_sun4, %g6
mov SUN4_PGSHIFT, %g5
set AC_CONTEXT, %g1 ! paranoia: set context to kernel
stba %g0, [%g1] ASI_CONTROL
- set memfault_sun4, %o1 ! ptr to our memfault routine
b start_havetype
mov CPU_SUN4, %g4
#else
@@ -2653,41 +3405,6 @@ is_sun4:
start_havetype:
/*
- * Back-patch the T_{TEXT,DATA}FAULT vectors to branch to the
- * correct memfault routine. There is a different memfault routine
- * for each processor. Here we carefully synthesize the offsets
- * and merge them into the "ba" instruction which already exists.
- */
- set KERNBASE, %o2 ! unmapped as yet
- sub %o1, %o2, %o1
- set _trapbase-KERNBASE, %o2 ! and store in trap table
- set 0xffc00000, %o4 ! 22bit offset in "bcc" instruction
-
- ! patch instr fetch fault vector to point to our memfault routine
- ld [%o2 + (16*T_TEXTFAULT+4)], %o3 ! fetch "ba" instruction
- and %o3, %o4, %o3 ! mask out garbage
- sub %o1, %o2, %o5 ! generate relative offset
- sub %o5, (16*T_TEXTFAULT+4), %o5
- srl %o5, 2, %o5 ! long-word offset
- or %o3, %o5, %o3 ! merge into "ba" instruction
- st %o3, [%o2 + (16*T_TEXTFAULT+4)]
-
- ! patch data fetch fault vector to point to our memfault routine
- ld [%o2 + (16*T_DATAFAULT+4)], %o3 ! fetch "ba" instruction
- and %o3, %o4, %o3 ! mask out garbage
- sub %o1, %o2, %o5 ! generate relative offset
- sub %o5, (16*T_DATAFAULT+4), %o5
- srl %o5, 2, %o5 ! long-word offset
- or %o3, %o5, %o3 ! merge into "ba" instruction
- st %o3, [%o2 + (16*T_DATAFAULT+4)]
-
- /*
- * XXX
- * We just modified the text segment. We should flush that cache
- * line!
- */
-
- /*
* Step 1: double map low RAM (addresses [0.._end-start-1])
* to KERNBASE (addresses [KERNBASE.._end-1]). None of these
* are `bad' aliases (since they are all on segment boundaries)
@@ -2746,17 +3463,16 @@ start_havetype:
stb %l1, [%l0]
b startmap_done
nop
-#endif /* SUN4C */
1:
+#endif /* SUN4C */
#if defined(SUN4)
cmp %g4, CPU_SUN4
bne 2f
#if defined(MMU_3L)
set AC_IDPROM+1, %l3
lduba [%l3] ASI_CONTROL, %l3
- cmp %l3, SUN4_400
+ cmp %l3, 0x24 ! XXX - SUN4_400
bne no_3mmu
- nop
add %l0, 2, %l0 ! get to proper half-word in RG space
add %l1, 2, %l1
lduha [%l0] ASI_REGMAP, %l4 ! regmap[highva] = regmap[lowva];
@@ -2775,6 +3491,7 @@ no_3mmu:
add %l3, %l0, %l0 ! (and lowva += segsz)
remap_done:
+
/*
* Now map the interrupt enable register and clear any interrupts,
* enabling NMIs. Note that we will not take NMIs until we change
@@ -2792,8 +3509,8 @@ remap_done:
nop; nop ! paranoia
stb %l1, [%l0]
b,a startmap_done
-#endif /* SUN4 */
2:
+#endif /* SUN4 */
#if defined(SUN4M)
cmp %g4, CPU_SUN4M ! skip for sun4m!
bne 3f
@@ -2860,10 +3577,10 @@ remap_notvik:
srl %l1, 22, %o2 ! note: 22 == RGSHIFT - 2
add %o1, %o2, %o1
sta %l4, [%o1] ASI_BYPASS
- b,a startmap_done
+ !b,a startmap_done
-#endif /* SUN4M */
3:
+#endif /* SUN4M */
! botch! We should blow up.
startmap_done:
@@ -2911,6 +3628,10 @@ startmap_done:
wr %l0, PSR_ET, %psr
nop; nop; nop
+ /* Export actual trapbase */
+ sethi %hi(_trapbase), %o0
+ st %g6, [%o0+%lo(_trapbase)]
+
/*
* Step 2: clear BSS. This may just be paranoia; the boot
* loader might already do it for us; but what the hell.
@@ -2941,7 +3662,7 @@ startmap_done:
call init_tables
st %o0, [%o1 + %lo(_nwindows)]
-#ifdef SUN4
+#if defined(SUN4)
/*
* Some sun4 models have fewer than 8 windows. For extra
* speed, we do not need to save/restore those windows
@@ -2957,18 +3678,15 @@ noplab: nop
set wb1, %l0
st %l1, [%l0 + 6*4]
st %l1, [%l0 + 7*4]
- set wb2, %l0
- st %l1, [%l0 + 6*4]
- st %l1, [%l0 + 7*4]
1:
#endif
/*
* Step 4: change the trap base register, now that our trap handlers
* will function (they need the tables we just set up).
+ * This depends on the fact that bzero does not use %g6.
*/
- set _trapbase, %l0
- wr %l0, 0, %tbr
+ wr %g6, 0, %tbr
nop; nop; nop ! paranoia
@@ -2984,17 +3702,7 @@ noplab: nop
*/
call _main
clr %o0 ! our frame arg is ignored
-
- /*
- * Here we finish up as in syscall, but simplified. We need to
- * fiddle pc and npc a bit, as execve() / setregs() have only set
- * npc, in anticipation that trap.c will advance past the trap
- * instruction; but we bypass that, so we must do it manually.
- */
- mov PSR_S, %l0 ! user psr (no need to load it)
- ld [%sp + CCFSZ + 8], %l1 ! pc = npc from execve
- b init_syscall_ret
- add %l1, 4, %l2 ! npc = pc+4
+ /*NOTREACHED*/
/*
* The following code is copied to the top of the user stack when each
@@ -3239,20 +3947,20 @@ _svr4_esigcode:
/*
* Primitives
- */
+ */
#ifdef GPROF
.globl mcount
#define ENTRY(x) \
- .globl _##x; _##x: ; \
+ .globl _/**/x; _/**/x: ; \
save %sp, -CCFSZ, %sp; \
call mcount; \
nop; \
restore
#else
-#define ENTRY(x) .globl _##x; _##x:
+#define ENTRY(x) .globl _/**/x; _/**/x:
#endif
-#define ALTENTRY(x) .globl _##x; _##x:
+#define ALTENTRY(x) .globl _/**/x; _/**/x:
/*
* getfp() - get stack frame pointer
@@ -3287,7 +3995,7 @@ ENTRY(copyinstr)
call _panic
or %lo(2f), %o0, %o0
2: .asciz "copyinstr"
- ALIGN
+ _ALIGN
/*
* copyoutstr(fromaddr, toaddr, maxlength, &lencopied)
@@ -3315,7 +4023,7 @@ ENTRY(copyoutstr)
call _panic
or %lo(2f), %o0, %o0
2: .asciz "copyoutstr"
- ALIGN
+ _ALIGN
Lcsdocopy:
! sethi %hi(_cpcb), %o4 ! (done earlier)
@@ -3395,10 +4103,10 @@ ENTRY(copystr)
or %lo(5f), %o0, %o0
5:
.asciz "copystr"
- ALIGN
+ _ALIGN
#endif
-/*
+/*
* Copyin(src, dst, len)
*
* Copy specified amount of data from user space into the kernel.
@@ -3458,25 +4166,6 @@ Lcopyfault:
jmp %g7 + 8
mov EFAULT, %o0
-/* Force the cpu to complete all pending store operations
- * and flush it's on chip write buffers. This is the sun4c/sun4
- * version and will only be called when servicing a memory
- * error so that we do not loop indefinately.
- */
-ENTRY(sun4c_sun4_storebuf_flush)
- set AC_CONTEXT, %o1
- lduba [%o1] ASI_CONTROL, %o0 ! load context reg
- stba %o0, [%o1] ASI_CONTROL ! store same value, flush begins
- nop
- nop
- nop
- /* Pipeline is now clear */
- nop
- nop
- nop
- /* Store buffer is empty, safe to return now. */
- retl
- nop
/*
* Write all user windows presently in the CPU back to the user's stack.
@@ -3615,7 +4304,7 @@ ENTRY(switchexit)
/*
* When no processes are on the runq, switch
- * idles here watiing for something to come ready.
+ * idles here waiting for something to come ready.
* The registers are set up as noted above.
*/
.globl idle
@@ -3644,7 +4333,7 @@ Lsw_panic_srun:
1: .asciz "switch rq"
2: .asciz "switch wchan"
3: .asciz "switch SRUN"
- ALIGN
+ _ALIGN
/*
* cpu_switch() picks a process to run and runs it, saving the current
@@ -3698,7 +4387,7 @@ ENTRY(cpu_switch)
st %g0, [%g7 + %lo(_curproc)] ! curproc = NULL;
wr %g1, 0, %psr ! (void) spl0();
nop; nop; nop ! paranoia
- wr %g1, PIL_CLOCK <<8 , %psr ! (void) splclock();
+ wr %g1, PIL_CLOCK << 8 , %psr ! (void) splclock();
Lsw_scan:
nop; nop; nop ! paranoia
@@ -3877,7 +4566,7 @@ Lsw_load:
ld [%o3 + VM_PMAP_CTX], %o0! if (vm->vm_pmap.pm_ctx != NULL)
tst %o0
bnz,a Lsw_havectx ! goto havecontext;
- ld [%o3 + VM_PMAP_CTXNUM], %o0
+ ld [%o3 + VM_PMAP_CTXNUM], %o0 ! load context number
/* p does not have a context: call ctx_alloc to get one */
save %sp, -CCFSZ, %sp
@@ -3888,7 +4577,7 @@ Lsw_load:
/* p does have a context: just switch to it */
Lsw_havectx:
-! ld [%o3 + VM_PMAP_CTXNUM], %o0 ! (done in delay slot)
+ ! context is in %o0
#if (defined(SUN4) || defined(SUN4C)) && defined(SUN4M)
sethi %hi(_cputyp), %o1 ! what cpu are we running on?
ld [%o1 + %lo(_cputyp)], %o1
@@ -3898,16 +4587,14 @@ Lsw_havectx:
#endif
#if defined(SUN4) || defined(SUN4C)
set AC_CONTEXT, %o1
- stba %o0, [%o1] ASI_CONTROL ! setcontext(vm->vm_pmap.pm_ctxnum);
retl
- nop
+ stba %o0, [%o1] ASI_CONTROL ! setcontext(vm->vm_pmap.pm_ctxnum);
#endif
1:
#if defined(SUN4M)
set SRMMU_CXR, %o1
- stba %o0, [%o1] ASI_SRMMU ! setcontext(vm->vm_pmap.pm_ctxnum);
retl
- nop
+ sta %o0, [%o1] ASI_SRMMU ! setcontext(vm->vm_pmap.pm_ctxnum);
#endif
Lsw_sameproc:
@@ -3923,21 +4610,10 @@ Lsw_sameproc:
/*
* Snapshot the current process so that stack frames are up to date.
- * This is called from two places:
- * - just before a crash dump, for the stack update;
- * - in cpu_fork(), before copying the kernel stack.
- * In the latter case the pcb and stack will be copied to the child,
- * and the child will be made runnable. Eventually switch() will run
- * it. When it does, we want its pcb_pc set so that we can appear
- * to return 1 from cpu_fork(), so we store the current sp and psr
- * in the given pcb, and set its pcb_pc to our return-1 code (offset
- * by -8 due to call/ret conventions). This is not useful in the crash
- * dump code but it is easiest to do it anyway.
+ * Only used just before a crash dump.
*/
ENTRY(snapshot)
st %o6, [%o0 + PCB_SP] ! save sp
- set 1f - 8, %o1 ! set child-return pc
- st %o1, [%o0 + PCB_PC]
rd %psr, %o1 ! save psr
st %o1, [%o0 + PCB_PSR]
@@ -3946,13 +4622,34 @@ ENTRY(snapshot)
* 7 of each. Minor tweak: the 7th restore is
* done after a ret.
*/
-wb2: SAVE; SAVE; SAVE; SAVE; SAVE; SAVE; SAVE
+ SAVE; SAVE; SAVE; SAVE; SAVE; SAVE; SAVE
restore; restore; restore; restore; restore; restore; ret; restore
-1: /* this is reached only after a child gets chosen in switch() */
- mov 1, %i0 ! return 1 from cpu_fork
- ret
- restore
+
+/*
+ * cpu_set_kpc() and cpu_fork() arrange for proc_trampoline() to run
+ * after after a process gets chosen in switch(). The stack frame will
+ * contain a function pointer in %l0, and an argument to pass to it in %l2.
+ *
+ * If the function *(%l0) returns, we arrange for an immediate return
+ * to user mode. This happens in two known cases: after execve(2) of init,
+ * and when returning a child to user mode after a fork(2).
+ */
+ENTRY(proc_trampoline)
+ call %l0 ! re-use current frame
+ mov %l1, %o0
+
+ /*
+ * Here we finish up as in syscall, but simplified. We need to
+ * fiddle pc and npc a bit, as execve() / setregs() /cpu_set_kpc()
+ * have only set npc, in anticipation that trap.c will advance past
+ * the trap instruction; but we bypass that, so we must do it manually.
+ */
+ mov PSR_S, %l0 ! user psr (no need to load it)
+ !?wr %g0, 2, %wim ! %wim = 2
+ ld [%sp + CCFSZ + 8], %l1 ! pc = tf->tf_npc from execve/fork
+ b return_from_syscall
+ add %l1, 4, %l2 ! npc = pc+4
/*
* {fu,su}{,i}{byte,word}
@@ -4109,10 +4806,6 @@ ENTRY(subyte)
*/
ENTRY(probeget)
! %o0 = addr, %o1 = (1,2,4)
- set KERNBASE, %o2
- cmp %o0, %o2 ! if addr < KERNBASE
- blu Lfsbadaddr ! go return error
- EMPTY
sethi %hi(_cpcb), %o2
ld [%o2 + %lo(_cpcb)], %o2 ! cpcb->pcb_onfault = Lfserr;
set Lfserr, %o5
@@ -4136,10 +4829,6 @@ ENTRY(probeget)
*/
ENTRY(probeset)
! %o0 = addr, %o1 = (1,2,4), %o2 = val
- set KERNBASE, %o3
- cmp %o0, %o3 ! if addr < KERNBASE
- blu Lfsbadaddr ! go return error
- EMPTY
sethi %hi(_cpcb), %o3
ld [%o3 + %lo(_cpcb)], %o3 ! cpcb->pcb_onfault = Lfserr;
set Lfserr, %o5
@@ -4702,10 +5391,36 @@ ENTRY(loadfpstate)
* ienab_bic(bic) int bic;
*
* Set and clear bits in the interrupt register.
+ */
+
+#if defined(SUN4M) && (defined(SUN4) || defined(SUN4C))
+ENTRY(ienab_bis)
+ sethi %hi(_cputyp), %o1
+ ld [%o1 + %lo(_cputyp)], %o1
+ cmp %o1, CPU_SUN4M
+ be,a _ienab_bis_4m
+ nop
+ b,a _ienab_bis_4c
+
+ENTRY(ienab_bic)
+ sethi %hi(_cputyp), %o1
+ ld [%o1 + %lo(_cputyp)], %o1
+ cmp %o1, CPU_SUN4M
+ be,a _ienab_bic_4m
+ nop
+ b,a _ienab_bic_4c
+#endif
+
+#if defined(SUN4) || defined(SUN4C)
+/*
* Since there are no read-modify-write instructions for this,
* and one of the interrupts is nonmaskable, we must disable traps.
*/
+#if defined(SUN4M)
+ENTRY(ienab_bis_4c)
+#else
ENTRY(ienab_bis)
+#endif
! %o0 = bits to set
rd %psr, %o2
wr %o2, PSR_ET, %psr ! disable traps
@@ -4719,7 +5434,11 @@ ENTRY(ienab_bis)
retl
nop
+#if defined(SUN4M)
+ENTRY(ienab_bic_4c)
+#else
ENTRY(ienab_bic)
+#endif
! %o0 = bits to clear
rd %psr, %o2
wr %o2, PSR_ET, %psr ! disable traps
@@ -4732,6 +5451,47 @@ ENTRY(ienab_bic)
nop
retl
nop
+#endif
+
+#if defined(SUN4M)
+/*
+ * sun4m has separate registers for clearing/setting the interrupt mask.
+ */
+#if defined(SUN4) || defined(SUN4C)
+ENTRY(ienab_bis_4m)
+#else
+ENTRY(ienab_bis)
+#endif
+ set ICR_SI_SET, %o1
+ retl
+ st %o0, [%o1]
+
+#if defined(SUN4) || defined(SUN4C)
+ENTRY(ienab_bic_4m)
+#else
+ENTRY(ienab_bic)
+#endif
+ set ICR_SI_CLR, %o1
+ retl
+ st %o0, [%o1]
+
+/*
+ * raise(cpu, level)
+ */
+ENTRY(raise)
+ ! *(ICR_PI_SET + cpu*_MAXNBPG) = PINTR_SINTRLEV(level)
+ sethi %hi(1 << 16), %o2
+ sll %o2, %o1, %o2
+ set ICR_PI_SET, %o1
+ set _MAXNBPG, %o3
+1:
+ subcc %o0, 1, %o0
+ bpos,a 1b
+ add %o1, %o3, %o1
+ retl
+ st %o2, [%o1]
+
+#endif /* SUN4M */
/*
* ffs(), using table lookup.
@@ -4805,15 +5565,14 @@ ENTRY(ffs)
* Vol 33 No 1.
*/
.data
- .globl __randseed
-__randseed:
+randseed:
.word 1
.text
ENTRY(random)
sethi %hi(16807), %o1
wr %o1, %lo(16807), %y
- sethi %hi(__randseed), %g1
- ld [%g1 + %lo(__randseed)], %o0
+ sethi %hi(randseed), %g1
+ ld [%g1 + %lo(randseed)], %o0
andcc %g0, 0, %o2
mulscc %o2, %o0, %o2
mulscc %o2, %o0, %o2
@@ -4842,13 +5601,13 @@ ENTRY(random)
bneg 1f
sethi %hi(0x7fffffff), %o1
retl
- st %o0, [%g1 + %lo(__randseed)]
+ st %o0, [%g1 + %lo(randseed)]
1:
or %o1, %lo(0x7fffffff), %o1
add %o0, 1, %o0
and %o1, %o0, %o0
retl
- st %o0, [%g1 + %lo(__randseed)]
+ st %o0, [%g1 + %lo(randseed)]
/*
* void lo_microtime(struct timeval *tv)
@@ -4872,37 +5631,86 @@ ENTRY(microtime)
#endif
sethi %hi(_time), %g2
sethi %hi(TIMERREG_VA), %g3
+
+/* blech, sun4m has microsecond counter at a different location */
+#if (defined(SUN4) || defined(SUN4C)) && !defined(SUN4M)
+#define r_aardvark %lo(TIMERREG_VA)
+#elif !(defined(SUN4) || defined(SUN4C)) && defined(SUN4M)
+#define r_aardvark %lo(TIMERREG_VA) + 4
+#else
+ sethi %hi(_cputyp), %g4
+ ld [%g4 + %lo(_cputyp)], %g4
+ or %g3, %lo(TIMERREG_VA), %g3
+ cmp %g4, CPU_SUN4M
+ be,a 1f
+ add %g3, 4, %g3
1:
+#define r_aardvark 0
+#endif
+2:
ldd [%g2+%lo(_time)], %o2 ! time.tv_sec & time.tv_usec
- ld [%g3+%lo(TIMERREG_VA)], %o4 ! usec counter
+ ld [%g3+r_aardvark], %o4 ! usec counter
ldd [%g2+%lo(_time)], %g4 ! see if time values changed
cmp %g4, %o2
- bne 1b ! if time.tv_sec changed
+ bne 2b ! if time.tv_sec changed
cmp %g5, %o3
- bne 1b ! if time.tv_usec changed
+ bne 2b ! if time.tv_usec changed
tst %o4
- bpos 2f ! reached limit?
+ bpos 3f ! reached limit?
srl %o4, TMR_SHIFT, %o4 ! convert counter to usec
sethi %hi(_tick), %g4 ! bump usec by 1 tick
ld [%g4+%lo(_tick)], %o1
set TMR_MASK, %g5
add %o1, %o3, %o3
and %o4, %g5, %o4
-2:
+3:
add %o4, %o3, %o3
set 1000000, %g5 ! normalize usec value
cmp %o3, %g5
- bl,a 3f
+ bl,a 4f
st %o2, [%o0] ! (should be able to std here)
add %o2, 1, %o2 ! overflow
sub %o3, %g5, %o3
st %o2, [%o0] ! (should be able to std here)
-3:
+4:
retl
st %o3, [%o0+4]
-#if defined(KGDB) || defined(DDB)
+/*
+ * delay function
+ *
+ * void delay(N) -- delay N microseconds
+ *
+ * Register usage: %o0 = "N" number of usecs to go (counts down to zero)
+ * %o1 = "timerblurb" (stays constant)
+ * %o2 = counter for 1 usec (counts down from %o1 to zero)
+ *
+ */
+
+ENTRY(delay) ! %o0 = n
+ subcc %o0, %g0, %g0
+ be 2f
+
+ sethi %hi(_timerblurb), %o1
+ ld [%o1 + %lo(_timerblurb)], %o1 ! %o1 = timerblurb
+
+ addcc %o1, %g0, %o2 ! %o2 = cntr (start @ %o1), clear CCs
+ ! first time through only
+
+ ! delay 1 usec
+1: bne 1b ! come back here if not done
+ subcc %o2, 1, %o2 ! %o2 = %o2 - 1 [delay slot]
+
+ subcc %o0, 1, %o0 ! %o0 = %o0 - 1
+ bne 1b ! done yet?
+ addcc %o1, %g0, %o2 ! reinit %o2 and CCs [delay slot]
+ ! harmless if not branching
+2:
+ retl ! return
+ nop ! [delay slot]
+
+#if defined(KGDB) || defined(DDB) || defined(DIAGNOSTIC)
/*
* Write all windows (user or otherwise), except the current one.
*
@@ -4939,7 +5747,7 @@ ENTRY(setjmp)
Lpanic_ljmp:
.asciz "longjmp botch"
- ALIGN
+ _ALIGN
ENTRY(longjmp)
addcc %o1, %g0, %g6 ! compute v ? v : 1 in a global register
@@ -4968,7 +5776,7 @@ Llongjmpbotch:
bge,a 3f
mov %o2, %sp ! it is OK, put it in place
b,a Llongjmpbotch
-3:
+3:
jmp %o3 + 8 ! success, return %g6
mov %g6, %o0
@@ -5005,7 +5813,7 @@ _intrnames:
.asciz "lev13"
.asciz "prof"
_eintrnames:
- ALIGN
+ _ALIGN
_intrcnt:
.skip 4*15
_eintrcnt:
diff --git a/sys/arch/sparc/sparc/locore2.c b/sys/arch/sparc/sparc/locore2.c
index 614dc6d707b..c44c604b200 100644
--- a/sys/arch/sparc/sparc/locore2.c
+++ b/sys/arch/sparc/sparc/locore2.c
@@ -1,4 +1,4 @@
-/* $NetBSD: locore2.c,v 1.5 1994/11/20 20:54:28 deraadt Exp $ */
+/* $NetBSD: locore2.c,v 1.6 1996/03/14 21:09:15 christos Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -50,9 +50,12 @@
*/
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/proc.h>
#include <sys/resourcevar.h>
+#include <machine/cpu.h>
+
int whichqs;
/*
@@ -81,6 +84,7 @@ setrunqueue(p)
* Remove process p from its run queue, which should be the one
* indicated by its priority. Calls should be made at splstatclock().
*/
+void
remrq(p)
register struct proc *p;
{
diff --git a/sys/arch/sparc/sparc/machdep.c b/sys/arch/sparc/sparc/machdep.c
index 9529ea31b0e..03dc4fc1a3b 100644
--- a/sys/arch/sparc/sparc/machdep.c
+++ b/sys/arch/sparc/sparc/machdep.c
@@ -1,4 +1,4 @@
-/* $NetBSD: machdep.c,v 1.45 1995/05/16 21:16:37 mycroft Exp $ */
+/* $NetBSD: machdep.c,v 1.64 1996/05/19 04:12:56 mrg Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -75,13 +75,16 @@
#include <sys/exec.h>
#include <sys/sysctl.h>
-#include <machine/bsd_openprom.h>
+#include <vm/vm.h>
+#include <vm/vm_kern.h>
+#include <vm/vm_page.h>
+
#include <machine/autoconf.h>
#include <machine/frame.h>
#include <machine/cpu.h>
-
-#include <vm/vm_kern.h>
-#include <vm/vm_page.h>
+#include <machine/pmap.h>
+#include <machine/oldmon.h>
+#include <machine/bsd_openprom.h>
#include <sparc/sparc/asm.h>
#include <sparc/sparc/cache.h>
@@ -109,7 +112,7 @@ int physmem;
extern struct msgbuf msgbuf;
struct msgbuf *msgbufp = &msgbuf;
-int msgbufmapped = 0; /* maps late; so that printf works early */
+int msgbufmapped = 0; /* not mapped until pmap_bootstrap */
/*
* safepri is a safe priority for sleep to set for a spin-wait
@@ -125,7 +128,9 @@ vm_offset_t dvma_base, dvma_end;
struct map *dvmamap;
static int ndvmamap; /* # of entries in dvmamap */
-caddr_t allocsys();
+caddr_t allocsys __P((caddr_t));
+void dumpsys __P((void));
+void stackdump __P((void));
/*
* Machine-dependent startup code
@@ -166,8 +171,10 @@ cpu_startup()
* and then give everything true virtual addresses.
*/
sz = (int)allocsys((caddr_t)0);
+
if ((v = (caddr_t)kmem_alloc(kernel_map, round_page(sz))) == 0)
panic("startup: no room for tables");
+
if (allocsys(v) - v != sz)
panic("startup: table size inconsistency");
@@ -176,12 +183,15 @@ cpu_startup()
* in that they usually occupy more virtual memory than physical.
*/
size = MAXBSIZE * nbuf;
+
buffer_map = kmem_suballoc(kernel_map, (vm_offset_t *)&buffers,
&maxaddr, size, TRUE);
+
minaddr = (vm_offset_t)buffers;
if (vm_map_find(buffer_map, vm_object_allocate(size), (vm_offset_t)0,
&minaddr, size, FALSE) != KERN_SUCCESS)
panic("startup: cannot allocate buffers");
+
base = bufpages / nbuf;
residual = bufpages % nbuf;
if (base >= MAXBSIZE) {
@@ -189,6 +199,7 @@ cpu_startup()
base = MAXBSIZE;
residual = 0;
}
+
for (i = 0; i < nbuf; i++) {
vm_size_t curbufsize;
vm_offset_t curbuf;
@@ -211,23 +222,15 @@ cpu_startup()
* limits the number of processes exec'ing at any time.
*/
exec_map = kmem_suballoc(kernel_map, &minaddr, &maxaddr,
- 16*NCARGS, TRUE);
+ 16*NCARGS, TRUE);
/*
* Allocate a map for physio. Others use a submap of the kernel
* map, but we want one completely separate, even though it uses
* the same pmap.
*/
-#if defined(SUN4M) && 0
- if (cputyp == CPU_SUN4M) {
- dvma_base = DVMA4M_BASE;
- dvma_end = (vm_offset_t)(0 - NBPG); /* want 4BG, but cant express */
- } else
-#endif
- {
- dvma_base = DVMA_BASE;
- dvma_end = DVMA_END;
- }
+ dvma_base = CPU_ISSUN4M ? DVMA4M_BASE : DVMA_BASE;
+ dvma_end = CPU_ISSUN4M ? DVMA4M_END : DVMA_END;
phys_map = vm_map_create(pmap_kernel(), dvma_base, dvma_end, 1);
if (phys_map == NULL)
panic("unable to create DVMA map");
@@ -261,7 +264,7 @@ cpu_startup()
#ifdef DEBUG
pmapdebug = opmapdebug;
#endif
- printf("avail mem = %d\n", ptoa(cnt.v_free_count));
+ printf("avail mem = %ld\n", ptoa(cnt.v_free_count));
printf("using %d buffers containing %d bytes of memory\n",
nbuf, bufpages * CLBYTES);
@@ -280,7 +283,7 @@ cpu_startup()
*/
pmap_enter(pmap_kernel(), MSGBUF_VA, 0x0, VM_PROT_READ|VM_PROT_WRITE, 1);
- if (cputyp == CPU_SUN4)
+ if (CPU_ISSUN4)
msgbufp = (struct msgbuf *)(MSGBUF_VA + 4096);
else
msgbufp = (struct msgbuf *)MSGBUF_VA;
@@ -320,7 +323,7 @@ allocsys(v)
valloc(msghdrs, struct msg, msginfo.msgtql);
valloc(msqids, struct msqid_ds, msginfo.msgmni);
#endif
-
+
/*
* Determine how many buffers to allocate (enough to
* hold 5% of total physical memory, but at least 16).
@@ -386,9 +389,9 @@ setregs(p, pack, stack, retval)
}
bzero((caddr_t)tf, sizeof *tf);
tf->tf_psr = psr;
- tf->tf_pc = pack->ep_entry & ~3;
+ tf->tf_npc = pack->ep_entry & ~3;
tf->tf_global[1] = (int)PS_STRINGS;
- tf->tf_global[2] = tf->tf_global[7] = tf->tf_npc = tf->tf_pc+4;
+ tf->tf_global[2] = tf->tf_global[7] = tf->tf_npc;
stack -= sizeof(struct rwindow);
tf->tf_out[6] = stack;
retval[1] = 0;
@@ -417,6 +420,7 @@ struct sigframe {
/*
* machine dependent system variables.
*/
+int
cpu_sysctl(name, namelen, oldp, oldlenp, newp, newlen, p)
int *name;
u_int namelen;
@@ -474,10 +478,10 @@ sendsig(catcher, sig, mask, code)
#ifdef DEBUG
if ((sigdebug & SDB_KSTACK) && p->p_pid == sigpid)
- printf("sendsig: %s[%d] sig %d newusp %x scp %x\n",
+ printf("sendsig: %s[%d] sig %d newusp %p scp %p\n",
p->p_comm, p->p_pid, sig, fp, &fp->sf_sc);
#endif
- /*
+ /*
* Now set up the signal frame. We build it in kernel space
* and then copy it out. We probably ought to just build it
* directly in user space....
@@ -527,7 +531,7 @@ sendsig(catcher, sig, mask, code)
}
#ifdef DEBUG
if (sigdebug & SDB_FOLLOW)
- printf("sendsig: %s[%d] sig %d scp %x\n",
+ printf("sendsig: %s[%d] sig %d scp %p\n",
p->p_comm, p->p_pid, sig, &fp->sf_sc);
#endif
/*
@@ -580,7 +584,7 @@ sys_sigreturn(p, v, retval)
sigexit(p, SIGILL);
#ifdef DEBUG
if (sigdebug & SDB_FOLLOW)
- printf("sigreturn: %s[%d], sigcntxp %x\n",
+ printf("sigreturn: %s[%d], sigcntxp %p\n",
p->p_comm, p->p_pid, SCARG(uap, sigcntxp));
#endif
scp = SCARG(uap, sigcntxp);
@@ -617,9 +621,12 @@ boot(howto)
{
int i;
static char str[4]; /* room for "-sd\0" */
- extern volatile void romhalt(void);
- extern volatile void romboot(char *);
- extern struct promvec *promvec;
+ extern int cold;
+
+ if (cold) {
+ printf("halted\n\n");
+ romhalt();
+ }
fb_unblank();
boothowto = howto;
@@ -641,17 +648,15 @@ boot(howto)
resettodr();
}
(void) splhigh(); /* ??? */
-
if (howto & RB_HALT) {
+ doshutdownhooks();
printf("halted\n\n");
-#if defined(SUN4M) || defined(SUN4C)
- if (cputyp==CPU_SUN4M || cputyp==CPU_SUN4C)
- *promvec->pv_synchook = NULL;
-#endif
romhalt();
}
if (howto & RB_DUMP)
dumpsys();
+
+ doshutdownhooks();
printf("rebooting\n\n");
i = 1;
if (howto & RB_SINGLE)
@@ -663,10 +668,6 @@ boot(howto)
str[i] = 0;
} else
str[0] = 0;
-#if defined(SUN4M) || defined(SUN4C)
- if (cputyp==CPU_SUN4M || cputyp==CPU_SUN4C)
- *promvec->pv_synchook = NULL;
-#endif
romboot(str);
/*NOTREACHED*/
}
@@ -675,6 +676,7 @@ u_long dumpmag = 0x8fca0101; /* magic number for savecore */
int dumpsize = 0; /* also for savecore */
long dumplo = 0;
+void
dumpconf()
{
register int nblks, nmem;
@@ -729,6 +731,7 @@ reserve_dumppages(p)
/*
* Write a crash dump.
*/
+void
dumpsys()
{
register int psize;
@@ -755,7 +758,7 @@ dumpsys()
dumpconf();
if (dumplo < 0)
return;
- printf("\ndumping to dev %x, offset %d\n", dumpdev, dumplo);
+ printf("\ndumping to dev %x, offset %ld\n", dumpdev, dumplo);
psize = (*bdevsw[major(dumpdev)].d_psize)(dumpdev);
printf("dump ");
@@ -767,10 +770,17 @@ dumpsys()
dump = bdevsw[major(dumpdev)].d_dump;
for (mp = pmemarr, nmem = npmemarr; --nmem >= 0; mp++) {
- register unsigned i, n;
+ register unsigned i = 0, n;
register maddr = mp->addr;
- for (i = 0; i < mp->len; i += n) {
+ if (maddr == 0) {
+ /* Skip first page at physical address 0 */
+ maddr += NBPG;
+ i += NBPG;
+ blkno += btodb(NBPG);
+ }
+
+ for (; i < mp->len; i += n) {
n = mp->len - i;
if (n > BYTES_PER_DUMP)
n = BYTES_PER_DUMP;
@@ -825,15 +835,16 @@ dumpsys()
* get the fp and dump the stack as best we can. don't leave the
* current stack page
*/
+void
stackdump()
{
- struct frame *fp = (struct frame *)getfp(), *sfp;
+ struct frame *fp = getfp(), *sfp;
sfp = fp;
- printf("Frame pointer is at 0x%x\n", fp);
+ printf("Frame pointer is at %p\n", fp);
printf("Call traceback:\n");
while (fp && ((u_long)fp >> PGSHIFT) == ((u_long)sfp >> PGSHIFT)) {
- printf(" pc = %x args = (%x, %x, %x, %x, %x, %x) fp = %x\n",
+ printf(" pc = %x args = (%x, %x, %x, %x, %x, %x, %x) fp = %p\n",
fp->fr_pc, fp->fr_arg[0], fp->fr_arg[1], fp->fr_arg[2],
fp->fr_arg[3], fp->fr_arg[4], fp->fr_arg[5], fp->fr_arg[6],
fp->fr_fp);
@@ -846,28 +857,28 @@ int bt2pmt[] = {
PMAP_OBIO,
PMAP_VME16,
PMAP_VME32,
- PMAP_OBIO,
PMAP_OBIO
};
/*
* Map an I/O device given physical address and size in bytes, e.g.,
*
- * mydev = (struct mydev *)mapdev(myioaddr, 0, 0,
- * sizeof(struct mydev), pmtype);
- *
+ * mydev = (struct mydev *)mapdev(myioaddr, 0,
+ * 0, sizeof(struct mydev), pmtype);
+ *
* See also machine/autoconf.h.
*/
void *
-mapdev(rr, virt, offset, size, bustype)
- register struct rom_reg *rr;
+mapdev(phys, virt, offset, size, bustype)
+ register struct rom_reg *phys;
register int offset, virt, size;
register int bustype;
{
- register vm_offset_t v, pa;
+ register vm_offset_t v;
+ register vm_offset_t pa;
register void *ret;
static vm_offset_t iobase;
- int pmtype;
+ unsigned int pmtype;
if (iobase == NULL)
iobase = IODEV_BASE;
@@ -883,19 +894,17 @@ mapdev(rr, virt, offset, size, bustype)
if (iobase > IODEV_END) /* unlikely */
panic("mapiodev");
}
- ret = (void *)(v | (((u_long)rr->rr_paddr + offset) & PGOFSET));
+ ret = (void *)(v | (((u_long)phys->rr_paddr + offset) & PGOFSET));
/* note: preserve page offset */
- pa = trunc_page(rr->rr_paddr + offset);
-#ifdef notyet
- pmtype = (cputyp == CPU_SUN4M)
- ? (rr->rr_iospace << PMAP_SHFT4M)
+
+ pa = trunc_page(phys->rr_paddr + offset);
+ pmtype = (CPU_ISSUN4M)
+ ? (phys->rr_iospace << PMAP_SHFT4M)
: bt2pmt[bustype];
-#else
- pmtype = bt2pmt[bustype];
-#endif
+
do {
pmap_enter(pmap_kernel(), v, pa | pmtype | PMAP_NC,
- VM_PROT_READ | VM_PROT_WRITE, 1);
+ VM_PROT_READ | VM_PROT_WRITE, 1);
v += PAGE_SIZE;
pa += PAGE_SIZE;
} while ((size -= PAGE_SIZE) > 0);
@@ -923,12 +932,10 @@ oldmon_w_trace(va)
u_long va;
{
u_long stop;
- extern u_long *par_err_reg;
- volatile u_long *memreg = (u_long *) par_err_reg;
struct frame *fp;
if (curproc)
- printf("curproc = %x, pid %d\n", curproc, curproc->p_pid);
+ printf("curproc = %p, pid %d\n", curproc, curproc->p_pid);
else
printf("no curproc\n");
@@ -939,19 +946,19 @@ oldmon_w_trace(va)
#define round_up(x) (( (x) + (NBPG-1) ) & (~(NBPG-1)) )
- printf("\nstack trace with sp = %x\n", va);
+ printf("\nstack trace with sp = %lx\n", va);
stop = round_up(va);
- printf("stop at %x\n", stop);
+ printf("stop at %lx\n", stop);
fp = (struct frame *) va;
while (round_up((u_long) fp) == stop) {
- printf(" %x(%x, %x, %x, %x, %x, %x) fp %x\n", fp->fr_pc,
- fp->fr_arg[0], fp->fr_arg[1], fp->fr_arg[2], fp->fr_arg[3],
+ printf(" %x(%x, %x, %x, %x, %x, %x, %x) fp %p\n", fp->fr_pc,
+ fp->fr_arg[0], fp->fr_arg[1], fp->fr_arg[2], fp->fr_arg[3],
fp->fr_arg[4], fp->fr_arg[5], fp->fr_arg[6], fp->fr_fp);
fp = fp->fr_fp;
if (fp == NULL)
break;
}
- printf("end of stack trace\n");
+ printf("end of stack trace\n");
}
void
@@ -968,7 +975,7 @@ oldmon_w_cmd(va, ar)
printf("w: case 4\n");
break;
default:
- printf("w: unknown case %d\n", va);
+ printf("w: unknown case %ld\n", va);
break;
}
break;
@@ -981,15 +988,21 @@ oldmon_w_cmd(va, ar)
}
#endif /* SUN4 */
-u_int
+int
ldcontrolb(addr)
caddr_t addr;
{
struct pcb *xpcb;
extern struct user *proc0paddr;
- u_long saveonfault, res;
+ u_long saveonfault;
+ int res;
int s;
+ if (CPU_ISSUN4M) {
+ printf("warning: ldcontrolb called in sun4m\n");
+ return 0;
+ }
+
s = splhigh();
if (curproc == NULL)
xpcb = (struct pcb *)proc0paddr;
@@ -997,7 +1010,7 @@ caddr_t addr;
xpcb = &curproc->p_addr->u_pcb;
saveonfault = (u_long)xpcb->pcb_onfault;
- res = xldcontrolb(addr, xpcb);
+ res = xldcontrolb(addr, xpcb);
xpcb->pcb_onfault = (caddr_t)saveonfault;
splx(s);
diff --git a/sys/arch/sparc/sparc/mem.c b/sys/arch/sparc/sparc/mem.c
index bef3cf01e07..ff531dca56f 100644
--- a/sys/arch/sparc/sparc/mem.c
+++ b/sys/arch/sparc/sparc/mem.c
@@ -1,4 +1,4 @@
-/* $NetBSD: mem.c,v 1.11 1995/05/29 23:57:16 pk Exp $ */
+/* $NetBSD: mem.c,v 1.13 1996/03/30 21:12:16 christos Exp $ */
/*
* Copyright (c) 1988 University of Utah.
@@ -45,14 +45,16 @@
*/
#include <sys/param.h>
-#include <sys/conf.h>
#include <sys/buf.h>
#include <sys/systm.h>
#include <sys/uio.h>
#include <sys/malloc.h>
+#include <sys/proc.h>
+#include <sys/conf.h>
#include <sparc/sparc/vaddrs.h>
#include <machine/eeprom.h>
+#include <machine/conf.h>
#include <vm/vm.h>
@@ -62,9 +64,10 @@ caddr_t zeropage;
/*ARGSUSED*/
int
-mmopen(dev, flag, mode)
+mmopen(dev, flag, mode, p)
dev_t dev;
int flag, mode;
+ struct proc *p;
{
return (0);
@@ -72,9 +75,10 @@ mmopen(dev, flag, mode)
/*ARGSUSED*/
int
-mmclose(dev, flag, mode)
+mmclose(dev, flag, mode, p)
dev_t dev;
int flag, mode;
+ struct proc *p;
{
return (0);
diff --git a/sys/arch/sparc/sparc/memreg.c b/sys/arch/sparc/sparc/memreg.c
index 8c2a3e95b6a..a2b5bf67ddc 100644
--- a/sys/arch/sparc/sparc/memreg.c
+++ b/sys/arch/sparc/sparc/memreg.c
@@ -1,4 +1,4 @@
-/* $NetBSD: memreg.c,v 1.7 1995/02/01 12:37:53 pk Exp $ */
+/* $NetBSD: memreg.c,v 1.13 1996/03/31 22:52:08 pk Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -12,6 +12,7 @@
* must display the following acknowledgement:
* This product includes software developed by the University of
* California, Lawrence Berkeley Laboratory.
+ * This product includes software developed by Harvard University.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
@@ -25,6 +26,7 @@
* must display the following acknowledgement:
* This product includes software developed by the University of
* California, Berkeley and its contributors.
+ * This product includes software developed by Harvard University.
* 4. Neither the name of the University nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
@@ -45,6 +47,7 @@
*/
#include <sys/param.h>
+#include <sys/systm.h>
#include <sys/device.h>
#include <machine/autoconf.h>
@@ -52,11 +55,26 @@
#include <sparc/sparc/memreg.h>
#include <sparc/sparc/vaddrs.h>
+#include <sparc/sparc/asm.h>
+
+#include <machine/reg.h> /* for trapframe */
+#include <machine/trap.h> /* for trap types */
static int memregmatch __P((struct device *, void *, void *));
static void memregattach __P((struct device *, struct device *, void *));
-struct cfdriver memregcd =
- { 0, "memreg", memregmatch, memregattach, DV_DULL, sizeof(struct device) };
+
+struct cfattach memreg_ca = {
+ sizeof(struct device), memregmatch, memregattach
+};
+
+struct cfdriver memreg_cd = {
+ 0, "memreg", DV_DULL
+};
+
+void memerr __P((int, u_int, u_int, u_int, u_int));
+#if defined(SUN4M)
+static void hardmemerr4m __P((int, u_int, u_int));
+#endif
/*
* The OPENPROM calls this "memory-error".
@@ -69,7 +87,7 @@ memregmatch(parent, vcf, aux)
struct cfdata *cf = vcf;
register struct confargs *ca = aux;
- if (cputyp == CPU_SUN4) {
+ if (CPU_ISSUN4) {
if (ca->ca_bustype == BUS_OBIO)
return (strcmp(cf->cf_driver->cd_name,
ca->ca_ra.ra_name) == 0);
@@ -87,14 +105,14 @@ memregattach(parent, self, aux)
register struct confargs *ca = aux;
register struct romaux *ra = &ca->ca_ra;
- if (cputyp == CPU_SUN4) {
+ if (CPU_ISSUN4) {
if (par_err_reg == NULL)
panic("memregattach");
ra->ra_vaddr = (caddr_t)par_err_reg;
} else {
- par_err_reg = ra->ra_vaddr ? (volatile int *)ra->ra_vaddr :
- (volatile int *)mapiodev(ra->ra_reg, 0, sizeof(int),
- ca->ca_bustype);
+ par_err_reg = ra->ra_vaddr ? (volatile u_int *)ra->ra_vaddr :
+ (volatile u_int *)mapiodev(ra->ra_reg, 0, sizeof(int),
+ ca->ca_bustype);
}
printf("\n");
}
@@ -106,14 +124,16 @@ memregattach(parent, self, aux)
* and take the page out of the page pool, but for now...
*/
-extern void sun4c_sun4_storebuf_flush(void);
-
void
memerr(issync, ser, sva, aer, ava)
- int issync, ser, sva, aer, ava;
+ int issync;
+ u_int ser, sva, aer, ava;
{
- sun4c_sun4_storebuf_flush(); /* prevent loops... */
- if (cputyp == CPU_SUN4) {
+
+ /* XXX Ugh! Clean up this switch and all the ifdefs! */
+ switch (cputyp) {
+#if defined(SUN4)
+ case CPU_SUN4:
if (par_err_reg) {
printf("mem err: ser=%b sva=%x\n", ser, SER_BITS, sva);
printf("parity error register = %b\n",
@@ -122,17 +142,155 @@ memerr(issync, ser, sva, aer, ava)
printf("mem err: ser=? sva=?\n");
printf("parity error register not mapped yet!\n"); /* XXX */
}
- } else {
+#ifdef DEBUG
+ callrom();
+#else
+ panic("memory error"); /* XXX */
+#endif
+ break;
+#endif /* Sun4 */
+
+#if defined(SUN4C)
+ case CPU_SUN4C:
printf("%ssync mem err: ser=%b sva=%x aer=%b ava=%x\n",
- issync ? "" : "a", ser, SER_BITS, sva, aer & 0xff,
- AER_BITS, ava);
+ issync ? "" : "a", ser, SER_BITS,
+ sva, aer & 0xff, AER_BITS, ava);
if (par_err_reg)
printf("parity error register = %b\n",
*par_err_reg, PER_BITS);
- }
#ifdef DEBUG
- callrom();
+ callrom();
#else
- panic("memory error"); /* XXX */
+ panic("memory error"); /* XXX */
#endif
+ break;
+#endif /* Sun4C */
+
+#if defined(SUN4M)
+ case CPU_SUN4M:
+ hardmemerr4m(2, ser, sva);
+ break;
+#endif /* Sun4M */
+ default:
+ break;
+ }
+}
+
+
+#if defined(SUN4M)
+/*
+ * hardmemerr4m: called upon fatal memory error. Print a message and panic.
+ * Note that issync is not really an indicator of whether or not the error
+ * was synchronous; if it is set, it means that the fsr/faddr pair correspond
+ * to the MMU's fault status register; if clear, they correspond to the
+ * HyperSPARC asynchronous error register. If issync==2, then both decodings
+ * of the error register are printed.
+ */
+
+static void
+hardmemerr4m(issync, fsr, faddr)
+ int issync;
+ u_int fsr, faddr;
+{
+ switch (issync) {
+ case 1:
+ if ((fsr & SFSR_FT) == SFSR_FT_NONE)
+ return;
+ panic("mem err: sfsr=%b sfaddr=%x", fsr, SFSR_BITS, faddr);
+ break;
+ case 0:
+ if (!(fsr & AFSR_AFO))
+ return;
+ panic("async (HS) mem err: afsr=%b afaddr=%x physaddr=%x%x",
+ fsr, AFSR_BITS, faddr,
+ (fsr & AFSR_AFA) >> AFSR_AFA_RSHIFT, faddr);
+ break;
+ default: /* unknown; print both decodings*/
+ panic("Unknown mem err: if sync, fsr=%b fva=%x; if async, fsr"
+ "=%b fa=%x pa=%x%x", fsr, SFSR_BITS, faddr, fsr,
+ AFSR_BITS, faddr, (fsr & AFSR_AFA) >> AFSR_AFA_RSHIFT,
+ faddr);
+ break;
+ }
+}
+
+/*
+ * Memerr4m: handle a non-trivial memory fault. These include HyperSPARC
+ * asynchronous faults, SuperSPARC store-buffer copyback failures, and
+ * data faults without a valid faulting VA. We try to retry the operation
+ * once, and then fail if we get called again.
+ */
+
+static int addrold = (int) 0xdeadbeef; /* We pick an unlikely address */
+static int addroldtop = (int) 0xdeadbeef;
+static int oldtype = -1;
+
+void
+memerr4m(type, sfsr, sfva, afsr, afva, tf)
+ register unsigned type;
+ register u_int sfsr;
+ register u_int sfva;
+ register u_int afsr;
+ register u_int afva;
+ register struct trapframe *tf;
+{
+ if ((afsr & AFSR_AFO) != 0) { /* HS async fault! */
+
+ printf("HyperSPARC async cache memory failure at phys 0x%x%x. "
+ "Ignoring.\n", (afsr & AFSR_AFA) >> AFSR_AFA_RSHIFT,
+ afva);
+
+ if (afva == addrold && (afsr & AFSR_AFA) == addroldtop)
+ hardmemerr4m(0, afsr, afva);
+ /* NOTREACHED */
+
+ oldtype = -1;
+ addrold = afva;
+ addroldtop = afsr & AFSR_AFA;
+
+ } else if (type == T_STOREBUFFAULT) {
+ /* We try to reenable the store buffers to force a retry */
+
+ printf("store buffer copy-back failure at 0x%x. Retrying...\n",
+ sfva);
+
+ if (oldtype == T_STOREBUFFAULT || addrold == sfva)
+ hardmemerr4m(1, sfsr, sfva);
+ /* NOTREACHED */
+
+ oldtype = T_STOREBUFFAULT;
+ addrold = sfva;
+
+ sta(SRMMU_PCR, ASI_SRMMU, lda(SRMMU_PCR, ASI_SRMMU) |
+ SRMMU_PCR_SB); /* reenable store buffer */
+
+ } else if (type == T_DATAFAULT && !(sfsr & SFSR_FAV)) { /* bizarre */
+ /* XXX: Should handle better. See SuperSPARC manual pg. 9-35 */
+
+ printf("warning: got data fault with no faulting address."
+ " Ignoring.\n");
+
+ if (oldtype == T_DATAFAULT)
+ hardmemerr4m(1, sfsr, sfva);
+ /* NOTREACHED */
+
+ oldtype = T_DATAFAULT;
+ } else if (type == 0) { /* NMI */
+ printf("ERROR: got NMI with sfsr=0x%b, sfva=0x%x, afsr=0x%b, "
+ "afaddr=0x%x. Retrying...\n",
+ sfsr,SFSR_BITS,sfva,afsr, AFSR_BITS,afva);
+ if (oldtype == 0 || addrold == sfva)
+ hardmemerr4m(1, sfsr, sfva); /* XXX: async? */
+ /* NOTREACHED */
+
+ oldtype = 0;
+ addrold = sfva;
+ } else /* something we don't know about?!? */ {
+ panic("memerr4m: unknown fatal memory error. type=%d, sfsr=%b,"
+ " sfva=%x, afsr=%b, afaddr=%x",
+ type, sfsr, SFSR_BITS, sfva, afsr, AFSR_BITS, afva);
+ }
+
+ return;
}
+#endif /* 4m */
diff --git a/sys/arch/sparc/sparc/memreg.h b/sys/arch/sparc/sparc/memreg.h
index 428a93f53c6..be25a8f76c0 100644
--- a/sys/arch/sparc/sparc/memreg.h
+++ b/sys/arch/sparc/sparc/memreg.h
@@ -1,4 +1,4 @@
-/* $NetBSD: memreg.h,v 1.2 1994/11/20 20:54:33 deraadt Exp $ */
+/* $NetBSD: memreg.h,v 1.4 1996/03/31 22:52:13 pk Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -48,7 +48,7 @@
* Sun-4c memory error register.
* The register is a single word.
*/
-volatile int *par_err_reg; /* virtual address; NULL if not yet mapped */
+volatile u_int *par_err_reg; /* virtual address; NULL if not yet mapped */
/*
* Bits in parity error register.
@@ -64,3 +64,11 @@ volatile int *par_err_reg; /* virtual address; NULL if not yet mapped */
#define PER_BYTE3 0x01 /* error occurred in byte 3 (bits 7..0) */
#define PER_BITS "\20\10ERR\7MULTI\6TEST\5ENABLE\4BYTE0\3BYTE1\2BYTE2\1BYTE3"
+
+
+
+/*
+ * sun4m ...
+ */
+struct trapframe;
+void memerr4m __P((unsigned, u_int, u_int, u_int, u_int, struct trapframe *));
diff --git a/sys/arch/sparc/sparc/openprom.c b/sys/arch/sparc/sparc/openprom.c
index b0e61849690..15c9ac1293f 100644
--- a/sys/arch/sparc/sparc/openprom.c
+++ b/sys/arch/sparc/sparc/openprom.c
@@ -1,4 +1,4 @@
-/* $NetBSD: openprom.c,v 1.5 1995/01/10 16:47:03 pk Exp $ */
+/* $NetBSD: openprom.c,v 1.8 1996/03/31 23:45:34 pk Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -50,32 +50,39 @@
#include <sys/fcntl.h>
#include <sys/ioctl.h>
#include <sys/malloc.h>
+#include <sys/conf.h>
#include <machine/bsd_openprom.h>
#include <machine/openpromio.h>
+#include <machine/autoconf.h>
+#include <machine/conf.h>
static int lastnode; /* speed hack */
extern int optionsnode; /* node ID of ROM's options */
-extern int findroot(); /* returns node ID of top node */
extern struct promvec *promvec;
+static int openpromcheckid __P((int, int));
+static int openpromgetstr __P((int, char *, char **));
+
int
-openpromopen(dev, flags, mode)
+openpromopen(dev, flags, mode, p)
dev_t dev;
int flags, mode;
+ struct proc *p;
{
#if defined(SUN4)
if (cputyp==CPU_SUN4)
return (ENODEV);
-#endif
+#endif
return (0);
}
int
-openpromclose(dev, flags, mode)
+openpromclose(dev, flags, mode, p)
dev_t dev;
int flags, mode;
+ struct proc *p;
{
return (0);
diff --git a/sys/arch/sparc/sparc/pmap.c b/sys/arch/sparc/sparc/pmap.c
index be83605c3ab..226bf7f6715 100644
--- a/sys/arch/sparc/sparc/pmap.c
+++ b/sys/arch/sparc/sparc/pmap.c
@@ -1,6 +1,8 @@
-/* $NetBSD: pmap.c,v 1.47 1995/07/05 18:52:32 pk Exp $ */
+/* $NetBSD: pmap.c,v 1.60.4.1 1996/06/12 20:36:30 pk Exp $ */
/*
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
*
@@ -10,12 +12,14 @@
*
* All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
+ * This product includes software developed by Harvard University.
* This product includes software developed by the University of
* California, Lawrence Berkeley Laboratory.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
+ *
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
@@ -23,8 +27,10 @@
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
- * This product includes software developed by the University of
- * California, Berkeley and its contributors.
+ * This product includes software developed by Aaron Brown and
+ * Harvard University.
+ * This product includes software developed by the University of
+ * California, Berkeley and its contributors.
* 4. Neither the name of the University nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
@@ -42,6 +48,7 @@
* SUCH DAMAGE.
*
* @(#)pmap.c 8.4 (Berkeley) 2/5/94
+ *
*/
/*
@@ -73,10 +80,9 @@
#ifdef DEBUG
#define PTE_BITS "\20\40V\37W\36S\35NC\33IO\32U\31M"
+#define PTE_BITS4M "\20\10C\7M\6R\5ACC3\4ACC2\3ACC1\2TYP2\1TYP1"
#endif
-extern struct promvec *promvec;
-
/*
* The SPARCstation offers us the following challenges:
*
@@ -85,10 +91,11 @@ extern struct promvec *promvec;
* This is a write-through cache on the 4c and a write-back cache
* on others.
*
- * 2. An MMU that acts like a cache. There is not enough space
- * in the MMU to map everything all the time. Instead, we need
+ * 2. (4/4c only) An MMU that acts like a cache. There is not enough
+ * space in the MMU to map everything all the time. Instead, we need
* to load MMU with the `working set' of translations for each
- * process.
+ * process. The sun4m does not act like a cache; tables are maintained
+ * in physical memory.
*
* 3. Segmented virtual and physical spaces. The upper 12 bits of
* a virtual address (the virtual segment) index a segment table,
@@ -97,10 +104,11 @@ extern struct promvec *promvec;
* next 5 or 6 bits of the virtual address---select the particular
* `Page Map Entry' for the page. We call the latter a PTE and
* call each Page Map Entry Group a pmeg (for want of a better name).
+ * Note that the sun4m has an unsegmented 36-bit physical space.
*
* Since there are no valid bits in the segment table, the only way
* to have an invalid segment is to make one full pmeg of invalid PTEs.
- * We use the last one (since the ROM does as well).
+ * We use the last one (since the ROM does as well) (sun4/4c only)
*
* 4. Discontiguous physical pages. The Mach VM expects physical pages
* to be in one sequential lump.
@@ -134,10 +142,13 @@ struct pmap_stats {
#define PDB_CTX_STEAL 0x0800
#define PDB_MMUREG_ALLOC 0x1000
#define PDB_MMUREG_STEAL 0x2000
-int pmapdebug = 0x0;
+#define PDB_CACHESTUFF 0x4000
+int pmapdebug = 0;
#endif
+#if 0
#define splpmap() splimp()
+#endif
/*
* First and last managed physical addresses.
@@ -168,10 +179,15 @@ struct pvlist {
/*
* Flags in pv_flags. Note that PV_MOD must be 1 and PV_REF must be 2
* since they must line up with the bits in the hardware PTEs (see pte.h).
+ * Sun4M bits are different (reversed), and at a different location in the
+ * pte. Now why did they do that?
*/
#define PV_MOD 1 /* page modified */
#define PV_REF 2 /* page referenced */
#define PV_NC 4 /* page cannot be cached */
+#define PV_REF4M 1 /* page referenced on sun4m */
+#define PV_MOD4M 2 /* page modified on 4m (why reversed?!?) */
+#define PV_C4M 4 /* page _can_ be cached on 4m */
/*efine PV_ALLF 7 ** all of the above */
struct pvlist *pv_table; /* array of entries, one per physical page */
@@ -184,6 +200,9 @@ struct pvlist *pv_table; /* array of entries, one per physical page */
* it is in the MMU, however; that is true iff pm_segmap[VA_VSEG(va)]
* does not point to the invalid PMEG.
*
+ * In the older SPARC architectures (pre-4m), page tables are cached in the
+ * MMU. The following discussion applies to these architectures:
+ *
* If a virtual segment is valid and loaded, the correct PTEs appear
* in the MMU only. If it is valid and unloaded, the correct PTEs appear
* in the pm_pte[VA_VSEG(va)] only. However, some effort is made to keep
@@ -214,6 +233,17 @@ struct pvlist *pv_table; /* array of entries, one per physical page */
* or locked. The LRU list is for user processes; the locked list is
* for kernel entries; both are doubly linked queues headed by `mmuhd's.
* The free list is a simple list, headed by a free list pointer.
+ *
+ * In the sun4m architecture using the SPARC Reference MMU (SRMMU), three
+ * levels of page tables are maintained in physical memory. We use the same
+ * structures as with the 3-level old-style MMU (pm_regmap, pm_segmap,
+ * rg_segmap, sg_pte, etc) to maintain kernel-edible page tables; we also
+ * build a parallel set of physical tables that can be used by the MMU.
+ * (XXX: This seems redundant, but is it necessary for the unified kernel?)
+ *
+ * If a virtual segment is valid, its entries will be in both parallel lists.
+ * If it is not valid, then its entry in the kernel tables will be zero, and
+ * its entry in the MMU tables will either be nonexistent or zero as well.
*/
struct mmuentry {
TAILQ_ENTRY(mmuentry) me_list; /* usage list link */
@@ -235,9 +265,16 @@ int reginval; /* the invalid region number */
#endif
/*
+ * (sun4/4c)
* A context is simply a small number that dictates which set of 4096
* segment map entries the MMU uses. The Sun 4c has eight such sets.
* These are alloted in an `almost MRU' fashion.
+ * (sun4m)
+ * A context is simply a small number that indexes the context table, the
+ * root-level page table mapping 4G areas. Each entry in this table points
+ * to a 1st-level region table. A SPARC reference MMU will usually use 16
+ * such contexts, but some offer as many as 64k contexts; the theoretical
+ * maximum is 2^32 - 1, but this would create overlarge context tables.
*
* Each context is either free or attached to a pmap.
*
@@ -255,9 +292,12 @@ union ctxinfo *ctx_freelist; /* context free list */
int ctx_kick; /* allocation rover when none free */
int ctx_kickdir; /* ctx_kick roves both directions */
+char *ctxbusyvector; /* [4m] tells what contexts are busy */
+int *ctx_phys_tbl; /* [4m] SRMMU-edible context table */
+
caddr_t vpage[2]; /* two reserved MD virtual pages */
caddr_t vmmap; /* one reserved MI vpage for /dev/mem */
-caddr_t vdumppages; /* 32KB worth of reserved dump pages */
+caddr_t vdumppages; /* 32KB worth of reserved dump pages */
#ifdef MMU_3L
smeg_t tregion;
@@ -266,6 +306,17 @@ struct pmap kernel_pmap_store; /* the kernel's pmap */
struct regmap kernel_regmap_store[NKREG]; /* the kernel's regmap */
struct segmap kernel_segmap_store[NKREG*NSEGRG];/* the kernel's segmaps */
+#ifdef SUN4M
+u_int *kernel_regtable_store; /* 1k of storage to map the kernel */
+u_int *kernel_segtable_store; /* 2k of storage to map the kernel */
+u_int *kernel_pagtable_store; /* 128k of storage to map the kernel */
+
+u_int *kernel_iopte_table; /* 64k of storage for iommu */
+u_int kernel_iopte_table_pa;
+
+static int cant_cache_pagetables = 0; /* 1 if page tables are uncachable */
+#endif
+
#define MA_SIZE 32 /* size of memory descriptor arrays */
struct memarr pmemarr[MA_SIZE];/* physical memory regions */
int npmemarr; /* number of entries in pmemarr */
@@ -276,81 +327,351 @@ int cpmemarr; /* pmap_next_page() state */
next free physical page */
/*static*/ vm_offset_t virtual_avail; /* first free virtual page number */
/*static*/ vm_offset_t virtual_end; /* last free virtual page number */
+#ifdef SUN4M
+#if 0 /* not needed anymore */
+static void * pmap_bootstrap_malloc __P((unsigned long, int, int));
+
+caddr_t minipool_start;
+caddr_t minipool_current;
+#define MINIPOOL_SIZE 8192 /* number of bytes to reserve for early malloc */
+ /* (must be multiple of pagesize) */
+#endif
+#endif
int mmu_has_hole;
vm_offset_t prom_vstart; /* For /dev/kmem */
vm_offset_t prom_vend;
-#ifdef SUN4
+#if defined(SUN4)
/*
- * segfixmask: on some systems (4/110) "getsegmap()" returns a partly
- * invalid value. getsegmap returns a 16 bit value on the sun4, but
- * only the first 8 or so bits are valid (the rest are *supposed* to
- * be zero. on the 4/110 the bits that are supposed to be zero are
- * all one instead. e.g. KERNBASE is usually mapped by pmeg number zero.
- * on a 4/300 getsegmap(KERNBASE) == 0x0000, but
+ * [sun4]: segfixmask: on some systems (4/110) "getsegmap()" returns a
+ * partly invalid value. getsegmap returns a 16 bit value on the sun4,
+ * but only the first 8 or so bits are valid (the rest are *supposed* to
+ * be zero. On the 4/110 the bits that are supposed to be zero are
+ * all one instead. e.g. KERNBASE is usually mapped by pmeg number zero.
+ * On a 4/300 getsegmap(KERNBASE) == 0x0000, but
* on a 4/100 getsegmap(KERNBASE) == 0xff00
*
- * this confuses mmu_reservemon() and causes it to not reserve the PROM's
- * pmegs. then the PROM's pmegs get used during autoconfig and everything
+ * This confuses mmu_reservemon() and causes it to not reserve the PROM's
+ * pmegs. Then the PROM's pmegs get used during autoconfig and everything
* falls apart! (not very fun to debug, BTW.)
*
* solution: mask the invalid bits in the getsetmap macro.
*/
static u_long segfixmask = 0xffffffff; /* all bits valid to start */
+#else
+#define segfixmask 0xffffffff /* It's in getsegmap's scope */
#endif
/*
* pseudo-functions for mnemonic value
- * NB: setsegmap should be stba for 4c, but stha works and makes the
- * code right for the Sun-4 as well.
*/
-#define getcontext() lduba(AC_CONTEXT, ASI_CONTROL)
-#define setcontext(c) stba(AC_CONTEXT, ASI_CONTROL, c)
-#if defined(SUN4) && !defined(SUN4C)
-#define getsegmap(va) (lduha(va, ASI_SEGMAP) & segfixmask)
-#define setsegmap(va, pmeg) stha(va, ASI_SEGMAP, pmeg)
+#define getcontext() (CPU_ISSUN4M \
+ ? lda(SRMMU_CXR, ASI_SRMMU) \
+ : lduba(AC_CONTEXT, ASI_CONTROL))
+#define setcontext(c) (CPU_ISSUN4M \
+ ? sta(SRMMU_CXR, ASI_SRMMU, c) \
+ : stba(AC_CONTEXT, ASI_CONTROL, c))
+
+#define getsegmap(va) (CPU_ISSUN4C \
+ ? lduba(va, ASI_SEGMAP) \
+ : (lduha(va, ASI_SEGMAP) & segfixmask))
+#define setsegmap(va, pmeg) (CPU_ISSUN4C \
+ ? stba(va, ASI_SEGMAP, pmeg) \
+ : stha(va, ASI_SEGMAP, pmeg))
+
+/* 3-level sun4 MMU only: */
+#define getregmap(va) ((unsigned)lduha((va)+2, ASI_REGMAP) >> 8)
+#define setregmap(va, smeg) stha((va)+2, ASI_REGMAP, (smeg << 8))
+
+#if defined(SUN4M)
+#define getpte4m(va) lda((va & 0xFFFFF000) | ASI_SRMMUFP_L3, \
+ ASI_SRMMUFP)
+void setpte4m __P((vm_offset_t va, int pte));
+void setptesw4m __P((struct pmap *pm, vm_offset_t va, int pte));
+u_int getptesw4m __P((struct pmap *pm, vm_offset_t va));
#endif
-#if !defined(SUN4) && defined(SUN4C)
-#define getsegmap(va) lduba(va, ASI_SEGMAP)
-#define setsegmap(va, pmeg) stba(va, ASI_SEGMAP, pmeg)
+
+#if defined(SUN4) || defined(SUN4C)
+#define getpte4(va) lda(va, ASI_PTE)
+#define setpte4(va, pte) sta(va, ASI_PTE, pte)
#endif
-#if defined(SUN4) && defined(SUN4C)
-#define getsegmap(va) (cputyp==CPU_SUN4C ? lduba(va, ASI_SEGMAP) \
- : (lduha(va, ASI_SEGMAP) & segfixmask))
-#define setsegmap(va, pmeg) (cputyp==CPU_SUN4C ? stba(va, ASI_SEGMAP, pmeg) \
- : stha(va, ASI_SEGMAP, pmeg))
+
+/* Function pointer messiness for supporting multiple sparc architectures
+ * within a single kernel: notice that there are two versions of many of the
+ * functions within this file/module, one for the sun4/sun4c and the other
+ * for the sun4m. For performance reasons (since things like pte bits don't
+ * map nicely between the two architectures), there are separate functions
+ * rather than unified functions which test the cputyp variable. If only
+ * one architecture is being used, then the non-suffixed function calls
+ * are macro-translated into the appropriate xxx4_4c or xxx4m call. If
+ * multiple architectures are defined, the calls translate to (*xxx_p),
+ * i.e. they indirect through function pointers initialized as appropriate
+ * to the run-time architecture in pmap_bootstrap. See also pmap.h.
+ */
+
+#if defined(SUN4M)
+static void mmu_setup4m_L1 __P((int, caddr_t, struct pmap *, caddr_t *));
+static void mmu_setup4m_L2 __P((int, caddr_t, struct regmap *, caddr_t *,
+ struct segmap *));
+static int mmu_setup4m_L3 __P((int, caddr_t));
+/*static*/ void mmu_reservemon4m __P((struct pmap *, caddr_t *));
+
+/*static*/ void pmap_rmk4m __P((struct pmap *, vm_offset_t, vm_offset_t,
+ int, int));
+/*static*/ void pmap_rmu4m __P((struct pmap *, vm_offset_t, vm_offset_t,
+ int, int));
+/*static*/ void pmap_enk4m __P((struct pmap *, vm_offset_t, vm_prot_t,
+ int, struct pvlist *, int));
+/*static*/ void pmap_enu4m __P((struct pmap *, vm_offset_t, vm_prot_t,
+ int, struct pvlist *, int));
+/*static*/ void pv_changepte4m __P((struct pvlist *, int, int));
+/*static*/ int pv_syncflags4m __P((struct pvlist *));
+/*static*/ int pv_link4m __P((struct pvlist *, struct pmap *, vm_offset_t));
+/*static*/ void pv_unlink4m __P((struct pvlist *, struct pmap *, vm_offset_t));
#endif
-#if defined(SUN4) && defined(MMU_3L)
-#define getregmap(va) ((unsigned)lduha(va+2, ASI_REGMAP) >> 8)
-#define setregmap(va, smeg) stha(va+2, ASI_REGMAP, (smeg << 8))
+
+#if defined(SUN4) || defined(SUN4C)
+/*static*/ void mmu_reservemon4_4c __P((int *, int *));
+/*static*/ void pmap_rmk4_4c __P((struct pmap *, vm_offset_t, vm_offset_t,
+ int, int));
+/*static*/ void pmap_rmu4_4c __P((struct pmap *, vm_offset_t, vm_offset_t,
+ int, int));
+/*static*/ void pmap_enk4_4c __P((struct pmap *, vm_offset_t, vm_prot_t,
+ int, struct pvlist *, int));
+/*static*/ void pmap_enu4_4c __P((struct pmap *, vm_offset_t, vm_prot_t,
+ int, struct pvlist *, int));
+/*static*/ void pv_changepte4_4c __P((struct pvlist *, int, int));
+/*static*/ int pv_syncflags4_4c __P((struct pvlist *));
+/*static*/ int pv_link4_4c __P((struct pvlist *, struct pmap *, vm_offset_t));
+/*static*/ void pv_unlink4_4c __P((struct pvlist *, struct pmap *, vm_offset_t));
#endif
-#define getpte(va) lda(va, ASI_PTE)
-#define setpte(va, pte) sta(va, ASI_PTE, pte)
+#if !defined(SUN4M) && (defined(SUN4) || defined(SUN4C))
+#define pmap_rmk pmap_rmk4_4c
+#define pmap_rmu pmap_rmu4_4c
+
+#elif defined(SUN4M) && !(defined(SUN4) || defined(SUN4C))
+#define pmap_rmk pmap_rmk4m
+#define pmap_rmu pmap_rmu4m
+
+#else /* must use function pointers */
+
+/* function pointer declarations */
+/* from pmap.h: */
+void (*pmap_clear_modify_p) __P((vm_offset_t pa));
+void (*pmap_clear_reference_p) __P((vm_offset_t pa));
+void (*pmap_copy_page_p) __P((vm_offset_t, vm_offset_t));
+void (*pmap_enter_p) __P((pmap_t,
+ vm_offset_t, vm_offset_t, vm_prot_t, boolean_t));
+vm_offset_t (*pmap_extract_p) __P((pmap_t, vm_offset_t));
+boolean_t (*pmap_is_modified_p) __P((vm_offset_t pa));
+boolean_t (*pmap_is_referenced_p) __P((vm_offset_t pa));
+void (*pmap_page_protect_p) __P((vm_offset_t, vm_prot_t));
+void (*pmap_protect_p) __P((pmap_t,
+ vm_offset_t, vm_offset_t, vm_prot_t));
+void (*pmap_zero_page_p) __P((vm_offset_t));
+void (*pmap_changeprot_p) __P((pmap_t, vm_offset_t,
+ vm_prot_t, int));
+/* local: */
+void (*pmap_rmk_p) __P((struct pmap *, vm_offset_t, vm_offset_t,
+ int, int));
+void (*pmap_rmu_p) __P((struct pmap *, vm_offset_t, vm_offset_t,
+ int, int));
+
+#define pmap_rmk (*pmap_rmk_p)
+#define pmap_rmu (*pmap_rmu_p)
-/*----------------------------------------------------------------*/
+#endif
-#define HWTOSW(pg) (pg)
-#define SWTOHW(pg) (pg)
+/* --------------------------------------------------------------*/
-#ifdef MMU_3L
-#define CTX_USABLE(pm,rp) ((pm)->pm_ctx && \
- (!mmu_3l || (rp)->rg_smeg != reginval))
+/*
+ * Next we have some Sun4m-specific routines which have no 4/4c
+ * counterparts, or which are 4/4c macros.
+ */
+
+#if defined(SUN4M)
+
+/* Macros which implement SRMMU TLB flushing/invalidation */
+
+#define tlb_flush_page(va) sta((va & ~0xfff) | ASI_SRMMUFP_L3, ASI_SRMMUFP,0)
+#define tlb_flush_segment(vreg, vseg) sta((vreg << RGSHIFT) | (vseg << SGSHIFT)\
+ | ASI_SRMMUFP_L2, ASI_SRMMUFP,0)
+#define tlb_flush_context() sta(ASI_SRMMUFP_L1, ASI_SRMMUFP, 0)
+#define tlb_flush_all() sta(ASI_SRMMUFP_LN, ASI_SRMMUFP, 0)
+
+static u_int VA2PA __P((caddr_t));
+
+/*
+ * VA2PA(addr) -- converts a virtual address to a physical address using
+ * the MMU's currently-installed page tables. As a side effect, the address
+ * translation used may cause the associated pte to be encached. The correct
+ * context for VA must be set before this is called.
+ *
+ * This routine should work with any level of mapping, as it is used
+ * during bootup to interact with the ROM's initial L1 mapping of the kernel.
+ */
+static __inline u_int
+VA2PA(addr)
+ register caddr_t addr;
+{
+ register u_int pte;
+
+ /* we'll use that handy SRMMU flush/probe! %%%: make consts below! */
+ /* Try each level in turn until we find a valid pte. Otherwise panic */
+
+ pte = lda(((u_int)addr & ~0xfff) | ASI_SRMMUFP_L3, ASI_SRMMUFP);
+ if ((pte & SRMMU_TETYPE) == SRMMU_TEPTE)
+ return (((pte & SRMMU_PPNMASK) << SRMMU_PPNPASHIFT) |
+ ((u_int)addr & 0xfff));
+
+ /* A `TLB Flush Entire' is required before any L0, L1 or L2 probe */
+ tlb_flush_all();
+
+ pte = lda(((u_int)addr & ~0xfff) | ASI_SRMMUFP_L2, ASI_SRMMUFP);
+ if ((pte & SRMMU_TETYPE) == SRMMU_TEPTE)
+ return (((pte & SRMMU_PPNMASK) << SRMMU_PPNPASHIFT) |
+ ((u_int)addr & 0x3ffff));
+ pte = lda(((u_int)addr & ~0xfff) | ASI_SRMMUFP_L1, ASI_SRMMUFP);
+ if ((pte & SRMMU_TETYPE) == SRMMU_TEPTE)
+ return (((pte & SRMMU_PPNMASK) << SRMMU_PPNPASHIFT) |
+ ((u_int)addr & 0xffffff));
+ pte = lda(((u_int)addr & ~0xfff) | ASI_SRMMUFP_L0, ASI_SRMMUFP);
+ if ((pte & SRMMU_TETYPE) == SRMMU_TEPTE)
+ return (((pte & SRMMU_PPNMASK) << SRMMU_PPNPASHIFT) |
+ ((u_int)addr & 0xffffffff));
+
+ panic("VA2PA: Asked to translate unmapped VA %p", addr);
+}
+
+/*
+ * Get the page table entry (PTE) for va by looking it up in the software
+ * page tables. These are the same tables that are used by the MMU; this
+ * routine allows easy access to the page tables even if the context
+ * corresponding to the table is not loaded or selected.
+ * This routine should NOT be used if there is any chance that the desired
+ * pte is in the TLB cache, since it will return stale data in that case.
+ * For that case, and for general use, use getpte4m, which is much faster
+ * and avoids walking in-memory page tables if the page is in the cache.
+ * Note also that this routine only works if a kernel mapping has been
+ * installed for the given page!
+ */
+__inline u_int
+getptesw4m(pm, va) /* Assumes L3 mapping! */
+ register struct pmap *pm;
+ register vm_offset_t va;
+{
+ register struct regmap *rm;
+ register struct segmap *sm;
+
+ rm = &pm->pm_regmap[VA_VREG(va)];
+#ifdef DEBUG
+ if (rm == NULL)
+ panic("getptesw4m: no regmap entry");
+#endif
+ sm = &rm->rg_segmap[VA_VSEG(va)];
+#ifdef DEBUG
+ if (sm == NULL)
+ panic("getptesw4m: no segmap");
+#endif
+ return (sm->sg_pte[VA_SUN4M_VPG(va)]); /* return pte */
+}
+
+/*
+ * Set the page table entry for va to pte. Only affects software MMU page-
+ * tables (the in-core pagetables read by the MMU). Ignores TLB, and
+ * thus should _not_ be called if the pte translation could be in the TLB.
+ * In this case, use setpte4m().
+ */
+__inline void
+setptesw4m(pm, va, pte)
+ register struct pmap *pm;
+ register vm_offset_t va;
+ register int pte;
+{
+ register struct regmap *rm;
+ register struct segmap *sm;
+
+ rm = &pm->pm_regmap[VA_VREG(va)];
+
+#ifdef DEBUG
+ if (pm->pm_regmap == NULL || rm == NULL)
+ panic("setpte4m: no regmap entry");
+#endif
+ sm = &rm->rg_segmap[VA_VSEG(va)];
+
+#ifdef DEBUG
+ if (rm->rg_segmap == NULL || sm == NULL || sm->sg_pte == NULL)
+ panic("setpte4m: no segmap for va %p", (caddr_t)va);
+#endif
+ sm->sg_pte[VA_SUN4M_VPG(va)] = pte; /* set new pte */
+}
+
+/* Set the page table entry for va to pte. Flushes cache. */
+__inline void
+setpte4m(va, pte)
+ register vm_offset_t va;
+ register int pte;
+{
+ register struct pmap *pm;
+ register struct regmap *rm;
+ register struct segmap *sm;
+ register union ctxinfo *c;
+
+ if (vactype != VAC_NONE)
+ cache_flush_page(va);
+
+ /*
+ * Now walk tables to find pte. We use ctxinfo to locate the pmap
+ * from the current context
+ */
+#ifdef DEBUG
+ if (ctxbusyvector[getcontext()] == 0)
+ panic("setpte4m: no pmap for current context (%d)",
+ getcontext());
+#endif
+ c = &ctxinfo[getcontext()];
+ pm = c->c_pmap;
+
+ /* Note: inline version of setptesw4m() */
+#ifdef DEBUG
+ if (pm->pm_regmap == NULL)
+ panic("setpte4m: no regmap entry");
+#endif
+ rm = &pm->pm_regmap[VA_VREG(va)];
+ sm = &rm->rg_segmap[VA_VSEG(va)];
+
+#ifdef DEBUG
+ if (rm->rg_segmap == NULL || sm == NULL || sm->sg_pte == NULL)
+ panic("setpte4m: no segmap for va %p", (caddr_t)va);
+#endif
+ tlb_flush_page(va);
+ sm->sg_pte[VA_SUN4M_VPG(va)] = pte; /* set new pte */
+}
+#endif /* 4m only */
+
+/*----------------------------------------------------------------*/
+
+#if defined(MMU_3L)
+#define CTX_USABLE(pm,rp) (CPU_ISSUN4M \
+ ? ((pm)->pm_ctx != NULL ) \
+ : ((pm)->pm_ctx != NULL && \
+ (!mmu_3l || (rp)->rg_smeg != reginval)))
#else
-#define CTX_USABLE(pm,rp) ((pm)->pm_ctx)
+#define CTX_USABLE(pm,rp) ((pm)->pm_ctx != NULL )
#endif
-#define GAP_WIDEN(pm,vr) do { \
- if (vr + 1 == pm->pm_gap_start) \
- pm->pm_gap_start = vr; \
- if (vr == pm->pm_gap_end) \
- pm->pm_gap_end = vr + 1;\
+#define GAP_WIDEN(pm,vr) do if (CPU_ISSUN4OR4C) { \
+ if (vr + 1 == pm->pm_gap_start) \
+ pm->pm_gap_start = vr; \
+ if (vr == pm->pm_gap_end) \
+ pm->pm_gap_end = vr + 1; \
} while (0)
-#define GAP_SHRINK(pm,vr) do { \
+#define GAP_SHRINK(pm,vr) do if (CPU_ISSUN4OR4C) { \
register int x; \
x = pm->pm_gap_start + (pm->pm_gap_end - pm->pm_gap_start) / 2; \
if (vr > x) { \
@@ -362,6 +683,18 @@ static u_long segfixmask = 0xffffffff; /* all bits valid to start */
} \
} while (0)
+static void sortm __P((struct memarr *, int));
+void ctx_alloc __P((struct pmap *));
+void ctx_free __P((struct pmap *));
+void pv_flushcache __P((struct pvlist *));
+void kvm_iocache __P((caddr_t, int));
+#ifdef DEBUG
+void pm_check __P((char *, struct pmap *));
+void pm_check_k __P((char *, struct pmap *));
+void pm_check_u __P((char *, struct pmap *));
+#endif
+
+
/*
* Sort a memory array by address.
*/
@@ -477,7 +810,7 @@ pmap_next_page(paddr)
* The returned index does NOT need to start at zero.
*
*/
-u_long
+int
pmap_page_index(pa)
vm_offset_t pa;
{
@@ -487,7 +820,7 @@ pmap_page_index(pa)
#ifdef DIAGNOSTIC
if (pa < avail_start || pa >= avail_end)
- panic("pmap_page_index: pa=0x%x", pa);
+ panic("pmap_page_index: pa=0x%lx", pa);
#endif
for (idx = 0, mp = pmemarr, nmem = npmemarr; --nmem >= 0; mp++) {
@@ -515,7 +848,8 @@ pmap_pa_exists(pa)
}
/* update pv_flags given a valid pte */
-#define MR(pte) (((pte) >> PG_M_SHIFT) & (PV_MOD | PV_REF))
+#define MR4_4C(pte) (((pte) >> PG_M_SHIFT) & (PV_MOD | PV_REF))
+#define MR4M(pte) (((pte) >> PG_M_SHIFT4M) & (PV_MOD4M | PV_REF4M))
/*----------------------------------------------------------------*/
@@ -528,22 +862,33 @@ pmap_pa_exists(pa)
* description of *free* virtual memory. Rather than invert this, we
* resort to two magic constants from the PROM vector description file.
*/
+#if defined(SUN4) || defined(SUN4C)
void
-mmu_reservemon(nrp, nsp)
+mmu_reservemon4_4c(nrp, nsp)
register int *nrp, *nsp;
{
- register u_int va, eva;
- register int mmureg, mmuseg, i, nr, ns, vr, lastvr;
+ register u_int va = 0, eva = 0;
+ register int mmuseg, i, nr, ns, vr, lastvr;
+#ifdef MMU_3L
+ register int mmureg;
+#endif
register struct regmap *rp;
+#if defined(SUN4M)
+ if (CPU_ISSUN4M) {
+ panic("mmu_reservemon called on Sun4M machine");
+ return;
+ }
+#endif
+
#if defined(SUN4)
- if (cputyp == CPU_SUN4) {
+ if (CPU_ISSUN4) {
prom_vstart = va = OLDMON_STARTVADDR;
prom_vend = eva = OLDMON_ENDVADDR;
}
#endif
#if defined(SUN4C)
- if (cputyp == CPU_SUN4C) {
+ if (CPU_ISSUN4C) {
prom_vstart = va = OPENPROM_STARTVADDR;
prom_vend = eva = OPENPROM_ENDVADDR;
}
@@ -592,12 +937,312 @@ mmu_reservemon(nrp, nsp)
/* PROM maps its memory user-accessible: fix it. */
for (i = NPTESG; --i >= 0; va += NBPG)
- setpte(va, getpte(va) | PG_S);
+ setpte4(va, getpte4(va) | PG_S);
}
*nsp = ns;
*nrp = nr;
return;
}
+#endif
+
+#if defined(SUN4M) /* Sun4M versions of above */
+
+/*
+ * Take the monitor's initial page table layout, convert it to 3rd-level pte's
+ * (it starts out as a L1 mapping), and install it along with a set of kernel
+ * mapping tables as the kernel's initial page table setup. Also create and
+ * enable a context table. I suppose we also want to block user-mode access
+ * to the new kernel/ROM mappings.
+ */
+
+/*
+ * mmu_reservemon4m(): Copies the existing (ROM) page tables to kernel space,
+ * converting any L1/L2 PTEs to L3 PTEs. Does *not* copy the L1 entry mapping
+ * the kernel at 0xf8000000 (KERNBASE) since we don't want to map 16M of
+ * physical memory for the kernel. Thus the kernel must be installed later!
+ * Also installs ROM mappings into the kernel pmap.
+ * NOTE: This also revokes all user-mode access to the mapped regions.
+ * NOTE ALSO: Must set up kernel pmap as all invalid before this is called.
+ */
+void
+mmu_reservemon4m(kpmap, kmemtop)
+ struct pmap *kpmap;
+ register caddr_t *kmemtop; /* Note: this is a *virtual* address! */
+{
+ unsigned int ctxtblptr;
+ int wierdviking = 0;
+ register int te;
+ register caddr_t tableptr;
+ unsigned int mmupcrsav;
+
+/*XXX-GCC!*/mmupcrsav = 0;
+
+ /*
+ * XXX: although the Sun4M can handle 36 bits of physical
+ * address space, we assume that all these page tables, etc
+ * are in the lower 4G (32-bits) of address space, i.e. out of I/O
+ * space. Eventually this should be changed to support the 36 bit
+ * physical addressing, in case some crazed ROM designer decides to
+ * stick the pagetables up there. In that case, we should use MMU
+ * transparent mode, (i.e. ASI 0x20 to 0x2f) to access
+ * physical memory.
+ */
+
+ ctxtblptr = (lda(SRMMU_CXTPTR,ASI_SRMMU) << SRMMU_PPNPASHIFT);
+
+ /* We're going to have to use MMU passthrough. If we're on a
+ * Viking MicroSparc without an mbus, we need to turn off traps
+ * and set the AC bit at 0x8000 in the MMU's control register. Ugh.
+ * XXX: Once we've done this, can we still access kernel vm?
+ */
+ if ((getpsr() & 0x40000000) && (!(lda(SRMMU_PCR,ASI_SRMMU) & 0x800))) {
+ wierdviking = 1;
+ sta(SRMMU_PCR, ASI_SRMMU, /* set MMU AC bit */
+ ((mmupcrsav = lda(SRMMU_PCR,ASI_SRMMU)) | SRMMU_PCR_AC));
+ }
+ if (ncontext < 1024) {
+ /* to keep supersparc happy, still align on 4K boundary */
+ ctx_phys_tbl = (int *)roundup((u_int) *kmemtop, 4096);
+ *kmemtop = (caddr_t)((u_int)ctx_phys_tbl + 4096);
+ qzero(ctx_phys_tbl, 4096);
+ } else {
+ ctx_phys_tbl = (int *)roundup((u_int) *kmemtop,
+ ncontext * sizeof(int));
+ *kmemtop = (caddr_t)((u_int)ctx_phys_tbl +
+ (ncontext * sizeof(int)));
+ qzero(ctx_phys_tbl, ncontext * sizeof(int));
+ }
+
+ te = lda(ctxtblptr, ASI_BYPASS);
+ switch (te & SRMMU_TETYPE) {
+ case SRMMU_TEINVALID:
+ ctx_phys_tbl[0] = SRMMU_TEINVALID;
+ panic("mmu_reservemon4m: no existing L0 mapping! (How are we running?");
+ break;
+ case SRMMU_TEPTE:
+#ifdef DEBUG
+ printf("mmu_reservemon4m: trying to remap 4G segment!\n");
+#endif
+ panic("mmu_reservemon4m: can't handle ROM 4G page size");
+ /* XXX: Should make this work, however stupid it is */
+ break;
+ case SRMMU_TEPTD:
+ tableptr = (caddr_t) roundup((u_int) *kmemtop,
+ SRMMU_L1SIZE * sizeof(long));
+ *kmemtop = tableptr + (SRMMU_L1SIZE * sizeof(long));
+ bzero(tableptr, *kmemtop - tableptr);
+ ctx_phys_tbl[0] = (VA2PA(tableptr) >> SRMMU_PPNPASHIFT) |
+ SRMMU_TEPTD;
+ kpmap->pm_reg_ptps = (int *)tableptr;
+ kpmap->pm_reg_ptps_pa = VA2PA(tableptr);
+ mmu_setup4m_L1(te, tableptr, kpmap, kmemtop);
+ break;
+ default:
+ panic("mmu_reservemon4m: unknown pagetable entry type");
+ }
+
+ if (wierdviking == 1) {
+ sta(SRMMU_PCR, ASI_SRMMU, mmupcrsav);
+ }
+}
+
+void
+mmu_setup4m_L1(regtblptd, newtableptr, kpmap, kmemtop)
+ int regtblptd; /* PTD for region table to be remapped */
+ caddr_t newtableptr; /* Virtual ptr to new region table */
+ struct pmap *kpmap;
+ register caddr_t *kmemtop;
+{
+ register unsigned int regtblrover;
+ register int i;
+ unsigned int te;
+ caddr_t segtblptr;
+ register caddr_t pagtblptr;
+ struct regmap *thisregmap = NULL;
+ struct segmap *segmaps;
+ int j, k;
+
+ /* Here we scan the region table to copy any entries which appear.
+ * We are only concerned with regions in kernel space and above
+ * (i.e. regions 0xf8 to 0xff). We also ignore region 0xf8, since
+ * that is the 16MB L1 mapping that the ROM used to map the kernel
+ * in initially. Later, we will rebuild a new L3 mapping for the
+ * kernel and install it before switching to the new pagetables.
+ */
+ regtblrover = ((regtblptd & ~SRMMU_TETYPE) << SRMMU_PPNPASHIFT) +
+ (VA_VREG(KERNBASE)+1) * sizeof(long); /* kernel only */
+
+ for (i = VA_VREG(KERNBASE) + 1; i < SRMMU_L1SIZE;
+ i++, regtblrover += sizeof(long)) {
+ te = lda(regtblrover, ASI_BYPASS);
+ switch(te & SRMMU_TETYPE) {
+ case SRMMU_TEINVALID:
+ ((int *)newtableptr)[i] = SRMMU_TEINVALID;
+ break;
+ case SRMMU_TEPTE:
+#ifdef DEBUG
+ printf("mmu_reservemon4m: converting region 0x%x from L1->L3\n",i);
+#endif
+ /* We must build segment and page tables, then
+ * fill them in
+ */
+ thisregmap = &kpmap->pm_regmap[i];
+ bzero(thisregmap, sizeof(struct regmap));
+ segmaps = &kernel_segmap_store[(i-VA_VREG(KERNBASE))*NSEGRG];
+ segtblptr = (caddr_t) roundup((u_int) *kmemtop,
+ SRMMU_L2SIZE * sizeof(long));
+ *kmemtop = segtblptr + (SRMMU_L2SIZE * sizeof(long));
+ bzero(segtblptr, *kmemtop - segtblptr);
+ bzero(segmaps, NSEGRG * sizeof(struct segmap));
+ thisregmap->rg_segmap = segmaps;
+ thisregmap->rg_seg_ptps = (int *)segtblptr;
+ ((int *)newtableptr)[i] = (VA2PA(segtblptr) >>
+ SRMMU_PPNPASHIFT) |
+ SRMMU_TEPTD;
+ for (j = 0; j < SRMMU_L2SIZE; j++) {
+ thisregmap->rg_nsegmap++;
+ pagtblptr = (caddr_t) roundup((u_int) *kmemtop,
+ SRMMU_L3SIZE * sizeof(int));
+ *kmemtop = segtblptr + (SRMMU_L3SIZE *
+ sizeof(long));
+ bzero(pagtblptr, *kmemtop - pagtblptr);
+ ((int *)segtblptr)[j] = (VA2PA(pagtblptr) >>
+ SRMMU_PPNPASHIFT) |
+ SRMMU_TEPTD;
+ segmaps[j].sg_pte = (int *)pagtblptr;
+
+ for (k = 0; k < SRMMU_L3SIZE; k++) {
+ segmaps[j].sg_npte++;
+ ((int *)pagtblptr)[k] =
+ (te & SRMMU_L1PPNMASK) |
+ (j << SRMMU_L2PPNSHFT) |
+ (k << SRMMU_L3PPNSHFT) |
+ (te & SRMMU_PGBITSMSK) |
+ ((te & SRMMU_PROT_MASK) |
+ PPROT_U2S_OMASK) |
+ SRMMU_TEPTE;
+ }
+ }
+ break;
+ case SRMMU_TEPTD:
+ thisregmap = &kpmap->pm_regmap[i];
+ bzero(thisregmap, sizeof(struct regmap));
+ segtblptr = (caddr_t) roundup((u_int)*kmemtop,
+ SRMMU_L2SIZE * sizeof(long));
+ *kmemtop = segtblptr + (SRMMU_L2SIZE * sizeof(long));
+ segmaps = &kernel_segmap_store[(i-(KERNBASE >> RGSHIFT))*NSEGRG];
+ bzero(segtblptr, *kmemtop - segtblptr);
+ bzero(segmaps, NSEGRG * sizeof(struct segmap));
+ thisregmap->rg_segmap = segmaps;
+ thisregmap->rg_seg_ptps = (int *)segtblptr;
+ ((int *)newtableptr)[i] = (VA2PA(segtblptr) >>
+ SRMMU_PPNPASHIFT) |
+ SRMMU_TEPTD;
+ mmu_setup4m_L2(te, segtblptr, thisregmap, kmemtop, segmaps);
+ break;
+ default:
+ panic("mmu_setup4m_L1: unknown pagetable entry type");
+ }
+ }
+}
+
+void
+mmu_setup4m_L2(segtblptd, newtableptr, pregmap, kmemtop, segmaps)
+ int segtblptd;
+ caddr_t newtableptr;
+ struct regmap *pregmap;
+ register caddr_t *kmemtop;
+ struct segmap *segmaps;
+{
+ register unsigned int segtblrover;
+ register int i, k;
+ unsigned int te;
+ register caddr_t pagtblptr;
+ struct segmap *thissegmap = NULL;
+
+ segtblrover = (segtblptd & ~SRMMU_TETYPE) << SRMMU_PPNPASHIFT;
+ for (i = 0; i < SRMMU_L2SIZE; i++, segtblrover += sizeof(long)) {
+ te = lda(segtblrover, ASI_BYPASS);
+ switch(te & SRMMU_TETYPE) {
+ case SRMMU_TEINVALID:
+ ((int *)newtableptr)[i] = SRMMU_TEINVALID;
+ break;
+ case SRMMU_TEPTE:
+#ifdef DEBUG
+ printf("mmu_reservemon4m: converting L2 entry at segment 0x%x to L3\n",i);
+#endif
+ pregmap->rg_nsegmap++;
+ /* We must build page tables and fill them in */
+ pagtblptr = (caddr_t) roundup((u_int) *kmemtop,
+ SRMMU_L3SIZE*sizeof(int));
+ *kmemtop = pagtblptr + (SRMMU_L3SIZE *
+ sizeof(int));
+ bzero(pagtblptr, *kmemtop - pagtblptr);
+ ((int *)newtableptr)[i] = (VA2PA(pagtblptr) >>
+ SRMMU_PPNPASHIFT) |
+ SRMMU_TEPTD;
+ segmaps[i].sg_pte = (int *) pagtblptr;
+
+ for (k = 0; k < SRMMU_L3SIZE; k++) {
+ segmaps[i].sg_npte++;
+ ((int *)pagtblptr)[k] =
+ (te & SRMMU_L1PPNMASK) |
+ (te & SRMMU_L2PPNMASK) |
+ (k << SRMMU_L3PPNSHFT) |
+ (te & SRMMU_PGBITSMSK) |
+ ((te & SRMMU_PROT_MASK) |
+ PPROT_U2S_OMASK) |
+ SRMMU_TEPTE;
+ }
+ break;
+ case SRMMU_TEPTD:
+ pregmap->rg_nsegmap++;
+ thissegmap = &segmaps[i];
+ pagtblptr = (caddr_t) roundup((u_int) *kmemtop,
+ SRMMU_L3SIZE*sizeof(int));
+ *kmemtop = pagtblptr + (SRMMU_L3SIZE * sizeof(int));
+ bzero(pagtblptr, *kmemtop - pagtblptr);
+ ((int *)newtableptr)[i] = (VA2PA(pagtblptr) >>
+ SRMMU_PPNPASHIFT) |
+ SRMMU_TEPTD;
+ thissegmap->sg_pte = (int *) pagtblptr;
+ thissegmap->sg_npte += mmu_setup4m_L3(te, pagtblptr);
+ break;
+ default:
+ panic("mmu_setup4m_L2: unknown pagetable entry type");
+ }
+ }
+}
+
+int
+mmu_setup4m_L3(pagtblptd, newtableptr)
+ register int pagtblptd;
+ register caddr_t newtableptr;
+{
+ register unsigned int pagtblrover;
+ register int i, n = 0;
+ register unsigned int te;
+
+ pagtblrover = (pagtblptd & ~SRMMU_TETYPE) << SRMMU_PPNPASHIFT;
+ for (i = 0; i < SRMMU_L3SIZE; i++, pagtblrover += sizeof(long)) {
+ te = lda(pagtblrover, ASI_BYPASS);
+ switch(te & SRMMU_TETYPE) {
+ case SRMMU_TEINVALID:
+ ((int *)newtableptr)[i] = SRMMU_TEINVALID;
+ break;
+ case SRMMU_TEPTE:
+ ((int *)newtableptr)[i] = te | PPROT_U2S_OMASK;
+ n++;
+ break;
+ case SRMMU_TEPTD:
+ panic("mmu_setup4m_L3: PTD found in L3 page table");
+ default:
+ panic("mmu_setup4m_L3: unknown pagetable entry type");
+ }
+ }
+ return n;
+}
+#endif /* defined SUN4M */
/*
* TODO: agree with the ROM on physical pages by taking them away
@@ -626,6 +1271,7 @@ void region_free __P((struct pmap *, u_int));
setcontext(new); \
}
+#if defined(SUN4) || defined(SUN4C) /* This is old sun MMU stuff */
/*
* Allocate an MMU entry (i.e., a PMEG).
* If necessary, steal one from someone else.
@@ -686,7 +1332,7 @@ me_alloc(mh, newpm, newvreg, newvseg)
panic("me_alloc: stealing from kernel");
#ifdef DEBUG
if (pmapdebug & (PDB_MMU_ALLOC | PDB_MMU_STEAL))
- printf("me_alloc: stealing pmeg %x from pmap %x\n",
+ printf("me_alloc: stealing pmeg %x from pmap %p\n",
me->me_cookie, pm);
#endif
/*
@@ -742,11 +1388,11 @@ me_alloc(mh, newpm, newvreg, newvseg)
*/
i = NPTESG;
do {
- tpte = getpte(va);
+ tpte = getpte4(va);
if ((tpte & (PG_V | PG_TYPE)) == (PG_V | PG_OBMEM)) {
- pa = ptoa(HWTOSW(tpte & PG_PFNUM));
+ pa = ptoa(tpte & PG_PFNUM);
if (managed(pa))
- pvhead(pa)->pv_flags |= MR(tpte);
+ pvhead(pa)->pv_flags |= MR4_4C(tpte);
}
*pte++ = tpte & ~(PG_U|PG_M);
va += NBPG;
@@ -799,7 +1445,7 @@ me_free(pm, pmeg)
#ifdef DEBUG
if (pmapdebug & PDB_MMU_ALLOC)
- printf("me_free: freeing pmeg %d from pmap %x\n",
+ printf("me_free: freeing pmeg %d from pmap %p\n",
me->me_cookie, pm);
if (me->me_cookie != pmeg)
panic("me_free: wrong mmuentry");
@@ -825,11 +1471,11 @@ if (getcontext() != 0) panic("me_free: ctx != 0");
}
i = NPTESG;
do {
- tpte = getpte(va);
+ tpte = getpte4(va);
if ((tpte & (PG_V | PG_TYPE)) == (PG_V | PG_OBMEM)) {
- pa = ptoa(HWTOSW(tpte & PG_PFNUM));
+ pa = ptoa(tpte & PG_PFNUM);
if (managed(pa))
- pvhead(pa)->pv_flags |= MR(tpte);
+ pvhead(pa)->pv_flags |= MR4_4C(tpte);
}
va += NBPG;
} while (--i > 0);
@@ -865,10 +1511,8 @@ region_alloc(mh, newpm, newvr)
{
register struct mmuentry *me;
register struct pmap *pm;
- register int i, va, pa;
int ctx;
struct regmap *rp;
- struct segmap *sp;
/* try free list first */
if ((me = region_freelist.tqh_first) != NULL) {
@@ -903,7 +1547,7 @@ region_alloc(mh, newpm, newvr)
panic("region_alloc: stealing from kernel");
#ifdef DEBUG
if (pmapdebug & (PDB_MMUREG_ALLOC | PDB_MMUREG_STEAL))
- printf("region_alloc: stealing smeg %x from pmap %x\n",
+ printf("region_alloc: stealing smeg %x from pmap %p\n",
me->me_cookie, pm);
#endif
/*
@@ -961,7 +1605,7 @@ region_free(pm, smeg)
#ifdef DEBUG
if (pmapdebug & PDB_MMUREG_ALLOC)
- printf("region_free: freeing smeg %x from pmap %x\n",
+ printf("region_free: freeing smeg %x from pmap %p\n",
me->me_cookie, pm);
if (me->me_cookie != smeg)
panic("region_free: wrong mmuentry");
@@ -1001,11 +1645,9 @@ region_free(pm, smeg)
int
mmu_pagein(pm, va, prot)
register struct pmap *pm;
- register int va;
- vm_prot_t prot;
+ register int va, prot;
{
register int *pte;
- register struct mmuentry *me;
register int vr, vs, pmeg, i, s, bits;
struct regmap *rp;
struct segmap *sp;
@@ -1051,7 +1693,7 @@ printf("mmu_pagein: kernel wants map at va %x, vr %d, vs %d\n", va, vr, vs);
/* return -1 if the fault is `hard', 0 if not */
if (sp->sg_pmeg != seginval)
- return (bits && (getpte(va) & bits) == bits ? -1 : 0);
+ return (bits && (getpte4(va) & bits) == bits ? -1 : 0);
/* reload segment: write PTEs into a new LRU entry */
va = VA_ROUNDDOWNTOSEG(va);
@@ -1060,12 +1702,62 @@ printf("mmu_pagein: kernel wants map at va %x, vr %d, vs %d\n", va, vr, vs);
setsegmap(va, pmeg);
i = NPTESG;
do {
- setpte(va, *pte++);
+ setpte4(va, *pte++);
va += NBPG;
} while (--i > 0);
splx(s);
return (1);
}
+#endif /* defined SUN4 or SUN4C */
+
+#if defined(DEBUG) && defined(SUN4M)
+/*
+ * `Page in' (load or inspect) an MMU entry; called on page faults.
+ * Returns -1 if the desired page was marked valid (in which case the
+ * fault must be a bus error or something), or 0 (segment loaded but
+ * PTE not valid, or segment not loaded at all).
+ *
+ * The SRMMU does not have the concept of `loading PMEGs into the MMU'.
+ * For now, we use it to debug certain sporadic and strange memory
+ * fault traps.
+ */
+int
+mmu_pagein4m(pm, va, prot)
+ register struct pmap *pm;
+ register int va, prot;
+{
+ register int pte;
+ register int bits;
+
+ pte = getpte4m(va);
+ if ((pte & SRMMU_TETYPE) != SRMMU_TEPTE)
+ /* pte not found == not valid */
+ return 0;
+
+ /*
+ * If pte has "correct" protection (according to bits), then
+ * something bizarre has happened, so we return -1 to generate
+ * a fault. Otherwise, we return 0 and let the VM code reload
+ * the page.
+ * XXX: Does this actually work, or will it cause a loop?
+ */
+
+ if (prot != VM_PROT_NONE)
+ bits = ((prot & VM_PROT_WRITE) ? PPROT_RWX_RWX : PPROT_RX_RX);
+ else
+ bits = 0;
+ /* If we're in kernelland, mask out user RWX */
+/* if (getcontext() == 0)
+ bits |= PPROT_S;
+*/
+ if (bits && (pte & bits) == bits) {
+ printf("pagein4m(%s[%d]): OOPS: prot=%x, va=%x, pte=%x, bits=%x\n",
+ curproc->p_comm, curproc->p_pid, prot, va, pte, bits);
+ return -1;
+ }
+ return 0;
+}
+#endif
/*
* Allocate a context. If necessary, steal one from someone else.
@@ -1084,14 +1776,17 @@ ctx_alloc(pm)
register int gap_start, gap_end;
register unsigned long va;
+/*XXX-GCC!*/gap_start=gap_end=0;
#ifdef DEBUG
if (pm->pm_ctx)
panic("ctx_alloc pm_ctx");
if (pmapdebug & PDB_CTX_ALLOC)
- printf("ctx_alloc(%x)\n", pm);
+ printf("ctx_alloc(%p)\n", pm);
#endif
- gap_start = pm->pm_gap_start;
- gap_end = pm->pm_gap_end;
+ if (CPU_ISSUN4OR4C) {
+ gap_start = pm->pm_gap_start;
+ gap_end = pm->pm_gap_end;
+ }
s = splpmap();
if ((c = ctx_freelist) != NULL) {
@@ -1111,67 +1806,134 @@ ctx_alloc(pm)
if (c->c_pmap == NULL)
panic("ctx_alloc cu_pmap");
if (pmapdebug & (PDB_CTX_ALLOC | PDB_CTX_STEAL))
- printf("ctx_alloc: steal context %x from %x\n",
+ printf("ctx_alloc: steal context %d from %p\n",
cnum, c->c_pmap);
#endif
c->c_pmap->pm_ctx = NULL;
doflush = (vactype != VAC_NONE);
- if (gap_start < c->c_pmap->pm_gap_start)
- gap_start = c->c_pmap->pm_gap_start;
- if (gap_end > c->c_pmap->pm_gap_end)
- gap_end = c->c_pmap->pm_gap_end;
+ if (CPU_ISSUN4OR4C) {
+ if (gap_start < c->c_pmap->pm_gap_start)
+ gap_start = c->c_pmap->pm_gap_start;
+ if (gap_end > c->c_pmap->pm_gap_end)
+ gap_end = c->c_pmap->pm_gap_end;
+ }
}
- setcontext(cnum);
c->c_pmap = pm;
pm->pm_ctx = c;
pm->pm_ctxnum = cnum;
- splx(s);
- if (doflush)
- cache_flush_context();
- /*
- * Write pmap's region (3-level MMU) or segment table into the MMU.
- *
- * Only write those entries that actually map something in this
- * context by maintaining a pair of region numbers in between
- * which the pmap has no valid mappings.
- *
- * If a context was just allocated from the free list, trust that
- * all its pmeg numbers are `seginval'. We make sure this is the
- * case initially in pmap_bootstrap(). Otherwise, the context was
- * freed by calling ctx_free() in pmap_release(), which in turn is
- * supposedly called only when all mappings have been removed.
- *
- * On the other hand, if the context had to be stolen from another
- * pmap, we possibly shrink the gap to be the disjuction of the new
- * and the previous map.
- */
+ if (CPU_ISSUN4OR4C) {
+ /*
+ * Write pmap's region (3-level MMU) or segment table into
+ * the MMU.
+ *
+ * Only write those entries that actually map something in
+ * this context by maintaining a pair of region numbers in
+ * between which the pmap has no valid mappings.
+ *
+ * If a context was just allocated from the free list, trust
+ * that all its pmeg numbers are `seginval'. We make sure this
+ * is the case initially in pmap_bootstrap(). Otherwise, the
+ * context was freed by calling ctx_free() in pmap_release(),
+ * which in turn is supposedly called only when all mappings
+ * have been removed.
+ *
+ * On the other hand, if the context had to be stolen from
+ * another pmap, we possibly shrink the gap to be the
+ * disjuction of the new and the previous map.
+ */
+ setcontext(cnum);
+ splx(s);
+ if (doflush)
+ cache_flush_context();
- rp = pm->pm_regmap;
- for (va = 0, i = NUREG; --i >= 0; ) {
- if (VA_VREG(va) >= gap_start) {
- va = VRTOVA(gap_end);
- i -= gap_end - gap_start;
- rp += gap_end - gap_start;
- if (i < 0)
- break;
- gap_start = NUREG; /* mustn't re-enter this branch */
- }
+ rp = pm->pm_regmap;
+ for (va = 0, i = NUREG; --i >= 0; ) {
+ if (VA_VREG(va) >= gap_start) {
+ va = VRTOVA(gap_end);
+ i -= gap_end - gap_start;
+ rp += gap_end - gap_start;
+ if (i < 0)
+ break;
+ /* mustn't re-enter this branch */
+ gap_start = NUREG;
+ }
#ifdef MMU_3L
- if (mmu_3l) {
- setregmap(va, rp++->rg_smeg);
- va += NBPRG;
- } else
+ if (mmu_3l) {
+ setregmap(va, rp++->rg_smeg);
+ va += NBPRG;
+ } else
#endif
- {
- register int j;
- register struct segmap *sp = rp->rg_segmap;
- for (j = NSEGRG; --j >= 0; va += NBPSG)
- setsegmap(va, sp?sp++->sg_pmeg:seginval);
- rp++;
+ {
+ register int j;
+ register struct segmap *sp = rp->rg_segmap;
+ for (j = NSEGRG; --j >= 0; va += NBPSG)
+ setsegmap(va,
+ sp?sp++->sg_pmeg:seginval);
+ rp++;
+ }
}
+
+ } else if (CPU_ISSUN4M) {
+
+ /*
+ * Reload page and context tables to activate the page tables
+ * for this context.
+ *
+ * The gap stuff isn't really needed in the Sun4m architecture,
+ * since we don't have to worry about excessive mappings (all
+ * mappings exist since the page tables must be complete for
+ * the mmu to be happy).
+ *
+ * If a context was just allocated from the free list, trust
+ * that all of its mmu-edible page tables are zeroed out
+ * (except for those associated with the kernel). We make
+ * sure this is the case initially in pmap_bootstrap() and
+ * pmap_init() (?).
+ * Otherwise, the context was freed by calling ctx_free() in
+ * pmap_release(), which in turn is supposedly called only
+ * when all mappings have been removed.
+ *
+ * XXX: Do we have to flush cache after reloading ctx tbl?
+ */
+
+ /*
+ * We install kernel mappings into the pmap here, since when
+ * the kernel expands it only propagates the expansion to pmaps
+ * corresponding to valid contexts. Thus it is possible (and
+ * it has happened!) that a pmap is created just before
+ * the kernel expands, but the pmap gets a context *after*
+ * the kernel expands, thus missing getting mappings.
+ */
+ qcopy(&pmap_kernel()->pm_reg_ptps[VA_VREG(KERNBASE)],
+ &pm->pm_reg_ptps[VA_VREG(KERNBASE)],
+ NKREG * sizeof(int));
+ /*
+ * We must also install the regmap/segmap/etc stuff for
+ * kernel maps.
+ */
+ qcopy(&pmap_kernel()->pm_regmap[VA_VREG(KERNBASE)],
+ &pm->pm_regmap[VA_VREG(KERNBASE)],
+ NKREG * sizeof(struct regmap));
+
+ ctxbusyvector[cnum] = 1; /* mark context as busy */
+#ifdef DEBUG
+ if (pm->pm_reg_ptps_pa == 0)
+ panic("ctx_alloc: no region table in current pmap");
+#endif
+ /*setcontext(0); * paranoia? can we modify curr. ctx? */
+ ctx_phys_tbl[cnum] =
+ (pm->pm_reg_ptps_pa >> SRMMU_PPNPASHIFT) | SRMMU_TEPTD;
+
+ setcontext(cnum);
+ if (doflush)
+ cache_flush_context();
+#if defined(SUN4M)
+ tlb_flush_context(); /* remove any remnant garbage from tlb */
+#endif
+ splx(s);
}
}
@@ -1189,16 +1951,35 @@ ctx_free(pm)
panic("ctx_free");
pm->pm_ctx = NULL;
oldc = getcontext();
+
if (vactype != VAC_NONE) {
newc = pm->pm_ctxnum;
CHANGE_CONTEXTS(oldc, newc);
cache_flush_context();
+#if defined(SUN4M)
+ if (CPU_ISSUN4M)
+ tlb_flush_context();
+#endif
setcontext(0);
} else {
+#if defined(SUN4M)
+ if (CPU_ISSUN4M)
+ tlb_flush_context();
+#endif
CHANGE_CONTEXTS(oldc, 0);
}
c->c_nextfree = ctx_freelist;
ctx_freelist = c;
+
+#if defined(SUN4M)
+ if (CPU_ISSUN4M) {
+ /* Map kernel back into unused context */
+ newc = pm->pm_ctxnum;
+ ctx_phys_tbl[newc] = ctx_phys_tbl[0];
+ if (newc)
+ ctxbusyvector[newc] = 0; /* mark as free */
+ }
+#endif
}
@@ -1221,15 +2002,18 @@ ctx_free(pm)
* It also copies ref and mod bits to the pvlist, on the theory that
* this might save work later. (XXX should test this theory)
*/
+
+#if defined(SUN4) || defined(SUN4C)
+
void
-pv_changepte(pv0, bis, bic)
+pv_changepte4_4c(pv0, bis, bic)
register struct pvlist *pv0;
register int bis, bic;
{
register int *pte;
register struct pvlist *pv;
register struct pmap *pm;
- register int va, vr, vs, i, flags;
+ register int va, vr, vs, flags;
int ctx, s;
struct regmap *rp;
struct segmap *sp;
@@ -1281,7 +2065,7 @@ if(pm==NULL)panic("pv_changepte 1");
continue;
setcontext(pm->pm_ctxnum);
/* XXX should flush only when necessary */
- tpte = getpte(va);
+ tpte = getpte4(va);
if (vactype != VAC_NONE && (tpte & PG_M))
cache_flush_page(va);
} else {
@@ -1293,13 +2077,12 @@ if(pm==NULL)panic("pv_changepte 1");
#endif
setsegmap(0, sp->sg_pmeg);
va = VA_VPG(va) << PGSHIFT;
- tpte = getpte(va);
+ tpte = getpte4(va);
}
if (tpte & PG_V)
- flags |= (tpte >> PG_M_SHIFT) &
- (PV_MOD|PV_REF);
+ flags |= (tpte >> PG_M_SHIFT) & (PV_MOD|PV_REF);
tpte = (tpte | bis) & ~bic;
- setpte(va, tpte);
+ setpte4(va, tpte);
if (pte != NULL) /* update software copy */
pte[VA_VPG(va)] = tpte;
}
@@ -1317,12 +2100,12 @@ if(pm==NULL)panic("pv_changepte 1");
* hence never need to adjust software copies.
*/
int
-pv_syncflags(pv0)
+pv_syncflags4_4c(pv0)
register struct pvlist *pv0;
{
register struct pvlist *pv;
register struct pmap *pm;
- register int tpte, va, vr, vs, pmeg, i, flags;
+ register int tpte, va, vr, vs, pmeg, flags;
int ctx, s;
struct regmap *rp;
struct segmap *sp;
@@ -1352,7 +2135,7 @@ pv_syncflags(pv0)
if (CTX_USABLE(pm,rp)) {
setcontext(pm->pm_ctxnum);
/* XXX should flush only when necessary */
- tpte = getpte(va);
+ tpte = getpte4(va);
if (vactype != VAC_NONE && (tpte & PG_M))
cache_flush_page(va);
} else {
@@ -1364,13 +2147,13 @@ pv_syncflags(pv0)
#endif
setsegmap(0, pmeg);
va = VA_VPG(va) << PGSHIFT;
- tpte = getpte(va);
+ tpte = getpte4(va);
}
if (tpte & (PG_M|PG_U) && tpte & PG_V) {
flags |= (tpte >> PG_M_SHIFT) &
(PV_MOD|PV_REF);
tpte &= ~(PG_M|PG_U);
- setpte(va, tpte);
+ setpte4(va, tpte);
}
}
pv0->pv_flags = flags;
@@ -1391,7 +2174,7 @@ pv_syncflags(pv0)
* in it to have PV_NC set, and we only remove one here.)
*/
/*static*/ void
-pv_unlink(pv, pm, va)
+pv_unlink4_4c(pv, pm, va)
register struct pvlist *pv;
register struct pmap *pm;
register vm_offset_t va;
@@ -1437,7 +2220,7 @@ pv_unlink(pv, pm, va)
if (BADALIAS(va, npv->pv_va))
return;
pv->pv_flags &= ~PV_NC;
- pv_changepte(pv, 0, PG_NC);
+ pv_changepte4_4c(pv, 0, PG_NC);
}
}
@@ -1447,7 +2230,7 @@ pv_unlink(pv, pm, va)
* be cached.
*/
/*static*/ int
-pv_link(pv, pm, va)
+pv_link4_4c(pv, pm, va)
register struct pvlist *pv;
register struct pmap *pm;
register vm_offset_t va;
@@ -1479,12 +2262,271 @@ pv_link(pv, pm, va)
if (BADALIAS(va, npv->pv_va)) {
#ifdef DEBUG
if (pmapdebug) printf(
- "pv_link: badalias: pid %d, %x<=>%x, pa %x\n",
+ "pv_link: badalias: pid %d, %lx<=>%x, pa %lx\n",
curproc?curproc->p_pid:-1, va, npv->pv_va,
vm_first_phys + (pv-pv_table)*NBPG);
#endif
pv->pv_flags |= PV_NC;
- pv_changepte(pv, ret = PG_NC, 0);
+ pv_changepte4_4c(pv, ret = PG_NC, 0);
+ break;
+ }
+ }
+ }
+ npv = (struct pvlist *)malloc(sizeof *npv, M_VMPVENT, M_WAITOK);
+ npv->pv_next = pv->pv_next;
+ npv->pv_pmap = pm;
+ npv->pv_va = va;
+ pv->pv_next = npv;
+ return (ret);
+}
+
+#endif /* sun4, sun4c code */
+
+#if defined(SUN4M) /* Sun4M versions of above */
+/*
+ * Walk the given pv list, and for each PTE, set or clear some bits
+ * (e.g., PG_W or PG_NC).
+ *
+ * As a special case, this never clears PG_W on `pager' pages.
+ * These, being kernel addresses, are always in hardware and have
+ * a context.
+ *
+ * This routine flushes the cache for any page whose PTE changes,
+ * as long as the process has a context; this is overly conservative.
+ * It also copies ref and mod bits to the pvlist, on the theory that
+ * this might save work later. (XXX should test this theory)
+ */
+void
+pv_changepte4m(pv0, bis, bic)
+ register struct pvlist *pv0;
+ register int bis, bic;
+{
+ register struct pvlist *pv;
+ register struct pmap *pm;
+ register int va, vr, flags;
+ int ctx, s;
+ struct regmap *rp;
+
+ write_user_windows(); /* paranoid? */
+
+ s = splpmap(); /* paranoid? */
+ if (pv0->pv_pmap == NULL) {
+ splx(s);
+ return;
+ }
+ ctx = getcontext();
+ flags = pv0->pv_flags;
+ for (pv = pv0; pv != NULL; pv = pv->pv_next) {
+ register int tpte;
+ pm = pv->pv_pmap;
+ if (pm == NULL)
+ panic("pv_changepte 1");
+ va = pv->pv_va;
+ vr = VA_VREG(va);
+ rp = &pm->pm_regmap[vr];
+ if (rp->rg_segmap == NULL)
+ panic("pv_changepte: no segments");
+
+ if (CTX_USABLE(pm,rp)) {
+ extern vm_offset_t pager_sva, pager_eva;
+
+ /*
+ * Bizarreness: we never clear PG_W on
+ * pager pages, nor set PG_C on DVMA pages.
+ */
+ if ((bic & PPROT_WRITE) &&
+ va >= pager_sva && va < pager_eva)
+ continue;
+ if ((bis & SRMMU_PG_C) &&
+ va >= DVMA_BASE && va < DVMA_END)
+ continue;
+ setcontext(pm->pm_ctxnum);
+ /* %%%: Do we need to always flush? */
+ tpte = getpte4m(va);
+ if (vactype != VAC_NONE && (tpte & SRMMU_PG_M))
+ cache_flush_page(va);
+ } else {
+ /* PTE that we want has no context. Use sw getpte*/
+ tpte = getptesw4m(pm, va);
+ }
+ if ((tpte & SRMMU_TETYPE) == SRMMU_TEPTE) /* i.e. if valid pte */
+ flags |= (tpte >> PG_M_SHIFT4M) & (PV_MOD4M|PV_REF4M|PV_C4M);
+
+ tpte = (tpte | bis) & ~bic;
+
+ if (CTX_USABLE(pm,rp))
+ setpte4m(va, tpte);
+ else
+ setptesw4m(pm, va, tpte);
+ }
+ pv0->pv_flags = flags;
+ setcontext(ctx);
+ splx(s);
+}
+
+/*
+ * Sync ref and mod bits in pvlist. If page has been ref'd or modified,
+ * update ref/mod bits in pvlist, and clear the hardware bits.
+ *
+ * Return the new flags.
+ */
+int
+pv_syncflags4m(pv0)
+ register struct pvlist *pv0;
+{
+ register struct pvlist *pv;
+ register struct pmap *pm;
+ register int tpte, va, vr, vs, flags;
+ int ctx, s;
+ struct regmap *rp;
+ struct segmap *sp;
+
+ write_user_windows(); /* paranoid? */
+
+ s = splpmap(); /* paranoid? */
+ if (pv0->pv_pmap == NULL) { /* paranoid */
+ splx(s);
+ return (0);
+ }
+ ctx = getcontext();
+ flags = pv0->pv_flags;
+ for (pv = pv0; pv != NULL; pv = pv->pv_next) {
+ pm = pv->pv_pmap;
+ va = pv->pv_va;
+ vr = VA_VREG(va);
+ vs = VA_VSEG(va);
+ rp = &pm->pm_regmap[vr];
+ if (rp->rg_segmap == NULL)
+ panic("pv_syncflags: no segments");
+ sp = &rp->rg_segmap[vs];
+
+ if (sp->sg_pte == NULL) /* invalid */
+ continue;
+
+ /*
+ * We need the PTE from memory as the TLB version will
+ * always have the SRMMU_PG_R bit on.
+ */
+ if (CTX_USABLE(pm,rp)) {
+ setcontext(pm->pm_ctxnum);
+ tlb_flush_page(va);
+ }
+ tpte = getptesw4m(pm, va);
+
+ if ((tpte & SRMMU_TETYPE) == SRMMU_TEPTE && /* if valid pte */
+ (tpte & (SRMMU_PG_M|SRMMU_PG_R))) { /* and mod/refd */
+ flags |= (tpte >> PG_M_SHIFT4M) &
+ (PV_MOD4M|PV_REF4M|PV_C4M);
+ tpte &= ~(SRMMU_PG_M | SRMMU_PG_R);
+ /* TLB has been invalidated, so just update memory */
+ setptesw4m(pm, va, tpte);
+ if (vactype != VAC_NONE &&
+ CTX_USABLE(pm,rp) && (tpte & SRMMU_PG_M))
+ cache_flush_page(va); /* XXX: do we need this?*/
+ }
+ }
+ pv0->pv_flags = flags;
+ setcontext(ctx);
+ splx(s);
+ return (flags);
+}
+
+void
+pv_unlink4m(pv, pm, va)
+ register struct pvlist *pv;
+ register struct pmap *pm;
+ register vm_offset_t va;
+{
+ register struct pvlist *npv;
+
+#ifdef DIAGNOSTIC
+ if (pv->pv_pmap == NULL)
+ panic("pv_unlink0");
+#endif
+ /*
+ * First entry is special (sigh).
+ */
+ npv = pv->pv_next;
+ if (pv->pv_pmap == pm && pv->pv_va == va) {
+ pmap_stats.ps_unlink_pvfirst++;
+ if (npv != NULL) {
+ pv->pv_next = npv->pv_next;
+ pv->pv_pmap = npv->pv_pmap;
+ pv->pv_va = npv->pv_va;
+ free(npv, M_VMPVENT);
+ } else
+ pv->pv_pmap = NULL;
+ } else {
+ register struct pvlist *prev;
+
+ for (prev = pv;; prev = npv, npv = npv->pv_next) {
+ pmap_stats.ps_unlink_pvsearch++;
+ if (npv == NULL)
+ panic("pv_unlink");
+ if (npv->pv_pmap == pm && npv->pv_va == va)
+ break;
+ }
+ prev->pv_next = npv->pv_next;
+ free(npv, M_VMPVENT);
+ }
+ if (!(pv->pv_flags & PV_C4M)) {
+ /*
+ * Not cached: check to see if we can fix that now.
+ */
+ va = pv->pv_va;
+ for (npv = pv->pv_next; npv != NULL; npv = npv->pv_next)
+ if (BADALIAS(va, npv->pv_va))
+ return;
+ pv->pv_flags |= PV_C4M;
+ pv_changepte4m(pv, SRMMU_PG_C, 0);
+ }
+}
+
+/*
+ * pv_link is the inverse of pv_unlink, and is used in pmap_enter.
+ * It returns SRMMU_PG_C if the (new) pvlist says that the address cannot
+ * be cached (i.e. its results must be (& ~)'d in.
+ */
+/*static*/ int
+pv_link4m(pv, pm, va)
+ register struct pvlist *pv;
+ register struct pmap *pm;
+ register vm_offset_t va;
+{
+ register struct pvlist *npv;
+ register int ret;
+
+ if (pv->pv_pmap == NULL) {
+ /* no pvlist entries yet */
+ pmap_stats.ps_enter_firstpv++;
+ pv->pv_next = NULL;
+ pv->pv_pmap = pm;
+ pv->pv_va = va;
+ pv->pv_flags |= PV_C4M;
+ return (0);
+ }
+ /*
+ * Before entering the new mapping, see if
+ * it will cause old mappings to become aliased
+ * and thus need to be `discached'.
+ */
+ ret = 0;
+ pmap_stats.ps_enter_secondpv++;
+ if (!(pv->pv_flags & PV_C4M)) {
+ /* already uncached, just stay that way */
+ ret = SRMMU_PG_C;
+ } else {
+ for (npv = pv; npv != NULL; npv = npv->pv_next) {
+ if (BADALIAS(va, npv->pv_va)) {
+#ifdef DEBUG
+ if (pmapdebug & PDB_CACHESTUFF) printf(
+ "pv_link: badalias: pid %d, %lx<=>%x, pa %lx\n",
+ curproc?curproc->p_pid:-1, va, npv->pv_va,
+ vm_first_phys + (pv-pv_table)*NBPG);
+#endif
+ pv->pv_flags &= ~PV_C4M;
+ pv_changepte4m(pv, 0, ret = SRMMU_PG_C);
+ /* cache_flush_page(va); XXX: needed? */
break;
}
}
@@ -1493,19 +2535,22 @@ pv_link(pv, pm, va)
npv->pv_next = pv->pv_next;
npv->pv_pmap = pm;
npv->pv_va = va;
+ npv->pv_flags |= (ret == SRMMU_PG_C ? 0 : PV_C4M);
pv->pv_next = npv;
return (ret);
}
+#endif
/*
* Walk the given list and flush the cache for each (MI) page that is
* potentially in the cache. Called only if vactype != VAC_NONE.
*/
+void
pv_flushcache(pv)
register struct pvlist *pv;
{
register struct pmap *pm;
- register int i, s, ctx;
+ register int s, ctx;
write_user_windows(); /* paranoia? */
@@ -1537,6 +2582,13 @@ pv_flushcache(pv)
int nptesg;
#endif
+#if defined(SUN4M)
+static void pmap_bootstrap4m __P((void));
+#endif
+#if defined(SUN4) || defined(SUN4C)
+static void pmap_bootstrap4_4c __P((int, int, int));
+#endif
+
/*
* Bootstrap the system enough to run with VM enabled.
*
@@ -1548,8 +2600,41 @@ void
pmap_bootstrap(nctx, nregion, nsegment)
int nsegment, nctx, nregion;
{
+
+ cnt.v_page_size = NBPG;
+ vm_set_page_size();
+
+#if defined(SUN4) && (defined(SUN4C) || defined(SUN4M))
+ /* In this case NPTESG is not a #define */
+ nptesg = (NBPSG >> pgshift);
+#endif
+
+ ncontext = nctx;
+
+#if defined(SUN4M)
+ if (CPU_ISSUN4M) {
+ pmap_bootstrap4m();
+ return;
+ }
+#endif
+#if defined(SUN4) || defined(SUN4C)
+ if (CPU_ISSUN4OR4C) {
+ pmap_bootstrap4_4c(nctx, nregion, nsegment);
+ return;
+ }
+#endif
+}
+
+#if defined(SUN4) || defined(SUN4C)
+void
+pmap_bootstrap4_4c(nctx, nregion, nsegment)
+ int nsegment, nctx, nregion;
+{
register union ctxinfo *ci;
- register struct mmuentry *mmuseg, *mmureg;
+ register struct mmuentry *mmuseg;
+#ifdef MMU_3L
+ register struct mmuentry *mmureg;
+#endif
struct regmap *rp;
register int i, j;
register int npte, zseg, vr, vs;
@@ -1563,7 +2648,6 @@ pmap_bootstrap(nctx, nregion, nsegment)
extern char *esym;
char *theend = end;
#endif
- extern caddr_t reserve_dumppages(caddr_t);
switch (cputyp) {
case CPU_SUN4C:
@@ -1579,11 +2663,6 @@ pmap_bootstrap(nctx, nregion, nsegment)
cnt.v_page_size = NBPG;
vm_set_page_size();
-#if defined(SUN4) && defined(SUN4C)
- /* In this case NPTESG is not a #define */
- nptesg = (NBPSG >> pgshift);
-#endif
-
#if defined(SUN4)
/*
* set up the segfixmask to mask off invalid bits
@@ -1598,7 +2677,21 @@ pmap_bootstrap(nctx, nregion, nsegment)
#endif
#endif
- ncontext = nctx;
+#if defined(SUN4M) /* We're in a dual-arch kernel. Setup 4/4c fn. ptrs */
+ pmap_clear_modify_p = pmap_clear_modify4_4c;
+ pmap_clear_reference_p = pmap_clear_reference4_4c;
+ pmap_copy_page_p = pmap_copy_page4_4c;
+ pmap_enter_p = pmap_enter4_4c;
+ pmap_extract_p = pmap_extract4_4c;
+ pmap_is_modified_p = pmap_is_modified4_4c;
+ pmap_is_referenced_p = pmap_is_referenced4_4c;
+ pmap_page_protect_p = pmap_page_protect4_4c;
+ pmap_protect_p = pmap_protect4_4c;
+ pmap_zero_page_p = pmap_zero_page4_4c;
+ pmap_changeprot_p = pmap_changeprot4_4c;
+ pmap_rmk_p = pmap_rmk4_4c;
+ pmap_rmu_p = pmap_rmu4_4c;
+#endif /* defined SUN4M */
/*
* Last segment is the `invalid' one (one PMEG of pte's with !pg_v).
@@ -1640,7 +2733,7 @@ pmap_bootstrap(nctx, nregion, nsegment)
* (otherwise L1-A crashes the machine!).
*/
- mmu_reservemon(&nregion, &nsegment);
+ mmu_reservemon4_4c(&nregion, &nsegment);
#ifdef MMU_3L
/* Reserve one region for temporary mappings */
@@ -1828,7 +2921,7 @@ pmap_bootstrap(nctx, nregion, nsegment)
* the final segment.
*/
for (p += npte << PGSHIFT; npte < NPTESG; npte++, p += NBPG)
- setpte(p, 0);
+ setpte4(p, 0);
#ifdef MMU_3L
if (mmu_3l) {
@@ -1889,9 +2982,521 @@ pmap_bootstrap(nctx, nregion, nsegment)
#endif
for (p = (caddr_t)trapbase; p < etext; p += NBPG)
- setpte(p, getpte(p) & mask);
+ setpte4(p, getpte4(p) & mask);
+ }
+}
+#endif
+
+#if defined(SUN4M) /* Sun4M version of pmap_bootstrap */
+/*
+ * Bootstrap the system enough to run with VM enabled on a Sun4M machine.
+ *
+ * Switches from ROM to kernel page tables, and sets up initial mappings.
+ */
+static void
+pmap_bootstrap4m(void)
+{
+ register int i, j;
+ caddr_t p, p2;
+ register caddr_t q;
+ register union ctxinfo *ci;
+ register struct memarr *mp;
+ struct regmap *rmapp = NULL;
+ struct segmap *smapp = NULL;
+ register int reg, seg;
+#if 0
+ int nkreg, nkseg, nkpag, kernsize, newpgs;
+#endif
+ caddr_t kphyssegtbl, kphyspagtbl = NULL;
+ int deadfill, deadspace;
+ extern char end[];
+ extern char etext[];
+#ifdef DDB
+ extern char *esym;
+ char *theend = end;
+#endif
+ extern caddr_t reserve_dumppages(caddr_t);
+
+#if defined(SUN4) || defined(SUN4C) /* setup 4M fn. ptrs for dual-arch kernel */
+ pmap_clear_modify_p = pmap_clear_modify4m;
+ pmap_clear_reference_p = pmap_clear_reference4m;
+ pmap_copy_page_p = pmap_copy_page4m;
+ pmap_enter_p = pmap_enter4m;
+ pmap_extract_p = pmap_extract4m;
+ pmap_is_modified_p = pmap_is_modified4m;
+ pmap_is_referenced_p = pmap_is_referenced4m;
+ pmap_page_protect_p = pmap_page_protect4m;
+ pmap_protect_p = pmap_protect4m;
+ pmap_zero_page_p = pmap_zero_page4m;
+ pmap_changeprot_p = pmap_changeprot4m;
+ pmap_rmk_p = pmap_rmk4m;
+ pmap_rmu_p = pmap_rmu4m;
+#endif /* defined Sun4/Sun4c */
+
+ /*
+ * Intialize the kernel pmap.
+ */
+ /* kernel_pmap_store.pm_ctxnum = 0; */
+ simple_lock_init(kernel_pmap_store.pm_lock);
+ kernel_pmap_store.pm_refcount = 1;
+ /* Set up pm_regmap for kernel to point NUREG *below* the beginning
+ * of kernel regmap storage. Since the kernel only uses regions
+ * above NUREG, we save storage space and can index kernel and
+ * user regions in the same way
+ */
+ kernel_pmap_store.pm_regmap = &kernel_regmap_store[-NUREG];
+ kernel_pmap_store.pm_reg_ptps = NULL;
+ kernel_pmap_store.pm_reg_ptps_pa = 0;
+ bzero(kernel_regmap_store, NKREG * sizeof(struct regmap));
+ bzero(kernel_segmap_store, NKREG * NSEGRG * sizeof(struct segmap));
+ for (i = NKREG; --i >= 0;) {
+ kernel_regmap_store[i].rg_segmap =
+ &kernel_segmap_store[i * NSEGRG];
+ kernel_regmap_store[i].rg_seg_ptps = NULL;
+ for (j = NSEGRG; --j >= 0;)
+ kernel_segmap_store[i * NSEGRG + j].sg_pte = NULL;
+ }
+
+ p = end; /* p points to top of kernel mem */
+#ifdef DDB
+ if (esym != 0)
+ theend = p = esym;
+#endif
+
+ /*
+ * Preserve the monitor ROM's reserved VM region, so that
+ * we can use L1-A or the monitor's debugger.
+ */
+
+ mmu_reservemon4m(&kernel_pmap_store, &p);
+
+ pmap_kernel()->pm_ctx = ctxinfo = ci = (union ctxinfo *)p;
+ p += ncontext * sizeof *ci;
+ bzero((caddr_t)ctxinfo, (u_int)p - (u_int)ctxinfo);
+ ctxbusyvector = p;
+ p += ncontext;
+ bzero(ctxbusyvector, ncontext);
+ ctxbusyvector[0] = 1; /* context 0 is always in use */
+
+ /*
+ * reserve memory for I/O pagetables. This takes 64k of memory
+ * since we want to have 64M of dvma space (this actually depends
+ * on the definition of DVMA4M_BASE...we may drop it back to 32M)
+ * but since the table must be aligned, we might end up using
+ * as much as 128K. (note 1024 = NBPG / sizeof(iopte_t))
+ *
+ * We optimize with some space saving song and dance to
+ * squeeze other pagetables in the dead space.
+ */
+#ifdef DEBUG
+ if ((0 - DVMA4M_BASE) % (16*1024*1024))
+ panic("pmap_bootstrap4m: invalid DVMA4M_BASE of 0x%x",DVMA4M_BASE);
+#endif
+
+ deadfill = 0;
+ p = (caddr_t) roundup((u_int) p, max(SRMMU_L1SIZE * sizeof(long),
+ max(SRMMU_L2SIZE * sizeof(long),
+ SRMMU_L3SIZE * sizeof(long))));
+
+ deadspace = (int) (((caddr_t)roundup((u_int)p,
+ (0 - DVMA4M_BASE) / 1024)) - p);
+
+ if (deadspace >= SRMMU_L3SIZE * sizeof(long) * NKREG * NSEGRG) {
+ p = (caddr_t) roundup((u_int)p, SRMMU_L3SIZE * sizeof(long));
+ kernel_pagtable_store = (u_int *)p;
+ p += ((SRMMU_L3SIZE * sizeof(long)) * NKREG) * NSEGRG;
+ bzero(kernel_pagtable_store,
+ p - (caddr_t) kernel_pagtable_store);
+ deadfill |= 4;
+ deadspace -= (int)(p - (caddr_t) kernel_pagtable_store);
+ }
+ if (deadspace >= SRMMU_L2SIZE * sizeof(long) * NKREG) {
+ p = (caddr_t) roundup((u_int)p, SRMMU_L2SIZE * sizeof(long));
+ kernel_segtable_store = (u_int *)p;
+ p += (SRMMU_L2SIZE * sizeof(long)) * NKREG;
+ bzero(kernel_segtable_store,
+ p - (caddr_t) kernel_segtable_store);
+ deadfill |= 2;
+ deadspace -= (int)(p - (caddr_t) kernel_segtable_store);
+ }
+ if (deadspace >= SRMMU_L1SIZE * sizeof(long)) {
+ p = (caddr_t) roundup((u_int)p, SRMMU_L1SIZE * sizeof(long));
+ kernel_regtable_store = (u_int *)p;
+ p += SRMMU_L1SIZE * sizeof(long);
+ bzero(kernel_regtable_store,
+ p - (caddr_t) kernel_regtable_store);
+ deadfill |= 1;
+ deadspace -= (int)(p - (caddr_t) kernel_regtable_store);
+ }
+ if (deadspace < 0)
+ printf("pmap_bootstrap4m: botch in memory-saver\n");
+
+ p = (caddr_t) roundup((u_int)p, (0 - DVMA4M_BASE) / 1024);
+ kernel_iopte_table = (u_int *)p;
+ kernel_iopte_table_pa = VA2PA((caddr_t)kernel_iopte_table);
+ p += (0 - DVMA4M_BASE) / 1024;
+ bzero(kernel_iopte_table, p - (caddr_t) kernel_iopte_table);
+
+ /*
+ * reserve memory for segment and page tables needed to map the entire
+ * kernel (from regions 0xf8 -> 0xff). This takes 130k of space, but
+ * unfortunately is necessary since pmap_enk *must* be able to enter
+ * a kernel mapping without resorting to malloc, or else the
+ * possibility of deadlock arises (pmap_enk4m is called to enter a
+ * mapping; it needs to malloc a page table; malloc then calls
+ * pmap_enk4m to enter the new malloc'd page; pmap_enk4m needs to
+ * malloc a page table to enter _that_ mapping; malloc deadlocks since
+ * it is already allocating that object).
+ *
+ * We only do this if it wasn't done above...
+ */
+ if (!(deadfill & 2)) {
+ p = (caddr_t) roundup((u_int)p, SRMMU_L2SIZE * sizeof(long));
+ kernel_segtable_store = (u_int *)p;
+ p += (SRMMU_L2SIZE * sizeof(long)) * NKREG;
+ bzero(kernel_segtable_store,
+ p - (caddr_t) kernel_segtable_store);
+ }
+ if (!(deadfill & 4)) {
+ p = (caddr_t) roundup((u_int)p, SRMMU_L3SIZE * sizeof(long));
+ kernel_pagtable_store = (u_int *)p;
+ p += ((SRMMU_L3SIZE * sizeof(long)) * NKREG) * NSEGRG;
+ bzero(kernel_pagtable_store,
+ p - (caddr_t) kernel_pagtable_store);
+ }
+ if (!(deadfill & 1)) {
+ p = (caddr_t) roundup((u_int)p, SRMMU_L1SIZE * sizeof(long));
+ kernel_regtable_store = (u_int *)p;
+ p += SRMMU_L1SIZE * sizeof(long);
+ bzero(kernel_regtable_store,
+ p - (caddr_t) kernel_regtable_store);
+ }
+
+#if 0
+ /* At this point, "p" is the highest location of kernel memory that
+ * we need to lock in/map initially. We now need to calculate the
+ * size of this kernel image and allocate memory for its page tables,
+ * which we won't set up until the end of this function. After this
+ * point, it is NOT POSSIBLE to allocate kernel physical memory
+ * by bumping up p!!!
+ */
+ p = (caddr_t) roundup((u_int)p,SRMMU_L1SIZE * sizeof(long));
+ kernsize = (u_int)p - KERNBASE;
+
+ /* We keep adding pages until the number of added pages is sufficient
+ * to hold the map for both the kernel and the new pages. This
+ * converges since it takes at most about 1152 bytes to map one page.
+ */
+ newpgs = 0;
+ do {
+ newpgs++;
+ nkreg = (kernsize + (newpgs * NBPG) + NBPRG - 1) / NBPRG;
+ nkseg = (kernsize + (newpgs * NBPG) + NBPSG - 1) / NBPSG;
+ nkpag = (kernsize + NBPG - 1) / NBPG + newpgs;
+ } while (((SRMMU_L1SIZE + (nkreg * SRMMU_L2SIZE) +
+ (nkseg * SRMMU_L3SIZE) + (nkpag)) * sizeof(long)) >
+ newpgs * NBPG);
+
+ kernsize += newpgs * NBPG;
+ p2 = p; /* Page tables go from p2 to p. */
+ p += newpgs * NBPG; /* p is now the _real_ top of kernel mem. */
+#endif
+
+ /*
+ * Set up the `constants' for the call to vm_init()
+ * in main(). All pages beginning at p (rounded up to
+ * the next whole page) and continuing through the number
+ * of available pages are free, but they start at a higher
+ * virtual address. This gives us two mappable MD pages
+ * for pmap_zero_page and pmap_copy_page, and one MI page
+ * for /dev/mem, all with no associated physical memory.
+ */
+ p = (caddr_t)(((u_int)p + NBPG - 1) & ~PGOFSET);
+ avail_start = (int)p - KERNBASE;
+ /*
+ * Grab physical memory list, so pmap_next_page() can do its bit.
+ */
+ npmemarr = makememarr(pmemarr, MA_SIZE, MEMARR_AVAILPHYS);
+ sortm(pmemarr, npmemarr);
+ if (pmemarr[0].addr != 0) {
+ printf("pmap_bootstrap: no kernel memory?!\n");
+ callrom();
+ }
+ avail_end = pmemarr[npmemarr-1].addr + pmemarr[npmemarr-1].len;
+ avail_next = avail_start;
+ for (physmem = 0, mp = pmemarr, j = npmemarr; --j >= 0; mp++)
+ physmem += btoc(mp->len);
+
+ i = (int)p;
+ vpage[0] = p, p += NBPG;
+ vpage[1] = p, p += NBPG;
+ vmmap = p, p += NBPG;
+ p = reserve_dumppages(p);
+
+ /*
+ * Allocate virtual memory for pv_table[], which will be mapped
+ * sparsely in pmap_init().
+ */
+ pv_table = (struct pvlist *)p;
+ p += round_page(sizeof(struct pvlist) * atop(avail_end - avail_start));
+
+ virtual_avail = (vm_offset_t)p;
+ virtual_end = VM_MAX_KERNEL_ADDRESS;
+
+ p = (caddr_t)i; /* retract to first free phys */
+
+ /* Set up the ctxinfo structures (freelist of contexts)
+ */
+ ci->c_pmap = pmap_kernel();
+ ctx_freelist = ci + 1;
+ for (i = 1; i < ncontext; i++) {
+ ci++;
+ ci->c_nextfree = ci + 1;
+ }
+ ci->c_nextfree = NULL;
+ ctx_kick = 0;
+ ctx_kickdir = -1;
+
+ /* Now map the kernel into our new set of page tables, then
+ * (finally) switch over to our running page tables.
+ * We map from KERNBASE to p into context 0's page tables (and
+ * the kernel pmap).
+ */
+#ifdef DEBUG /* Sanity checks */
+ if ((u_int)p % NBPG != 0)
+ panic("pmap_bootstrap4m: p misaligned?!?");
+ if (KERNBASE % NBPRG != 0)
+ panic("pmap_bootstrap4m: KERNBASE not region-aligned");
+#endif
+ if (pmap_kernel()->pm_reg_ptps == NULL) {
+#ifdef DEBUG
+ printf("pmap_bootstrap4m: no kernel regiontable; creating.\n");
+#endif
+ pmap_kernel()->pm_reg_ptps = (int *) kernel_regtable_store;
+ pmap_kernel()->pm_reg_ptps_pa =
+ VA2PA((caddr_t)pmap_kernel()->pm_reg_ptps);
+ p2 = (caddr_t) pmap_kernel()->pm_reg_ptps +
+ (SRMMU_L1SIZE * sizeof(int));
+ bzero(pmap_kernel()->pm_reg_ptps, SRMMU_L1SIZE * sizeof(long));
+ }
+ reg = -1;
+ seg = -1;
+ for (q = (caddr_t) KERNBASE; q < p; q += NBPG) {
+ if ((VA_VSEG(q) % NSEGRG == 0) && VA_VREG(q) != reg) {
+ /* entering new region; install & build segtbl
+ * XXX: WE TRASH ANY EXISTING MAPPINGS IN THE KERNEL
+ * REGION. SHOULD BE FIXED!
+ */
+ reg = VA_VREG(q);
+ rmapp = &(pmap_kernel()->pm_regmap[reg]);
+ kphyssegtbl = (caddr_t)
+ &kernel_segtable_store[(reg - VA_VREG(KERNBASE)) *
+ SRMMU_L2SIZE];
+ bzero(kphyssegtbl, SRMMU_L2SIZE * sizeof(long));
+
+ ((int *)pmap_kernel()->pm_reg_ptps)[VA_VREG(q)] =
+ (VA2PA(kphyssegtbl) >> SRMMU_PPNPASHIFT) |
+ SRMMU_TEPTD;
+
+ rmapp->rg_seg_ptps = (int *)kphyssegtbl;
+
+ if (rmapp->rg_segmap == NULL)
+ rmapp->rg_segmap = &kernel_segmap_store[(reg -
+ VA_VREG(KERNBASE)) * NSEGRG];
+ }
+ if (((VA_VPG(q) % NPTESG) == 0) && VA_VSEG(q) != seg) {
+ rmapp->rg_nsegmap++;
+ /* Entering new segment. XXX: trashes existing maps */
+ seg = VA_VSEG(q);
+ smapp = &(rmapp->rg_segmap[seg]);
+ kphyspagtbl = (caddr_t)
+ &kernel_pagtable_store[(((reg-VA_VREG(KERNBASE))*
+ NSEGRG) + seg)*SRMMU_L3SIZE];
+ bzero(kphyspagtbl, SRMMU_L3SIZE * sizeof(long));
+
+ rmapp->rg_seg_ptps[VA_VSEG(q)] =
+ (VA2PA(kphyspagtbl) >> SRMMU_PPNPASHIFT) |
+ SRMMU_TEPTD;
+ smapp->sg_pte = (int *) kphyspagtbl;
+ }
+ /* Now install entry for current page. Cache and write-protect
+ * kernel text.
+ */
+ smapp->sg_npte++;
+ if (q == (caddr_t) KERNBASE) {
+ /* Must map in message buffer in low page. */
+ ((int *)kphyspagtbl)[VA_VPG(q)] = PPROT_N_RWX |
+ SRMMU_PG_C | SRMMU_TEPTE;
+ continue;
+ }
+ else if (q < (caddr_t) (KERNBASE+0x4000)) {
+ /* Special case: retain interrupt register mapping */
+ ((int *)kphyspagtbl)[VA_VPG(q)] =
+ getpte4m((vm_offset_t)q);
+ }
+ if (q >= (caddr_t) trapbase && q < etext)
+ ((int *)kphyspagtbl)[VA_VPG(q)] =
+ (VA2PA(q) >> SRMMU_PPNPASHIFT) |
+ PPROT_N_RX | SRMMU_PG_C | SRMMU_TEPTE;
+ else
+ ((int *)kphyspagtbl)[VA_VPG(q)] =
+ (VA2PA(q) >> SRMMU_PPNPASHIFT) |
+ PPROT_N_RWX | SRMMU_PG_C | SRMMU_TEPTE;
+ }
+
+ /*
+ * We must also create a segment table for region 0xfe and any
+ * needed page tables necessary to map the special (important)
+ * devices that get mapped at the beginning of this I/O segment.
+ * This is necessary since we have to map the interrupt registers
+ * at the start of region 0xfe in bootstrap().
+ *
+ * Since we've statically allocated space to map the entire kernel,
+ * we might as well pre-wire the mappings to save time in pmap_enter.
+ * This also gets around nasty problems with caching of L1/L2 ptp's.
+ *
+ * XXX WHY DO WE HAVE THIS CACHING PROBLEM WITH L1/L2 PTPS????? %%%
+ */
+ for (reg = VA_VREG(KERNBASE); reg < NKREG+VA_VREG(KERNBASE); reg++) {
+ rmapp = &(pmap_kernel()->pm_regmap[reg]);
+
+ if (((int *)pmap_kernel()->pm_reg_ptps)[reg] == NULL) {
+ kphyssegtbl = (caddr_t)
+ &kernel_segtable_store[(reg - VA_VREG(KERNBASE)) *
+ SRMMU_L2SIZE];
+ bzero(kphyssegtbl, SRMMU_L2SIZE * sizeof(long));
+ ((int *)pmap_kernel()->pm_reg_ptps)[reg] =
+ (VA2PA(kphyssegtbl) >> SRMMU_PPNPASHIFT) |
+ SRMMU_TEPTD;
+
+ rmapp->rg_seg_ptps = (int *)kphyssegtbl;
+
+ if (rmapp->rg_segmap == NULL)
+ rmapp->rg_segmap = &kernel_segmap_store
+ [(reg - VA_VREG(KERNBASE)) * NSEGRG];
+ }
+ for (seg = 0; seg < NSEGRG; seg++) {
+ if (rmapp->rg_seg_ptps[seg] == NULL) {
+ rmapp->rg_nsegmap++;
+
+ smapp = &(rmapp->rg_segmap[seg]);
+ kphyspagtbl = (caddr_t)
+ &kernel_pagtable_store[(((reg-
+ VA_VREG(KERNBASE))*
+ NSEGRG) + seg)*
+ SRMMU_L3SIZE];
+ bzero(kphyspagtbl,
+ SRMMU_L3SIZE * sizeof(long));
+
+ rmapp->rg_seg_ptps[seg] =
+ (VA2PA(kphyspagtbl) >> SRMMU_PPNPASHIFT) |
+ SRMMU_TEPTD;
+ smapp->sg_pte = (int *) kphyspagtbl;
+ }
+ }
+ }
+
+ /*
+ * We also install the kernel mapping into all other contexts by
+ * copying the context 0 L1 PTP from ctx_phys_tbl[0] into the
+ * remainder of the context table (i.e. we share the kernel page-
+ * tables). Each user pmap automatically gets the kernel mapped
+ * into it when it is created, but we do this extra step early on
+ * in case some twit decides to switch to a context with no user
+ * pmap associated with it.
+ */
+ for (i = 1; i < ncontext; i++)
+ ctx_phys_tbl[i] = ctx_phys_tbl[0];
+
+ /*
+ * Now switch to kernel pagetables (finally!)
+ */
+#ifdef DEBUG
+ printf("pmap_bootstrap: installing kernel page tables...");
+#endif
+ setcontext(0); /* paranoia? %%%: Make 0x3 a define! below */
+
+ /* Enable MMU tablewalk caching, flush TLB */
+
+ i = lda(SRMMU_PCR, ASI_SRMMU);
+ switch(mmumod) {
+/* case SUN4M_MMU_MS: */ /* These have the same model # as SS */
+ case SUN4M_MMU_SS:
+ if ((cpumod & 0xf0) == (SUN4M_SS & 0xf0))
+ sta(SRMMU_PCR, ASI_SRMMU, (i | SRMMU_PCR_TC));
+ else /* microsparc */
+ printf("(if this doesn't work, "
+ "fix pmap_bootstrap4m in pmap.c)");
+ break;
+ case SUN4M_MMU_HS:
+ printf("(if this doesn't work, fix pmap_bootstrap4m in pmap.c)");
+ sta(SRMMU_PCR, ASI_SRMMU, (i | SRMMU_PCR_C) & ~SRMMU_PCR_CE);
+ /* above: CHECKFIX %%% below: add microsparc*/
+ break;
+ case SUN4M_MMU_MS1:
+ printf("(if this doesn't work, fix pmap_bootstrap4m in pmap.c)");
+ break;
+ default:
+ panic("Unimplemented MMU architecture %d",mmumod);
+ }
+
+ tlb_flush_all();
+
+ sta(SRMMU_CXTPTR, ASI_SRMMU,
+ (VA2PA((caddr_t)ctx_phys_tbl) >> SRMMU_PPNPASHIFT) & ~0x3);
+
+ tlb_flush_all();
+
+#ifdef DEBUG
+ printf("done.\n");
+#endif
+
+ /*
+ * On SuperSPARC machines without a MXCC, we *cannot* cache the
+ * page tables.
+ * XXX: this needs to be cleaned up with the cpu_softc stuff...
+ */
+ if (mmumod == SUN4M_MMU_SS &&
+ (lda(SRMMU_PCR, ASI_SRMMU) & SRMMU_PCR_MB)) {
+
+ int bytes, numpages;
+ cant_cache_pagetables = 1;
+
+#define DO_THE_MATH(math) \
+ bytes = (math); \
+ numpages = (bytes >> PGSHIFT) + (bytes % NBPG ? 1 : 0);
+
+ DO_THE_MATH(SRMMU_L3SIZE * sizeof(long) * NKREG * NSEGRG);
+#ifdef DEBUG
+ printf("pmap_bootstrap4m: uncaching %d PT pages at 0x%lx\n",
+ numpages, (long)kernel_pagtable_store);
+#endif
+ kvm_uncache((caddr_t)kernel_pagtable_store, numpages);
+
+ DO_THE_MATH(SRMMU_L2SIZE * sizeof(long) * NKREG);
+#ifdef DEBUG
+ printf("pmap_bootstrap4m: uncaching %d ST pages at 0x%lx\n",
+ numpages, (long)kernel_segtable_store);
+#endif
+ kvm_uncache((caddr_t)kernel_segtable_store, numpages);
+
+ DO_THE_MATH(SRMMU_L1SIZE * sizeof(long));
+#ifdef DEBUG
+ printf("pmap_bootstrap4m: uncaching %d RT pages at 0x%lx\n",
+ numpages, (long)kernel_regtable_store);
+#endif
+ kvm_uncache((caddr_t)kernel_regtable_store, numpages);
+
+#undef DO_THE_MATH
}
+
+#ifdef DEBUG
+ printf("\n"); /* Might as well make it pretty... */
+#endif
+
+ /* All done! */
}
+#endif /* defined sun4m */
void
pmap_init()
@@ -1900,7 +3505,7 @@ pmap_init()
int pass1, nmem;
register struct memarr *mp;
vm_offset_t sva, va, eva;
- vm_offset_t pa;
+ vm_offset_t pa = 0;
if (PAGE_SIZE != NBPG)
panic("pmap_init: CLSIZE!=1");
@@ -1932,17 +3537,23 @@ pass2:
len = sizeof(struct pvlist) * atop(len);
if (addr < avail_start || addr >= avail_end)
- panic("pmap_init: unmanaged address: 0x%x", addr);
+ panic("pmap_init: unmanaged address: 0x%lx", addr);
va = (vm_offset_t)&pv_table[atop(addr - avail_start)];
sva = trunc_page(va);
+
if (sva < eva) {
-#ifdef DEBUG
+#if defined(DEBUG) && !defined(SUN4M)
+ /*
+ * crowded chunks are normal on SS20s; don't clutter
+ * screen with messages
+ */
printf("note: crowded chunk at 0x%x\n", mp->addr);
#endif
sva += PAGE_SIZE;
if (sva < eva)
- panic("pmap_init: sva(%x) < eva(%x)", sva, eva);
+ panic("pmap_init: sva(%lx) < eva(%lx)",
+ sva, eva);
}
eva = round_page(va + len);
if (pass1) {
@@ -2005,7 +3616,7 @@ pmap_create(size)
pm = (struct pmap *)malloc(sizeof *pm, M_VMPMAP, M_WAITOK);
#ifdef DEBUG
if (pmapdebug & PDB_CREATE)
- printf("pmap_create: created %x\n", pm);
+ printf("pmap_create: created %p\n", pm);
#endif
bzero((caddr_t)pm, sizeof *pm);
pmap_pinit(pm);
@@ -2020,30 +3631,60 @@ void
pmap_pinit(pm)
register struct pmap *pm;
{
- register int i, size;
+ register int size;
void *urp;
#ifdef DEBUG
if (pmapdebug & PDB_CREATE)
- printf("pmap_pinit(%x)\n", pm);
+ printf("pmap_pinit(%p)\n", pm);
#endif
size = NUREG * sizeof(struct regmap);
+
pm->pm_regstore = urp = malloc(size, M_VMPMAP, M_WAITOK);
- bzero((caddr_t)urp, size);
+ qzero((caddr_t)urp, size);
/* pm->pm_ctx = NULL; */
simple_lock_init(&pm->pm_lock);
pm->pm_refcount = 1;
-#ifdef MMU_3L
- TAILQ_INIT(&pm->pm_reglist);
-#endif
- TAILQ_INIT(&pm->pm_seglist);
pm->pm_regmap = urp;
+
+ if (CPU_ISSUN4OR4C) {
+ TAILQ_INIT(&pm->pm_seglist);
#ifdef MMU_3L
- if (mmu_3l)
- for (i = NUREG; --i >= 0;)
+ TAILQ_INIT(&pm->pm_reglist);
+ if (mmu_3l)
+ for (i = NUREG; --i >= 0;)
pm->pm_regmap[i].rg_smeg = reginval;
#endif
+ }
+#if defined(SUN4M)
+ else {
+ /*
+ * We must allocate and initialize hardware-readable (MMU)
+ * pagetables. We must also map the kernel regions into this
+ * pmap's pagetables, so that we can access the kernel from
+ * user mode!
+ *
+ * Note: pm->pm_regmap's have been zeroed already, so we don't
+ * need to explicitly mark them as invalid (a null
+ * rg_seg_ptps pointer indicates invalid for the 4m)
+ */
+ urp = malloc(SRMMU_L1SIZE * sizeof(int), M_VMPMAP, M_WAITOK);
+ if (cant_cache_pagetables)
+ kvm_uncache(urp,
+ ((SRMMU_L1SIZE*sizeof(int))+NBPG-1)/NBPG);
+
+#ifdef DEBUG
+ if ((u_int) urp % (SRMMU_L1SIZE * sizeof(int)))
+ panic("pmap_pinit: malloc() not giving aligned memory");
+#endif
+ pm->pm_reg_ptps = urp;
+ pm->pm_reg_ptps_pa = VA2PA(urp);
+ qzero(urp, SRMMU_L1SIZE * sizeof(int));
+
+ }
+#endif
+
pm->pm_gap_end = VA_VREG(VM_MAXUSER_ADDRESS);
return;
@@ -2063,7 +3704,7 @@ pmap_destroy(pm)
return;
#ifdef DEBUG
if (pmapdebug & PDB_DESTROY)
- printf("pmap_destroy(%x)\n", pm);
+ printf("pmap_destroy(%p)\n", pm);
#endif
simple_lock(&pm->pm_lock);
count = --pm->pm_refcount;
@@ -2087,22 +3728,27 @@ pmap_release(pm)
#ifdef DEBUG
if (pmapdebug & PDB_DESTROY)
- printf("pmap_release(%x)\n", pm);
+ printf("pmap_release(%p)\n", pm);
#endif
+
+ if (CPU_ISSUN4OR4C) {
#ifdef MMU_3L
- if (pm->pm_reglist.tqh_first)
- panic("pmap_release: region list not empty");
+ if (pm->pm_reglist.tqh_first)
+ panic("pmap_release: region list not empty");
#endif
- if (pm->pm_seglist.tqh_first)
- panic("pmap_release: segment list not empty");
- if ((c = pm->pm_ctx) != NULL) {
- if (pm->pm_ctxnum == 0)
- panic("pmap_release: releasing kernel");
- ctx_free(pm);
+ if (pm->pm_seglist.tqh_first)
+ panic("pmap_release: segment list not empty");
+
+ if ((c = pm->pm_ctx) != NULL) {
+ if (pm->pm_ctxnum == 0)
+ panic("pmap_release: releasing kernel");
+ ctx_free(pm);
+ }
}
splx(s);
+
#ifdef DEBUG
-{
+if (pmapdebug) {
int vs, vr;
for (vr = 0; vr < NUREG; vr++) {
struct regmap *rp = &pm->pm_regmap[vr];
@@ -2129,6 +3775,17 @@ pmap_release(pm)
#endif
if (pm->pm_regstore)
free(pm->pm_regstore, M_VMPMAP);
+
+ if (CPU_ISSUN4M) {
+ if ((c = pm->pm_ctx) != NULL) {
+ if (pm->pm_ctxnum == 0)
+ panic("pmap_release: releasing kernel");
+ ctx_free(pm);
+ }
+ free(pm->pm_reg_ptps, M_VMPMAP);
+ pm->pm_reg_ptps = NULL;
+ pm->pm_reg_ptps_pa = 0;
+ }
}
/*
@@ -2146,11 +3803,6 @@ pmap_reference(pm)
}
}
-static void pmap_rmk __P((struct pmap *, vm_offset_t, vm_offset_t,
- int, int));
-static void pmap_rmu __P((struct pmap *, vm_offset_t, vm_offset_t,
- int, int));
-
/*
* Remove the given range of mapping entries.
* The starting and ending addresses are already rounded to pages.
@@ -2171,7 +3823,7 @@ pmap_remove(pm, va, endva)
#ifdef DEBUG
if (pmapdebug & PDB_REMOVE)
- printf("pmap_remove(%x, %x, %x)\n", pm, va, endva);
+ printf("pmap_remove(%p, %lx, %lx)\n", pm, va, endva);
#endif
if (pm == pmap_kernel()) {
@@ -2225,9 +3877,11 @@ pmap_remove(pm, va, endva)
* These are egregiously complicated routines.
*/
+#if defined(SUN4) || defined(SUN4C)
+
/* remove from kernel */
-static void
-pmap_rmk(pm, va, endva, vr, vs)
+/*static*/ void
+pmap_rmk4_4c(pm, va, endva, vr, vs)
register struct pmap *pm;
register vm_offset_t va, endva;
register int vr, vs;
@@ -2274,24 +3928,24 @@ pmap_rmk(pm, va, endva, vr, vs)
perpage = (vactype != VAC_NONE);
}
while (va < endva) {
- tpte = getpte(va);
+ tpte = getpte4(va);
if ((tpte & PG_V) == 0) {
- va += PAGE_SIZE;
+ va += NBPG;
continue;
}
if ((tpte & PG_TYPE) == PG_OBMEM) {
/* if cacheable, flush page as needed */
if (perpage && (tpte & PG_NC) == 0)
cache_flush_page(va);
- i = ptoa(HWTOSW(tpte & PG_PFNUM));
+ i = ptoa(tpte & PG_PFNUM);
if (managed(i)) {
pv = pvhead(i);
- pv->pv_flags |= MR(tpte);
- pv_unlink(pv, pm, va);
+ pv->pv_flags |= MR4_4C(tpte);
+ pv_unlink4_4c(pv, pm, va);
}
}
nleft--;
- setpte(va, 0);
+ setpte4(va, 0);
va += NBPG;
}
@@ -2326,6 +3980,105 @@ pmap_rmk(pm, va, endva, vr, vs)
}
}
+#endif /* sun4, sun4c */
+
+#if defined(SUN4M) /* 4M version of pmap_rmk */
+/* remove from kernel (4m)*/
+/*static*/ void
+pmap_rmk4m(pm, va, endva, vr, vs)
+ register struct pmap *pm;
+ register vm_offset_t va, endva;
+ register int vr, vs;
+{
+ register int i, tpte, perpage, npg;
+ register struct pvlist *pv;
+ register int nleft;
+ struct regmap *rp;
+ struct segmap *sp;
+
+ rp = &pm->pm_regmap[vr];
+ sp = &rp->rg_segmap[vs];
+
+ if (rp->rg_nsegmap == 0)
+ return;
+
+#ifdef DEBUG
+ if (rp->rg_segmap == NULL)
+ panic("pmap_rmk: no segments");
+#endif
+
+ if ((nleft = sp->sg_npte) == 0)
+ return;
+
+#ifdef DEBUG
+ if (sp->sg_pte == NULL || rp->rg_seg_ptps == NULL)
+ panic("pmap_rmk: segment/region does not exist");
+ if (pm->pm_ctx == NULL)
+ panic("pmap_rmk: lost context");
+#endif
+
+ setcontext(0);
+ /* decide how to flush cache */
+ npg = (endva - va) >> PGSHIFT;
+ if (npg > PMAP_RMK_MAGIC) {
+ /* flush the whole segment */
+ perpage = 0;
+ if (vactype != VAC_NONE)
+ cache_flush_segment(vr, vs);
+ } else {
+ /* flush each page individually; some never need flushing */
+ perpage = (vactype != VAC_NONE);
+ }
+ while (va < endva) {
+ tpte = getpte4m(va);
+ if ((tpte & SRMMU_TETYPE) != SRMMU_TEPTE) {
+ va += NBPG;
+ continue;
+ }
+ if ((tpte & SRMMU_PGTYPE) == PG_SUN4M_OBMEM) {
+ /* if cacheable, flush page as needed */
+ if (perpage && (tpte & SRMMU_PG_C))
+ cache_flush_page(va);
+ i = ptoa((tpte & SRMMU_PPNMASK) >> SRMMU_PPNSHIFT);
+ if (managed(i)) {
+ pv = pvhead(i);
+ pv->pv_flags |= MR4M(tpte);
+ pv_unlink4m(pv, pm, va);
+ }
+ }
+ nleft--;
+ setpte4m(va, SRMMU_TEINVALID);
+ va += NBPG;
+ }
+
+ /*
+ * If the segment is all gone, remove it from everyone and
+ * flush the TLB.
+ */
+ if ((sp->sg_npte = nleft) == 0) {
+ va = VSTOVA(vr,vs); /* retract */
+
+ tlb_flush_segment(vr, vs); /* Paranoia? */
+
+ /*
+ * We need to free the segment table. The problem is that
+ * we can't free the initial (bootstrap) mapping, so
+ * we have to explicitly check for this case (ugh).
+ */
+ if (va < virtual_avail) {
+#ifdef DEBUG
+ printf("pmap_rmk4m: attempt to free base kernel alloc\n");
+#endif
+ /* sp->sg_pte = NULL; */
+ sp->sg_npte = 0;
+ return;
+ }
+ /* no need to free the table; it is statically allocated */
+ qzero(sp->sg_pte, SRMMU_L3SIZE * sizeof(long));
+ }
+ /* if we're done with a region, leave it wired */
+}
+#endif /* sun4m */
/*
* Just like pmap_rmk_magic, but we have a different threshold.
* Note that this may well deserve further tuning work.
@@ -2336,9 +4089,11 @@ pmap_rmk(pm, va, endva, vr, vs)
#define PMAP_RMU_MAGIC 4 /* if > magic, use cache_flush_segment */
#endif
+#if defined(SUN4) || defined(SUN4C)
+
/* remove from user */
-static void
-pmap_rmu(pm, va, endva, vr, vs)
+/*static*/ void
+pmap_rmu4_4c(pm, va, endva, vr, vs)
register struct pmap *pm;
register vm_offset_t va, endva;
register int vr, vs;
@@ -2371,16 +4126,16 @@ pmap_rmu(pm, va, endva, vr, vs)
/*
* PTEs are not in MMU. Just invalidate software copies.
*/
- for (; va < endva; pte++, va += PAGE_SIZE) {
+ for (; va < endva; pte++, va += NBPG) {
tpte = *pte;
if ((tpte & PG_V) == 0) {
/* nothing to remove (braindead VM layer) */
continue;
}
if ((tpte & PG_TYPE) == PG_OBMEM) {
- i = ptoa(HWTOSW(tpte & PG_PFNUM));
+ i = ptoa(tpte & PG_PFNUM);
if (managed(i))
- pv_unlink(pvhead(i), pm, va);
+ pv_unlink4_4c(pvhead(i), pm, va);
}
nleft--;
*pte = 0;
@@ -2433,23 +4188,23 @@ pmap_rmu(pm, va, endva, vr, vs)
pteva = VA_VPG(va) << PGSHIFT;
perpage = 0;
}
- for (; va < endva; pteva += PAGE_SIZE, va += PAGE_SIZE) {
- tpte = getpte(pteva);
+ for (; va < endva; pteva += NBPG, va += NBPG) {
+ tpte = getpte4(pteva);
if ((tpte & PG_V) == 0)
continue;
if ((tpte & PG_TYPE) == PG_OBMEM) {
/* if cacheable, flush page as needed */
if (perpage && (tpte & PG_NC) == 0)
cache_flush_page(va);
- i = ptoa(HWTOSW(tpte & PG_PFNUM));
+ i = ptoa(tpte & PG_PFNUM);
if (managed(i)) {
pv = pvhead(i);
- pv->pv_flags |= MR(tpte);
- pv_unlink(pv, pm, va);
+ pv->pv_flags |= MR4_4C(tpte);
+ pv_unlink4_4c(pv, pm, va);
}
}
nleft--;
- setpte(pteva, 0);
+ setpte4(pteva, 0);
#define PMAP_PTESYNC
#ifdef PMAP_PTESYNC
pte0[VA_VPG(pteva)] = 0;
@@ -2498,6 +4253,109 @@ if (pm->pm_ctx == NULL) {
}
}
+#endif /* sun4,4c */
+
+#if defined(SUN4M) /* 4M version of pmap_rmu */
+/* remove from user */
+/*static*/ void
+pmap_rmu4m(pm, va, endva, vr, vs)
+ register struct pmap *pm;
+ register vm_offset_t va, endva;
+ register int vr, vs;
+{
+ register int *pte0, i, tpte, perpage, npg;
+ register struct pvlist *pv;
+ register int nleft;
+ struct regmap *rp;
+ struct segmap *sp;
+
+ rp = &pm->pm_regmap[vr];
+ if (rp->rg_nsegmap == 0)
+ return;
+ if (rp->rg_segmap == NULL)
+ panic("pmap_rmu: no segments");
+
+ sp = &rp->rg_segmap[vs];
+ if ((nleft = sp->sg_npte) == 0)
+ return;
+ if (sp->sg_pte == NULL)
+ panic("pmap_rmu: no pages");
+
+ pte0 = sp->sg_pte;
+
+ /*
+ * Invalidate PTE in MMU pagetables. Flush cache if necessary.
+ */
+ if (CTX_USABLE(pm,rp)) {
+ /* process has a context, must flush cache */
+ setcontext(pm->pm_ctxnum);
+ if (vactype != VAC_NONE) {
+ npg = (endva - va) >> PGSHIFT;
+ if (npg > PMAP_RMU_MAGIC) {
+ perpage = 0; /* flush the whole segment */
+ cache_flush_segment(vr, vs);
+ } else
+ perpage = 1;
+ } else
+ perpage = 0;
+ } else {
+ /* no context; cache flush unnecessary */
+ perpage = 0;
+ }
+ for (; va < endva; va += NBPG) {
+ /* Note: we use sw pagetables here since pages have been
+ * flushed already. This avoids over-zealous cache flushing.
+ */
+ if (CTX_USABLE(pm,rp)) { /* %%% XXX: Performance hit? */
+ tpte = getpte4m(va); /* should we just flush seg? */
+ tlb_flush_page(va);
+ } else
+ tpte = getptesw4m(pm, va);
+ if ((tpte & SRMMU_TETYPE) != SRMMU_TEPTE)
+ continue;
+ if ((tpte & SRMMU_PGTYPE) == PG_SUN4M_OBMEM) {
+ /* if cacheable, flush page as needed */
+ if (perpage && (tpte & SRMMU_PG_C))
+ cache_flush_page(va);
+ i = ptoa((tpte & SRMMU_PPNMASK) >> SRMMU_PPNSHIFT);
+ if (managed(i)) {
+ pv = pvhead(i);
+ pv->pv_flags |= MR4M(tpte);
+ pv_unlink4m(pv, pm, va);
+ }
+ }
+ nleft--;
+ setptesw4m(pm, va, SRMMU_TEINVALID); /* Update pagetables */
+ }
+
+ /*
+ * If the segment is all gone, and the context is loaded, give
+ * the segment back.
+ */
+ if ((sp->sg_npte = nleft) == 0 /* ??? && pm->pm_ctx != NULL*/) {
+#ifdef DEBUG
+ if (pm->pm_ctx == NULL) {
+ printf("pmap_rmu: no context here...");
+ }
+#endif
+ va = VSTOVA(vr,vs); /* retract */
+
+ tlb_flush_segment(vr, vs); /* Paranoia? */
+ rp->rg_seg_ptps[vs] = SRMMU_TEINVALID;
+ free(pte0, M_VMPMAP);
+ sp->sg_pte = NULL;
+
+ if (--rp->rg_nsegmap == 0) {
+ free(rp->rg_segmap, M_VMPMAP);
+ rp->rg_segmap = NULL;
+ free(rp->rg_seg_ptps, M_VMPMAP);
+ pm->pm_reg_ptps[vr] = SRMMU_TEINVALID;
+ GAP_WIDEN(pm,vr);
+ }
+ }
+}
+#endif /* sun4m */
+
/*
* Lower (make more strict) the protection on the specified
* physical page.
@@ -2506,24 +4364,26 @@ if (pm->pm_ctx == NULL) {
* (in which case we do the dirty work here), or it is going from
* to read-only (in which case pv_changepte does the trick).
*/
+
+#if defined(SUN4) || defined(SUN4C)
void
-pmap_page_protect(pa, prot)
+pmap_page_protect4_4c(pa, prot)
vm_offset_t pa;
vm_prot_t prot;
{
register struct pvlist *pv, *pv0, *npv;
register struct pmap *pm;
register int va, vr, vs, pteva, tpte;
- register int flags, nleft, i, s, ctx, doflush;
+ register int flags, nleft, i, s, ctx;
struct regmap *rp;
struct segmap *sp;
#ifdef DEBUG
if (!pmap_pa_exists(pa))
- panic("pmap_page_protect: no such address: %x", pa);
+ panic("pmap_page_protect: no such address: %lx", pa);
if ((pmapdebug & PDB_CHANGEPROT) ||
(pmapdebug & PDB_REMOVE && prot == VM_PROT_NONE))
- printf("pmap_page_protect(%x, %x)\n", pa, prot);
+ printf("pmap_page_protect(%lx, %x)\n", pa, prot);
#endif
/*
* Skip unmanaged pages, or operations that do not take
@@ -2534,7 +4394,7 @@ pmap_page_protect(pa, prot)
return;
write_user_windows(); /* paranoia */
if (prot & VM_PROT_READ) {
- pv_changepte(pvhead(pa), 0, PG_W);
+ pv_changepte4_4c(pvhead(pa), 0, PG_W);
return;
}
@@ -2607,13 +4467,13 @@ pmap_page_protect(pa, prot)
pteva = VA_VPG(va) << PGSHIFT;
}
- tpte = getpte(pteva);
+ tpte = getpte4(pteva);
if ((tpte & PG_V) == 0)
panic("pmap_page_protect !PG_V");
- flags |= MR(tpte);
+ flags |= MR4_4C(tpte);
if (nleft) {
- setpte(pteva, 0);
+ setpte4(pteva, 0);
#ifdef PMAP_PTESYNC
if (sp->sg_pte != NULL)
sp->sg_pte[VA_VPG(pteva)] = 0;
@@ -2692,12 +4552,12 @@ pmap_page_protect(pa, prot)
* fairly easy.
*/
void
-pmap_protect(pm, sva, eva, prot)
+pmap_protect4_4c(pm, sva, eva, prot)
register struct pmap *pm;
vm_offset_t sva, eva;
vm_prot_t prot;
{
- register int va, nva, vr, vs, pteva;
+ register int va, nva, vr, vs;
register int s, ctx;
struct regmap *rp;
struct segmap *sp;
@@ -2758,14 +4618,14 @@ if (nva == 0) panic("pmap_protect: last segment"); /* cannot happen */
*/
setcontext(pm->pm_ctxnum);
for (; va < nva; va += NBPG) {
- tpte = getpte(va);
+ tpte = getpte4(va);
pmap_stats.ps_npg_prot_all++;
if ((tpte & (PG_W|PG_TYPE)) ==
(PG_W|PG_OBMEM)) {
pmap_stats.ps_npg_prot_actual++;
if (vactype != VAC_NONE)
cache_flush_page(va);
- setpte(va, tpte & ~PG_W);
+ setpte4(va, tpte & ~PG_W);
}
}
} else {
@@ -2784,7 +4644,7 @@ if (nva == 0) panic("pmap_protect: last segment"); /* cannot happen */
setsegmap(0, sp->sg_pmeg);
pteva = VA_VPG(va) << PGSHIFT;
for (; va < nva; pteva += NBPG, va += NBPG)
- setpte(pteva, getpte(pteva) & ~PG_W);
+ setpte4(pteva, getpte4(pteva) & ~PG_W);
}
}
}
@@ -2799,24 +4659,25 @@ if (nva == 0) panic("pmap_protect: last segment"); /* cannot happen */
* is changing.
*/
void
-pmap_changeprot(pm, va, prot, wired)
+pmap_changeprot4_4c(pm, va, prot, wired)
register struct pmap *pm;
register vm_offset_t va;
vm_prot_t prot;
int wired;
{
- register int vr, vs, tpte, newprot, ctx, i, s;
+ register int vr, vs, tpte, newprot, ctx, s;
struct regmap *rp;
struct segmap *sp;
#ifdef DEBUG
if (pmapdebug & PDB_CHANGEPROT)
- printf("pmap_changeprot(%x, %x, %x, %x)\n",
+ printf("pmap_changeprot(%p, %lx, %x, %x)\n",
pm, va, prot, wired);
#endif
write_user_windows(); /* paranoia */
+ va &= ~(NBPG-1);
if (pm == pmap_kernel())
newprot = prot & VM_PROT_WRITE ? PG_S|PG_W : PG_S;
else
@@ -2856,18 +4717,12 @@ pmap_changeprot(pm, va, prot, wired)
if (CTX_USABLE(pm,rp)) {
/* use current context; flush writeback cache */
setcontext(pm->pm_ctxnum);
- tpte = getpte(va);
+ tpte = getpte4(va);
if ((tpte & PG_PROT) == newprot) {
setcontext(ctx);
goto useless;
}
-
- /*
- * the latter check deals with a writethrough cache
- * problem on the 4/300
- */
- if ((vactype==VAC_WRITEBACK ||
- (vactype==VAC_WRITETHROUGH && cputyp==CPU_SUN4)) &&
+ if (vactype == VAC_WRITEBACK &&
(tpte & (PG_U|PG_NC|PG_TYPE)) == (PG_U|PG_OBMEM))
cache_flush_page((int)va);
} else {
@@ -2879,16 +4734,304 @@ pmap_changeprot(pm, va, prot, wired)
#endif
setsegmap(0, sp->sg_pmeg);
va = VA_VPG(va) << PGSHIFT;
- tpte = getpte(va);
+ tpte = getpte4(va);
if ((tpte & PG_PROT) == newprot) {
setcontext(ctx);
goto useless;
}
}
tpte = (tpte & ~PG_PROT) | newprot;
- setpte(va, tpte);
+ setpte4(va, tpte);
+ setcontext(ctx);
+ }
+ splx(s);
+ return;
+
+useless:
+ /* only wiring changed, and we ignore wiring */
+ pmap_stats.ps_useless_changeprots++;
+ splx(s);
+}
+
+#endif /* sun4, 4c */
+
+#if defined(SUN4M) /* 4M version of protection routines above */
+/*
+ * Lower (make more strict) the protection on the specified
+ * physical page.
+ *
+ * There are only two cases: either the protection is going to 0
+ * (in which case we do the dirty work here), or it is going
+ * to read-only (in which case pv_changepte does the trick).
+ */
+void
+pmap_page_protect4m(pa, prot)
+ vm_offset_t pa;
+ vm_prot_t prot;
+{
+ register struct pvlist *pv, *pv0, *npv;
+ register struct pmap *pm;
+ register int va, vr, vs, tpte;
+ register int flags, nleft, s, ctx;
+ struct regmap *rp;
+ struct segmap *sp;
+
+#ifdef DEBUG
+ if (!pmap_pa_exists(pa))
+ panic("pmap_page_protect: no such address: 0x%lx", pa);
+ if ((pmapdebug & PDB_CHANGEPROT) ||
+ (pmapdebug & PDB_REMOVE && prot == VM_PROT_NONE))
+ printf("pmap_page_protect(%lx, %x)\n", pa, prot);
+#endif
+ /*
+ * Skip unmanaged pages, or operations that do not take
+ * away write permission.
+ */
+ if (!managed(pa) || prot & VM_PROT_WRITE)
+ return;
+ write_user_windows(); /* paranoia */
+ if (prot & VM_PROT_READ) {
+ pv_changepte4m(pvhead(pa), 0, PPROT_WRITE);
+ return;
+ }
+
+ /*
+ * Remove all access to all people talking to this page.
+ * Walk down PV list, removing all mappings.
+ * The logic is much like that for pmap_remove,
+ * but we know we are removing exactly one page.
+ */
+ pv = pvhead(pa);
+ s = splpmap();
+ if ((pm = pv->pv_pmap) == NULL) {
+ splx(s);
+ return;
+ }
+ ctx = getcontext();
+ pv0 = pv;
+ flags = pv->pv_flags /*| PV_C4M*/; /* %%%: ???? */
+ for (;; pm = pv->pv_pmap) {
+ va = pv->pv_va;
+ vr = VA_VREG(va);
+ vs = VA_VSEG(va);
+ rp = &pm->pm_regmap[vr];
+ if (rp->rg_nsegmap == 0)
+ panic("pmap_remove_all: empty vreg");
+ sp = &rp->rg_segmap[vs];
+ if ((nleft = sp->sg_npte) == 0)
+ panic("pmap_remove_all: empty vseg");
+ nleft--;
+ sp->sg_npte = nleft;
+
+ /* Invalidate PTE in MMU pagetables. Flush cache if necessary */
+ if (CTX_USABLE(pm,rp)) { /* Must flush */
+ setcontext(pm->pm_ctxnum);
+ tpte = getpte4m(va);
+ if (vactype != VAC_NONE)
+ cache_flush_page(va);
+ tlb_flush_page(va);
+ } else
+ tpte = getptesw4m(pm, va);
+
+ if ((tpte & SRMMU_TETYPE) != SRMMU_TEPTE)
+ panic("pmap_page_protect !PG_V");
+ flags |= MR4M(tpte);
+
+ if (nleft)
+ setptesw4m(pm, va, SRMMU_TEINVALID);
+ else {
+ if (pm == pmap_kernel()) {
+ tlb_flush_segment(vr, vs); /* Paranoid? */
+ if (va < virtual_avail) {
+#ifdef DEBUG
+ printf("pmap_rmk4m: attempt to free "
+ "base kernel allocation\n");
+#endif
+ goto nextpv;
+ }
+ /* no need to free the table; it is static */
+ qzero(sp->sg_pte, SRMMU_L3SIZE * sizeof(int));
+
+ /* if we're done with a region, leave it */
+
+ } else { /* User mode mapping */
+ if (CTX_USABLE(pm,rp))
+ tlb_flush_segment(vr, vs);
+ rp->rg_seg_ptps[vs] = SRMMU_TEINVALID;
+ free(sp->sg_pte, M_VMPMAP);
+ sp->sg_pte = NULL;
+
+ if (--rp->rg_nsegmap == 0) {
+ free(rp->rg_segmap, M_VMPMAP);
+ rp->rg_segmap = NULL;
+ free(rp->rg_seg_ptps, M_VMPMAP);
+ pm->pm_reg_ptps[vr] = SRMMU_TEINVALID;
+ GAP_WIDEN(pm,vr);
+ }
+ }
+ }
+ nextpv:
+ npv = pv->pv_next;
+ if (pv != pv0)
+ free(pv, M_VMPVENT);
+ if ((pv = npv) == NULL)
+ break;
+ }
+ pv0->pv_pmap = NULL;
+ pv0->pv_next = NULL; /* ? */
+ pv0->pv_flags = flags;
+ setcontext(ctx);
+ splx(s);
+}
+
+/*
+ * Lower (make more strict) the protection on the specified
+ * range of this pmap.
+ *
+ * There are only two cases: either the protection is going to 0
+ * (in which case we call pmap_remove to do the dirty work), or
+ * it is going from read/write to read-only. The latter is
+ * fairly easy.
+ */
+void
+pmap_protect4m(pm, sva, eva, prot)
+ register struct pmap *pm;
+ vm_offset_t sva, eva;
+ vm_prot_t prot;
+{
+ register int va, nva, vr, vs;
+ register int s, ctx;
+ struct regmap *rp;
+ struct segmap *sp;
+
+ if (pm == NULL || prot & VM_PROT_WRITE)
+ return;
+
+ if ((prot & VM_PROT_READ) == 0) {
+ pmap_remove(pm, sva, eva);
+ return;
+ }
+
+ write_user_windows();
+ ctx = getcontext();
+ s = splpmap();
+ simple_lock(&pm->pm_lock);
+
+ for (va = sva; va < eva;) {
+ vr = VA_VREG(va);
+ vs = VA_VSEG(va);
+ rp = &pm->pm_regmap[vr];
+ nva = VSTOVA(vr,vs + 1);
+ if (nva == 0) /* XXX */
+ panic("pmap_protect: last segment"); /* cannot happen(why?)*/
+ if (nva > eva)
+ nva = eva;
+ if (rp->rg_nsegmap == 0) {
+ va = nva;
+ continue;
+ }
+#ifdef DEBUG
+ if (rp->rg_segmap == NULL)
+ panic("pmap_protect: no segments");
+#endif
+ sp = &rp->rg_segmap[vs];
+ if (sp->sg_npte == 0) {
+ va = nva;
+ continue;
+ }
+#ifdef DEBUG
+ if (sp->sg_pte == NULL)
+ panic("pmap_protect: no pages");
+#endif
+ /* in MMU: take away write bits from MMU PTEs */
+ if (CTX_USABLE(pm,rp)) {
+ /*
+ * Flush cache so that any existing cache
+ * tags are updated. This is really only
+ * needed for PTEs that lose PG_W.
+ */
+ setcontext(pm->pm_ctxnum);
+ for (; va < nva; va += NBPG) {
+ register int tpte = getpte4m(va);
+ pmap_stats.ps_npg_prot_all++;
+ if ((tpte & (PPROT_WRITE|SRMMU_PGTYPE)) ==
+ (PPROT_WRITE|PG_SUN4M_OBMEM)) {
+ pmap_stats.ps_npg_prot_actual++;
+ if (vactype != VAC_NONE)
+ cache_flush_page(va);
+ setpte4m(va, tpte & ~PPROT_WRITE);
+ }
+ }
+ } else {
+ /*
+ * No context, hence not cached;
+ * just update PTEs.
+ */
+ setcontext(0);
+ for (; va < nva; va += NBPG) {
+ register int tpte = getptesw4m(pm, va);
+ pmap_stats.ps_npg_prot_all++;
+ if ((tpte & (PPROT_WRITE)))
+ pmap_stats.ps_npg_prot_actual++;
+ setptesw4m(pm, va, tpte & ~PPROT_WRITE);
+ }
+ }
+ }
+ simple_unlock(&pm->pm_lock);
+ splx(s);
+ setcontext(ctx);
+}
+
+/*
+ * Change the protection and/or wired status of the given (MI) virtual page.
+ * XXX: should have separate function (or flag) telling whether only wiring
+ * is changing.
+ */
+void
+pmap_changeprot4m(pm, va, prot, wired)
+ register struct pmap *pm;
+ register vm_offset_t va;
+ vm_prot_t prot;
+ int wired;
+{
+ register int tpte, newprot, ctx, s;
+
+#ifdef DEBUG
+ if (pmapdebug & PDB_CHANGEPROT)
+ printf("pmap_changeprot(%p, %lx, %x, %x)\n",
+ pm, va, prot, wired);
+#endif
+
+ write_user_windows(); /* paranoia */
+
+ va &= ~(NBPG-1);
+ if (pm == pmap_kernel())
+ newprot = prot & VM_PROT_WRITE ? PPROT_N_RWX : PPROT_N_RX;
+ else
+ newprot = prot & VM_PROT_WRITE ? PPROT_RWX_RWX : PPROT_RX_RX;
+
+ pmap_stats.ps_changeprots++;
+
+ s = splpmap(); /* conservative */
+ ctx = getcontext();
+ if (pm->pm_ctx) {
+ setcontext(pm->pm_ctxnum);
+ tpte = getpte4m(va);
+ if (vactype == VAC_WRITEBACK &&
+ (tpte & SRMMU_PGTYPE) == PG_SUN4M_OBMEM)
+ cache_flush_page(va); /* XXX: paranoia? */
+ } else {
+ tpte = getptesw4m(pm, va);
+ }
+ if ((tpte & SRMMU_PROT_MASK) == newprot) {
setcontext(ctx);
+ goto useless;
}
+ if (pm->pm_ctx)
+ setpte4m(va, (tpte & ~SRMMU_PROT_MASK) | newprot);
+ else
+ setptesw4m(pm, va, (tpte & ~SRMMU_PROT_MASK) | newprot);
+ setcontext(ctx);
splx(s);
return;
@@ -2897,6 +5040,7 @@ useless:
pmap_stats.ps_useless_changeprots++;
splx(s);
}
+#endif /* 4m */
/*
* Insert (MI) physical page pa at virtual address va in the given pmap.
@@ -2910,8 +5054,11 @@ useless:
* changing protections and/or wiring on an existing mapping.
* XXX should have different entry points for changing!
*/
+
+#if defined(SUN4) || defined(SUN4C)
+
void
-pmap_enter(pm, va, pa, prot, wired)
+pmap_enter4_4c(pm, va, pa, prot, wired)
register struct pmap *pm;
vm_offset_t va, pa;
vm_prot_t prot;
@@ -2925,7 +5072,7 @@ pmap_enter(pm, va, pa, prot, wired)
if (VA_INHOLE(va)) {
#ifdef DEBUG
- printf("pmap_enter: pm %x, va %x, pa %x: in MMU hole\n",
+ printf("pmap_enter: pm %p, va %lx, pa %lx: in MMU hole\n",
pm, va, pa);
#endif
return;
@@ -2933,7 +5080,7 @@ pmap_enter(pm, va, pa, prot, wired)
#ifdef DEBUG
if (pmapdebug & PDB_ENTER)
- printf("pmap_enter(%x, %x, %x, %x, %x)\n",
+ printf("pmap_enter(%p, %lx, %lx, %x, %x)\n",
pm, va, pa, prot, wired);
#endif
@@ -2947,27 +5094,27 @@ pmap_enter(pm, va, pa, prot, wired)
if ((pteproto & PG_TYPE) == PG_OBMEM && managed(pa)) {
#ifdef DIAGNOSTIC
if (!pmap_pa_exists(pa))
- panic("pmap_enter: no such address: %x", pa);
+ panic("pmap_enter: no such address: %lx", pa);
#endif
- pteproto |= SWTOHW(atop(pa));
pv = pvhead(pa);
} else {
- pteproto |= atop(pa) & PG_PFNUM;
pv = NULL;
}
+ pteproto |= atop(pa) & PG_PFNUM;
if (prot & VM_PROT_WRITE)
pteproto |= PG_W;
ctx = getcontext();
if (pm == pmap_kernel())
- pmap_enk(pm, va, prot, wired, pv, pteproto | PG_S);
+ pmap_enk4_4c(pm, va, prot, wired, pv, pteproto | PG_S);
else
- pmap_enu(pm, va, prot, wired, pv, pteproto);
+ pmap_enu4_4c(pm, va, prot, wired, pv, pteproto);
setcontext(ctx);
}
/* enter new (or change existing) kernel mapping */
-pmap_enk(pm, va, prot, wired, pv, pteproto)
+void
+pmap_enk4_4c(pm, va, prot, wired, pv, pteproto)
register struct pmap *pm;
vm_offset_t va;
vm_prot_t prot;
@@ -3003,7 +5150,7 @@ pmap_enk(pm, va, prot, wired, pv, pteproto)
};
}
#endif
- if (sp->sg_pmeg != seginval && (tpte = getpte(va)) & PG_V) {
+ if (sp->sg_pmeg != seginval && (tpte = getpte4(va)) & PG_V) {
register int addr;
/* old mapping exists, and is of the same pa type */
@@ -3017,7 +5164,7 @@ pmap_enk(pm, va, prot, wired, pv, pteproto)
if ((tpte & PG_TYPE) == PG_OBMEM) {
#ifdef DEBUG
-printf("pmap_enk: changing existing va=>pa entry: va %x, pteproto %x\n",
+printf("pmap_enk: changing existing va=>pa entry: va %lx, pteproto %x\n",
va, pteproto);
#endif
/*
@@ -3025,9 +5172,9 @@ printf("pmap_enk: changing existing va=>pa entry: va %x, pteproto %x\n",
* If old pa was managed, remove from pvlist.
* If old page was cached, flush cache.
*/
- addr = ptoa(HWTOSW(tpte & PG_PFNUM));
+ addr = ptoa(tpte & PG_PFNUM);
if (managed(addr))
- pv_unlink(pvhead(addr), pm, va);
+ pv_unlink4_4c(pvhead(addr), pm, va);
if ((tpte & PG_NC) == 0) {
setcontext(0); /* ??? */
if (vactype != VAC_NONE)
@@ -3045,7 +5192,7 @@ printf("pmap_enk: changing existing va=>pa entry: va %x, pteproto %x\n",
* unique (hence will never cause a second call to malloc).
*/
if (pv != NULL)
- pteproto |= pv_link(pv, pm, va);
+ pteproto |= pv_link4_4c(pv, pm, va);
if (sp->sg_pmeg == seginval) {
register int tva;
@@ -3080,18 +5227,19 @@ printf("pmap_enk: changing existing va=>pa entry: va %x, pteproto %x\n",
tva = VA_ROUNDDOWNTOSEG(va);
i = NPTESG;
do {
- setpte(tva, 0);
+ setpte4(tva, 0);
tva += NBPG;
} while (--i > 0);
}
/* ptes kept in hardware only */
- setpte(va, pteproto);
+ setpte4(va, pteproto);
splx(s);
}
/* enter new (or change existing) user mapping */
-pmap_enu(pm, va, prot, wired, pv, pteproto)
+void
+pmap_enu4_4c(pm, va, prot, wired, pv, pteproto)
register struct pmap *pm;
vm_offset_t va;
vm_prot_t prot;
@@ -3140,7 +5288,7 @@ printf("pmap_enter: segment filled during sleep\n"); /* can this happen? */
free(sp, M_VMPMAP);
goto rretry;
}
- bzero((caddr_t)sp, size);
+ qzero((caddr_t)sp, size);
rp->rg_segmap = sp;
rp->rg_nsegmap = 0;
for (i = NSEGRG; --i >= 0;)
@@ -3164,7 +5312,7 @@ printf("pmap_enter: pte filled during sleep\n"); /* can this happen? */
if (sp->sg_pmeg != seginval)
panic("pmap_enter: new ptes, but not seginval");
#endif
- bzero((caddr_t)pte, size);
+ qzero((caddr_t)pte, size);
sp->sg_pte = pte;
sp->sg_npte = 1;
rp->rg_nsegmap++;
@@ -3178,7 +5326,7 @@ printf("pmap_enter: pte filled during sleep\n"); /* can this happen? */
/* hardware pte */
if (CTX_USABLE(pm,rp)) {
setcontext(pm->pm_ctxnum);
- tpte = getpte(va);
+ tpte = getpte4(va);
doflush = 1;
} else {
setcontext(0);
@@ -3188,7 +5336,7 @@ printf("pmap_enter: pte filled during sleep\n"); /* can this happen? */
setregmap(0, tregion);
#endif
setsegmap(0, pmeg);
- tpte = getpte(VA_VPG(va) << PGSHIFT);
+ tpte = getpte4(VA_VPG(va) << PGSHIFT);
}
}
if (tpte & PG_V) {
@@ -3200,7 +5348,7 @@ printf("pmap_enter: pte filled during sleep\n"); /* can this happen? */
/* just changing prot and/or wiring */
splx(s);
/* caller should call this directly: */
- pmap_changeprot(pm, va, prot, wired);
+ pmap_changeprot4_4c(pm, va, prot, wired);
if (wired)
pm->pm_stats.wired_count++;
else
@@ -3215,9 +5363,9 @@ printf("pmap_enter: pte filled during sleep\n"); /* can this happen? */
/*printf("%s[%d]: pmap_enu: changing existing va(%x)=>pa entry\n",
curproc->p_comm, curproc->p_pid, va);*/
if ((tpte & PG_TYPE) == PG_OBMEM) {
- addr = ptoa(HWTOSW(tpte & PG_PFNUM));
+ addr = ptoa(tpte & PG_PFNUM);
if (managed(addr))
- pv_unlink(pvhead(addr), pm, va);
+ pv_unlink4_4c(pvhead(addr), pm, va);
if (vactype != VAC_NONE &&
doflush && (tpte & PG_NC) == 0)
cache_flush_page((int)va);
@@ -3235,7 +5383,7 @@ curproc->p_comm, curproc->p_pid, va);*/
}
if (pv != NULL)
- pteproto |= pv_link(pv, pm, va);
+ pteproto |= pv_link4_4c(pv, pm, va);
/*
* Update hardware & software PTEs.
@@ -3254,7 +5402,7 @@ curproc->p_comm, curproc->p_pid, va);*/
setsegmap(0, pmeg);
va = VA_VPG(va) << PGSHIFT;
}
- setpte(va, pteproto);
+ setpte4(va, pteproto);
}
/* update software copy */
pte += VA_VPG(va);
@@ -3263,6 +5411,330 @@ curproc->p_comm, curproc->p_pid, va);*/
splx(s);
}
+#endif /*sun4,4c*/
+
+#if defined(SUN4M) /* Sun4M versions of enter routines */
+/*
+ * Insert (MI) physical page pa at virtual address va in the given pmap.
+ * NB: the pa parameter includes type bits PMAP_OBIO, PMAP_NC as necessary.
+ *
+ * If pa is not in the `managed' range it will not be `bank mapped'.
+ * This works during bootstrap only because the first 4MB happens to
+ * map one-to-one.
+ *
+ * There may already be something else there, or we might just be
+ * changing protections and/or wiring on an existing mapping.
+ * XXX should have different entry points for changing!
+ */
+
+void
+pmap_enter4m(pm, va, pa, prot, wired)
+ register struct pmap *pm;
+ vm_offset_t va, pa;
+ vm_prot_t prot;
+ int wired;
+{
+ register struct pvlist *pv;
+ register int pteproto, ctx;
+
+ if (pm == NULL)
+ return;
+
+#ifdef DEBUG
+ if (pmapdebug & PDB_ENTER)
+ printf("pmap_enter(%p, %lx, %lx, %x, %x)\n",
+ pm, va, pa, prot, wired);
+#endif
+
+ /* Initialise pteproto with cache bit */
+ pteproto = (pa & PMAP_NC) == 0 ? SRMMU_PG_C : 0;
+
+ if (pa & PMAP_TYPE4M) { /* this page goes in an iospace */
+ if (cpumod == SUN4M_MS)
+ panic("pmap_enter4m: attempt to use 36-bit iospace on"
+ " MicroSPARC");
+ pteproto |= (pa & PMAP_TYPE4M) << PMAP_PTESHFT4M;
+ }
+
+ /* Make sure we get a pte with appropriate perms! */
+ pteproto |= SRMMU_TEPTE | PPROT_RX_RX;
+
+ pa &= ~PMAP_TNC;
+ /*
+ * Set up prototype for new PTE. Cannot set PG_NC from PV_NC yet
+ * since the pvlist no-cache bit might change as a result of the
+ * new mapping.
+ */
+ if ((pteproto & SRMMU_PGTYPE) == PG_SUN4M_OBMEM && managed(pa)) {
+#ifdef DIAGNOSTIC
+ if (!pmap_pa_exists(pa))
+ panic("pmap_enter: no such address: %lx", pa);
+#endif
+ pv = pvhead(pa);
+ } else {
+ pv = NULL;
+ }
+ pteproto |= (atop(pa) << SRMMU_PPNSHIFT);
+
+ if (prot & VM_PROT_WRITE)
+ pteproto |= PPROT_WRITE;
+
+ ctx = getcontext();
+
+ if (pm == pmap_kernel())
+ pmap_enk4m(pm, va, prot, wired, pv, pteproto | PPROT_S);
+ else
+ pmap_enu4m(pm, va, prot, wired, pv, pteproto);
+
+ setcontext(ctx);
+}
+
+/* enter new (or change existing) kernel mapping */
+void
+pmap_enk4m(pm, va, prot, wired, pv, pteproto)
+ register struct pmap *pm;
+ vm_offset_t va;
+ vm_prot_t prot;
+ int wired;
+ register struct pvlist *pv;
+ register int pteproto;
+{
+ register int vr, vs, tpte, s;
+ struct regmap *rp;
+ struct segmap *sp;
+
+#ifdef DEBUG
+ if (va < KERNBASE)
+ panic("pmap_enk4m: can't enter va 0x%lx below KERNBASE",va);
+#endif
+ vr = VA_VREG(va);
+ vs = VA_VSEG(va);
+ rp = &pm->pm_regmap[vr];
+ sp = &rp->rg_segmap[vs];
+
+ s = splpmap(); /* XXX way too conservative */
+
+ if (rp->rg_seg_ptps == NULL) /* enter new region */
+ panic("pmap_enk4m: missing kernel region table for va %lx",va);
+
+ if (((tpte = getpte4m(va)) & SRMMU_TETYPE) == SRMMU_TEPTE) {
+ register int addr;
+
+ /* old mapping exists, and is of the same pa type */
+
+ if ((tpte & SRMMU_PPNMASK) == (pteproto & SRMMU_PPNMASK)) {
+ /* just changing protection and/or wiring */
+ splx(s);
+ pmap_changeprot(pm, va, prot, wired);
+ return;
+ }
+
+ if ((tpte & SRMMU_PGTYPE) == PG_SUN4M_OBMEM) {
+#ifdef DEBUG
+printf("pmap_enk4m: changing existing va=>pa entry: va %lx, pteproto %x, "
+ "oldpte %x\n", va, pteproto, tpte);
+#endif
+ /*
+ * Switcheroo: changing pa for this va.
+ * If old pa was managed, remove from pvlist.
+ * If old page was cached, flush cache.
+ */
+ addr = ptoa((tpte & SRMMU_PPNMASK) >> SRMMU_PPNSHIFT);
+ if (managed(addr))
+ pv_unlink4m(pvhead(addr), pm, va);
+ if (tpte & SRMMU_PG_C) {
+ setcontext(0); /* ??? */
+ if (vactype != VAC_NONE)
+ cache_flush_page((int)va);
+ }
+ }
+ } else {
+ /* adding new entry */
+ sp->sg_npte++;
+ }
+
+ /*
+ * If the new mapping is for a managed PA, enter into pvlist.
+ * Note that the mapping for a malloc page will always be
+ * unique (hence will never cause a second call to malloc).
+ */
+ if (pv != NULL)
+ pteproto &= ~(pv_link4m(pv, pm, va));
+
+ if (sp->sg_pte == NULL) /* If no existing pagetable */
+ panic("pmap_enk4m: missing segment table for va 0x%lx",va);
+
+ /* ptes kept in hardware only */
+ setpte4m(va, pteproto);
+
+ splx(s);
+}
+
+/* enter new (or change existing) user mapping */
+void
+pmap_enu4m(pm, va, prot, wired, pv, pteproto)
+ register struct pmap *pm;
+ vm_offset_t va;
+ vm_prot_t prot;
+ int wired;
+ register struct pvlist *pv;
+ register int pteproto;
+{
+ register int vr, vs, *pte, tpte, s, doflush;
+ struct regmap *rp;
+ struct segmap *sp;
+
+ write_user_windows(); /* XXX conservative */
+ vr = VA_VREG(va);
+ vs = VA_VSEG(va);
+ rp = &pm->pm_regmap[vr];
+ s = splpmap(); /* XXX conservative */
+
+#ifdef DEBUG
+ if (pm->pm_gap_end < pm->pm_gap_start) {
+ printf("pmap_enu: gap_start %x, gap_end %x",
+ pm->pm_gap_start, pm->pm_gap_end);
+ panic("pmap_enu: gap botch");
+ }
+#endif
+
+rretry:
+ if (rp->rg_segmap == NULL) {
+ /* definitely a new mapping */
+ register int size = NSEGRG * sizeof (struct segmap);
+
+ sp = (struct segmap *)malloc((u_long)size, M_VMPMAP, M_WAITOK);
+ if (rp->rg_segmap != NULL) {
+#ifdef DEBUG
+printf("pmap_enu4m: segment filled during sleep\n"); /* can this happen? */
+#endif
+ free(sp, M_VMPMAP);
+ goto rretry;
+ }
+ qzero((caddr_t)sp, size);
+ rp->rg_segmap = sp;
+ rp->rg_nsegmap = 0;
+ rp->rg_seg_ptps = NULL;
+ }
+rgretry:
+ if (rp->rg_seg_ptps == NULL) {
+ /* Need a segment table */
+ register int size;
+ register caddr_t tblp;
+ size = SRMMU_L2SIZE * sizeof(long);
+ tblp = malloc(size, M_VMPMAP, M_WAITOK);
+ if (rp->rg_seg_ptps != NULL) {
+#ifdef DEBUG
+printf("pmap_enu4m: bizarre segment table fill during sleep\n");
+#endif
+ free(tblp,M_VMPMAP);
+ goto rgretry;
+ }
+ if (cant_cache_pagetables)
+ kvm_uncache(tblp, (size+NBPG-1)/NBPG);
+
+ rp->rg_seg_ptps = (int *)tblp;
+ qzero(tblp, size);
+ pm->pm_reg_ptps[vr] =
+ (VA2PA(tblp) >> SRMMU_PPNPASHIFT) | SRMMU_TEPTD;
+ }
+
+ sp = &rp->rg_segmap[vs];
+
+sretry:
+ if ((pte = sp->sg_pte) == NULL) {
+ /* definitely a new mapping */
+ register int size = SRMMU_L3SIZE * sizeof(*pte);
+
+ pte = (int *)malloc((u_long)size, M_VMPMAP, M_WAITOK);
+ if (sp->sg_pte != NULL) {
+printf("pmap_enter: pte filled during sleep\n"); /* can this happen? */
+ free(pte, M_VMPMAP);
+ goto sretry;
+ }
+ if (cant_cache_pagetables)
+ kvm_uncache((caddr_t)pte, (size+NBPG-1)/NBPG);
+
+ qzero((caddr_t)pte, size);
+ sp->sg_pte = pte;
+ sp->sg_npte = 1;
+ rp->rg_nsegmap++;
+ rp->rg_seg_ptps[vs] =
+ (VA2PA((caddr_t)pte) >> SRMMU_PPNPASHIFT) | SRMMU_TEPTD;
+ } else {
+ /* might be a change: fetch old pte */
+ doflush = 0;
+
+ if (CTX_USABLE(pm,rp)) {
+ setcontext(pm->pm_ctxnum);
+ tpte = getpte4m(va);
+ doflush = 1;
+ } else {
+ tpte = getptesw4m(pm, va);
+ }
+ if ((tpte & SRMMU_TETYPE) == SRMMU_TEPTE) {
+ register int addr;
+
+ /* old mapping exists, and is of the same pa type */
+ if ((tpte & SRMMU_PPNMASK) ==
+ (pteproto & SRMMU_PPNMASK)) {
+ /* just changing prot and/or wiring */
+ splx(s);
+ /* caller should call this directly: */
+ pmap_changeprot4m(pm, va, prot, wired);
+ if (wired)
+ pm->pm_stats.wired_count++;
+ else
+ pm->pm_stats.wired_count--;
+ return;
+ }
+ /*
+ * Switcheroo: changing pa for this va.
+ * If old pa was managed, remove from pvlist.
+ * If old page was cached, flush cache.
+ */
+#ifdef DEBUG
+if (pmapdebug & PDB_ENTER)
+printf("%s[%d]: pmap_enu: changing existing va(%x)=>pa(pte=%x) entry\n",
+curproc->p_comm, curproc->p_pid, (int)va, (int)pte);
+#endif
+ if ((tpte & SRMMU_PGTYPE) == PG_SUN4M_OBMEM) {
+ addr = ptoa( (tpte & SRMMU_PPNMASK) >>
+ SRMMU_PPNSHIFT);
+ if (managed(addr))
+ pv_unlink4m(pvhead(addr), pm, va);
+ if (vactype != VAC_NONE &&
+ doflush && (tpte & SRMMU_PG_C))
+ cache_flush_page((int)va);
+ }
+ } else {
+ /* adding new entry */
+ sp->sg_npte++;
+
+ /*
+ * Increment counters
+ */
+ if (wired)
+ pm->pm_stats.wired_count++;
+ }
+ }
+ if (pv != NULL)
+ pteproto &= ~(pv_link4m(pv, pm, va));
+
+ /*
+ * Update hardware & software PTEs.
+ */
+ if (CTX_USABLE(pm,rp)) {
+ setcontext(pm->pm_ctxnum);
+ setpte4m(va, pteproto);
+ } else
+ setptesw4m(pm, va, pteproto);
+ /* XXX: restore previous context here? */
+
+ splx(s);
+}
+#endif /* sun4m */
+
/*
* Change the wiring attribute for a map/virtual-address pair.
*/
@@ -3282,8 +5754,10 @@ pmap_change_wiring(pm, va, wired)
* with the given map/virtual_address pair.
* GRR, the vm code knows; we should not have to do this!
*/
+
+#if defined(SUN4) || defined(SUN4C)
vm_offset_t
-pmap_extract(pm, va)
+pmap_extract4_4c(pm, va)
register struct pmap *pm;
vm_offset_t va;
{
@@ -3309,16 +5783,16 @@ pmap_extract(pm, va)
register int ctx = getcontext();
if (CTX_USABLE(pm,rp)) {
- setcontext(pm->pm_ctxnum);
- tpte = getpte(va);
+ CHANGE_CONTEXTS(ctx, pm->pm_ctxnum);
+ tpte = getpte4(va);
} else {
- setcontext(0);
+ CHANGE_CONTEXTS(ctx, 0);
#ifdef MMU_3L
if (mmu_3l)
setregmap(0, tregion);
#endif
setsegmap(0, sp->sg_pmeg);
- tpte = getpte(VA_VPG(va) << PGSHIFT);
+ tpte = getpte4(VA_VPG(va) << PGSHIFT);
}
setcontext(ctx);
} else {
@@ -3335,9 +5809,48 @@ pmap_extract(pm, va)
return (0);
}
tpte &= PG_PFNUM;
- tpte = HWTOSW(tpte);
+ tpte = tpte;
return ((tpte << PGSHIFT) | (va & PGOFSET));
}
+#endif /*4,4c*/
+
+#if defined(SUN4M) /* 4m version of pmap_extract */
+/*
+ * Extract the physical page address associated
+ * with the given map/virtual_address pair.
+ * GRR, the vm code knows; we should not have to do this!
+ */
+vm_offset_t
+pmap_extract4m(pm, va)
+ register struct pmap *pm;
+ vm_offset_t va;
+{
+ register int tpte, ctx;
+
+ if (pm == NULL) {
+ printf("pmap_extract: null pmap\n");
+ return (0);
+ }
+
+ if (pm->pm_ctx) {
+ ctx = getcontext();
+ CHANGE_CONTEXTS(ctx, pm->pm_ctxnum);
+ tpte = getpte4m(va);
+#ifdef DEBUG
+ if ((tpte & SRMMU_TETYPE) != SRMMU_TEPTE) {
+ printf("pmap_extract: invalid pte of type %d\n",
+ tpte & SRMMU_TETYPE);
+ return (0);
+ }
+#endif
+ setcontext(ctx);
+ } else
+ tpte = getptesw4m(pm, va);
+
+
+ return (ptoa((tpte & SRMMU_PPNMASK) >> SRMMU_PPNSHIFT) | VA_OFF(va));
+}
+#endif /* sun4m */
/*
* Copy the range specified by src_addr/len
@@ -3354,6 +5867,24 @@ pmap_copy(dst_pmap, src_pmap, dst_addr, len, src_addr)
vm_size_t len;
vm_offset_t src_addr;
{
+#if 0
+ if (CPU_ISSUN4M) {
+ register int i, pte;
+ for (i = 0; i < len/NBPG; i++) {
+ pte = getptesw4m(src_pmap, src_addr);
+ pmap_enter(dst_pmap, dst_addr,
+ ptoa((pte & SRMMU_PPNMASK) >>
+ SRMMU_PPNSHIFT) |
+ VA_OFF(src_addr),
+ (pte & PPROT_WRITE)
+ ? VM_PROT_WRITE| VM_PROT_READ
+ : VM_PROT_READ,
+ 0);
+ src_addr += NBPG;
+ dst_addr += NBPG;
+ }
+ }
+#endif
}
/*
@@ -3364,6 +5895,10 @@ pmap_copy(dst_pmap, src_pmap, dst_addr, len, src_addr)
void
pmap_update()
{
+#if defined(SUN4M)
+ if (CPU_ISSUN4M)
+ tlb_flush_all(); /* %%%: Extreme Paranoia? */
+#endif
}
/*
@@ -3381,18 +5916,20 @@ pmap_collect(pm)
{
}
+#if defined(SUN4) || defined(SUN4C)
+
/*
* Clear the modify bit for the given physical page.
*/
void
-pmap_clear_modify(pa)
+pmap_clear_modify4_4c(pa)
register vm_offset_t pa;
{
register struct pvlist *pv;
if ((pa & (PMAP_TNC & ~PMAP_NC)) == 0 && managed(pa)) {
pv = pvhead(pa);
- (void) pv_syncflags(pv);
+ (void) pv_syncflags4_4c(pv);
pv->pv_flags &= ~PV_MOD;
}
}
@@ -3401,14 +5938,14 @@ pmap_clear_modify(pa)
* Tell whether the given physical page has been modified.
*/
int
-pmap_is_modified(pa)
+pmap_is_modified4_4c(pa)
register vm_offset_t pa;
{
register struct pvlist *pv;
if ((pa & (PMAP_TNC & ~PMAP_NC)) == 0 && managed(pa)) {
pv = pvhead(pa);
- if (pv->pv_flags & PV_MOD || pv_syncflags(pv) & PV_MOD)
+ if (pv->pv_flags & PV_MOD || pv_syncflags4_4c(pv) & PV_MOD)
return (1);
}
return (0);
@@ -3418,14 +5955,14 @@ pmap_is_modified(pa)
* Clear the reference bit for the given physical page.
*/
void
-pmap_clear_reference(pa)
+pmap_clear_reference4_4c(pa)
vm_offset_t pa;
{
register struct pvlist *pv;
if ((pa & (PMAP_TNC & ~PMAP_NC)) == 0 && managed(pa)) {
pv = pvhead(pa);
- (void) pv_syncflags(pv);
+ (void) pv_syncflags4_4c(pv);
pv->pv_flags &= ~PV_REF;
}
}
@@ -3434,18 +5971,97 @@ pmap_clear_reference(pa)
* Tell whether the given physical page has been referenced.
*/
int
-pmap_is_referenced(pa)
+pmap_is_referenced4_4c(pa)
vm_offset_t pa;
{
register struct pvlist *pv;
if ((pa & (PMAP_TNC & ~PMAP_NC)) == 0 && managed(pa)) {
pv = pvhead(pa);
- if (pv->pv_flags & PV_REF || pv_syncflags(pv) & PV_REF)
+ if (pv->pv_flags & PV_REF || pv_syncflags4_4c(pv) & PV_REF)
return (1);
}
return (0);
}
+#endif /*4,4c*/
+
+#if defined(SUN4M)
+
+/*
+ * 4m versions of bit test/set routines
+ *
+ * Note that the 4m-specific routines should eventually service these
+ * requests from their page tables, and the whole pvlist bit mess should
+ * be dropped for the 4m (unless this causes a performance hit from
+ * tracing down pagetables/regmap/segmaps).
+ */
+
+/*
+ * Clear the modify bit for the given physical page.
+ */
+void
+pmap_clear_modify4m(pa) /* XXX %%%: Should service from swpagetbl for 4m */
+ register vm_offset_t pa;
+{
+ register struct pvlist *pv;
+
+ if ((pa & (PMAP_TNC & ~PMAP_NC)) == 0 && managed(pa)) {
+ pv = pvhead(pa);
+ (void) pv_syncflags4m(pv);
+ pv->pv_flags &= ~PV_MOD4M;
+ }
+}
+
+/*
+ * Tell whether the given physical page has been modified.
+ */
+int
+pmap_is_modified4m(pa) /* Test performance with SUN4M && SUN4/4C. XXX */
+ register vm_offset_t pa;
+{
+ register struct pvlist *pv;
+
+ if ((pa & (PMAP_TNC & ~PMAP_NC)) == 0 && managed(pa)) {
+ pv = pvhead(pa);
+ if (pv->pv_flags & PV_MOD4M || pv_syncflags4m(pv) & PV_MOD4M)
+ return(1);
+ }
+ return (0);
+}
+
+/*
+ * Clear the reference bit for the given physical page.
+ */
+void
+pmap_clear_reference4m(pa)
+ vm_offset_t pa;
+{
+ register struct pvlist *pv;
+
+ if ((pa & (PMAP_TNC & ~PMAP_NC)) == 0 && managed(pa)) {
+ pv = pvhead(pa);
+ (void) pv_syncflags4m(pv);
+ pv->pv_flags &= ~PV_REF4M;
+ }
+}
+
+/*
+ * Tell whether the given physical page has been referenced.
+ */
+int
+pmap_is_referenced4m(pa)
+ vm_offset_t pa;
+{
+ register struct pvlist *pv;
+
+ if ((pa & (PMAP_TNC & ~PMAP_NC)) == 0 && managed(pa)) {
+ pv = pvhead(pa);
+ if (pv->pv_flags & PV_REF4M || pv_syncflags4m(pv) & PV_REF4M)
+ return(1);
+ }
+ return (0);
+}
+#endif /* 4m */
/*
* Make the specified pages (by pmap, offset) pageable (or not) as requested.
@@ -3472,8 +6088,11 @@ pmap_pageable(pm, start, end, pageable)
* We avoid stomping on the cache.
* XXX might be faster to use destination's context and allow cache to fill?
*/
+
+#if defined(SUN4) || defined(SUN4C)
+
void
-pmap_zero_page(pa)
+pmap_zero_page4_4c(pa)
register vm_offset_t pa;
{
register caddr_t va;
@@ -3487,14 +6106,13 @@ pmap_zero_page(pa)
*/
if (vactype != VAC_NONE)
pv_flushcache(pvhead(pa));
- pte = PG_V | PG_S | PG_W | PG_NC | SWTOHW(atop(pa));
- } else
- pte = PG_V | PG_S | PG_W | PG_NC | (atop(pa) & PG_PFNUM);
+ }
+ pte = PG_V | PG_S | PG_W | PG_NC | (atop(pa) & PG_PFNUM);
va = vpage[0];
- setpte(va, pte);
+ setpte4(va, pte);
qzero(va, NBPG);
- setpte(va, 0);
+ setpte4(va, 0);
}
/*
@@ -3507,7 +6125,7 @@ pmap_zero_page(pa)
* the processor.
*/
void
-pmap_copy_page(src, dst)
+pmap_copy_page4_4c(src, dst)
vm_offset_t src, dst;
{
register caddr_t sva, dva;
@@ -3516,28 +6134,101 @@ pmap_copy_page(src, dst)
if (managed(src)) {
if (vactype == VAC_WRITEBACK)
pv_flushcache(pvhead(src));
- spte = PG_V | PG_S | SWTOHW(atop(src));
- } else
- spte = PG_V | PG_S | (atop(src) & PG_PFNUM);
+ }
+ spte = PG_V | PG_S | (atop(src) & PG_PFNUM);
if (managed(dst)) {
/* similar `might not be necessary' comment applies */
if (vactype != VAC_NONE)
pv_flushcache(pvhead(dst));
- dpte = PG_V | PG_S | PG_W | PG_NC | SWTOHW(atop(dst));
- } else
- dpte = PG_V | PG_S | PG_W | PG_NC | (atop(dst) & PG_PFNUM);
+ }
+ dpte = PG_V | PG_S | PG_W | PG_NC | (atop(dst) & PG_PFNUM);
sva = vpage[0];
dva = vpage[1];
- setpte(sva, spte);
- setpte(dva, dpte);
+ setpte4(sva, spte);
+ setpte4(dva, dpte);
qcopy(sva, dva, NBPG); /* loads cache, so we must ... */
if (vactype != VAC_NONE)
cache_flush_page((int)sva);
- setpte(sva, 0);
- setpte(dva, 0);
+ setpte4(sva, 0);
+ setpte4(dva, 0);
}
+#endif /* 4, 4c */
+
+#if defined(SUN4M) /* Sun4M version of copy/zero routines */
+/*
+ * Fill the given MI physical page with zero bytes.
+ *
+ * We avoid stomping on the cache.
+ * XXX might be faster to use destination's context and allow cache to fill?
+ */
+void
+pmap_zero_page4m(pa)
+ register vm_offset_t pa;
+{
+ register caddr_t va;
+ register int pte;
+
+ if (((pa & (PMAP_TNC & ~PMAP_NC)) == 0) && managed(pa)) {
+ /*
+ * The following might not be necessary since the page
+ * is being cleared because it is about to be allocated,
+ * i.e., is in use by no one.
+ */
+ if (vactype != VAC_NONE)
+ pv_flushcache(pvhead(pa));
+ }
+ pte = ~SRMMU_PG_C & (SRMMU_TEPTE | PPROT_S | PPROT_WRITE |
+ (atop(pa) << SRMMU_PPNSHIFT));
+ va = vpage[0];
+ setpte4m((vm_offset_t) va, pte);
+ qzero(va, NBPG);
+ setpte4m((vm_offset_t) va, SRMMU_TEINVALID);
+}
+
+/*
+ * Copy the given MI physical source page to its destination.
+ *
+ * We avoid stomping on the cache as above (with same `XXX' note).
+ * We must first flush any write-back cache for the source page.
+ * We go ahead and stomp on the kernel's virtual cache for the
+ * source page, since the cache can read memory MUCH faster than
+ * the processor.
+ */
+void
+pmap_copy_page4m(src, dst)
+ vm_offset_t src, dst;
+{
+ register caddr_t sva, dva;
+ register int spte, dpte;
+
+ if (managed(src)) {
+ if (vactype == VAC_WRITEBACK)
+ pv_flushcache(pvhead(src));
+ }
+ spte = SRMMU_TEPTE | SRMMU_PG_C | PPROT_S |
+ (atop(src) << SRMMU_PPNSHIFT);
+
+ if (managed(dst)) {
+ /* similar `might not be necessary' comment applies */
+ if (vactype != VAC_NONE)
+ pv_flushcache(pvhead(dst));
+ }
+ dpte = ~SRMMU_PG_C & (SRMMU_TEPTE | PPROT_S | PPROT_WRITE |
+ (atop(dst) << SRMMU_PPNSHIFT));
+
+ sva = vpage[0];
+ dva = vpage[1];
+ setpte4m((vm_offset_t) sva, spte);
+ setpte4m((vm_offset_t) dva, dpte);
+ qcopy(sva, dva, NBPG); /* loads cache, so we must ... */
+ if (vactype != VAC_NONE)
+ cache_flush_page((int)sva);
+ setpte4m((vm_offset_t) sva, SRMMU_TEINVALID);
+ setpte4m((vm_offset_t) dva, SRMMU_TEINVALID);
+}
+#endif /* Sun4M */
/*
* Turn a cdevsw d_mmap value into a byte address for pmap_enter.
@@ -3558,20 +6249,36 @@ pmap_phys_address(x)
* We just assert PG_NC for each PTE; the addresses must reside
* in locked kernel space. A cache flush is also done.
*/
+void
kvm_uncache(va, npages)
register caddr_t va;
register int npages;
{
register int pte;
-
- for (; --npages >= 0; va += NBPG) {
- pte = getpte(va);
- if ((pte & PG_V) == 0)
- panic("kvm_uncache !pg_v");
- pte |= PG_NC;
- setpte(va, pte);
- if (vactype != VAC_NONE && (pte & PG_TYPE) == PG_OBMEM)
- cache_flush_page((int)va);
+ if (CPU_ISSUN4M) {
+#if defined(SUN4M)
+ for (; --npages >= 0; va += NBPG) {
+ pte = getpte4m((vm_offset_t) va);
+ if ((pte & SRMMU_TETYPE) != SRMMU_TEPTE)
+ panic("kvm_uncache: table entry not pte");
+ pte &= ~SRMMU_PG_C;
+ setpte4m((vm_offset_t) va, pte);
+ if (vactype != VAC_NONE && (pte & PG_TYPE) == PG_OBMEM)
+ cache_flush_page((int)va);
+ }
+#endif
+ } else {
+#if defined(SUN4) || defined(SUN4C)
+ for (; --npages >= 0; va += NBPG) {
+ pte = getpte4(va);
+ if ((pte & PG_V) == 0)
+ panic("kvm_uncache !pg_v");
+ pte |= PG_NC;
+ setpte4(va, pte);
+ if (vactype != VAC_NONE && (pte & PG_TYPE) == PG_OBMEM)
+ cache_flush_page((int)va);
+ }
+#endif
}
}
@@ -3581,19 +6288,25 @@ kvm_uncache(va, npages)
* We just assert PG_NC for each PTE; the addresses must reside
* in locked kernel space. A cache flush is also done.
*/
+void
kvm_iocache(va, npages)
register caddr_t va;
register int npages;
{
- register int pte;
+#ifdef SUN4M
+ if (CPU_ISSUN4M) /* %%%: Implement! */
+ panic("kvm_iocache: 4m iocache not implemented");
+#endif
+#if defined(SUN4) || defined(SUN4C)
for (; --npages >= 0; va += NBPG) {
- pte = getpte(va);
+ register int pte = getpte4(va);
if ((pte & PG_V) == 0)
panic("kvm_iocache !pg_v");
pte |= PG_IOC;
- setpte(va, pte);
+ setpte4(va, pte);
}
+#endif
}
int
@@ -3619,62 +6332,52 @@ pmap_count_ptes(pm)
}
/*
- * Find first virtual address >= va that doesn't cause
- * a cache alias on physical address pa.
+ * Find first virtual address >= *va that is
+ * least likely to cause cache aliases.
+ * (This will just seg-align mappings.)
*/
-vm_offset_t
-pmap_prefer(pa, va)
- register vm_offset_t pa;
- register vm_offset_t va;
+void
+pmap_prefer(foff, vap)
+ register vm_offset_t foff;
+ register vm_offset_t *vap;
{
- register struct pvlist *pv;
- register long m, d;
+ register vm_offset_t va = *vap;
+ register long d, m;
+
+ if (VA_INHOLE(va))
+ va = MMU_HOLE_END;
m = CACHE_ALIAS_DIST;
if (m == 0) /* m=0 => no cache aliasing */
- return (va);
-
- if (pa == (vm_offset_t)-1) {
- /*
- * Do not consider physical address. Just return
- * a cache aligned address.
- */
- if (VA_INHOLE(va))
- va = MMU_HOLE_END;
-
- /* XXX - knowledge about `exec' formats; can we get by without? */
- va -= USRTEXT;
- va = (va + m - 1) & ~(m - 1);
- return (va + USRTEXT);
- }
-
- if ((pa & (PMAP_TNC & ~PMAP_NC)) || !managed(pa))
- return va;
-
- pv = pvhead(pa);
- if (pv->pv_pmap == NULL) {
- /* Unusable, tell caller to try another one */
- return (vm_offset_t)-1;
- }
-
- d = (long)(pv->pv_va & (m - 1)) - (long)(va & (m - 1));
- if (d < 0)
- va += m;
- va += d;
+ return;
- return va;
+ d = foff - va;
+ d &= (m - 1);
+ *vap = va + d;
}
+void
pmap_redzone()
{
- setpte(KERNBASE, 0);
+#if defined(SUN4M)
+ if (CPU_ISSUN4M) {
+ setpte4m(KERNBASE, 0);
+ return;
+ }
+#endif
+#if defined(SUN4) || defined(SUN4C)
+ if (CPU_ISSUN4OR4C) {
+ setpte4(KERNBASE, 0);
+ return;
+ }
+#endif
}
#ifdef DEBUG
/*
* Check consistency of a pmap (time consuming!).
*/
-int
+void
pm_check(s, pm)
char *s;
struct pmap *pm;
@@ -3685,7 +6388,7 @@ pm_check(s, pm)
pm_check_u(s, pm);
}
-int
+void
pm_check_u(s, pm)
char *s;
struct pmap *pm;
@@ -3694,6 +6397,24 @@ pm_check_u(s, pm)
struct segmap *sp;
int n, vs, vr, j, m, *pte;
+ if (pm->pm_regmap == NULL)
+ panic("%s: CHK(pmap %p): no region mapping", s, pm);
+
+#if defined(SUN4M)
+ if (CPU_ISSUN4M &&
+ (pm->pm_reg_ptps == NULL ||
+ pm->pm_reg_ptps_pa != VA2PA((caddr_t)pm->pm_reg_ptps)))
+ panic("%s: CHK(pmap %p): no SRMMU region table or bad pa: tblva=%p, tblpa=0x%x",
+ s, pm, pm->pm_reg_ptps, pm->pm_reg_ptps_pa);
+
+ if (CPU_ISSUN4M && pm->pm_ctx != NULL &&
+ (ctx_phys_tbl[pm->pm_ctxnum] != ((VA2PA((caddr_t)pm->pm_reg_ptps)
+ >> SRMMU_PPNPASHIFT) |
+ SRMMU_TEPTD)))
+ panic("%s: CHK(pmap %p): SRMMU region table at %x not installed "
+ "for context %d", s, pm, pm->pm_reg_ptps_pa, pm->pm_ctxnum);
+#endif
+
for (vr = 0; vr < NUREG; vr++) {
rp = &pm->pm_regmap[vr];
if (rp->rg_nsegmap == 0)
@@ -3701,24 +6422,44 @@ pm_check_u(s, pm)
if (rp->rg_segmap == NULL)
panic("%s: CHK(vr %d): nsegmap = %d; sp==NULL",
s, vr, rp->rg_nsegmap);
+#if defined(SUN4M)
+ if (CPU_ISSUN4M && rp->rg_seg_ptps == NULL)
+ panic("%s: CHK(vr %d): nsegmap=%d; no SRMMU segment table",
+ s, vr, rp->rg_nsegmap);
+ if (CPU_ISSUN4M &&
+ pm->pm_reg_ptps[vr] != ((VA2PA((caddr_t)rp->rg_seg_ptps) >>
+ SRMMU_PPNPASHIFT) | SRMMU_TEPTD))
+ panic("%s: CHK(vr %d): SRMMU segtbl not installed",s,vr);
+#endif
if ((unsigned int)rp < KERNBASE)
- panic("%s: rp=%x", s, rp);
+ panic("%s: rp=%p", s, rp);
n = 0;
for (vs = 0; vs < NSEGRG; vs++) {
sp = &rp->rg_segmap[vs];
if ((unsigned int)sp < KERNBASE)
- panic("%s: sp=%x", s, sp);
+ panic("%s: sp=%p", s, sp);
if (sp->sg_npte != 0) {
n++;
if (sp->sg_pte == NULL)
panic("%s: CHK(vr %d, vs %d): npte=%d, "
"pte=NULL", s, vr, vs, sp->sg_npte);
-
+#if defined(SUN4M)
+ if (CPU_ISSUN4M &&
+ rp->rg_seg_ptps[vs] !=
+ ((VA2PA((caddr_t)sp->sg_pte)
+ >> SRMMU_PPNPASHIFT) |
+ SRMMU_TEPTD))
+ panic("%s: CHK(vr %d, vs %d): SRMMU page "
+ "table not installed correctly",s,vr,
+ vs);
+#endif
pte=sp->sg_pte;
m = 0;
for (j=0; j<NPTESG; j++,pte++)
- if (*pte & PG_V)
- m++;
+ if ((CPU_ISSUN4M
+ ?((*pte & SRMMU_TETYPE) == SRMMU_TEPTE)
+ :(*pte & PG_V)))
+ m++;
if (m != sp->sg_npte)
/*if (pmapdebug & 0x10000)*/
printf("%s: user CHK(vr %d, vs %d): "
@@ -3731,18 +6472,33 @@ pm_check_u(s, pm)
"# of pte's: %d, should be %d",
s, vr, rp->rg_nsegmap, n);
}
- return 0;
+ return;
}
-int
-pm_check_k(s, pm)
+void
+pm_check_k(s, pm) /* Note: not as extensive as pm_check_u. */
char *s;
struct pmap *pm;
{
struct regmap *rp;
- struct segmap *sp;
int vr, vs, n;
+ if (pm->pm_regmap == NULL)
+ panic("%s: CHK(pmap %p): no region mapping", s, pm);
+
+#if defined(SUN4M)
+ if (CPU_ISSUN4M &&
+ (pm->pm_reg_ptps == NULL ||
+ pm->pm_reg_ptps_pa != VA2PA((caddr_t)pm->pm_reg_ptps)))
+ panic("%s: CHK(pmap %p): no SRMMU region table or bad pa: tblva=%p, tblpa=%x",
+ s, pm, pm->pm_reg_ptps, pm->pm_reg_ptps_pa);
+
+ if (CPU_ISSUN4M &&
+ (ctx_phys_tbl[0] != ((VA2PA((caddr_t)pm->pm_reg_ptps) >>
+ SRMMU_PPNPASHIFT) | SRMMU_TEPTD)))
+ panic("%s: CHK(pmap %p): SRMMU region table at %x not installed "
+ "for context %d", s, pm, pm->pm_reg_ptps_pa, 0);
+#endif
for (vr = NUREG; vr < NUREG+NKREG; vr++) {
rp = &pm->pm_regmap[vr];
if (rp->rg_segmap == NULL)
@@ -3750,6 +6506,15 @@ pm_check_k(s, pm)
s, vr, rp->rg_nsegmap);
if (rp->rg_nsegmap == 0)
continue;
+#if defined(SUN4M)
+ if (CPU_ISSUN4M && rp->rg_seg_ptps == NULL)
+ panic("%s: CHK(vr %d): nsegmap=%d; no SRMMU segment table",
+ s, vr, rp->rg_nsegmap);
+ if (CPU_ISSUN4M &&
+ pm->pm_reg_ptps[vr] != ((VA2PA((caddr_t)rp->rg_seg_ptps) >>
+ SRMMU_PPNPASHIFT) | SRMMU_TEPTD))
+ panic("%s: CHK(vr %d): SRMMU segtbl not installed",s,vr);
+#endif
n = 0;
for (vs = 0; vs < NSEGRG; vs++) {
if (rp->rg_segmap[vs].sg_npte)
@@ -3760,7 +6525,7 @@ pm_check_k(s, pm)
"# of pte's: %d, should be %d\n",
s, vr, rp->rg_nsegmap, n);
}
- return 0;
+ return;
}
#endif
@@ -3772,30 +6537,40 @@ pm_check_k(s, pm)
int
pmap_dumpsize()
{
- return btoc(((seginval + 1) * NPTESG * sizeof(int)) +
- sizeof(seginval) +
- sizeof(pmemarr) +
- sizeof(kernel_segmap_store));
+ if (CPU_ISSUN4M) /* No need to dump on 4m; its in-core. */
+ return (0);
+
+ if (CPU_ISSUN4OR4C)
+ return btoc(((seginval + 1) * NPTESG * sizeof(int)) +
+ sizeof(seginval) +
+ sizeof(pmemarr) +
+ sizeof(kernel_segmap_store));
+ return 0;
}
/*
* Write the mmu contents to the dump device.
* This gets appended to the end of a crash dump since
- * there is no in-core copy of kernel memory mappings.
+ * there is no in-core copy of kernel memory mappings on a 4/4c machine.
*/
int
pmap_dumpmmu(dump, blkno)
register daddr_t blkno;
register int (*dump) __P((dev_t, daddr_t, caddr_t, size_t));
{
+#if defined(SUN4C) || defined(SUN4)
register int pmeg;
- register int addr; /* unused kernel virtual address */
register int i;
register int *pte, *ptend;
- register int error;
+ register int error = 0;
register int *kp;
int buffer[dbtob(1) / sizeof(int)];
+#endif
+
+ if (CPU_ISSUN4M) /* No need to dump on 4m; its in-core. */
+ return (0);
+#if defined(SUN4C) || defined(SUN4)
/*
* dump page table entries
*
@@ -3805,6 +6580,7 @@ pmap_dumpmmu(dump, blkno)
* in the MMU. This fixed segment number is used in the virtual
* address argument to getpte().
*/
+
setcontext(0);
/*
@@ -3818,12 +6594,12 @@ pmap_dumpmmu(dump, blkno)
setsegmap(va, pmeg);
i = NPTESG;
do {
- *pte++ = getpte(va);
+ *pte++ = getpte4(va);
if (pte >= ptend) {
/*
* Note that we'll dump the last block
* the last time through the loops because
- * all the PMEGs occupy 32KB which is
+ * all the PMEGs occupy 32KB which is
* a multiple of the block size.
*/
error = (*dump)(dumpdev, blkno,
@@ -3865,4 +6641,180 @@ pmap_dumpmmu(dump, blkno)
error = (*dump)(dumpdev, blkno++, (caddr_t)buffer, dbtob(1));
return (error);
+#endif
+}
+
+#ifdef EXTREME_DEBUG
+
+static void test_region __P((int, int, int));
+
+void
+debug_pagetables()
+{
+ register int i;
+ register int *regtbl;
+ register int te;
+
+ printf("\nncontext=%d. ",ncontext);
+ printf("Context table is at va 0x%x. Level 0 PTP: 0x%x\n",
+ ctx_phys_tbl, ctx_phys_tbl[0]);
+ printf("Context 0 region table is at va 0x%x, pa 0x%x. Contents:\n",
+ pmap_kernel()->pm_reg_ptps, pmap_kernel()->pm_reg_ptps_pa);
+
+ regtbl = pmap_kernel()->pm_reg_ptps;
+
+ printf("PROM vector is at 0x%x\n",promvec);
+ printf("PROM reboot routine is at 0x%x\n",promvec->pv_reboot);
+ printf("PROM abort routine is at 0x%x\n",promvec->pv_abort);
+ printf("PROM halt routine is at 0x%x\n",promvec->pv_halt);
+
+ printf("Testing region 0xfe: ");
+ test_region(0xfe,0,16*1024*1024);
+ printf("Testing region 0xff: ");
+ test_region(0xff,0,16*1024*1024);
+ printf("Testing kernel region 0xf8: ");
+ test_region(0xf8, 4096, avail_start);
+ cngetc();
+
+ for (i = 0; i < SRMMU_L1SIZE; i++) {
+ te = regtbl[i];
+ if ((te & SRMMU_TETYPE) == SRMMU_TEINVALID)
+ continue;
+ printf("Region 0x%x: PTE=0x%x <%s> L2PA=0x%x kernL2VA=0x%x\n",
+ i, te, ((te & SRMMU_TETYPE) == SRMMU_TEPTE ? "pte" :
+ ((te & SRMMU_TETYPE) == SRMMU_TEPTD ? "ptd" :
+ ((te & SRMMU_TETYPE) == SRMMU_TEINVALID ?
+ "invalid" : "reserved"))),
+ (te & ~0x3) << SRMMU_PPNPASHIFT,
+ pmap_kernel()->pm_regmap[i].rg_seg_ptps);
+ }
+ printf("Press q to halt...\n");
+ if (cngetc()=='q')
+ callrom();
+}
+
+static u_int
+VA2PAsw(ctx, addr, pte)
+ register int ctx;
+ register caddr_t addr;
+ int *pte;
+{
+ register int *curtbl;
+ register int curpte;
+
+#ifdef EXTREME_EXTREME_DEBUG
+ printf("Looking up addr 0x%x in context 0x%x\n",addr,ctx);
+#endif
+ /* L0 */
+ *pte = curpte = ctx_phys_tbl[ctx];
+#ifdef EXTREME_EXTREME_DEBUG
+ printf("Got L0 pte 0x%x\n",pte);
+#endif
+ if ((curpte & SRMMU_TETYPE) == SRMMU_TEPTE) {
+ return (((curpte & SRMMU_PPNMASK) << SRMMU_PPNPASHIFT) |
+ ((u_int)addr & 0xffffffff));
+ }
+ if ((curpte & SRMMU_TETYPE) != SRMMU_TEPTD) {
+ printf("Bad context table entry 0x%x for context 0x%x\n",
+ curpte, ctx);
+ return 0;
+ }
+ /* L1 */
+ curtbl = ((curpte & ~0x3) << 4) | (0xf8 << RGSHIFT); /* correct for krn*/
+ *pte = curpte = curtbl[VA_VREG(addr)];
+#ifdef EXTREME_EXTREME_DEBUG
+ printf("L1 table at 0x%x.\nGot L1 pte 0x%x\n",curtbl,curpte);
+#endif
+ if ((curpte & SRMMU_TETYPE) == SRMMU_TEPTE)
+ return (((curpte & SRMMU_PPNMASK) << SRMMU_PPNPASHIFT) |
+ ((u_int)addr & 0xffffff));
+ if ((curpte & SRMMU_TETYPE) != SRMMU_TEPTD) {
+ printf("Bad region table entry 0x%x for region 0x%x\n",
+ curpte, VA_VREG(addr));
+ return 0;
+ }
+ /* L2 */
+ curtbl = ((curpte & ~0x3) << 4) | (0xf8 << RGSHIFT); /* correct for krn*/
+ *pte = curpte = curtbl[VA_VSEG(addr)];
+#ifdef EXTREME_EXTREME_DEBUG
+ printf("L2 table at 0x%x.\nGot L2 pte 0x%x\n",curtbl,curpte);
+#endif
+ if ((curpte & SRMMU_TETYPE) == SRMMU_TEPTE)
+ return (((curpte & SRMMU_PPNMASK) << SRMMU_PPNPASHIFT) |
+ ((u_int)addr & 0x3ffff));
+ if ((curpte & SRMMU_TETYPE) != SRMMU_TEPTD) {
+ printf("Bad segment table entry 0x%x for reg 0x%x, seg 0x%x\n",
+ curpte, VA_VREG(addr), VA_VSEG(addr));
+ return 0;
+ }
+ /* L3 */
+ curtbl = ((curpte & ~0x3) << 4) | (0xf8 << RGSHIFT); /* correct for krn*/
+ *pte = curpte = curtbl[VA_VPG(addr)];
+#ifdef EXTREME_EXTREME_DEBUG
+ printf("L3 table at 0x%x.\nGot L3 pte 0x%x\n",curtbl,curpte);
+#endif
+ if ((curpte & SRMMU_TETYPE) == SRMMU_TEPTE)
+ return (((curpte & SRMMU_PPNMASK) << SRMMU_PPNPASHIFT) |
+ ((u_int)addr & 0xfff));
+ else {
+ printf("Bad L3 pte 0x%x for reg 0x%x, seg 0x%x, pg 0x%x\n",
+ curpte, VA_VREG(addr), VA_VSEG(addr), VA_VPG(addr));
+ return 0;
+ }
+ printf("Bizarreness with address 0x%x!\n",addr);
+}
+
+void test_region(reg, start, stop)
+ register int reg;
+ register int start, stop;
+{
+ register int i;
+ register int addr;
+ register int pte;
+ int ptesw;
+/* int cnt=0;
+*/
+
+ for (i = start; i < stop; i+= NBPG) {
+ addr = (reg << RGSHIFT) | i;
+ pte=lda(((u_int)(addr)) | ASI_SRMMUFP_LN, ASI_SRMMUFP);
+ if (pte) {
+/* printf("Valid address 0x%x\n",addr);
+ if (++cnt == 20) {
+ cngetc();
+ cnt=0;
+ }
+*/
+ if (VA2PA(addr) != VA2PAsw(0,addr,&ptesw)) {
+ printf("Mismatch at address 0x%x.\n",addr);
+ if (cngetc()=='q') break;
+ }
+ if (reg == 0xf8) /* kernel permissions are different */
+ continue;
+ if ((pte&SRMMU_PROT_MASK)!=(ptesw&SRMMU_PROT_MASK)) {
+ printf("Mismatched protections at address "
+ "0x%x; pte=0x%x, ptesw=0x%x\n",
+ addr,pte,ptesw);
+ if (cngetc()=='q') break;
+ }
+ }
+ }
+ printf("done.\n");
+}
+
+
+void print_fe_map(void)
+{
+ u_int i, pte;
+
+ printf("map of region 0xfe:\n");
+ for (i = 0xfe000000; i < 0xff000000; i+=4096) {
+ if (((pte = getpte4m(i)) & SRMMU_TETYPE) != SRMMU_TEPTE)
+ continue;
+ printf("0x%x -> 0x%x%x (pte %x)\n", i, pte >> 28,
+ (pte & ~0xff) << 4, pte);
+ }
+ printf("done\n");
}
+
+#endif
diff --git a/sys/arch/sparc/sparc/process_machdep.c b/sys/arch/sparc/sparc/process_machdep.c
index 0f20e1858e2..3ea7669ede6 100644
--- a/sys/arch/sparc/sparc/process_machdep.c
+++ b/sys/arch/sparc/sparc/process_machdep.c
@@ -1,4 +1,4 @@
-/* $NetBSD: process_machdep.c,v 1.5 1994/11/20 20:54:37 deraadt Exp $ */
+/* $NetBSD: process_machdep.c,v 1.6 1996/03/14 21:09:26 christos Exp $ */
/*
* Copyright (c) 1993 The Regents of the University of California.
@@ -97,6 +97,7 @@ process_write_regs(p, regs)
int
process_sstep(p, sstep)
struct proc *p;
+ int sstep;
{
if (sstep)
return EINVAL;
diff --git a/sys/arch/sparc/sparc/svr4_machdep.c b/sys/arch/sparc/sparc/svr4_machdep.c
index 5a572604c7b..a16e126ab1c 100644
--- a/sys/arch/sparc/sparc/svr4_machdep.c
+++ b/sys/arch/sparc/sparc/svr4_machdep.c
@@ -1,4 +1,4 @@
-/* $NetBSD: svr4_machdep.c,v 1.13 1996/01/07 19:47:27 christos Exp $ */
+/* $NetBSD: svr4_machdep.c,v 1.17.4.1 1996/06/11 01:46:42 jtc Exp $ */
/*
* Copyright (c) 1994 Christos Zoulas
@@ -74,7 +74,7 @@ svr4_printcontext(fun, uc)
svr4_greg_t *r = uc->uc_mcontext.greg;
struct svr4_sigaltstack *s = &uc->uc_stack;
- printf("%s at %x\n", fun, uc);
+ printf("%s at %p\n", fun, uc);
printf("Regs: ");
printf("PSR = %x ", r[SVR4_SPARC_PSR]);
@@ -98,12 +98,10 @@ svr4_printcontext(fun, uc)
printf("O7 = %x ", r[SVR4_SPARC_O7]);
printf("\n");
- printf("Signal Stack: sp %x, size %d, flags %x\n",
+ printf("Signal Stack: sp %p, size %d, flags %x\n",
s->ss_sp, s->ss_size, s->ss_flags);
- printf("Signal mask: %x\n", uc->uc_sigmask);
-
- printf("Flags: %x\n", uc->uc_flags);
+ printf("Flags: %lx\n", uc->uc_flags);
}
#endif
@@ -114,15 +112,21 @@ svr4_getcontext(p, uc, mask, oonstack)
int mask, oonstack;
{
struct trapframe *tf = (struct trapframe *)p->p_md.md_tf;
- struct sigacts *psp = p->p_sigacts;
svr4_greg_t *r = uc->uc_mcontext.greg;
struct svr4_sigaltstack *s = &uc->uc_stack;
- struct sigaltstack *sf = &psp->ps_sigstk;
+#ifdef FPU_CONTEXT
+ svr4_fregset_t *f = &uc->uc_mcontext.freg;
+ struct fpstate *fps = p->p_md.md_fpstate;
+#endif
+
+ write_user_windows();
+ if (rwindow_save(p))
+ sigexit(p, SIGILL);
bzero(uc, sizeof(struct svr4_ucontext));
/*
- * Set the general purpose registers
+ * Get the general purpose registers
*/
r[SVR4_SPARC_PSR] = tf->tf_psr;
r[SVR4_SPARC_PC] = tf->tf_pc;
@@ -144,20 +148,50 @@ svr4_getcontext(p, uc, mask, oonstack)
r[SVR4_SPARC_O6] = tf->tf_out[6];
r[SVR4_SPARC_O7] = tf->tf_out[7];
+#ifdef FPU_CONTEXT
/*
- * Set the signal stack
+ * Get the floating point registers
*/
- bsd_to_svr4_sigaltstack(sf, s);
+ bcopy(fps->fs_regs, f->fpu_regs, sizeof(fps->fs_regs));
+ f->fp_nqsize = sizeof(struct fp_qentry);
+ f->fp_nqel = fps->fs_qsize;
+ f->fp_fsr = fps->fs_fsr;
+ if (f->fp_q != NULL) {
+ size_t sz = f->fp_nqel * f->fp_nqsize;
+ if (sz > sizeof(fps->fs_queue)) {
+#ifdef DIAGNOSTIC
+ printf("getcontext: fp_queue too large\n");
+#endif
+ return;
+ }
+ if (copyout(fps->fs_queue, f->fp_q, sz) != 0) {
+#ifdef DIAGNOSTIC
+ printf("getcontext: copy of fp_queue failed %d\n",
+ error);
+#endif
+ return;
+ }
+ }
+ f->fp_busy = 0; /* XXX: How do we determine that? */
+#endif
/*
- * Set the signal mask
+ * Set the signal stack to something reasonable
+ */
+ /* XXX: Don't really know what to do with this */
+ s->ss_sp = (char *) ((r[SVR4_SPARC_SP] & ~0xfff) - 8192);
+ s->ss_size = 8192;
+ s->ss_flags = 0;
+
+ /*
+ * Get the signal mask
*/
bsd_to_svr4_sigset(&mask, &uc->uc_sigmask);
/*
- * Set the flags
+ * Get the flags
*/
- uc->uc_flags = SVR4_UC_ALL;
+ uc->uc_flags = SVR4_UC_CPU|SVR4_UC_SIGMASK|SVR4_UC_STACK;
#ifdef DEBUG_SVR4
svr4_printcontext("getcontext", uc);
@@ -181,21 +215,20 @@ svr4_setcontext(p, uc)
struct proc *p;
struct svr4_ucontext *uc;
{
- struct sigcontext *scp, context;
struct sigacts *psp = p->p_sigacts;
register struct trapframe *tf;
svr4_greg_t *r = uc->uc_mcontext.greg;
struct svr4_sigaltstack *s = &uc->uc_stack;
struct sigaltstack *sf = &psp->ps_sigstk;
int mask;
+#ifdef FPU_CONTEXT
+ svr4_fregset_t *f = &uc->uc_mcontext.freg;
+ struct fpstate *fps = p->p_md.md_fpstate;
+#endif
#ifdef DEBUG_SVR4
svr4_printcontext("setcontext", uc);
#endif
- /*
- * XXX:
- * Should we check the value of flags to determine what to restore?
- */
write_user_windows();
if (rwindow_save(p))
@@ -203,7 +236,7 @@ svr4_setcontext(p, uc)
#ifdef DEBUG
if (sigdebug & SDB_FOLLOW)
- printf("svr4_setcontext: %s[%d], svr4_ucontext %x\n",
+ printf("svr4_setcontext: %s[%d], svr4_ucontext %p\n",
p->p_comm, p->p_pid, uc);
#endif
@@ -212,50 +245,86 @@ svr4_setcontext(p, uc)
/*
* Restore register context.
*/
- /*
- * Only the icc bits in the psr are used, so it need not be
- * verified. pc and npc must be multiples of 4. This is all
- * that is required; if it holds, just do it.
- */
- if (((r[SVR4_SPARC_PC] | r[SVR4_SPARC_nPC]) & 3) != 0) {
- printf("pc or npc are not multiples of 4!\n");
- return EINVAL;
+ if (uc->uc_flags & SVR4_UC_CPU) {
+ /*
+ * Only the icc bits in the psr are used, so it need not be
+ * verified. pc and npc must be multiples of 4. This is all
+ * that is required; if it holds, just do it.
+ */
+ if (((r[SVR4_SPARC_PC] | r[SVR4_SPARC_nPC]) & 3) != 0) {
+ printf("pc or npc are not multiples of 4!\n");
+ return EINVAL;
+ }
+
+ /* take only psr ICC field */
+ tf->tf_psr = (tf->tf_psr & ~PSR_ICC) |
+ (r[SVR4_SPARC_PSR] & PSR_ICC);
+ tf->tf_pc = r[SVR4_SPARC_PC];
+ tf->tf_npc = r[SVR4_SPARC_nPC];
+ tf->tf_y = r[SVR4_SPARC_Y];
+
+ /* Restore everything */
+ tf->tf_global[1] = r[SVR4_SPARC_G1];
+ tf->tf_global[2] = r[SVR4_SPARC_G2];
+ tf->tf_global[3] = r[SVR4_SPARC_G3];
+ tf->tf_global[4] = r[SVR4_SPARC_G4];
+ tf->tf_global[5] = r[SVR4_SPARC_G5];
+ tf->tf_global[6] = r[SVR4_SPARC_G6];
+ tf->tf_global[7] = r[SVR4_SPARC_G7];
+
+ tf->tf_out[0] = r[SVR4_SPARC_O0];
+ tf->tf_out[1] = r[SVR4_SPARC_O1];
+ tf->tf_out[2] = r[SVR4_SPARC_O2];
+ tf->tf_out[3] = r[SVR4_SPARC_O3];
+ tf->tf_out[4] = r[SVR4_SPARC_O4];
+ tf->tf_out[5] = r[SVR4_SPARC_O5];
+ tf->tf_out[6] = r[SVR4_SPARC_O6];
+ tf->tf_out[7] = r[SVR4_SPARC_O7];
}
- /* take only psr ICC field */
- tf->tf_psr = (tf->tf_psr & ~PSR_ICC) | (r[SVR4_SPARC_PSR] & PSR_ICC);
- tf->tf_pc = r[SVR4_SPARC_PC];
- tf->tf_npc = r[SVR4_SPARC_nPC];
- tf->tf_y = r[SVR4_SPARC_Y];
-
- /* Restore everything */
- tf->tf_global[1] = r[SVR4_SPARC_G1];
- tf->tf_global[2] = r[SVR4_SPARC_G2];
- tf->tf_global[3] = r[SVR4_SPARC_G3];
- tf->tf_global[4] = r[SVR4_SPARC_G4];
- tf->tf_global[5] = r[SVR4_SPARC_G5];
- tf->tf_global[6] = r[SVR4_SPARC_G6];
- tf->tf_global[7] = r[SVR4_SPARC_G7];
-
- tf->tf_out[0] = r[SVR4_SPARC_O0];
- tf->tf_out[1] = r[SVR4_SPARC_O1];
- tf->tf_out[2] = r[SVR4_SPARC_O2];
- tf->tf_out[3] = r[SVR4_SPARC_O3];
- tf->tf_out[4] = r[SVR4_SPARC_O4];
- tf->tf_out[5] = r[SVR4_SPARC_O5];
- tf->tf_out[6] = r[SVR4_SPARC_O6];
- tf->tf_out[7] = r[SVR4_SPARC_O7];
- /*
- * restore signal stack
- */
- svr4_to_bsd_sigaltstack(s, sf);
+#ifdef FPU_CONTEXT
+ if (uc->uc_flags & SVR4_UC_FPU) {
+ /*
+ * Set the floating point registers
+ */
+ int error;
+ size_t sz = f->fp_nqel * f->fp_nqsize;
+ if (sz > sizeof(fps->fs_queue)) {
+#ifdef DIAGNOSTIC
+ printf("setcontext: fp_queue too large\n");
+#endif
+ return EINVAL;
+ }
+ bcopy(f->fpu_regs, fps->fs_regs, sizeof(fps->fs_regs));
+ fps->fs_qsize = f->fp_nqel;
+ fps->fs_fsr = f->fp_fsr;
+ if (f->fp_q != NULL) {
+ if ((error = copyin(f->fp_q, fps->fs_queue,
+ f->fp_nqel * f->fp_nqsize)) != 0) {
+#ifdef DIAGNOSTIC
+ printf("setcontext: copy of fp_queue failed\n");
+#endif
+ return error;
+ }
+ }
+ }
+#endif
- /*
- * restore signal mask
- */
- svr4_to_bsd_sigset(&uc->uc_sigmask, &mask);
- p->p_sigmask = mask & ~sigcantmask;
+ if (uc->uc_flags & SVR4_UC_STACK) {
+ /*
+ * restore signal stack
+ */
+ svr4_to_bsd_sigaltstack(s, sf);
+ }
+
+ if (uc->uc_flags & SVR4_UC_SIGMASK) {
+ /*
+ * restore signal mask
+ */
+ svr4_to_bsd_sigset(&uc->uc_sigmask, &mask);
+ p->p_sigmask = mask & ~sigcantmask;
+ }
return EJUSTRETURN;
}
@@ -280,6 +349,10 @@ svr4_getsiginfo(si, sig, code, addr)
si->si_trap = code;
switch (code) {
+ case T_RESET:
+ si->si_code = 0;
+ break;
+
case T_TEXTFAULT:
si->si_code = SVR4_BUS_ADRALN;
break;
@@ -372,7 +445,7 @@ svr4_getsiginfo(si, sig, code, addr)
default:
si->si_code = 0;
#ifdef DIAGNOSTIC
- printf("sig %d code %d\n", sig, code);
+ printf("sig %d code %ld\n", sig, code);
panic("svr4_getsiginfo");
#endif
break;
@@ -399,7 +472,6 @@ svr4_sendsig(catcher, sig, mask, code)
struct svr4_sigframe *fp, frame;
struct sigacts *psp = p->p_sigacts;
int oonstack, oldsp, newsp, addr;
- svr4_greg_t* r = frame.sf_uc.uc_mcontext.greg;
extern char svr4_sigcode[], svr4_esigcode[];
@@ -424,7 +496,7 @@ svr4_sendsig(catcher, sig, mask, code)
*/
fp = (struct svr4_sigframe *) ((int) (fp - 1) & ~7);
- /*
+ /*
* Build the argument list for the signal handler.
*/
svr4_getsiginfo(&frame.sf_si, sig, code, (caddr_t) tf->tf_pc);
@@ -502,15 +574,30 @@ svr4_trap(type, p)
break;
case T_SVR4_GETHRTIME:
- uprintf("T_SVR4_GETHRTIME\n");
- break;
-
+ /*
+ * this list like gethrtime(3). To implement this
+ * correctly we need a timer that does not get affected
+ * adjtime(), or settimeofday(). For now we use
+ * microtime, and convert to nanoseconds...
+ */
+ /*FALLTHROUGH*/
case T_SVR4_GETHRVTIME:
- uprintf("T_SVR4_GETHRVTIME\n");
+ /*
+ * This is like gethrvtime(3). Since we don't have lwp
+ * we massage microtime() output
+ */
+ {
+ struct timeval tv;
+
+ microtime(&tv);
+ tf->tf_out[0] = tv.tv_sec;
+ tf->tf_out[1] = tv.tv_usec * 1000;
+ }
break;
case T_SVR4_GETHRESTIME:
{
+ /* I assume this is like gettimeofday(3) */
struct timeval tv;
microtime(&tv);
diff --git a/sys/arch/sparc/sparc/swapgeneric.c b/sys/arch/sparc/sparc/swapgeneric.c
index c2518a0bdf4..493b04bc198 100644
--- a/sys/arch/sparc/sparc/swapgeneric.c
+++ b/sys/arch/sparc/sparc/swapgeneric.c
@@ -1,4 +1,4 @@
-/* $NetBSD: swapgeneric.c,v 1.7 1995/02/16 21:42:52 pk Exp $ */
+/* $NetBSD: swapgeneric.c,v 1.8 1996/03/14 21:09:32 christos Exp $ */
/*-
* Copyright (c) 1994
@@ -42,7 +42,7 @@
#include <sys/param.h>
#include <sys/conf.h>
-int (*mountroot)() = NULL; /* tells autoconf.c that we are "generic" */
+int (*mountroot) __P((void *)) = NULL; /* tells autoconf.c that we are "generic" */
dev_t rootdev = NODEV;
dev_t dumpdev = NODEV;
diff --git a/sys/arch/sparc/sparc/sys_machdep.c b/sys/arch/sparc/sparc/sys_machdep.c
index 28f00072d14..e83270eaf3b 100644
--- a/sys/arch/sparc/sparc/sys_machdep.c
+++ b/sys/arch/sparc/sparc/sys_machdep.c
@@ -1,4 +1,4 @@
-/* $NetBSD: sys_machdep.c,v 1.6 1995/10/07 06:26:10 mycroft Exp $ */
+/* $NetBSD: sys_machdep.c,v 1.7 1996/03/14 21:09:33 christos Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -62,6 +62,7 @@
#ifdef TRACE
int nvualarm;
+int
vtrace(p, uap, retval)
struct proc *p;
register struct vtrace_args /* {
@@ -103,6 +104,7 @@ vtrace(p, uap, retval)
return (0);
}
+void
vdoualarm(arg)
int arg;
{
@@ -114,6 +116,7 @@ vdoualarm(arg)
}
#endif
+int
sys_sysarch(p, v, retval)
struct proc *p;
void *v;
diff --git a/sys/arch/sparc/sparc/timerreg.h b/sys/arch/sparc/sparc/timerreg.h
index ae90cd8c237..26bc2a57ca3 100644
--- a/sys/arch/sparc/sparc/timerreg.h
+++ b/sys/arch/sparc/sparc/timerreg.h
@@ -1,4 +1,4 @@
-/* $NetBSD: timerreg.h,v 1.2 1994/11/20 20:54:41 deraadt Exp $ */
+/* $NetBSD: timerreg.h,v 1.5 1996/05/02 18:17:33 pk Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -66,18 +66,53 @@
* 1, 2, 3, 1, 2, 3, ...
*
* and if we want to divide by N we must set the limit register to N+1.
+ *
+ * Sun-4m counters/timer registers are similar, with these exceptions:
+ *
+ * - the limit and counter registers have changed positions..
+ * - both limit and counter registers are 22 bits wide, but
+ * they count in 500ns increments (bit 9 being the least
+ * significant bit).
+ *
+ * Note that we still use the `sun4c' masks and shifts to compute
+ * the bit pattern, given the tick period in microseconds, resulting
+ * in a limit value that is 1 too high. This means that (with HZ=100)
+ * the clock will err on the slow side by 500ns/10ms (or 0.00005 %).
+ * We dont bother.
+ *
+ */
+#ifndef _LOCORE
+struct timer_4 {
+ volatile int t_counter; /* counter reg */
+ volatile int t_limit; /* limit reg */
+};
+
+struct timerreg_4 {
+ struct timer_4 t_c10; /* counter that interrupts at ipl 10 */
+ struct timer_4 t_c14; /* counter that interrupts at ipl 14 */
+};
+
+struct timer_4m { /* counter that interrupts at ipl 10 */
+ volatile int t_limit; /* limit register */
+ volatile int t_counter; /* counter register */
+ volatile int t_limit_nr; /* limit reg, non-resetting */
+ volatile int t_reserved;
+ volatile int t_cfg; /* a configuration register */
+/*
+ * Note: The SparcClassic manual only defines this one bit
+ * I suspect there are more in multi-processor machines.
*/
-#ifndef LOCORE
-struct timer {
- int t_counter; /* counter reg */
- int t_limit; /* limit reg */
+#define TMR_CFG_USER 1
};
-struct timerreg {
- struct timer t_c10; /* counter that interrupts at ipl 10 */
- struct timer t_c14; /* counter that interrupts at ipl 14 */
+struct counter_4m { /* counter that interrupts at ipl 14 */
+ volatile int t_limit; /* limit register */
+ volatile int t_counter; /* counter register */
+ volatile int t_limit_nr; /* limit reg, non-resetting */
+ volatile int t_ss; /* Start/Stop register */
+#define TMR_USER_RUN 1
};
-#endif
+#endif /* _LOCORE */
#define TMR_LIMIT 0x80000000 /* counter reached its limit */
#define TMR_SHIFT 10 /* shift to obtain microseconds */
@@ -86,5 +121,5 @@ struct timerreg {
/* Compute a limit that causes the timer to fire every n microseconds. */
#define tmr_ustolim(n) (((n) + 1) << TMR_SHIFT)
-#include <sparc/sparc/vaddrs.h>
-#define TIMERREG ((volatile struct timerreg *)TIMERREG_VA)
+/*efine TMR_SHIFT4M 9 -* shift to obtain microseconds */
+/*efine tmr_ustolim(n) (((2*(n)) + 1) << TMR_SHIFT4M)*/
diff --git a/sys/arch/sparc/sparc/trap.c b/sys/arch/sparc/sparc/trap.c
index 2925b0adea2..e44f163ccb8 100644
--- a/sys/arch/sparc/sparc/trap.c
+++ b/sys/arch/sparc/sparc/trap.c
@@ -1,6 +1,8 @@
-/* $NetBSD: trap.c,v 1.33 1995/07/04 22:57:35 christos Exp $ */
+/* $NetBSD: trap.c,v 1.42.4.2 1996/07/03 00:06:40 jtc Exp $ */
/*
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
*
@@ -12,6 +14,7 @@
* must display the following acknowledgement:
* This product includes software developed by the University of
* California, Lawrence Berkeley Laboratory.
+ * This product includes software developed by Harvard University.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
@@ -25,6 +28,7 @@
* must display the following acknowledgement:
* This product includes software developed by the University of
* California, Berkeley and its contributors.
+ * This product includes software developed by Harvard University.
* 4. Neither the name of the University nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
@@ -59,18 +63,35 @@
#include <sys/ktrace.h>
#endif
+#include <vm/vm.h>
#include <vm/vm_kern.h>
+#include <sparc/sparc/asm.h>
#include <machine/cpu.h>
#include <machine/ctlreg.h>
-#include <machine/frame.h>
#include <machine/trap.h>
+#include <machine/instr.h>
+#include <machine/pmap.h>
+
+#ifdef DDB
+#include <machine/db_machdep.h>
+#else
+#include <machine/frame.h>
+#endif
+#ifdef COMPAT_SVR4
+#include <machine/svr4_machdep.h>
+#endif
+
+#include <sparc/fpu/fpu_extern.h>
+#include <sparc/sparc/memreg.h>
#define offsetof(s, f) ((int)&((s *)0)->f)
extern int cold;
+#ifdef DEBUG
int rwindow_debug = 0;
+#endif
/*
* Initial FPU state is all registers == all 1s, everything else == all 0s.
@@ -162,12 +183,22 @@ const char *trap_type[] = {
#define N_TRAP_TYPES (sizeof trap_type / sizeof *trap_type)
+static __inline void userret __P((struct proc *, int, u_quad_t));
+void trap __P((unsigned, int, int, struct trapframe *));
+static __inline void share_fpu __P((struct proc *, struct trapframe *));
+void mem_access_fault __P((unsigned, int, u_int, int, int, struct trapframe *));
+void mem_access_fault4m __P((unsigned, u_int, u_int, u_int, u_int, struct trapframe *));
+void syscall __P((register_t, struct trapframe *, register_t));
+
/*
* Define the code needed before returning to user mode, for
* trap, mem_access_fault, and syscall.
*/
-static inline void
-userret(struct proc *p, int pc, u_quad_t oticks)
+static __inline void
+userret(p, pc, oticks)
+ struct proc *p;
+ int pc;
+ u_quad_t oticks;
{
int sig;
@@ -215,7 +246,10 @@ userret(struct proc *p, int pc, u_quad_t oticks)
* the ktrsysret() in syscall(). Actually, it is likely that the
* ktrsysret should occur before the call to userret.
*/
-static inline void share_fpu(struct proc *p, struct trapframe *tf) {
+static __inline void share_fpu(p, tf)
+ struct proc *p;
+ struct trapframe *tf;
+{
if ((tf->tf_psr & PSR_EF) != 0 && fpproc != p)
tf->tf_psr &= ~PSR_EF;
}
@@ -224,6 +258,7 @@ static inline void share_fpu(struct proc *p, struct trapframe *tf) {
* Called from locore.s trap handling, for non-MMU-related traps.
* (MMU-related traps go through mem_access_fault, below.)
*/
+void
trap(type, psr, pc, tf)
register unsigned type;
register int psr, pc;
@@ -252,6 +287,17 @@ trap(type, psr, pc, tf)
}
}
#endif
+#ifdef DIAGNOSTIC
+ /*
+ * Currently, we allow DIAGNOSTIC kernel code to
+ * flush the windows to record stack traces.
+ */
+ if (type == T_FLUSHWIN) {
+ write_all_windows();
+ ADVANCE;
+ return;
+ }
+#endif
/*
* Storing %fsr in cpu_attach will cause this trap
* even though the fpu has been enabled, if and only
@@ -272,14 +318,21 @@ trap(type, psr, pc, tf)
switch (type) {
default:
+#if defined(SUN4M)
+ if (type == 0x29)
+ /* Mysterious trap 29.. for now print&signal process */
+ goto badtrap;
+#endif
if (type < 0x80) {
dopanic:
printf("trap type 0x%x: pc=%x npc=%x psr=%b\n",
- type, pc, tf->tf_npc, psr, PSR_BITS);
+ type, pc, tf->tf_npc, psr, PSR_BITS);
panic(type < N_TRAP_TYPES ? trap_type[type] : T);
/* NOTREACHED */
}
+#if defined(COMPAT_SVR4) || defined(SUN4M)
badtrap:
+#endif
/* the following message is gratuitous */
/* ... but leave it in until we find anything */
printf("%s[%d]: unimplemented software trap 0x%x\n",
@@ -367,19 +420,17 @@ badtrap:
* nsaved to -1. If we decide to deliver a signal on
* our way out, we will clear nsaved.
*/
-#ifdef DIAGNOSTIC
if (pcb->pcb_uw || pcb->pcb_nsaved)
panic("trap T_RWRET 1");
+#ifdef DEBUG
if (rwindow_debug)
printf("%s[%d]: rwindow: pcb<-stack: %x\n",
- p->p_comm, p->p_pid, tf->tf_out[6]);
-#endif /* DIAGNOSTIC */
+ p->p_comm, p->p_pid, tf->tf_out[6]);
+#endif
if (read_rw(tf->tf_out[6], &pcb->pcb_rw[0]))
sigexit(p, SIGILL);
-#ifdef DIAGNOSTIC
if (pcb->pcb_nsaved)
panic("trap T_RWRET 2");
-#endif /* DIAGNOSTIC */
pcb->pcb_nsaved = -1; /* mark success */
break;
@@ -393,25 +444,23 @@ badtrap:
* in the pcb. The restore's window may still be in
* the cpu; we need to force it out to the stack.
*/
-#ifdef DIAGNOSTIC
+#ifdef DEBUG
if (rwindow_debug)
printf("%s[%d]: rwindow: T_WINUF 0: pcb<-stack: %x\n",
- p->p_comm, p->p_pid, tf->tf_out[6]);
-#endif /* DIAGNOSTIC */
+ p->p_comm, p->p_pid, tf->tf_out[6]);
+#endif
write_user_windows();
if (rwindow_save(p) || read_rw(tf->tf_out[6], &pcb->pcb_rw[0]))
sigexit(p, SIGILL);
-#ifdef DIAGNOSTIC
+#ifdef DEBUG
if (rwindow_debug)
printf("%s[%d]: rwindow: T_WINUF 1: pcb<-stack: %x\n",
- p->p_comm, p->p_pid, pcb->pcb_rw[0].rw_in[6]);
-#endif /* DIAGNOSTIC */
+ p->p_comm, p->p_pid, pcb->pcb_rw[0].rw_in[6]);
+#endif
if (read_rw(pcb->pcb_rw[0].rw_in[6], &pcb->pcb_rw[1]))
sigexit(p, SIGILL);
-#ifdef DIAGNOSTIC
if (pcb->pcb_nsaved)
panic("trap T_WINUF");
-#endif /* DIAGNOSTIC */
pcb->pcb_nsaved = -1; /* mark success */
break;
@@ -518,24 +567,24 @@ rwindow_save(p)
}
if (i == 0)
return (0);
-#ifdef DIAGNOSTIC
- if(rwindow_debug)
+#ifdef DEBUG
+ if (rwindow_debug)
printf("%s[%d]: rwindow: pcb->stack:", p->p_comm, p->p_pid);
-#endif /* DIAGNOSTIC */
+#endif
do {
-#ifdef DIAGNOSTIC
- if(rwindow_debug)
+#ifdef DEBUG
+ if (rwindow_debug)
printf(" %x", rw[1].rw_in[6]);
-#endif /* DIAGNOSTIC */
+#endif
if (copyout((caddr_t)rw, (caddr_t)rw[1].rw_in[6],
sizeof *rw))
return (-1);
rw++;
} while (--i > 0);
-#ifdef DIAGNOSTIC
- if(rwindow_debug)
+#ifdef DEBUG
+ if (rwindow_debug)
printf("\n");
-#endif /* DIAGNOSTIC */
+#endif
pcb->pcb_nsaved = 0;
return (0);
}
@@ -545,6 +594,7 @@ rwindow_save(p)
* and then erasing any pcb tracks. Otherwise we might try to write
* the registers into the new process after the exec.
*/
+void
kill_user_windows(p)
struct proc *p;
{
@@ -564,6 +614,7 @@ kill_user_windows(p)
* more than one `cause'. But we do not care what the cause, here;
* we just want to page in the page and try again.
*/
+void
mem_access_fault(type, ser, v, pc, psr, tf)
register unsigned type;
register int ser;
@@ -571,12 +622,13 @@ mem_access_fault(type, ser, v, pc, psr, tf)
register int pc, psr;
register struct trapframe *tf;
{
+#if defined(SUN4) || defined(SUN4C)
register struct proc *p;
register struct vmspace *vm;
register vm_offset_t va;
register int rv;
vm_prot_t ftype;
- int onfault, mmucode;
+ int onfault;
u_quad_t sticks;
cnt.v_trap++;
@@ -598,28 +650,13 @@ mem_access_fault(type, ser, v, pc, psr, tf)
if (VA_INHOLE(v))
goto fault;
ftype = ser & SER_WRITE ? VM_PROT_READ|VM_PROT_WRITE : VM_PROT_READ;
- if(ftype == VM_PROT_READ && (ser & SER_PROT) && type != T_TEXTFAULT) {
- /* If this is an ldstub or swap instruction
- * then we are about to fault in a loop forever
- * as only the read part of the fault will be
- * reported by the mmu.
- */
- int error, insn;
-
- error = copyin((caddr_t) pc, &insn, sizeof(int));
- if(!error) {
- insn = (insn >> 16); /* high word */
- insn &= 0xc168;
- if(insn == 0xc068)
- ftype |= VM_PROT_WRITE;
- }
- }
va = trunc_page(v);
if (psr & PSR_PS) {
extern char Lfsbail[];
if (type == T_TEXTFAULT) {
(void) splhigh();
- printf("text fault: pc=%x ser=%b\n", pc, ser, SER_BITS);
+ printf("text fault: pc=%x ser=%b\n", pc,
+ ser, SER_BITS);
panic("kernel fault");
/* NOTREACHED */
}
@@ -651,7 +688,8 @@ mem_access_fault(type, ser, v, pc, psr, tf)
* that got bumped out via LRU replacement.
*/
vm = p->p_vmspace;
- rv = mmu_pagein(&vm->vm_pmap, va, ftype);
+ rv = mmu_pagein(&vm->vm_pmap, va,
+ ser & SER_WRITE ? VM_PROT_WRITE : VM_PROT_READ);
if (rv < 0)
goto fault;
if (rv > 0)
@@ -697,7 +735,7 @@ kfault:
if (!onfault) {
(void) splhigh();
printf("data fault: pc=%x addr=%x ser=%b\n",
- pc, v, ser, SER_BITS);
+ pc, v, ser, SER_BITS);
panic("kernel fault");
/* NOTREACHED */
}
@@ -712,7 +750,280 @@ out:
userret(p, pc, sticks);
share_fpu(p, tf);
}
+#endif /* Sun4/Sun4C */
+}
+
+#if defined(SUN4M) /* 4m version of mem_access_fault() follows */
+
+static int tfaultaddr = (int) 0xdeadbeef;
+
+#ifdef DEBUG
+int dfdebug = 0;
+#endif
+
+void
+mem_access_fault4m(type, sfsr, sfva, afsr, afva, tf)
+ register unsigned type;
+ register u_int sfsr;
+ register u_int sfva;
+ register u_int afsr;
+ register u_int afva;
+ register struct trapframe *tf;
+{
+ register int pc, psr;
+ register struct proc *p;
+ register struct vmspace *vm;
+ register vm_offset_t va;
+ register int rv;
+ vm_prot_t ftype;
+ int onfault;
+ u_quad_t sticks;
+#if DEBUG
+static int lastdouble;
+#endif
+
+ cnt.v_trap++;
+ if ((p = curproc) == NULL) /* safety check */
+ p = &proc0;
+ sticks = p->p_sticks;
+
+ pc = tf->tf_pc; /* These are needed below */
+ psr = tf->tf_psr;
+
+ /*
+ * Our first priority is handling serious faults, such as
+ * parity errors or async faults that might have come through here.
+ * If afsr & AFSR_AFO != 0, then we're on a HyperSPARC and we
+ * got an async fault. We pass it on to memerr4m. Similarly, if
+ * the trap was T_STOREBUFFAULT, we pass it on to memerr4m.
+ * If we have a data fault, but SFSR_FAV is not set in the sfsr,
+ * then things are really bizarre, and we treat it as a hard
+ * error and pass it on to memerr4m. See pg. 9-35 in the SuperSPARC
+ * user's guide for more info, and for a possible solution which we
+ * don't implement here.
+ */
+ if ((afsr & AFSR_AFO) != 0 || type == T_STOREBUFFAULT ||
+ (type == T_DATAFAULT && !(sfsr & SFSR_FAV))) {
+ memerr4m(type, sfsr, sfva, afsr, afva, tf);
+ /*
+ * If we get here, exit the trap handler and wait for the
+ * trap to reoccur
+ */
+ goto out;
+ }
+
+ /*
+ * Figure out what to pass the VM code. We cannot ignore the sfva
+ * register on text faults, since this might be a trap on an
+ * alternate-ASI access to code space. However, if we're on a
+ * supersparc, we can't help using PC, since we don't get a VA in
+ * sfva.
+ * Kernel faults are somewhat different: text faults are always
+ * illegal, and data faults are extra complex. User faults must
+ * set p->p_md.md_tf, in case we decide to deliver a signal. Check
+ * for illegal virtual addresses early since those can induce more
+ * faults.
+ * All translation faults are illegal, and result in a SIGSEGV
+ * being delivered to the running process (or a kernel panic, for
+ * a kernel fault). We check the translation first to make sure
+ * it is not spurious.
+ * Also, note that in the case where we have an overwritten
+ * text fault (OW==1, AT==2,3), we attempt to service the
+ * second (overwriting) fault, then restart the instruction
+ * (which is from the first fault) and allow the first trap
+ * to reappear. XXX is this right? It will probably change...
+ */
+ if ((sfsr & SFSR_FT) == SFSR_FT_NONE)
+ goto out; /* No fault. Why were we called? */
+
+ /*
+ * This next section is a mess since some chips use sfva, and others
+ * don't on text faults. We want to use sfva where possible, since
+ * we _could_ be dealing with an ASI 0x8,0x9 data access to text space,
+ * which would trap as a text fault, at least on a HyperSPARC. Ugh.
+ * XXX: Find out about MicroSPARCs.
+ */
+
+ if (type == T_TEXTFAULT && mmumod == SUN4M_MMU_SS &&
+ (cpumod & 0xf0) == (SUN4M_SS) && (sfsr & SFSR_FAV)) {
+ sfva = pc; /* can't trust fav on supersparc/text fault */
+ } else if (type == T_TEXTFAULT && mmumod != SUN4M_MMU_HS) {
+ sfva = pc;
+ } else if (!(sfsr & SFSR_FAV)) {
+#ifdef DEBUG
+ if (type != T_TEXTFAULT)
+ printf("mem_access_fault: got fault without valid SFVA\n");
+ if (mmumod == SUN4M_MMU_HS)
+ printf("mem_access_fault: got fault without valid SFVA on "
+ "HyperSPARC!\n");
+#endif
+ if (type == T_TEXTFAULT)
+ sfva = pc;
+ else
+ goto fault;
+ }
+
+ if ((sfsr & SFSR_FT) == SFSR_FT_TRANSERR) {
+ /* Translation errors are always fatal, as they indicate
+ * a corrupt translation (page) table heirarchy.
+ */
+ if (tfaultaddr == sfva) /* Prevent infinite loops w/a static */
+ goto fault;
+ tfaultaddr = sfva;
+ if ((lda((sfva & 0xFFFFF000) | ASI_SRMMUFP_LN, ASI_SRMMUFP) &
+ SRMMU_TETYPE) != SRMMU_TEPTE)
+ goto fault; /* Translation bad */
+ else goto out; /* Translation OK, retry operation */
+ }
+
+ va = trunc_page(sfva);
+
+#ifdef DEBUG
+ if (lastdouble) {
+ printf("stacked tfault @ %x (pc %x); sfsr %x", sfva, pc, sfsr);
+ lastdouble = 0;
+ if (curproc == NULL)
+ printf("NULL proc\n");
+ else
+ printf("pid %d(%s); sigmask %x, sigcatch %x\n",
+ curproc->p_pid, curproc->p_comm,
+ curproc->p_sigmask, curproc->p_sigcatch);
+ }
+#endif
+ if (((sfsr & SFSR_AT_TEXT) || type == T_TEXTFAULT) &&
+ !(sfsr & SFSR_AT_STORE) && (sfsr & SFSR_OW)) {
+ if (psr & PSR_PS) /* never allow in kernel */
+ goto kfault;
+ /*
+ * Double text fault. The evil "case 5" from the HS manual...
+ * Attempt to handle early fault. Ignores ASI 8,9 issue...may
+ * do a useless VM read.
+ * XXX: Is this really necessary?
+ */
+#ifdef DEBUG
+ if (dfdebug) {
+ lastdouble = 1;
+ printf("mem_access_fault: double text fault @ %x (pc %x); sfsr %x",
+ sfva, pc, sfsr);
+ if (curproc == NULL)
+ printf("NULL proc\n");
+ else
+ printf(" pid %d(%s); sigmask %x, sigcatch %x\n",
+ curproc->p_pid, curproc->p_comm,
+ curproc->p_sigmask, curproc->p_sigcatch);
+ }
+#endif
+ if (mmumod == SUN4M_MMU_HS) { /* On HS, we have va for both */
+ if (vm_fault(kernel_map, trunc_page(pc),
+ VM_PROT_READ, 0) != KERN_SUCCESS)
+#ifdef DEBUG
+ printf("mem_access_fault: "
+ "can't pagein 1st text fault.\n")
+#endif
+ ;
+ }
+ }
+
+ /* Now munch on protections... */
+
+ ftype = sfsr & SFSR_AT_STORE ? VM_PROT_READ|VM_PROT_WRITE:VM_PROT_READ;
+ if (psr & PSR_PS) {
+ extern char Lfsbail[];
+ if (sfsr & SFSR_AT_TEXT || type == T_TEXTFAULT) {
+ (void) splhigh();
+ printf("text fault: pc=%x sfsr=%b sfva=%x\n", pc,
+ sfsr, SFSR_BITS, sfva);
+ panic("kernel fault");
+ /* NOTREACHED */
+ }
+ /*
+ * If this was an access that we shouldn't try to page in,
+ * resume at the fault handler without any action.
+ */
+ if (p->p_addr && p->p_addr->u_pcb.pcb_onfault == Lfsbail)
+ goto kfault;
+
+ /*
+ * During autoconfiguration, faults are never OK unless
+ * pcb_onfault is set. Once running normally we must allow
+ * exec() to cause copy-on-write faults to kernel addresses.
+ */
+ if (cold)
+ goto kfault;
+ if (va >= KERNBASE) {
+ if (vm_fault(kernel_map, va, ftype, 0) == KERN_SUCCESS)
+ return;
+ goto kfault;
+ }
+ } else
+ p->p_md.md_tf = tf;
+
+ vm = p->p_vmspace;
+#ifdef DEBUG
+ /*
+ * mmu_pagein returns -1 if the page is already valid, in which
+ * case we have a hard fault.. now why would *that* happen?
+ * But it happens sporadically, and vm_fault() seems to clear it..
+ */
+ rv = mmu_pagein4m(&vm->vm_pmap, va,
+ sfsr & SFSR_AT_STORE ? VM_PROT_WRITE : VM_PROT_READ);
+ if (rv < 0)
+ printf(" sfsr=%x(FT=%x,AT=%x,LVL=%x), sfva=%x, pc=%x, psr=%x\n",
+ sfsr, (sfsr >> 2) & 7, (sfsr >> 5) & 7, (sfsr >> 8) & 3,
+ sfva, pc, psr);
+ if (rv > 0)
+ panic("mmu_pagein4m returns %d", rv);
+#endif
+
+ /* alas! must call the horrible vm code */
+ rv = vm_fault(&vm->vm_map, (vm_offset_t)va, ftype, FALSE);
+
+ /*
+ * If this was a stack access we keep track of the maximum
+ * accessed stack size. Also, if vm_fault gets a protection
+ * failure it is due to accessing the stack region outside
+ * the current limit and we need to reflect that as an access
+ * error.
+ */
+ if ((caddr_t)va >= vm->vm_maxsaddr) {
+ if (rv == KERN_SUCCESS) {
+ unsigned nss = clrnd(btoc(USRSTACK - va));
+ if (nss > vm->vm_ssize)
+ vm->vm_ssize = nss;
+ } else if (rv == KERN_PROTECTION_FAILURE)
+ rv = KERN_INVALID_ADDRESS;
+ }
+ if (rv != KERN_SUCCESS) {
+ /*
+ * Pagein failed. If doing copyin/out, return to onfault
+ * address. Any other page fault in kernel, die; if user
+ * fault, deliver SIGSEGV.
+ */
+fault:
+ if (psr & PSR_PS) {
+kfault:
+ onfault = p->p_addr ?
+ (int)p->p_addr->u_pcb.pcb_onfault : 0;
+ if (!onfault) {
+ (void) splhigh();
+ printf("data fault: pc=%x addr=%x sfsr=%b\n",
+ pc, sfva, sfsr, SFSR_BITS);
+ panic("kernel fault");
+ /* NOTREACHED */
+ }
+ tf->tf_pc = onfault;
+ tf->tf_npc = onfault + 4;
+ return;
+ }
+ trapsignal(p, SIGSEGV, (u_int)sfva);
+ }
+out:
+ if ((psr & PSR_PS) == 0) {
+ userret(p, pc, sticks);
+ share_fpu(p, tf);
+ }
}
+#endif
/*
* System calls. `pc' is just a copy of tf->tf_pc.
@@ -722,9 +1033,11 @@ out:
* `save' effect of each trap). They are, however, the %o registers of the
* thing that made the system call, and are named that way here.
*/
+void
syscall(code, tf, pc)
- register_t code, pc;
+ register_t code;
register struct trapframe *tf;
+ register_t pc;
{
register int i, nsys, *ap, nap;
register struct sysent *callp;
@@ -735,7 +1048,9 @@ syscall(code, tf, pc)
} args;
register_t rval[2];
u_quad_t sticks;
+#ifdef DIAGNOSTIC
extern struct pcb *cpcb;
+#endif
cnt.v_syscall++;
p = curproc;
@@ -783,7 +1098,7 @@ syscall(code, tf, pc)
break;
}
- if (code < 0 || code >= nsys)
+ if (code < 0 || code >= nsys)
callp += p->p_emul->e_nosys;
else {
callp += code;
@@ -814,18 +1129,7 @@ syscall(code, tf, pc)
rval[1] = tf->tf_out[1];
error = (*callp->sy_call)(p, &args, rval);
if (error == 0) {
- /*
- * If fork succeeded and we are the child, our stack
- * has moved and the pointer tf is no longer valid,
- * and p is wrong. Compute the new trapframe pointer.
- * (The trap frame invariably resides at the
- * tippity-top of the u. area.)
- */
- p = curproc;
- tf = (struct trapframe *)
- ((caddr_t)p->p_addr + USPACE - sizeof(*tf));
-/* this is done earlier: */
-/* p->p_md.md_tf = tf; */
+ /* Note: fork() does not return here in the child */
tf->tf_out[0] = rval[0];
tf->tf_out[1] = rval[1];
if (new) {
@@ -862,3 +1166,22 @@ bad:
#endif
share_fpu(p, tf);
}
+
+/*
+ * Process the tail end of a fork() for the child.
+ */
+void
+child_return(p)
+ struct proc *p;
+{
+
+ /*
+ * Return values in the frame set by cpu_fork().
+ */
+ userret(p, p->p_md.md_tf->tf_pc, 0);
+#ifdef KTRACE
+ if (KTRPOINT(p, KTR_SYSRET))
+ ktrsysret(p->p_tracep,
+ (p->p_flag & P_PPWAIT) ? SYS_vfork : SYS_fork, 0, 0);
+#endif
+}
diff --git a/sys/arch/sparc/sparc/vaddrs.h b/sys/arch/sparc/sparc/vaddrs.h
index e6d02bf0c95..48cad5f52e8 100644
--- a/sys/arch/sparc/sparc/vaddrs.h
+++ b/sys/arch/sparc/sparc/vaddrs.h
@@ -1,6 +1,8 @@
-/* $NetBSD: vaddrs.h,v 1.5 1994/12/06 08:34:14 deraadt Exp $ */
+/* $NetBSD: vaddrs.h,v 1.7 1996/05/16 15:57:28 abrown Exp $ */
/*
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
*
@@ -25,6 +27,7 @@
* must display the following acknowledgement:
* This product includes software developed by the University of
* California, Berkeley and its contributors.
+ * This product includes software developed by Harvard University.
* 4. Neither the name of the University nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
@@ -59,24 +62,57 @@
* the Zilog ZSCC serial port chips to be mapped at fixed VAs to make
* microtime() and the zs hardware interrupt handlers faster.
*
+ * [sun4/sun4c:]
* Ideally, we should map the interrupt enable register here as well,
* but that would require allocating pmegs in locore.s, so instead we
- * use one of the two `wasted' pages at KERNBASE+2*NBPG (see locore.s).
+ * use one of the two `wasted' pages at KERNBASE+_MAXNBPG (see locore.s).
*/
#ifndef IODEV_0
#define IODEV_0 0xfe000000 /* must match VM_MAX_KERNEL_ADDRESS */
+#define _MAXNBPG 8192 /* fixed VAs, independent of actual NBPG */
+#define _MAXNCPU 4 /* fixed VA allocation allows 4 CPUs */
+
+/* [4m:] interrupt and counter registers take (1 + NCPU) pages. */
+
#define TIMERREG_VA (IODEV_0)
-#define ZS0_VA (IODEV_0 + 1*NBPG)
-#define ZS1_VA (IODEV_0 + 2*NBPG)
-#define AUXREG_VA (IODEV_0 + 3*NBPG)
-#define TMPMAP_VA (IODEV_0 + 4*NBPG)
-#define MSGBUF_VA (IODEV_0 + 5*NBPG)
-#define IODEV_BASE (IODEV_0 + 6*NBPG)
+#define COUNTERREG_VA ( TIMERREG_VA + _MAXNBPG*_MAXNCPU) /* [4m] */
+#define ZS0_VA (COUNTERREG_VA + _MAXNBPG)
+#define ZS1_VA ( ZS0_VA + _MAXNBPG)
+#define AUXREG_VA ( ZS1_VA + _MAXNBPG)
+#define TMPMAP_VA ( AUXREG_VA + _MAXNBPG)
+#define MSGBUF_VA ( TMPMAP_VA + _MAXNBPG)
+#define PI_INTR_VA ( MSGBUF_VA + _MAXNBPG) /* [4m] */
+#define SI_INTR_VA ( PI_INTR_VA + _MAXNBPG*_MAXNCPU) /* [4m] */
+#define IODEV_BASE ( SI_INTR_VA + _MAXNBPG)
#define IODEV_END 0xff000000 /* 16 MB of iospace */
#define DVMA_BASE 0xfff00000
#define DVMA_END 0xfffc0000
+/*
+ * The next constant defines the amount of reserved DVMA space on the
+ * Sun4m. The amount of space *must* be a multiple of 16MB, and thus
+ * (((u_int)0) - DVMA4M_BASE) must be divisible by 16*1024*1024!
+ * Note that pagetables must be allocated at a cost of 1k per MB of DVMA
+ * space, plus severe alignment restrictions. So don't make DVMA4M_BASE too
+ * low (max space = 2G).
+ *
+ * Since DVMA space overlaps with normal kernel address space (notably
+ * the device mappings and the PROM), we don't want to put any DVMA
+ * mappings where any of this useful stuff is (i.e. if we dvma_malloc
+ * a buffer, we want to still have a SRMMU mapping to it, and we can't
+ * have that if its on top of kernel code). Thus the last two
+ * constants define the actual DVMA addresses used. These can be anything
+ * as long as they are within the bounds setup by the first 2 constants.
+ * This is especially important on MP systems with cache coherency: to
+ * avoid consistency problems, DVMA addresses must map to the same place
+ * in both processor and IOMMU space.
+ */
+#define DVMA4M_BASE 0xfc000000 /* can change subject to above rule */
+#define DVMA4M_TOP 0xffffffff /* do not modify */
+#define DVMA4M_START 0xfd000000 /* 16M of DVMA */
+#define DVMA4M_END 0xfe000000 /* XXX is this enough? */
+
#endif /* IODEV_0 */
diff --git a/sys/arch/sparc/sparc/vm_machdep.c b/sys/arch/sparc/sparc/vm_machdep.c
index a4c3c9fa4c5..0a0bc838935 100644
--- a/sys/arch/sparc/sparc/vm_machdep.c
+++ b/sys/arch/sparc/sparc/vm_machdep.c
@@ -1,6 +1,8 @@
-/* $NetBSD: vm_machdep.c,v 1.18 1995/12/11 12:44:39 pk Exp $ */
+/* $NetBSD: vm_machdep.c,v 1.27.4.1 1996/08/02 21:35:20 jtc Exp $ */
/*
+ * Copyright (c) 1996
+ * The President and Fellows of Harvard College. All rights reserved.
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
*
@@ -12,6 +14,7 @@
* must display the following acknowledgement:
* This product includes software developed by the University of
* California, Lawrence Berkeley Laboratory.
+ * This product includes software developed by Harvard University.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
@@ -23,6 +26,7 @@
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
+ * This product includes software developed by Harvard University.
* This product includes software developed by the University of
* California, Berkeley and its contributors.
* 4. Neither the name of the University nor the names of its contributors
@@ -53,21 +57,24 @@
#include <sys/buf.h>
#include <sys/exec.h>
#include <sys/vnode.h>
+#include <sys/map.h>
#include <vm/vm.h>
#include <vm/vm_kern.h>
#include <machine/cpu.h>
#include <machine/frame.h>
+#include <machine/trap.h>
#include <sparc/sparc/cache.h>
/*
* Move pages from one kernel virtual address to another.
*/
+void
pagemove(from, to, size)
register caddr_t from, to;
- int size;
+ size_t size;
{
register vm_offset_t pa;
@@ -108,6 +115,7 @@ dvma_malloc(len, kaddr, flags)
vm_offset_t kva;
vm_offset_t dva;
+ len = round_page(len);
kva = (vm_offset_t)malloc(len, M_DEVBUF, flags);
if (kva == NULL)
return (NULL);
@@ -129,7 +137,7 @@ dvma_free(dva, len, kaddr)
{
vm_offset_t kva = *(vm_offset_t *)kaddr;
- dvma_mapout((vm_offset_t)dva, kva, len);
+ dvma_mapout((vm_offset_t)dva, kva, round_page(len));
free((void *)kva, M_DEVBUF);
}
@@ -147,13 +155,19 @@ dvma_mapin(map, va, len, canwait)
register int npf, s;
register vm_offset_t pa;
long off, pn;
+#if defined(SUN4M)
+ extern int has_iocache;
+#endif
off = (int)va & PGOFSET;
va -= off;
len = round_page(len + off);
npf = btoc(len);
- kvm_uncache((caddr_t)va, len >> PGSHIFT);
+#if defined(SUN4M)
+ if (!has_iocache)
+ kvm_uncache((caddr_t)va, len >> PGSHIFT);
+#endif
s = splimp();
for (;;) {
@@ -179,25 +193,25 @@ dvma_mapin(map, va, len, canwait)
panic("dvma_mapin: null page frame");
pa = trunc_page(pa);
-#if defined(SUN4M) && 0
+#if defined(SUN4M)
if (cputyp == CPU_SUN4M) {
iommu_enter(tva, pa);
} else
#endif
{
- /*
+ /*
* pmap_enter distributes this mapping to all
* contexts... maybe we should avoid this extra work
- */
+ */
#ifdef notyet
#if defined(SUN4)
if (have_iocache)
pa |= PG_IOC;
#endif
#endif
- pmap_enter(pmap_kernel(), tva,
+ pmap_enter(pmap_kernel(), tva,
pa | PMAP_NC,
- VM_PROT_READ|VM_PROT_WRITE, 1);
+ VM_PROT_READ|VM_PROT_WRITE, 1);
}
tva += PAGE_SIZE;
@@ -209,7 +223,7 @@ dvma_mapin(map, va, len, canwait)
/*
* Remove double map of `va' in DVMA space at `kva'.
*/
-int
+void
dvma_mapout(kva, va, len)
vm_offset_t kva, va;
int len;
@@ -220,12 +234,12 @@ dvma_mapout(kva, va, len)
kva -= off;
len = round_page(len + off);
-#if defined(SUN4M) && 0
+#if defined(SUN4M)
if (cputyp == CPU_SUN4M)
iommu_remove(kva, len);
else
#endif
- pmap_remove(pmap_kernel(), kva, kva + len);
+ pmap_remove(pmap_kernel(), kva, kva + len);
s = splimp();
rmfree(dvmamap, btoc(len), vtorc(kva));
@@ -239,8 +253,11 @@ dvma_mapout(kva, va, len)
/*
* Map an IO request into kernel virtual address space.
*/
-vmapbuf(bp)
+/*ARGSUSED*/
+void
+vmapbuf(bp, sz)
register struct buf *bp;
+ vm_size_t sz;
{
register vm_offset_t addr, kva, pa;
register vm_size_t size, off;
@@ -281,8 +298,11 @@ vmapbuf(bp)
/*
* Free the io map addresses associated with this IO operation.
*/
-vunmapbuf(bp)
+/*ARGSUSED*/
+void
+vunmapbuf(bp, sz)
register struct buf *bp;
+ vm_size_t sz;
{
register vm_offset_t kva = (vm_offset_t)bp->b_data;
register vm_size_t size, off;
@@ -308,39 +328,34 @@ vunmapbuf(bp)
/*
* Finish a fork operation, with process p2 nearly set up.
- * Copy and update the kernel stack and pcb, making the child
- * ready to run, and marking it so that it can return differently
- * than the parent. Returns 1 in the child process, 0 in the parent.
+ * Copy and update the pcb, making the child ready to run, and marking
+ * it so that it can return differently than the parent.
*
* This function relies on the fact that the pcb is
* the first element in struct user.
*/
+void
cpu_fork(p1, p2)
register struct proc *p1, *p2;
{
register struct pcb *opcb = &p1->p_addr->u_pcb;
register struct pcb *npcb = &p2->p_addr->u_pcb;
- register u_int sp, topframe, off, ssize;
+ register struct trapframe *tf2;
+ register struct rwindow *rp;
/*
- * Save all the registers to p1's stack or, in the case of
+ * Save all user registers to p1's stack or, in the case of
* user registers and invalid stack pointers, to opcb.
- * snapshot() also sets the given pcb's pcb_sp and pcb_psr
- * to the current %sp and %psr, and sets pcb_pc to a stub
- * which returns 1. We then copy the whole pcb to p2;
- * when switch() selects p2 to run, it will run at the stub,
- * rather than at the copying code below, and cpu_fork
- * will return 1.
- *
- * Note that the order `*npcb = *opcb, snapshot(npcb)' is wrong,
- * as user registers might then wind up only in opcb.
- * We could call save_user_windows first,
- * but that would only save 3 stores anyway.
+ * We then copy the whole pcb to p2; when switch() selects p2
+ * to run, it will run at the `proc_trampoline' stub, rather
+ * than returning at the copying code below.
*
* If process p1 has an FPU state, we must copy it. If it is
* the FPU user, we must save the FPU state first.
*/
- snapshot(opcb);
+
+ write_user_windows();
+ opcb->pcb_psr = getpsr();
bcopy((caddr_t)opcb, (caddr_t)npcb, sizeof(struct pcb));
if (p1->p_md.md_fpstate) {
if (p1 == fpproc)
@@ -353,30 +368,81 @@ cpu_fork(p1, p2)
p2->p_md.md_fpstate = NULL;
/*
- * Copy the active part of the kernel stack,
- * then adjust each kernel sp -- the frame pointer
- * in the top frame is a user sp -- in the child's copy,
- * including the initial one in the child's pcb.
+ * Setup (kernel) stack frame that will by-pass the child
+ * out of the kernel. (The trap frame invariably resides at
+ * the tippity-top of the u. area.)
*/
- sp = npcb->pcb_sp; /* points to old kernel stack */
- ssize = (u_int)opcb + USPACE - sp;
- if (ssize >= USPACE - sizeof(struct pcb))
- panic("cpu_fork 1");
- off = (u_int)npcb - (u_int)opcb;
- qcopy((caddr_t)sp, (caddr_t)sp + off, ssize);
- sp += off;
- npcb->pcb_sp = sp;
- topframe = (u_int)npcb + TOPFRAMEOFF;
- while (sp < topframe)
- sp = ((struct rwindow *)sp)->rw_in[6] += off;
- if (sp != topframe)
- panic("cpu_fork 2");
+ tf2 = p2->p_md.md_tf = (struct trapframe *)
+ ((int)npcb + USPACE - sizeof(*tf2));
+
+ /* Copy parent's trapframe */
+ *tf2 = *(struct trapframe *)((int)opcb + USPACE - sizeof(*tf2));
+
+ /* Duplicate efforts of syscall(), but slightly differently */
+ if (tf2->tf_global[1] & SYSCALL_G2RFLAG) {
+ /* jmp %g2 (or %g7, deprecated) on success */
+ tf2->tf_npc = tf2->tf_global[2];
+ } else {
+ /*
+ * old system call convention: clear C on success
+ * note: proc_trampoline() sets a fresh psr when
+ * returning to user mode.
+ */
+ /*tf2->tf_psr &= ~PSR_C; -* success */
+ }
+
+ /* Set return values in child mode */
+ tf2->tf_out[0] = 0;
+ tf2->tf_out[1] = 1;
+
+ /* Construct kernel frame to return to in cpu_switch() */
+ rp = (struct rwindow *)((u_int)npcb + TOPFRAMEOFF);
+ rp->rw_local[0] = (int)child_return; /* Function to call */
+ rp->rw_local[1] = (int)p2; /* and its argument */
+
+ npcb->pcb_pc = (int)proc_trampoline - 8;
+ npcb->pcb_sp = (int)rp;
+ npcb->pcb_psr &= ~PSR_CWP; /* Run in window #0 */
+ npcb->pcb_wim = 1; /* Fence at window #1 */
+
+}
+
+/*
+ * cpu_set_kpc:
+ *
+ * Arrange for in-kernel execution of a process to continue at the
+ * named pc, as if the code at that address were called as a function
+ * with the current process's process pointer as an argument.
+ *
+ * Note that it's assumed that when the named process returns,
+ * we immediately return to user mode.
+ *
+ * (Note that cpu_fork(), above, uses an open-coded version of this.)
+ */
+void
+cpu_set_kpc(p, pc)
+ struct proc *p;
+ void (*pc) __P((struct proc *));
+{
+ struct pcb *pcb;
+ struct rwindow *rp;
+
+ pcb = &p->p_addr->u_pcb;
+
+ rp = (struct rwindow *)((u_int)pcb + TOPFRAMEOFF);
+ rp->rw_local[0] = (int)pc; /* Function to call */
+ rp->rw_local[1] = (int)p; /* and its argument */
+
/*
- * This might be unnecessary, but it may be possible for the child
- * to run in ptrace or sendsig before it returns from fork.
+ * Frob PCB:
+ * - arrange to return to proc_trampoline() from cpu_switch()
+ * - point it at the stack frame constructed above
+ * - make it run in a clear set of register windows
*/
- p2->p_md.md_tf = (struct trapframe *)((int)p1->p_md.md_tf + off);
- return (0);
+ pcb->pcb_pc = (int)proc_trampoline - 8;
+ pcb->pcb_sp = (int)rp;
+ pcb->pcb_psr &= ~PSR_CWP; /* Run in window #0 */
+ pcb->pcb_wim = 1; /* Fence at window #1 */
}
/*
@@ -416,7 +482,6 @@ cpu_coredump(p, vp, cred, chdr)
struct core *chdr;
{
int error;
- register struct user *up = p->p_addr;
struct md_coredump md_core;
struct coreseg cseg;
diff --git a/sys/arch/sparc/stand/Makefile.inc b/sys/arch/sparc/stand/Makefile.inc
index a083047b0d1..5a960d5b0cb 100644
--- a/sys/arch/sparc/stand/Makefile.inc
+++ b/sys/arch/sparc/stand/Makefile.inc
@@ -1,4 +1,4 @@
-# $NetBSD: Makefile.inc,v 1.4 1995/10/09 23:25:55 pk Exp $
+# $NetBSD: Makefile.inc,v 1.5 1996/02/01 22:33:01 mycroft Exp $
.if !defined(__stand_makefile_inc)
__stand_makefile_inc=1
@@ -22,7 +22,7 @@ DEFS+= -DSTANDALONE -DRELOC=${RELOC} -DSUN4 -DSUN4C -DSUN_BOOTPARAMS
CFLAGS+= -O2 ${INCLUDES} ${DEFS}
srt0.o: srt0.S
- ${CC} ${CFLAGS} -DLOCORE -c ${.IMPSRC}
+ ${CC} ${CFLAGS} -D_LOCORE -c ${.IMPSRC}
cleandir:
rm -rf lib machine
diff --git a/sys/arch/sparc/stand/binstall.sh b/sys/arch/sparc/stand/binstall.sh
index 552a18ef41c..1b2db4158af 100644
--- a/sys/arch/sparc/stand/binstall.sh
+++ b/sys/arch/sparc/stand/binstall.sh
@@ -1,5 +1,5 @@
#!/bin/sh
-# $NetBSD: binstall.sh,v 1.1.2.1 1995/10/13 19:55:03 pk Exp $
+# $NetBSD: binstall.sh,v 1.3 1996/04/07 20:00:12 thorpej Exp $
#
vecho () {
@@ -59,7 +59,7 @@ fi
WHAT=$1
DEST=$2
-if [ "`sysctl -n hw.model | cut -b1-5`" = "SUN/4" ]; then
+if [ "`sysctl -n hw.model | cut -b1-5`" = "SUN-4" ]; then
KARCH=sun4
else
KARCH=sun4c
@@ -106,7 +106,7 @@ case $WHAT in
;;
"net")
- TARGET=$DEST/boot.sparc.openbsd.$KARCH
+ TARGET=$DEST/boot.sparc.netbsd.$KARCH
vecho Target: $TARGET
$DOIT dd if=${MDEC}/boot of=$TARGET skip=$SKIP bs=32
;;
diff --git a/sys/arch/sparc/stand/boot.c b/sys/arch/sparc/stand/boot.c
index 7a696f208e7..5b423db0ad2 100644
--- a/sys/arch/sparc/stand/boot.c
+++ b/sys/arch/sparc/stand/boot.c
@@ -39,7 +39,6 @@
#include <sys/reboot.h>
#include <a.out.h>
#include <stand.h>
-#include <paths.h>
#include "promdev.h"
@@ -51,6 +50,7 @@ int netif_debug;
/*
* Boot device is derived from ROM provided information.
*/
+#define DEFAULT_KERNEL "netbsd"
extern char *version;
unsigned long esym;
@@ -69,11 +69,11 @@ main()
prom_init();
- printf(">> OpenBSD BOOT [%s]\n", version);
+ printf(">> NetBSD BOOT [%s]\n", version);
file = prom_bootfile;
if (file == 0 || *file == 0)
- file = _PATH_UNIX;
+ file = DEFAULT_KERNEL;
for (;;) {
if (prom_boothow & RB_ASKNAME) {
diff --git a/sys/arch/sparc/stand/bootxx/Makefile b/sys/arch/sparc/stand/bootxx/Makefile
index 7769be5223b..fb0900ac655 100644
--- a/sys/arch/sparc/stand/bootxx/Makefile
+++ b/sys/arch/sparc/stand/bootxx/Makefile
@@ -1,4 +1,4 @@
-# $NetBSD: Makefile,v 1.4.2.1 1995/10/13 19:55:36 pk Exp $
+# $NetBSD: Makefile,v 1.5 1995/10/10 20:07:54 pk Exp $
R= ..
.PATH: ${.CURDIR}/${R}
diff --git a/sys/arch/sparc/stand/installboot.c b/sys/arch/sparc/stand/installboot.c
index e92313f5c69..79dd8964036 100644
--- a/sys/arch/sparc/stand/installboot.c
+++ b/sys/arch/sparc/stand/installboot.c
@@ -1,4 +1,4 @@
-/* $NetBSD$ */
+/* $NetBSD: installboot.c,v 1.11 1995/11/08 09:09:20 pk Exp $ */
/*
* Copyright (c) 1994 Paul Kranenburg
diff --git a/sys/arch/sparc/stand/net.c b/sys/arch/sparc/stand/net.c
index f70c1834793..e67b22ac6f4 100644
--- a/sys/arch/sparc/stand/net.c
+++ b/sys/arch/sparc/stand/net.c
@@ -1,4 +1,4 @@
-/* $NetBSD: net.c,v 1.2 1995/09/18 21:31:46 pk Exp $ */
+/* $NetBSD: net.c,v 1.3 1996/05/04 19:36:01 pk Exp $ */
/*
* Copyright (c) 1995 Gordon W. Ross
@@ -153,5 +153,7 @@ net_mountroot()
printf("root addr=%s path=%s\n", inet_ntoa(rootip), rootpath);
/* Get the NFS file handle (mount). */
- return nfs_mount(netdev_sock, rootip, rootpath);
+ if (nfs_mount(netdev_sock, rootip, rootpath) != 0)
+ return (errno);
+ return 0;
}
diff --git a/sys/arch/sparc/stand/promdev.c b/sys/arch/sparc/stand/promdev.c
index 1404ce4f20d..94ab155eff0 100644
--- a/sys/arch/sparc/stand/promdev.c
+++ b/sys/arch/sparc/stand/promdev.c
@@ -1,4 +1,4 @@
-/* $NetBSD: promdev.c,v 1.14.2.1 1995/10/13 19:53:55 pk Exp $ */
+/* $NetBSD: promdev.c,v 1.16 1995/11/14 15:04:01 pk Exp $ */
/*
* Copyright (c) 1993 Paul Kranenburg
diff --git a/sys/arch/sparc/stand/srt0.S b/sys/arch/sparc/stand/srt0.S
index b31e322a86d..e5248292b84 100644
--- a/sys/arch/sparc/stand/srt0.S
+++ b/sys/arch/sparc/stand/srt0.S
@@ -1,4 +1,4 @@
-/* $NetBSD: srt0.S,v 1.5 1995/12/13 23:35:18 pk Exp $ */
+/* $NetBSD: srt0.S,v 1.5.4.2 1996/07/17 01:51:46 jtc Exp $ */
/*
* Copyright (c) 1994 Paul Kranenburg
@@ -102,13 +102,16 @@ start:
call _bzero
sub %o1, %o0, %o1
+#if 0
/*
- * Enable interrupts.
+ * Enable interrupts above level 11. This enables "L1-A", but
+ * avoids spurious interrupt bites from most other devices
*/
rd %psr, %o0
andn %o0, PSR_PIL, %o0
- wr %g0, %o0, %psr
+ wr %o0, 0xb00, %psr ! (11 << 8)
nop; nop; nop
+#endif
/*
* Set CPU type that we are running on.
diff --git a/sys/arch/sparc/stand/version.c b/sys/arch/sparc/stand/version.c
index f288f3059da..1217ce7aba7 100644
--- a/sys/arch/sparc/stand/version.c
+++ b/sys/arch/sparc/stand/version.c
@@ -38,4 +38,4 @@
* 1.4 add oldmon support and network support.
*/
-char *version = "$Revision: 1.1 $";
+char *version = "$Revision: 1.2 $";