summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJason Wright <jason@cvs.openbsd.org>2002-07-26 04:24:45 +0000
committerJason Wright <jason@cvs.openbsd.org>2002-07-26 04:24:45 +0000
commit65db6d275234574e446f57317e0af2aab04d9e0e (patch)
tree96d4dcbba179fbbe30935351752a8986c725a52d
parent28de560906046cf5e208353689d62bad05c43a79 (diff)
add support for hardware accelerated fill/copy; from NetBSD (committed from the acceptably fast non-X11 console of a U1)
-rw-r--r--sys/dev/sbus/cgsix.c373
1 files changed, 367 insertions, 6 deletions
diff --git a/sys/dev/sbus/cgsix.c b/sys/dev/sbus/cgsix.c
index de4fe1a21d6..94ace48b927 100644
--- a/sys/dev/sbus/cgsix.c
+++ b/sys/dev/sbus/cgsix.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: cgsix.c,v 1.22 2002/07/25 19:04:46 miod Exp $ */
+/* $OpenBSD: cgsix.c,v 1.23 2002/07/26 04:24:44 jason Exp $ */
/*
* Copyright (c) 2001 Jason L. Wright (jason@thought.net)
@@ -78,6 +78,7 @@ union bt_cmap {
#define CGSIX_THC_OFFSET 0x301000
#define CGSIX_THC_SIZE (sizeof(u_int32_t) * 640)
#define CGSIX_FBC_OFFSET 0x700000
+#define CGSIX_FBC_SIZE 0x1000
#define CGSIX_TEC_OFFSET 0x701000
#define CGSIX_TEC_SIZE (sizeof(u_int32_t) * 3)
#define CGSIX_VID_OFFSET 0x800000
@@ -110,6 +111,93 @@ union bt_cmap {
#define FHC_TESTY_MASK 0x0000000f /* test window Y */
#define FHC_TESTY_SHIFT 0
+#define CG6_FBC_MODE 0x004 /* mode setting */
+#define CG6_FBC_CLIP 0x008 /* ??? */
+#define CG6_FBC_S 0x010 /* global status */
+#define CG6_FBC_DRAW 0x014 /* drawing pipeline status */
+#define CG6_FBC_BLIT 0x018 /* blitter status */
+#define CG6_FBC_X0 0x080 /* blitter, src llx */
+#define CG6_FBC_Y0 0x084 /* blitter, src lly */
+#define CG6_FBC_X1 0x090 /* blitter, src urx */
+#define CG6_FBC_Y1 0x094 /* blitter, src ury */
+#define CG6_FBC_X2 0x0a0 /* blitter, dst llx */
+#define CG6_FBC_Y2 0x0a4 /* blitter, dst lly */
+#define CG6_FBC_X3 0x0b0 /* blitter, dst urx */
+#define CG6_FBC_Y3 0x0b4 /* blitter, dst ury */
+#define CG6_FBC_OFFX 0x0c0 /* x offset for drawing */
+#define CG6_FBC_OFFY 0x0c4 /* y offset for drawing */
+#define CG6_FBC_CLIPMINX 0x0e0 /* clip rectangle llx */
+#define CG6_FBC_CLIPMINY 0x0e4 /* clip rectangle lly */
+#define CG6_FBC_CLIPMAXX 0x0f0 /* clip rectangle urx */
+#define CG6_FBC_CLIPMAXY 0x0f4 /* clip rectangle ury */
+#define CG6_FBC_FG 0x100 /* fg value for rop */
+#define CG6_FBC_ALU 0x108 /* operation */
+#define CG6_FBC_ARECTX 0x900 /* rectangle drawing, x coord */
+#define CG6_FBC_ARECTY 0x904 /* rectangle drawing, y coord */
+
+#define FBC_MODE_MASK ( \
+ 0x00200000 /* GX_BLIT_SRC */ \
+ | 0x00020000 /* GX_MODE_COLOR8 */ \
+ | 0x00008000 /* GX_DRAW_RENDER */ \
+ | 0x00002000 /* GX_BWRITE0_ENABLE */ \
+ | 0x00001000 /* GX_BWRITE1_DISABLE */ \
+ | 0x00000200 /* GX_BREAD_0 */ \
+ | 0x00000080 /* GX_BDISP_0 */ \
+)
+#define FBC_MODE_VAL ( \
+ 0x00300000 /* GX_BLIT_ALL */ \
+ | 0x00060000 /* GX_MODE_ALL */ \
+ | 0x00018000 /* GX_DRAW_ALL */ \
+ | 0x00006000 /* GX_BWRITE0_ALL */ \
+ | 0x00001800 /* GX_BWRITE1_ALL */ \
+ | 0x00000600 /* GX_BREAD_ALL */ \
+ | 0x00000180 /* GX_BDISP_ALL */ \
+)
+
+#define FBC_S_GXINPROGRESS 0x10000000 /* drawing in progress */
+
+#define FBC_BLIT_UNKNOWN 0x80000000 /* ??? */
+#define FBC_BLIT_GXFULL 0x20000000 /* queue is full */
+
+#define FBC_DRAW_UNKNOWN 0x80000000 /* ??? */
+#define FBC_DRAW_GXFULL 0x20000000
+
+/* Value for the alu register for screen-to-screen copies */
+#define FBC_ALU_COPY ( \
+ 0x80000000 /* GX_PLANE_ONES (ignore planemask register) */ \
+ | 0x20000000 /* GX_PIXEL_ONES (ignore pixelmask register) */ \
+ | 0x00800000 /* GX_ATTR_SUPP (function unknown) */ \
+ | 0x00000000 /* GX_RAST_BOOL (function unknown) */ \
+ | 0x00000000 /* GX_PLOT_PLOT (function unknown) */ \
+ | 0x08000000 /* GX_PATTERN_ONES (ignore pattern) */ \
+ | 0x01000000 /* GX_POLYG_OVERLAP (unsure - handle overlap?) */ \
+ | 0x0000cccc /* ALU = src */ \
+)
+
+/* Value for the alu register for region fills */
+#define FBC_ALU_FILL ( \
+ 0x80000000 /* GX_PLANE_ONES (ignore planemask register) */ \
+ | 0x20000000 /* GX_PIXEL_ONES (ignore pixelmask register) */ \
+ | 0x00800000 /* GX_ATTR_SUPP (function unknown) */ \
+ | 0x00000000 /* GX_RAST_BOOL (function unknown) */ \
+ | 0x00000000 /* GX_PLOT_PLOT (function unknown) */ \
+ | 0x08000000 /* GX_PATTERN_ONES (ignore pattern) */ \
+ | 0x01000000 /* GX_POLYG_OVERLAP (unsure - handle overlap?) */ \
+ | 0x0000ff00 /* ALU = fg color */ \
+)
+
+/* Value for the alu register for toggling an area */
+#define FBC_ALU_FLIP ( \
+ 0x80000000 /* GX_PLANE_ONES (ignore planemask register) */ \
+ | 0x20000000 /* GX_PIXEL_ONES (ignore pixelmask register) */ \
+ | 0x00800000 /* GX_ATTR_SUPP (function unknown) */ \
+ | 0x00000000 /* GX_RAST_BOOL (function unknown) */ \
+ | 0x00000000 /* GX_PLOT_PLOT (function unknown) */ \
+ | 0x08000000 /* GX_PATTERN_ONES (ignore pattern) */ \
+ | 0x01000000 /* GX_POLYG_OVERLAP (unsure - handle overlap?) */ \
+ | 0x00005555 /* ALU = ~dst */ \
+)
+
#define CG6_TEC_MV 0x0 /* matrix stuff */
#define CG6_TEC_CLIP 0x4 /* clipping stuff */
#define CG6_TEC_VDC 0x8 /* ??? */
@@ -150,6 +238,7 @@ struct cgsix_softc {
bus_space_handle_t sc_thc_regs;
bus_space_handle_t sc_tec_regs;
bus_space_handle_t sc_vid_regs;
+ bus_space_handle_t sc_fbc_regs;
struct rasops_info sc_rasops;
int sc_nscreens;
int sc_width, sc_height, sc_depth, sc_linebytes;
@@ -181,6 +270,11 @@ struct cgsix_softc {
#define FHC_WRITE(sc,v) \
bus_space_write_4((sc)->sc_bustag, (sc)->sc_fhc_regs, CG6_FHC, (v))
+#define FBC_READ(sc,r) \
+ bus_space_read_4((sc)->sc_bustag, (sc)->sc_fbc_regs, (r))
+#define FBC_WRITE(sc,r,v) \
+ bus_space_write_4((sc)->sc_bustag, (sc)->sc_fbc_regs, (r), (v))
+
#define BT_WRITE(sc, reg, val) \
bus_space_write_4((sc)->sc_bustag, (sc)->sc_bt_regs, (reg), (val))
#define BT_READ(sc, reg) \
@@ -225,6 +319,12 @@ void cgsix_hardreset(struct cgsix_softc *);
void cgsix_burner(void *, u_int, u_int);
int cgsix_intr(void *);
static int a2int(char *, int);
+void cgsix_ras_init(struct cgsix_softc *);
+void cgsix_ras_copyrows(void *, int, int, int);
+void cgsix_ras_copycols(void *, int, int, int, int);
+void cgsix_ras_erasecols(void *, int, int, int, long int);
+void cgsix_ras_eraserows(void *, int, int, long int);
+void cgsix_ras_do_cursor(struct rasops_info *);
struct wsdisplay_accessops cgsix_accessops = {
cgsix_ioctl,
@@ -280,25 +380,25 @@ cgsixattach(parent, self, aux)
}
/*
- * Map just BT, FHC, THC, and video RAM.
+ * Map just BT, FHC, FBC, THC, and video RAM.
*/
if (sbus_bus_map(sa->sa_bustag, sa->sa_reg[0].sbr_slot,
sa->sa_reg[0].sbr_offset + CGSIX_BT_OFFSET,
- CGSIX_BT_SIZE, BUS_SPACE_MAP_LINEAR, 0, &sc->sc_bt_regs) != 0) {
+ CGSIX_BT_SIZE, 0, 0, &sc->sc_bt_regs) != 0) {
printf(": cannot map bt registers\n");
goto fail_bt;
}
if (sbus_bus_map(sa->sa_bustag, sa->sa_reg[0].sbr_slot,
sa->sa_reg[0].sbr_offset + CGSIX_FHC_OFFSET,
- CGSIX_FHC_SIZE, BUS_SPACE_MAP_LINEAR, 0, &sc->sc_fhc_regs) != 0) {
+ CGSIX_FHC_SIZE, 0, 0, &sc->sc_fhc_regs) != 0) {
printf(": cannot map fhc registers\n");
goto fail_fhc;
}
if (sbus_bus_map(sa->sa_bustag, sa->sa_reg[0].sbr_slot,
sa->sa_reg[0].sbr_offset + CGSIX_THC_OFFSET,
- CGSIX_THC_SIZE, BUS_SPACE_MAP_LINEAR, 0, &sc->sc_thc_regs) != 0) {
+ CGSIX_THC_SIZE, 0, 0, &sc->sc_thc_regs) != 0) {
printf(": cannot map thc registers\n");
goto fail_thc;
}
@@ -312,11 +412,18 @@ cgsixattach(parent, self, aux)
if (sbus_bus_map(sa->sa_bustag, sa->sa_reg[0].sbr_slot,
sa->sa_reg[0].sbr_offset + CGSIX_TEC_OFFSET,
- CGSIX_TEC_SIZE, BUS_SPACE_MAP_LINEAR, 0, &sc->sc_tec_regs) != 0) {
+ CGSIX_TEC_SIZE, 0, 0, &sc->sc_tec_regs) != 0) {
printf(": cannot map tec registers\n");
goto fail_tec;
}
+ if (sbus_bus_map(sa->sa_bustag, sa->sa_reg[0].sbr_slot,
+ sa->sa_reg[0].sbr_offset + CGSIX_FBC_OFFSET,
+ CGSIX_FBC_SIZE, 0, 0, &sc->sc_fbc_regs) != 0) {
+ printf(": cannot map fbc registers\n");
+ goto fail_fbc;
+ }
+
if ((sc->sc_ih = bus_intr_establish(sa->sa_bustag, sa->sa_pri,
IPL_TTY, 0, cgsix_intr, sc)) == NULL) {
printf(": couldn't establish interrupt, pri %d\n", sa->sa_pri);
@@ -360,6 +467,15 @@ cgsixattach(parent, self, aux)
rasops_init(&sc->sc_rasops,
a2int(getpropstring(optionsnode, "screen-#rows"), 34),
a2int(getpropstring(optionsnode, "screen-#columns"), 80));
+ sc->sc_rasops.ri_hw = sc;
+ sc->sc_rasops.ri_ops.copyrows = cgsix_ras_copyrows;
+ sc->sc_rasops.ri_ops.copycols = cgsix_ras_copycols;
+ sc->sc_rasops.ri_ops.eraserows = cgsix_ras_eraserows;
+ sc->sc_rasops.ri_ops.erasecols = cgsix_ras_erasecols;
+ sc->sc_rasops.ri_do_cursor = cgsix_ras_do_cursor;
+#if 0
+ cgsix_ras_init(sc);
+#endif
cgsix_stdscreen.nrows = sc->sc_rasops.ri_rows;
cgsix_stdscreen.ncols = sc->sc_rasops.ri_cols;
@@ -402,6 +518,8 @@ cgsixattach(parent, self, aux)
return;
fail_intr:
+ bus_space_unmap(sa->sa_bustag, sc->sc_fbc_regs, CGSIX_FBC_SIZE);
+fail_fbc:
bus_space_unmap(sa->sa_bustag, sc->sc_tec_regs, CGSIX_TEC_SIZE);
fail_tec:
bus_space_unmap(sa->sa_bustag, sc->sc_vid_regs, CGSIX_VID_SIZE);
@@ -786,3 +904,246 @@ cgsix_intr(vsc)
cgsix_loadcmap_immediate(sc, 0, 256);
return (1);
}
+
+#define CG6_BLIT_WAIT(sc) \
+ while ((FBC_READ(sc, CG6_FBC_BLIT) & \
+ (FBC_BLIT_UNKNOWN|FBC_BLIT_GXFULL)) == \
+ (FBC_BLIT_UNKNOWN|FBC_BLIT_GXFULL))
+#define CG6_DRAW_WAIT(sc) \
+ while ((FBC_READ(sc, CG6_FBC_DRAW) & \
+ (FBC_DRAW_UNKNOWN|FBC_DRAW_GXFULL)) == \
+ (FBC_DRAW_UNKNOWN|FBC_DRAW_GXFULL))
+#define CG6_DRAIN(sc) \
+ while (FBC_READ(sc, CG6_FBC_S) & FBC_S_GXINPROGRESS)
+
+void
+cgsix_ras_init(sc)
+ struct cgsix_softc *sc;
+{
+ u_int32_t m;
+
+ CG6_DRAIN(sc);
+ m = FBC_READ(sc, CG6_FBC_MODE);
+ m &= ~FBC_MODE_MASK;
+ m |= FBC_MODE_VAL;
+ FBC_WRITE(sc, CG6_FBC_MODE, m);
+}
+
+void
+cgsix_ras_copyrows(cookie, src, dst, n)
+ void *cookie;
+ int src, dst, n;
+{
+ struct rasops_info *ri = cookie;
+ struct cgsix_softc *sc = ri->ri_hw;
+
+ if (dst == src)
+ return;
+ if (src < 0) {
+ n += src;
+ src = 0;
+ }
+ if (src + n > ri->ri_rows)
+ n = ri->ri_rows - src;
+ if (dst < 0) {
+ n += dst;
+ dst = 0;
+ }
+ if (dst + n > ri->ri_rows)
+ n = ri->ri_rows - dst;
+ if (n <= 0)
+ return;
+ n *= ri->ri_font->fontheight;
+ src *= ri->ri_font->fontheight;
+ dst *= ri->ri_font->fontheight;
+
+ FBC_WRITE(sc, CG6_FBC_CLIP, 0);
+ FBC_WRITE(sc, CG6_FBC_S, 0);
+ FBC_WRITE(sc, CG6_FBC_OFFX, 0);
+ FBC_WRITE(sc, CG6_FBC_OFFY, 0);
+ FBC_WRITE(sc, CG6_FBC_CLIPMINX, 0);
+ FBC_WRITE(sc, CG6_FBC_CLIPMINY, 0);
+ FBC_WRITE(sc, CG6_FBC_CLIPMAXX, ri->ri_width - 1);
+ FBC_WRITE(sc, CG6_FBC_CLIPMAXY, ri->ri_height - 1);
+ FBC_WRITE(sc, CG6_FBC_ALU, FBC_ALU_COPY);
+ FBC_WRITE(sc, CG6_FBC_X0, ri->ri_xorigin);
+ FBC_WRITE(sc, CG6_FBC_Y0, ri->ri_yorigin + src);
+ FBC_WRITE(sc, CG6_FBC_X1, ri->ri_xorigin + ri->ri_emuwidth - 1);
+ FBC_WRITE(sc, CG6_FBC_Y1, ri->ri_yorigin + src + n - 1);
+ FBC_WRITE(sc, CG6_FBC_X2, ri->ri_xorigin);
+ FBC_WRITE(sc, CG6_FBC_Y2, ri->ri_yorigin + dst);
+ FBC_WRITE(sc, CG6_FBC_X3, ri->ri_xorigin + ri->ri_emuwidth - 1);
+ FBC_WRITE(sc, CG6_FBC_Y3, ri->ri_yorigin + dst + n - 1);
+ CG6_BLIT_WAIT(sc);
+ CG6_DRAIN(sc);
+}
+
+void
+cgsix_ras_copycols(cookie, row, src, dst, n)
+ void *cookie;
+ int row, src, dst, n;
+{
+ struct rasops_info *ri = cookie;
+ struct cgsix_softc *sc = ri->ri_hw;
+
+ if (dst == src)
+ return;
+ if ((row < 0) || (row >= ri->ri_rows))
+ return;
+ if (src < 0) {
+ n += src;
+ src = 0;
+ }
+ if (src + n > ri->ri_cols)
+ n = ri->ri_cols - src;
+ if (dst < 0) {
+ n += dst;
+ dst = 0;
+ }
+ if (dst + n > ri->ri_cols)
+ n = ri->ri_cols - dst;
+ if (n <= 0)
+ return;
+ n *= ri->ri_font->fontwidth;
+ src *= ri->ri_font->fontwidth;
+ dst *= ri->ri_font->fontwidth;
+ row *= ri->ri_font->fontheight;
+
+ FBC_WRITE(sc, CG6_FBC_CLIP, 0);
+ FBC_WRITE(sc, CG6_FBC_S, 0);
+ FBC_WRITE(sc, CG6_FBC_OFFX, 0);
+ FBC_WRITE(sc, CG6_FBC_OFFY, 0);
+ FBC_WRITE(sc, CG6_FBC_CLIPMINX, 0);
+ FBC_WRITE(sc, CG6_FBC_CLIPMINY, 0);
+ FBC_WRITE(sc, CG6_FBC_CLIPMAXX, ri->ri_width - 1);
+ FBC_WRITE(sc, CG6_FBC_CLIPMAXY, ri->ri_height - 1);
+ FBC_WRITE(sc, CG6_FBC_ALU, FBC_ALU_COPY);
+ FBC_WRITE(sc, CG6_FBC_X0, ri->ri_xorigin + src);
+ FBC_WRITE(sc, CG6_FBC_Y0, ri->ri_yorigin + row);
+ FBC_WRITE(sc, CG6_FBC_X1, ri->ri_xorigin + src + n - 1);
+ FBC_WRITE(sc, CG6_FBC_Y1,
+ ri->ri_yorigin + row + ri->ri_font->fontheight - 1);
+ FBC_WRITE(sc, CG6_FBC_X2, ri->ri_xorigin + dst);
+ FBC_WRITE(sc, CG6_FBC_Y2, ri->ri_yorigin + row);
+ FBC_WRITE(sc, CG6_FBC_X3, ri->ri_xorigin + dst + n - 1);
+ FBC_WRITE(sc, CG6_FBC_Y3,
+ ri->ri_yorigin + row + ri->ri_font->fontheight - 1);
+ CG6_BLIT_WAIT(sc);
+ CG6_DRAIN(sc);
+}
+
+void
+cgsix_ras_erasecols(cookie, row, col, n, attr)
+ void *cookie;
+ int row, col, n;
+ long int attr;
+{
+ struct rasops_info *ri = cookie;
+ struct cgsix_softc *sc = ri->ri_hw;
+
+ if ((row < 0) || (row >= ri->ri_rows))
+ return;
+ if (col < 0) {
+ n += col;
+ col = 0;
+ }
+ if (col + n > ri->ri_cols)
+ n = ri->ri_cols - col;
+ if (n <= 0)
+ return;
+ n *= ri->ri_font->fontwidth;
+ col *= ri->ri_font->fontwidth;
+ row *= ri->ri_font->fontheight;
+
+ FBC_WRITE(sc, CG6_FBC_CLIP, 0);
+ FBC_WRITE(sc, CG6_FBC_S, 0);
+ FBC_WRITE(sc, CG6_FBC_OFFX, 0);
+ FBC_WRITE(sc, CG6_FBC_OFFY, 0);
+ FBC_WRITE(sc, CG6_FBC_CLIPMINX, 0);
+ FBC_WRITE(sc, CG6_FBC_CLIPMINY, 0);
+ FBC_WRITE(sc, CG6_FBC_CLIPMAXX, ri->ri_width - 1);
+ FBC_WRITE(sc, CG6_FBC_CLIPMAXY, ri->ri_height - 1);
+ FBC_WRITE(sc, CG6_FBC_ALU, FBC_ALU_FILL);
+ FBC_WRITE(sc, CG6_FBC_FG, ri->ri_devcmap[(attr >> 16) & 0xf]);
+ FBC_WRITE(sc, CG6_FBC_ARECTY, ri->ri_yorigin + row);
+ FBC_WRITE(sc, CG6_FBC_ARECTX, ri->ri_xorigin + col);
+ FBC_WRITE(sc, CG6_FBC_ARECTY,
+ ri->ri_yorigin + row + ri->ri_font->fontheight - 1);
+ FBC_WRITE(sc, CG6_FBC_ARECTX, ri->ri_xorigin + col + n - 1);
+ CG6_DRAW_WAIT(sc);
+ CG6_DRAIN(sc);
+}
+
+void
+cgsix_ras_eraserows(cookie, row, n, attr)
+ void *cookie;
+ int row, n;
+ long int attr;
+{
+ struct rasops_info *ri = cookie;
+ struct cgsix_softc *sc = ri->ri_hw;
+
+ if (row < 0) {
+ n += row;
+ row = 0;
+ }
+ if (row + n > ri->ri_rows)
+ n = ri->ri_rows - row;
+ if (n <= 0)
+ return;
+
+ FBC_WRITE(sc, CG6_FBC_CLIP, 0);
+ FBC_WRITE(sc, CG6_FBC_S, 0);
+ FBC_WRITE(sc, CG6_FBC_OFFX, 0);
+ FBC_WRITE(sc, CG6_FBC_OFFY, 0);
+ FBC_WRITE(sc, CG6_FBC_CLIPMINX, 0);
+ FBC_WRITE(sc, CG6_FBC_CLIPMINY, 0);
+ FBC_WRITE(sc, CG6_FBC_CLIPMAXX, ri->ri_width - 1);
+ FBC_WRITE(sc, CG6_FBC_CLIPMAXY, ri->ri_height - 1);
+ FBC_WRITE(sc, CG6_FBC_ALU, FBC_ALU_FILL);
+ FBC_WRITE(sc, CG6_FBC_FG, ri->ri_devcmap[(attr >> 16) & 0xf]);
+ if ((n == ri->ri_rows) && (ri->ri_flg & RI_FULLCLEAR)) {
+ FBC_WRITE(sc, CG6_FBC_ARECTY, 0);
+ FBC_WRITE(sc, CG6_FBC_ARECTX, 0);
+ FBC_WRITE(sc, CG6_FBC_ARECTY, ri->ri_height - 1);
+ FBC_WRITE(sc, CG6_FBC_ARECTX, ri->ri_width - 1);
+ } else {
+ row *= ri->ri_font->fontheight;
+ FBC_WRITE(sc, CG6_FBC_ARECTY, ri->ri_yorigin + row);
+ FBC_WRITE(sc, CG6_FBC_ARECTX, ri->ri_xorigin);
+ FBC_WRITE(sc, CG6_FBC_ARECTY,
+ ri->ri_yorigin + row + (n * ri->ri_font->fontheight) - 1);
+ FBC_WRITE(sc, CG6_FBC_ARECTX,
+ ri->ri_xorigin + ri->ri_emuwidth - 1);
+ }
+ CG6_DRAW_WAIT(sc);
+ CG6_DRAIN(sc);
+}
+
+void
+cgsix_ras_do_cursor(ri)
+ struct rasops_info *ri;
+{
+ struct cgsix_softc *sc = ri->ri_hw;
+ int row, col;
+
+ row = ri->ri_crow * ri->ri_font->fontheight;
+ col = ri->ri_ccol * ri->ri_font->fontwidth;
+ FBC_WRITE(sc, CG6_FBC_CLIP, 0);
+ FBC_WRITE(sc, CG6_FBC_S, 0);
+ FBC_WRITE(sc, CG6_FBC_OFFX, 0);
+ FBC_WRITE(sc, CG6_FBC_OFFY, 0);
+ FBC_WRITE(sc, CG6_FBC_CLIPMINX, 0);
+ FBC_WRITE(sc, CG6_FBC_CLIPMINY, 0);
+ FBC_WRITE(sc, CG6_FBC_CLIPMAXX, ri->ri_width - 1);
+ FBC_WRITE(sc, CG6_FBC_CLIPMAXY, ri->ri_height - 1);
+ FBC_WRITE(sc, CG6_FBC_ALU, FBC_ALU_FLIP);
+ FBC_WRITE(sc, CG6_FBC_ARECTY, ri->ri_yorigin + row);
+ FBC_WRITE(sc, CG6_FBC_ARECTX, ri->ri_xorigin + col);
+ FBC_WRITE(sc, CG6_FBC_ARECTY,
+ ri->ri_yorigin + row + ri->ri_font->fontheight - 1);
+ FBC_WRITE(sc, CG6_FBC_ARECTX,
+ ri->ri_xorigin + col + ri->ri_font->fontwidth - 1);
+ CG6_DRAW_WAIT(sc);
+ CG6_DRAIN(sc);
+}