Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/dev/rasops Add width-optimized putchar functions for 32-...
details: https://anonhg.NetBSD.org/src/rev/e523b0aaa9d7
branches: trunk
changeset: 462136:e523b0aaa9d7
user: rin <rin%NetBSD.org@localhost>
date: Thu Jul 25 15:12:47 2019 +0000
description:
Add width-optimized putchar functions for 32-bpp, that work fine on
both little- and big-endian machines.
WIP genfb(4) driver for mac68k becomes 1.5 times faster!
diffstat:
sys/dev/rasops/rasops32.c | 477 +++++++++++++++++++++++++++++++++++++++++++++-
1 files changed, 474 insertions(+), 3 deletions(-)
diffs (truncated from 510 to 300 lines):
diff -r 2c9d98aa3a4b -r e523b0aaa9d7 sys/dev/rasops/rasops32.c
--- a/sys/dev/rasops/rasops32.c Thu Jul 25 15:06:07 2019 +0000
+++ b/sys/dev/rasops/rasops32.c Thu Jul 25 15:12:47 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: rasops32.c,v 1.35 2019/07/25 03:02:44 rin Exp $ */
+/* $NetBSD: rasops32.c,v 1.36 2019/07/25 15:12:47 rin Exp $ */
/*-
* Copyright (c) 1999 The NetBSD Foundation, Inc.
@@ -30,7 +30,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: rasops32.c,v 1.35 2019/07/25 03:02:44 rin Exp $");
+__KERNEL_RCSID(0, "$NetBSD: rasops32.c,v 1.36 2019/07/25 15:12:47 rin Exp $");
#include "opt_rasops.h"
@@ -44,6 +44,30 @@
static void rasops32_putchar(void *, int, int, u_int, long attr);
static void rasops32_putchar_aa(void *, int, int, u_int, long attr);
+#ifndef RASOPS_SMALL
+static void rasops32_putchar8(void *, int, int, u_int, long);
+static void rasops32_putchar12(void *, int, int, u_int, long);
+static void rasops32_putchar16(void *, int, int, u_int, long);
+static void rasops32_makestamp(struct rasops_info *, long);
+
+/*
+ * 4x1 stamp for optimized character blitting
+ */
+static uint32_t stamp[64];
+static long stamp_attr;
+static int stamp_mutex; /* XXX see note in readme */
+#endif
+
+/*
+ * offset = STAMP_SHIFT(fontbits, nibble #) & STAMP_MASK
+ * destination uint32_t[0] = STAMP_READ(offset)
+ * destination uint32_t[1] = STAMP_READ(offset + 4)
+ * destination uint32_t[2] = STAMP_READ(offset + 8)
+ * destination uint32_t[3] = STAMP_READ(offset + 12)
+ */
+#define STAMP_SHIFT(fb, n) ((n) ? (fb) : (fb) << 4)
+#define STAMP_MASK (0xf << 4)
+#define STAMP_READ(o) (*(uint32_t *)((char *)stamp + (o)))
/*
* Initialize a 'rasops_info' descriptor for this depth.
@@ -63,8 +87,25 @@
if (FONT_IS_ALPHA(ri->ri_font)) {
ri->ri_ops.putchar = rasops32_putchar_aa;
- } else
+ return;
+ }
+
+ switch (ri->ri_font->fontwidth) {
+#ifndef RASOPS_SMALL
+ case 8:
+ ri->ri_ops.putchar = rasops32_putchar8;
+ break;
+ case 12:
+ ri->ri_ops.putchar = rasops32_putchar12;
+ break;
+ case 16:
+ ri->ri_ops.putchar = rasops32_putchar16;
+ break;
+#endif
+ default:
ri->ri_ops.putchar = rasops32_putchar;
+ break;
+ }
}
/*
@@ -243,3 +284,433 @@
*rp++ = clr[1];
}
}
+
+#ifndef RASOPS_SMALL
+/*
+ * Recompute the blitting stamp.
+ */
+static void
+rasops32_makestamp(struct rasops_info *ri, long attr)
+{
+ uint32_t fg, bg;
+ int i;
+
+ fg = ri->ri_devcmap[((uint32_t)attr >> 24) & 0xf];
+ bg = ri->ri_devcmap[((uint32_t)attr >> 16) & 0xf];
+ stamp_attr = attr;
+
+ for (i = 0; i < 64; i += 4) {
+ stamp[i + 0] = (i & 32 ? fg : bg);
+ stamp[i + 1] = (i & 16 ? fg : bg);
+ stamp[i + 2] = (i & 8 ? fg : bg);
+ stamp[i + 3] = (i & 4 ? fg : bg);
+ }
+}
+
+/*
+ * Put a single character. This is for 8-pixel wide fonts.
+ */
+static void
+rasops32_putchar8(void *cookie, int row, int col, u_int uc, long attr)
+{
+ struct rasops_info *ri = (struct rasops_info *)cookie;
+ struct wsdisplay_font *font = PICK_FONT(ri, uc);
+ int height, so, fs;
+ uint32_t *rp, *hrp = NULL;
+ uint8_t *fr;
+
+ hrp = NULL; /* XXX GCC */
+
+#ifdef RASOPS_CLIPPING
+ /* Catches 'row < 0' case too */
+ if ((unsigned)row >= (unsigned)ri->ri_rows)
+ return;
+
+ if ((unsigned)col >= (unsigned)ri->ri_cols)
+ return;
+#endif
+
+ /* check if character fits into font limits */
+ if (!CHAR_IN_FONT(uc, font))
+ return;
+
+ /* Can't risk remaking the stamp if it's already in use */
+ if (stamp_mutex++) {
+ stamp_mutex--;
+ rasops32_putchar(cookie, row, col, uc, attr);
+ return;
+ }
+
+ /* Recompute stamp? */
+ if (attr != stamp_attr)
+ rasops32_makestamp(ri, attr);
+
+ rp = (uint32_t *)(ri->ri_bits + row*ri->ri_yscale + col*ri->ri_xscale);
+ if (ri->ri_hwbits)
+ hrp = (uint32_t *)(ri->ri_hwbits + row*ri->ri_yscale +
+ col*ri->ri_xscale);
+
+ height = font->fontheight;
+
+ if (uc == ' ') {
+ while (height--) {
+ rp[0] = rp[1] = rp[2] = rp[3] =
+ rp[4] = rp[5] = rp[6] = rp[7] = stamp[0];
+ DELTA(rp, ri->ri_stride, uint32_t *);
+ if (ri->ri_hwbits) {
+ hrp[0] = hrp[1] = hrp[2] = hrp[3] =
+ hrp[4] = hrp[5] = hrp[6] = hrp[7] = stamp[0];
+ DELTA(hrp, ri->ri_stride, uint32_t *);
+ }
+ }
+ } else {
+ fr = FONT_GLYPH(uc, font, ri);
+ fs = font->stride;
+
+ while (height--) {
+ uint32_t tmp0, tmp1, tmp2, tmp3;
+
+ so = STAMP_SHIFT(fr[0], 1) & STAMP_MASK;
+ tmp0 = STAMP_READ(so);
+ tmp1 = STAMP_READ(so + 4);
+ tmp2 = STAMP_READ(so + 8);
+ tmp3 = STAMP_READ(so + 12);
+ rp[0] = tmp0;
+ rp[1] = tmp1;
+ rp[2] = tmp2;
+ rp[3] = tmp3;
+ if (ri->ri_hwbits) {
+ hrp[0] = tmp0;
+ hrp[1] = tmp1;
+ hrp[2] = tmp2;
+ hrp[3] = tmp3;
+ }
+
+ so = STAMP_SHIFT(fr[0], 0) & STAMP_MASK;
+ tmp0 = STAMP_READ(so);
+ tmp1 = STAMP_READ(so + 4);
+ tmp2 = STAMP_READ(so + 8);
+ tmp3 = STAMP_READ(so + 12);
+ rp[4] = tmp0;
+ rp[5] = tmp1;
+ rp[6] = tmp2;
+ rp[7] = tmp3;
+ if (ri->ri_hwbits) {
+ hrp[4] = tmp0;
+ hrp[5] = tmp1;
+ hrp[6] = tmp2;
+ hrp[7] = tmp3;
+ }
+
+ fr += fs;
+ DELTA(rp, ri->ri_stride, uint32_t *);
+ if (ri->ri_hwbits)
+ DELTA(hrp, ri->ri_stride, uint32_t *);
+ }
+ }
+
+ /* Do underline */
+ if ((attr & WSATTR_UNDERLINE) != 0) {
+ DELTA(rp, -(ri->ri_stride << 1), uint32_t *);
+ rp[0] = rp[1] = rp[2] = rp[3] =
+ rp[4] = rp[5] = rp[6] = rp[7] = stamp[60];
+ if (ri->ri_hwbits)
+ hrp[0] = hrp[1] = hrp[2] = hrp[3] =
+ hrp[4] = hrp[5] = hrp[6] = hrp[7] = stamp[60];
+ }
+
+ stamp_mutex--;
+}
+
+/*
+ * Put a single character. This is for 12-pixel wide fonts.
+ */
+static void
+rasops32_putchar12(void *cookie, int row, int col, u_int uc, long attr)
+{
+ struct rasops_info *ri = (struct rasops_info *)cookie;
+ struct wsdisplay_font *font = PICK_FONT(ri, uc);
+ int height, so, fs;
+ uint32_t *rp, *hrp = NULL;
+ uint8_t *fr;
+
+ hrp = NULL; /* XXX GCC */
+
+#ifdef RASOPS_CLIPPING
+ /* Catches 'row < 0' case too */
+ if ((unsigned)row >= (unsigned)ri->ri_rows)
+ return;
+
+ if ((unsigned)col >= (unsigned)ri->ri_cols)
+ return;
+#endif
+
+ /* check if character fits into font limits */
+ if (!CHAR_IN_FONT(uc, font))
+ return;
+
+ /* Can't risk remaking the stamp if it's already in use */
+ if (stamp_mutex++) {
+ stamp_mutex--;
+ rasops32_putchar(cookie, row, col, uc, attr);
+ return;
+ }
+
+ /* Recompute stamp? */
+ if (attr != stamp_attr)
+ rasops32_makestamp(ri, attr);
+
+ rp = (uint32_t *)(ri->ri_bits + row*ri->ri_yscale + col*ri->ri_xscale);
+ if (ri->ri_hwbits)
+ hrp = (uint32_t *)(ri->ri_hwbits + row*ri->ri_yscale +
+ col*ri->ri_xscale);
+
+ height = font->fontheight;
+
+ if (uc == ' ') {
+ while (height--) {
+ rp[0] = rp[1] = rp[2] = rp[3] =
+ rp[4] = rp[5] = rp[6] = rp[7] =
+ rp[8] = rp[9] = rp[10] = rp[11] = stamp[0];
+ DELTA(rp, ri->ri_stride, uint32_t *);
+ if (ri->ri_hwbits) {
+ hrp[0] = hrp[1] = hrp[2] = hrp[3] =
+ hrp[4] = hrp[5] = hrp[6] = hrp[7] =
+ hrp[8] = hrp[9] = hrp[10] = hrp[11] =
+ stamp[0];
+ DELTA(hrp, ri->ri_stride, uint32_t *);
+ }
+ }
+ } else {
+ fr = FONT_GLYPH(uc, font, ri);
+ fs = font->stride;
+
+ while (height--) {
+ uint32_t tmp0, tmp1, tmp2, tmp3;
+
+ so = STAMP_SHIFT(fr[0], 1) & STAMP_MASK;
+ tmp0 = STAMP_READ(so);
+ tmp1 = STAMP_READ(so + 4);
+ tmp2 = STAMP_READ(so + 8);
+ tmp3 = STAMP_READ(so + 12);
+ rp[0] = tmp0;
+ rp[1] = tmp1;
+ rp[2] = tmp2;
+ rp[3] = tmp3;
+ if (ri->ri_hwbits) {
+ hrp[0] = tmp0;
+ hrp[1] = tmp1;
+ hrp[2] = tmp2;
+ hrp[3] = tmp3;
+ }
+
Home |
Main Index |
Thread Index |
Old Index