#include <sys/param.h>
#include <sys/systm.h>
#include <sys/buf.h>
#include <sys/device.h>
#include <sys/ioctl.h>
#include <sys/conf.h>
#include <uvm/uvm_extern.h>
#include <machine/autoconf.h>
#include <machine/bus.h>
#include <machine/pmap.h>
#include <machine/cpu.h>
#include <machine/conf.h>
#include <dev/wscons/wsconsio.h>
#include <dev/wscons/wsdisplayvar.h>
#include <dev/rasops/rasops.h>
#include <machine/fbvar.h>
#include <dev/sbus/sbusvar.h>
#include <dev/sbus/cgtwelvereg.h>
#include <dev/cons.h>
struct cgtwelve_softc {
struct sunfb sc_sunfb;
bus_space_tag_t sc_bustag;
bus_addr_t sc_paddr;
volatile struct cgtwelve_dpu *sc_dpu;
volatile struct cgtwelve_apu *sc_apu;
volatile struct cgtwelve_dac *sc_ramdac;
volatile u_char *sc_overlay;
volatile u_long *sc_inten;
int sc_highres;
int sc_nscreens;
int sc_isconsole;
};
int cgtwelve_ioctl(void *, u_long, caddr_t, int, struct proc *);
paddr_t cgtwelve_mmap(void *, off_t, int);
void cgtwelve_reset(struct cgtwelve_softc *, int);
void cgtwelve_prom(struct cgtwelve_softc *);
static __inline__ void cgtwelve_ramdac_wraddr(struct cgtwelve_softc *sc,
u_int32_t addr);
struct wsdisplay_accessops cgtwelve_accessops = {
.ioctl = cgtwelve_ioctl,
.mmap = cgtwelve_mmap
};
int cgtwelvematch(struct device *, void *, void *);
void cgtwelveattach(struct device *, struct device *, void *);
int cgtwelveactivate(struct device *, int);
const struct cfattach cgtwelve_ca = {
sizeof(struct cgtwelve_softc), cgtwelvematch, cgtwelveattach,
NULL, cgtwelveactivate
};
struct cfdriver cgtwelve_cd = {
NULL, "cgtwelve", DV_DULL
};
int
cgtwelvematch(struct device *parent, void *vcf, void *aux)
{
struct cfdata *cf = vcf;
struct sbus_attach_args *sa = aux;
if (strcmp(cf->cf_driver->cd_name, sa->sa_name) != 0)
return (0);
return (1);
}
void
cgtwelveattach(struct device *parent, struct device *self, void *args)
{
struct cgtwelve_softc *sc = (struct cgtwelve_softc *)self;
struct sbus_attach_args *sa = args;
bus_space_tag_t bt;
bus_space_handle_t bh;
int node;
char *ps;
bt = sa->sa_bustag;
node = sa->sa_node;
printf(": %s", getpropstring(node, "model"));
ps = getpropstring(node, "dev_id");
if (*ps != '\0')
printf(" (%s)", ps);
printf("\n");
sc->sc_isconsole = node == fbnode;
if (sa->sa_nreg == 0) {
printf("%s: no SBus registers!\n", self->dv_xname);
return;
}
sc->sc_bustag = bt;
if (sbus_bus_map(bt, sa->sa_slot, sa->sa_offset +
CG12_OFF_DPU, sizeof(struct cgtwelve_dpu),
BUS_SPACE_MAP_LINEAR, 0, &bh) != 0) {
printf("%s: can't map DPU registers\n", self->dv_xname);
return;
}
sc->sc_dpu = bus_space_vaddr(bt, bh);
if (sbus_bus_map(bt, sa->sa_slot, sa->sa_offset +
CG12_OFF_APU, sizeof(struct cgtwelve_apu),
BUS_SPACE_MAP_LINEAR, 0, &bh) != 0) {
printf("%s: can't map APU registers\n", self->dv_xname);
return;
}
sc->sc_apu = bus_space_vaddr(bt, bh);
if (sbus_bus_map(bt, sa->sa_slot, sa->sa_offset +
CG12_OFF_DAC, sizeof(struct cgtwelve_dac),
BUS_SPACE_MAP_LINEAR, 0, &bh) != 0) {
printf("%s: can't map RAMDAC registers\n", self->dv_xname);
return;
}
sc->sc_ramdac = bus_space_vaddr(bt, bh);
fb_setsize(&sc->sc_sunfb, 1, CG12_WIDTH, CG12_HEIGHT,
node, 0);
sc->sc_sunfb.sf_depth = 1;
sc->sc_sunfb.sf_linebytes = sc->sc_sunfb.sf_width / 8;
sc->sc_sunfb.sf_fbsize = sc->sc_sunfb.sf_height *
sc->sc_sunfb.sf_linebytes;
sc->sc_highres = sc->sc_sunfb.sf_width == CG12_WIDTH_HR;
if (sbus_bus_map(bt, sa->sa_slot, sa->sa_offset +
(sc->sc_highres ? CG12_OFF_OVERLAY0_HR : CG12_OFF_OVERLAY0),
round_page(sc->sc_highres ? CG12_SIZE_OVERLAY_HR :
CG12_SIZE_OVERLAY), BUS_SPACE_MAP_LINEAR, 0, &bh) != 0) {
printf("%s: can't map overlay plane\n", self->dv_xname);
return;
}
sc->sc_overlay = bus_space_vaddr(bt, bh);
if (sbus_bus_map(bt, sa->sa_slot, sa->sa_offset +
(sc->sc_highres ? CG12_OFF_INTEN_HR : CG12_OFF_INTEN),
round_page(sc->sc_highres ? CG12_SIZE_COLOR24_HR :
CG12_SIZE_COLOR24), BUS_SPACE_MAP_LINEAR, 0, &bh) != 0) {
printf("%s: can't map color plane\n", self->dv_xname);
return;
}
sc->sc_inten = bus_space_vaddr(bt, bh);
sc->sc_paddr = sbus_bus_addr(bt, sa->sa_slot, sa->sa_offset +
(sc->sc_highres ? CG12_OFF_INTEN_HR : CG12_OFF_INTEN));
sc->sc_sunfb.sf_depth = 0;
cgtwelve_reset(sc, 1);
sc->sc_sunfb.sf_ro.ri_bits = (void *)sc->sc_overlay;
sc->sc_sunfb.sf_ro.ri_hw = sc;
fbwscons_init(&sc->sc_sunfb, 0, sc->sc_isconsole);
if (sc->sc_isconsole)
fbwscons_console_init(&sc->sc_sunfb, -1);
printf("%s: %dx%d", self->dv_xname,
sc->sc_sunfb.sf_width, sc->sc_sunfb.sf_height);
ps = getpropstring(node, "ucoderev");
if (*ps != '\0')
printf(", microcode rev. %s", ps);
printf("\n");
fbwscons_attach(&sc->sc_sunfb, &cgtwelve_accessops, sc->sc_isconsole);
}
int
cgtwelveactivate(struct device *self, int act)
{
struct cgtwelve_softc *sc = (struct cgtwelve_softc *)self;
int ret = 0;
switch (act) {
case DVACT_POWERDOWN:
if (sc->sc_isconsole)
cgtwelve_prom(sc);
break;
}
return (ret);
}
int
cgtwelve_ioctl(void *dev, u_long cmd, caddr_t data, int flags, struct proc *p)
{
struct cgtwelve_softc *sc = dev;
struct wsdisplay_fbinfo *wdf;
switch (cmd) {
case WSDISPLAYIO_GTYPE:
*(u_int *)data = WSDISPLAY_TYPE_SUNCG12;
break;
case WSDISPLAYIO_GINFO:
wdf = (struct wsdisplay_fbinfo *)data;
wdf->height = sc->sc_sunfb.sf_height;
wdf->width = sc->sc_sunfb.sf_width;
wdf->depth = 32;
wdf->stride = sc->sc_sunfb.sf_linebytes * 32;
wdf->offset = 0;
wdf->cmsize = 0;
break;
case WSDISPLAYIO_GETSUPPORTEDDEPTH:
*(u_int *)data = WSDISPLAYIO_DEPTH_24_32;
break;
case WSDISPLAYIO_LINEBYTES:
*(u_int *)data = sc->sc_sunfb.sf_linebytes * 32;
break;
case WSDISPLAYIO_GETCMAP:
case WSDISPLAYIO_PUTCMAP:
break;
case WSDISPLAYIO_SMODE:
if (*(int *)data == WSDISPLAYIO_MODE_EMUL) {
cgtwelve_reset(sc, 1);
} else {
cgtwelve_reset(sc, 32);
}
break;
case WSDISPLAYIO_SVIDEO:
case WSDISPLAYIO_GVIDEO:
break;
default:
return (-1);
}
return (0);
}
void
cgtwelve_reset(struct cgtwelve_softc *sc, int depth)
{
u_int32_t c;
if (sc->sc_sunfb.sf_depth != depth) {
if (depth == 1) {
sc->sc_apu->hpage = sc->sc_highres ?
CG12_HPAGE_ENABLE_HR : CG12_HPAGE_ENABLE;
sc->sc_apu->haccess = CG12_HACCESS_ENABLE;
sc->sc_dpu->pln_sl_host = CG12_PLN_SL_ENABLE;
sc->sc_dpu->pln_rd_msk_host = CG12_PLN_RD_ENABLE;
sc->sc_dpu->pln_wr_msk_host = CG12_PLN_WR_ENABLE;
memset((void *)sc->sc_overlay, 0xff, sc->sc_highres ?
CG12_SIZE_ENABLE_HR : CG12_SIZE_ENABLE);
sc->sc_apu->hpage = sc->sc_highres ?
CG12_HPAGE_OVERLAY_HR : CG12_HPAGE_OVERLAY;
sc->sc_apu->haccess = CG12_HACCESS_OVERLAY;
sc->sc_dpu->pln_sl_host = CG12_PLN_SL_OVERLAY;
sc->sc_dpu->pln_rd_msk_host = CG12_PLN_RD_OVERLAY;
sc->sc_dpu->pln_wr_msk_host = CG12_PLN_WR_OVERLAY;
cgtwelve_ramdac_wraddr(sc, 0);
sc->sc_ramdac->color = 0x00000000;
for (c = 1; c < 256; c++)
sc->sc_ramdac->color = 0x00ffffff;
} else {
sc->sc_apu->hpage = sc->sc_highres ?
CG12_HPAGE_OVERLAY_HR : CG12_HPAGE_OVERLAY;
sc->sc_apu->haccess = CG12_HACCESS_OVERLAY;
sc->sc_dpu->pln_sl_host = CG12_PLN_SL_OVERLAY;
sc->sc_dpu->pln_rd_msk_host = CG12_PLN_RD_OVERLAY;
sc->sc_dpu->pln_wr_msk_host = CG12_PLN_WR_OVERLAY;
bzero((void *)sc->sc_overlay, sc->sc_highres ?
CG12_SIZE_OVERLAY_HR : CG12_SIZE_OVERLAY);
sc->sc_apu->hpage = sc->sc_highres ?
CG12_HPAGE_ENABLE_HR : CG12_HPAGE_ENABLE;
sc->sc_apu->haccess = CG12_HACCESS_ENABLE;
sc->sc_dpu->pln_sl_host = CG12_PLN_SL_ENABLE;
sc->sc_dpu->pln_rd_msk_host = CG12_PLN_RD_ENABLE;
sc->sc_dpu->pln_wr_msk_host = CG12_PLN_WR_ENABLE;
bzero((void *)sc->sc_overlay, sc->sc_highres ?
CG12_SIZE_ENABLE_HR : CG12_SIZE_ENABLE);
sc->sc_apu->hpage = sc->sc_highres ?
CG12_HPAGE_24BIT_HR : CG12_HPAGE_24BIT;
sc->sc_apu->haccess = CG12_HACCESS_24BIT;
sc->sc_dpu->pln_sl_host = CG12_PLN_SL_24BIT;
sc->sc_dpu->pln_rd_msk_host = CG12_PLN_RD_24BIT;
sc->sc_dpu->pln_wr_msk_host = CG12_PLN_WR_24BIT;
memset((void *)sc->sc_inten, 0x00ffffff,
sc->sc_highres ?
CG12_SIZE_COLOR24_HR : CG12_SIZE_COLOR24);
cgtwelve_ramdac_wraddr(sc, 0);
for (c = 0; c < 256; c++)
sc->sc_ramdac->color = c | (c << 8) | (c << 16);
}
}
sc->sc_sunfb.sf_depth = depth;
}
paddr_t
cgtwelve_mmap(void *v, off_t offset, int prot)
{
struct cgtwelve_softc *sc = v;
if (offset & PGOFSET || offset < 0)
return (-1);
if (offset < sc->sc_sunfb.sf_fbsize * 32) {
return (bus_space_mmap(sc->sc_bustag, sc->sc_paddr, offset,
prot, BUS_SPACE_MAP_LINEAR));
}
return (-1);
}
static __inline__ void
cgtwelve_ramdac_wraddr(struct cgtwelve_softc *sc, u_int32_t addr)
{
sc->sc_ramdac->addr_lo = (addr & 0xff);
sc->sc_ramdac->addr_hi = ((addr >> 8) & 0xff);
}
void
cgtwelve_prom(struct cgtwelve_softc *sc)
{
extern struct consdev consdev_prom;
if (sc->sc_sunfb.sf_depth != 1) {
cgtwelve_reset(sc, 1);
cn_tab = &consdev_prom;
}
}