summaryrefslogtreecommitdiff
path: root/sys/arch/hp300/dev/grf.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch/hp300/dev/grf.c')
-rw-r--r--sys/arch/hp300/dev/grf.c675
1 files changed, 675 insertions, 0 deletions
diff --git a/sys/arch/hp300/dev/grf.c b/sys/arch/hp300/dev/grf.c
new file mode 100644
index 00000000000..15f34e180a7
--- /dev/null
+++ b/sys/arch/hp300/dev/grf.c
@@ -0,0 +1,675 @@
+/* $NetBSD: grf.c,v 1.14 1995/04/22 20:25:42 christos Exp $ */
+
+/*
+ * Copyright (c) 1988 University of Utah.
+ * Copyright (c) 1990, 1993
+ * The Regents of the University of California. All rights reserved.
+ *
+ * This code is derived from software contributed to Berkeley by
+ * the Systems Programming Group of the University of Utah Computer
+ * Science Department.
+ *
+ * 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 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.
+ *
+ * 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.
+ *
+ * from: Utah $Hdr: grf.c 1.36 93/08/13$
+ *
+ * @(#)grf.c 8.4 (Berkeley) 1/12/94
+ */
+
+/*
+ * Graphics display driver for HP 300/400/700/800 machines.
+ * This is the hardware-independent portion of the driver.
+ * Hardware access is through the machine dependent grf switch routines.
+ */
+
+#include "grf.h"
+#if NGRF > 0
+
+#include <sys/param.h>
+#include <sys/proc.h>
+#include <sys/ioctl.h>
+#include <sys/file.h>
+#include <sys/malloc.h>
+#include <sys/vnode.h>
+#include <sys/mman.h>
+
+#include <hp300/dev/grfioctl.h>
+#include <hp300/dev/grfvar.h>
+#include <hp300/dev/grfreg.h>
+
+#include <machine/cpu.h>
+
+#ifdef COMPAT_HPUX
+#include <hp300/hpux/hpux.h>
+extern struct emul emul_hpux;
+#endif
+
+#include <vm/vm.h>
+#include <vm/vm_kern.h>
+#include <vm/vm_page.h>
+#include <vm/vm_pager.h>
+
+#include <miscfs/specfs/specdev.h>
+
+#include <ite.h>
+#if NITE == 0
+#define iteon(u,f)
+#define iteoff(u,f)
+#endif
+
+struct grf_softc grf_softc[NGRF];
+
+#ifdef DEBUG
+int grfdebug = 0;
+#define GDB_DEVNO 0x01
+#define GDB_MMAP 0x02
+#define GDB_IOMAP 0x04
+#define GDB_LOCK 0x08
+#endif
+
+/*ARGSUSED*/
+grfopen(dev, flags)
+ dev_t dev;
+ int flags;
+{
+ int unit = GRFUNIT(dev);
+ register struct grf_softc *gp = &grf_softc[unit];
+ int error = 0;
+
+ if (unit >= NGRF || (gp->g_flags & GF_ALIVE) == 0)
+ return(ENXIO);
+ if ((gp->g_flags & (GF_OPEN|GF_EXCLUDE)) == (GF_OPEN|GF_EXCLUDE))
+ return(EBUSY);
+#ifdef COMPAT_HPUX
+ /*
+ * XXX: cannot handle both HPUX and BSD processes at the same time
+ */
+ if (curproc->p_emul == &emul_hpux)
+ if (gp->g_flags & GF_BSDOPEN)
+ return(EBUSY);
+ else
+ gp->g_flags |= GF_HPUXOPEN;
+ else
+ if (gp->g_flags & GF_HPUXOPEN)
+ return(EBUSY);
+ else
+ gp->g_flags |= GF_BSDOPEN;
+#endif
+ /*
+ * First open.
+ * XXX: always put in graphics mode.
+ */
+ error = 0;
+ if ((gp->g_flags & GF_OPEN) == 0) {
+ gp->g_flags |= GF_OPEN;
+ error = grfon(dev);
+ }
+ return(error);
+}
+
+/*ARGSUSED*/
+grfclose(dev, flags)
+ dev_t dev;
+ int flags;
+{
+ register struct grf_softc *gp = &grf_softc[GRFUNIT(dev)];
+
+ (void) grfoff(dev);
+#ifdef COMPAT_HPUX
+ (void) grfunlock(gp);
+#endif
+ gp->g_flags &= GF_ALIVE;
+ return(0);
+}
+
+/*ARGSUSED*/
+grfioctl(dev, cmd, data, flag, p)
+ dev_t dev;
+ int cmd, flag;
+ caddr_t data;
+ struct proc *p;
+{
+ register struct grf_softc *gp = &grf_softc[GRFUNIT(dev)];
+ int error;
+
+#ifdef COMPAT_HPUX
+ if (p->p_emul == &emul_hpux)
+ return(hpuxgrfioctl(dev, cmd, data, flag, p));
+#endif
+ error = 0;
+ switch (cmd) {
+
+ case GRFIOCGINFO:
+ bcopy((caddr_t)&gp->g_display, data, sizeof(struct grfinfo));
+ break;
+
+ case GRFIOCON:
+ error = grfon(dev);
+ break;
+
+ case GRFIOCOFF:
+ error = grfoff(dev);
+ break;
+
+ case GRFIOCMAP:
+ error = grfmap(dev, (caddr_t *)data, p);
+ break;
+
+ case GRFIOCUNMAP:
+ error = grfunmap(dev, *(caddr_t *)data, p);
+ break;
+
+ default:
+ error = EINVAL;
+ break;
+
+ }
+ return(error);
+}
+
+/*ARGSUSED*/
+grfselect(dev, rw)
+ dev_t dev;
+ int rw;
+{
+ if (rw == FREAD)
+ return(0);
+ return(1);
+}
+
+/*ARGSUSED*/
+grfmmap(dev, off, prot)
+ dev_t dev;
+ int off, prot;
+{
+ return(grfaddr(&grf_softc[GRFUNIT(dev)], off));
+}
+
+grfon(dev)
+ dev_t dev;
+{
+ int unit = GRFUNIT(dev);
+ struct grf_softc *gp = &grf_softc[unit];
+
+ /*
+ * XXX: iteoff call relies on devices being in same order
+ * as ITEs and the fact that iteoff only uses the minor part
+ * of the dev arg.
+ */
+ iteoff(unit, 3);
+ return((*gp->g_sw->gd_mode)(gp,
+ (dev&GRFOVDEV) ? GM_GRFOVON : GM_GRFON,
+ (caddr_t)0));
+}
+
+grfoff(dev)
+ dev_t dev;
+{
+ int unit = GRFUNIT(dev);
+ struct grf_softc *gp = &grf_softc[unit];
+ int error;
+
+ (void) grfunmap(dev, (caddr_t)0, curproc);
+ error = (*gp->g_sw->gd_mode)(gp,
+ (dev&GRFOVDEV) ? GM_GRFOVOFF : GM_GRFOFF,
+ (caddr_t)0);
+ /* XXX: see comment for iteoff above */
+ iteon(unit, 2);
+ return(error);
+}
+
+grfaddr(gp, off)
+ struct grf_softc *gp;
+ register int off;
+{
+ register struct grfinfo *gi = &gp->g_display;
+
+ /* control registers */
+ if (off >= 0 && off < gi->gd_regsize)
+ return(((u_int)gi->gd_regaddr + off) >> PGSHIFT);
+
+ /* frame buffer */
+ if (off >= gi->gd_regsize && off < gi->gd_regsize+gi->gd_fbsize) {
+ off -= gi->gd_regsize;
+ return(((u_int)gi->gd_fbaddr + off) >> PGSHIFT);
+ }
+ /* bogus */
+ return(-1);
+}
+
+/*
+ * HP-UX compatibility routines
+ */
+#ifdef COMPAT_HPUX
+
+/*ARGSUSED*/
+hpuxgrfioctl(dev, cmd, data, flag, p)
+ dev_t dev;
+ int cmd, flag;
+ caddr_t data;
+ struct proc *p;
+{
+ register struct grf_softc *gp = &grf_softc[GRFUNIT(dev)];
+ int error;
+
+ error = 0;
+ switch (cmd) {
+
+ case GCID:
+ *(int *)data = gp->g_display.gd_id;
+ break;
+
+ case GCON:
+ error = grfon(dev);
+ break;
+
+ case GCOFF:
+ error = grfoff(dev);
+ break;
+
+ case GCLOCK:
+ error = grflock(gp, 1);
+ break;
+
+ case GCUNLOCK:
+ error = grfunlock(gp);
+ break;
+
+ case GCAON:
+ case GCAOFF:
+ break;
+
+ /* GCSTATIC is implied by our implementation */
+ case GCSTATIC_CMAP:
+ case GCVARIABLE_CMAP:
+ break;
+
+ /* map in control regs and frame buffer */
+ case GCMAP:
+ error = grfmap(dev, (caddr_t *)data, p);
+ break;
+
+ case GCUNMAP:
+ error = grfunmap(dev, *(caddr_t *)data, p);
+ /* XXX: HP-UX uses GCUNMAP to get rid of GCSLOT memory */
+ if (error)
+ error = grflckunmmap(dev, *(caddr_t *)data);
+ break;
+
+ case GCSLOT:
+ {
+ struct grf_slot *sp = (struct grf_slot *)data;
+
+ sp->slot = grffindpid(gp);
+ if (sp->slot) {
+ error = grflckmmap(dev, (caddr_t *)&sp->addr);
+ if (error && gp->g_pid) {
+ free((caddr_t)gp->g_pid, M_DEVBUF);
+ gp->g_pid = NULL;
+ }
+ } else
+ error = EINVAL; /* XXX */
+ break;
+ }
+
+ case GCDESCRIBE:
+ error = (*gp->g_sw->gd_mode)(gp, GM_DESCRIBE, data);
+ break;
+
+ /*
+ * XXX: only used right now to map in rbox control registers
+ * Will be replaced in the future with a real IOMAP interface.
+ */
+ case IOMAPMAP:
+ error = iommap(dev, (caddr_t *)data);
+#if 0
+ /*
+ * It may not be worth kludging this (using p_devtmp) to
+ * make this work. It was an undocumented side-effect
+ * in HP-UX that the mapped address was the return value
+ * of the ioctl. The only thing I remember that counted
+ * on this behavior was the rbox X10 server.
+ */
+ if (!error)
+ u.u_r.r_val1 = *(int *)data; /* XXX: this sux */
+#endif
+ break;
+
+ case IOMAPUNMAP:
+ error = iounmmap(dev, *(caddr_t *)data);
+ break;
+
+ default:
+ error = EINVAL;
+ break;
+ }
+ return(error);
+}
+
+grflock(gp, block)
+ register struct grf_softc *gp;
+ int block;
+{
+ struct proc *p = curproc; /* XXX */
+ int error;
+ extern char devioc[];
+
+#ifdef DEBUG
+ if (grfdebug & GDB_LOCK)
+ printf("grflock(%d): dev %x flags %x lockpid %x\n",
+ p->p_pid, gp-grf_softc, gp->g_flags,
+ gp->g_lockp ? gp->g_lockp->p_pid : -1);
+#endif
+ if (gp->g_pid) {
+#ifdef DEBUG
+ if (grfdebug & GDB_LOCK)
+ printf(" lockpslot %d lockslot %d lock[lockslot] %d\n",
+ gp->g_lock->gl_lockslot, gp->g_lockpslot,
+ gp->g_lock->gl_locks[gp->g_lockpslot]);
+#endif
+ gp->g_lock->gl_lockslot = 0;
+ if (gp->g_lock->gl_locks[gp->g_lockpslot] == 0) {
+ gp->g_lockp = NULL;
+ gp->g_lockpslot = 0;
+ }
+ }
+ if (gp->g_lockp) {
+ if (gp->g_lockp == p)
+ return(EBUSY);
+ if (!block)
+ return(OEAGAIN);
+ do {
+ gp->g_flags |= GF_WANTED;
+ if (error = tsleep((caddr_t)&gp->g_flags,
+ (PZERO+1) | PCATCH, devioc, 0))
+ return (error);
+ } while (gp->g_lockp);
+ }
+ gp->g_lockp = p;
+ if (gp->g_pid) {
+ int slot = grffindpid(gp);
+
+#ifdef DEBUG
+ if (grfdebug & GDB_LOCK)
+ printf(" slot %d\n", slot);
+#endif
+ gp->g_lockpslot = gp->g_lock->gl_lockslot = slot;
+ gp->g_lock->gl_locks[slot] = 1;
+ }
+ return(0);
+}
+
+grfunlock(gp)
+ register struct grf_softc *gp;
+{
+#ifdef DEBUG
+ if (grfdebug & GDB_LOCK)
+ printf("grfunlock(%d): dev %x flags %x lockpid %d\n",
+ curproc->p_pid, gp-grf_softc, gp->g_flags,
+ gp->g_lockp ? gp->g_lockp->p_pid : -1);
+#endif
+ if (gp->g_lockp != curproc)
+ return(EBUSY);
+ if (gp->g_pid) {
+#ifdef DEBUG
+ if (grfdebug & GDB_LOCK)
+ printf(" lockpslot %d lockslot %d lock[lockslot] %d\n",
+ gp->g_lock->gl_lockslot, gp->g_lockpslot,
+ gp->g_lock->gl_locks[gp->g_lockpslot]);
+#endif
+ gp->g_lock->gl_locks[gp->g_lockpslot] = 0;
+ gp->g_lockpslot = gp->g_lock->gl_lockslot = 0;
+ }
+ if (gp->g_flags & GF_WANTED) {
+ wakeup((caddr_t)&gp->g_flags);
+ gp->g_flags &= ~GF_WANTED;
+ }
+ gp->g_lockp = NULL;
+ return(0);
+}
+
+/*
+ * Convert a BSD style minor devno to HPUX style.
+ * We cannot just create HPUX style nodes as they require 24 bits
+ * of minor device number and we only have 8.
+ * XXX: This may give the wrong result for remote stats of other
+ * machines where device 10 exists.
+ */
+grfdevno(dev)
+ dev_t dev;
+{
+ int unit = GRFUNIT(dev);
+ struct grf_softc *gp = &grf_softc[unit];
+ int newdev;
+
+ if (unit >= NGRF || (gp->g_flags&GF_ALIVE) == 0)
+ return(bsdtohpuxdev(dev));
+ /* magic major number */
+ newdev = 12 << 24;
+ /* now construct minor number */
+ if (gp->g_display.gd_regaddr != (caddr_t)GRFIADDR) {
+ int sc = patosc(gp->g_display.gd_regaddr);
+ newdev |= (sc << 16) | 0x200;
+ }
+ if (dev & GRFIMDEV)
+ newdev |= 0x02;
+ else if (dev & GRFOVDEV)
+ newdev |= 0x01;
+#ifdef DEBUG
+ if (grfdebug & GDB_DEVNO)
+ printf("grfdevno: dev %x newdev %x\n", dev, newdev);
+#endif
+ return(newdev);
+}
+
+#endif /* COMPAT_HPUX */
+
+grfmap(dev, addrp, p)
+ dev_t dev;
+ caddr_t *addrp;
+ struct proc *p;
+{
+ struct grf_softc *gp = &grf_softc[GRFUNIT(dev)];
+ int len, error;
+ struct vnode vn;
+ struct specinfo si;
+ int flags;
+
+#ifdef DEBUG
+ if (grfdebug & GDB_MMAP)
+ printf("grfmap(%d): addr %x\n", p->p_pid, *addrp);
+#endif
+ len = gp->g_display.gd_regsize + gp->g_display.gd_fbsize;
+ flags = MAP_SHARED;
+ if (*addrp)
+ flags |= MAP_FIXED;
+ else
+ *addrp = (caddr_t)0x1000000; /* XXX */
+ vn.v_type = VCHR; /* XXX */
+ vn.v_specinfo = &si; /* XXX */
+ vn.v_rdev = dev; /* XXX */
+ error = vm_mmap(&p->p_vmspace->vm_map, (vm_offset_t *)addrp,
+ (vm_size_t)len, VM_PROT_ALL, VM_PROT_ALL,
+ flags, (caddr_t)&vn, 0);
+ if (error == 0)
+ (void) (*gp->g_sw->gd_mode)(gp, GM_MAP, *addrp);
+ return(error);
+}
+
+grfunmap(dev, addr, p)
+ dev_t dev;
+ caddr_t addr;
+ struct proc *p;
+{
+ struct grf_softc *gp = &grf_softc[GRFUNIT(dev)];
+ vm_size_t size;
+ int rv;
+
+#ifdef DEBUG
+ if (grfdebug & GDB_MMAP)
+ printf("grfunmap(%d): dev %x addr %x\n", p->p_pid, dev, addr);
+#endif
+ if (addr == 0)
+ return(EINVAL); /* XXX: how do we deal with this? */
+ (void) (*gp->g_sw->gd_mode)(gp, GM_UNMAP, 0);
+ size = round_page(gp->g_display.gd_regsize + gp->g_display.gd_fbsize);
+ rv = vm_deallocate(&p->p_vmspace->vm_map, (vm_offset_t)addr, size);
+ return(rv == KERN_SUCCESS ? 0 : EINVAL);
+}
+
+#ifdef COMPAT_HPUX
+iommap(dev, addrp)
+ dev_t dev;
+ caddr_t *addrp;
+{
+
+#ifdef DEBUG
+ if (grfdebug & (GDB_MMAP|GDB_IOMAP))
+ printf("iommap(%d): addr %x\n", curproc->p_pid, *addrp);
+#endif
+ return(EINVAL);
+}
+
+iounmmap(dev, addr)
+ dev_t dev;
+ caddr_t addr;
+{
+ int unit = minor(dev);
+
+#ifdef DEBUG
+ if (grfdebug & (GDB_MMAP|GDB_IOMAP))
+ printf("iounmmap(%d): id %d addr %x\n",
+ curproc->p_pid, unit, addr);
+#endif
+ return(0);
+}
+
+/*
+ * Processes involved in framebuffer mapping via GCSLOT are recorded in
+ * an array of pids. The first element is used to record the last slot used
+ * (for faster lookups). The remaining elements record up to GRFMAXLCK-1
+ * process ids. Returns a slot number between 1 and GRFMAXLCK or 0 if no
+ * slot is available.
+ */
+grffindpid(gp)
+ struct grf_softc *gp;
+{
+ register short pid, *sp;
+ register int i, limit;
+ int ni;
+
+ if (gp->g_pid == NULL) {
+ gp->g_pid = (short *)
+ malloc(GRFMAXLCK * sizeof(short), M_DEVBUF, M_WAITOK);
+ bzero((caddr_t)gp->g_pid, GRFMAXLCK * sizeof(short));
+ }
+ pid = curproc->p_pid;
+ ni = limit = gp->g_pid[0];
+ for (i = 1, sp = &gp->g_pid[1]; i <= limit; i++, sp++) {
+ if (*sp == pid)
+ goto done;
+ if (*sp == 0)
+ ni = i;
+ }
+ i = ni;
+ if (i < limit) {
+ gp->g_pid[i] = pid;
+ goto done;
+ }
+ if (++i == GRFMAXLCK)
+ return(0);
+ gp->g_pid[0] = i;
+ gp->g_pid[i] = pid;
+done:
+#ifdef DEBUG
+ if (grfdebug & GDB_LOCK)
+ printf("grffindpid(%d): slot %d of %d\n",
+ pid, i, gp->g_pid[0]);
+#endif
+ return(i);
+}
+
+grfrmpid(gp)
+ struct grf_softc *gp;
+{
+ register short pid, *sp;
+ register int limit, i;
+ int mi;
+
+ if (gp->g_pid == NULL || (limit = gp->g_pid[0]) == 0)
+ return;
+ pid = curproc->p_pid;
+ limit = gp->g_pid[0];
+ mi = 0;
+ for (i = 1, sp = &gp->g_pid[1]; i <= limit; i++, sp++) {
+ if (*sp == pid)
+ *sp = 0;
+ else if (*sp)
+ mi = i;
+ }
+ i = mi;
+ if (i < limit)
+ gp->g_pid[0] = i;
+#ifdef DEBUG
+ if (grfdebug & GDB_LOCK)
+ printf("grfrmpid(%d): slot %d of %d\n",
+ pid, sp-gp->g_pid, gp->g_pid[0]);
+#endif
+}
+
+grflckmmap(dev, addrp)
+ dev_t dev;
+ caddr_t *addrp;
+{
+#ifdef DEBUG
+ struct proc *p = curproc; /* XXX */
+
+ if (grfdebug & (GDB_MMAP|GDB_LOCK))
+ printf("grflckmmap(%d): addr %x\n",
+ p->p_pid, *addrp);
+#endif
+ return(EINVAL);
+}
+
+grflckunmmap(dev, addr)
+ dev_t dev;
+ caddr_t addr;
+{
+#ifdef DEBUG
+ int unit = minor(dev);
+
+ if (grfdebug & (GDB_MMAP|GDB_LOCK))
+ printf("grflckunmmap(%d): id %d addr %x\n",
+ curproc->p_pid, unit, addr);
+#endif
+ return(EINVAL);
+}
+#endif /* COMPAT_HPUX */
+
+#endif /* NGRF > 0 */