summaryrefslogtreecommitdiffstats
path: root/drivers/video/fbcon-iplan2p4.c
diff options
context:
space:
mode:
authorRalf Baechle <ralf@linux-mips.org>1997-12-06 23:51:34 +0000
committerRalf Baechle <ralf@linux-mips.org>1997-12-06 23:51:34 +0000
commit230e5ab6a084ed50470f101934782dbf54b0d06b (patch)
tree5dd821c8d33f450470588e7a543f74bf74306e9e /drivers/video/fbcon-iplan2p4.c
parentc9b1c8a64c6444d189856f1e26bdcb8b4cd0113a (diff)
Merge with Linux 2.1.67.
Diffstat (limited to 'drivers/video/fbcon-iplan2p4.c')
-rw-r--r--drivers/video/fbcon-iplan2p4.c474
1 files changed, 474 insertions, 0 deletions
diff --git a/drivers/video/fbcon-iplan2p4.c b/drivers/video/fbcon-iplan2p4.c
new file mode 100644
index 000000000..693269f7b
--- /dev/null
+++ b/drivers/video/fbcon-iplan2p4.c
@@ -0,0 +1,474 @@
+/*
+ * linux/drivers/video/iplan2p4.c -- Low level frame buffer operations for
+ * interleaved bitplanes à la Atari (4
+ * planes, 2 bytes interleave)
+ *
+ * Created 5 Apr 1997 by Geert Uytterhoeven
+ *
+ * This file is subject to the terms and conditions of the GNU General Public
+ * License. See the file COPYING in the main directory of this archive for
+ * more details.
+ */
+
+#include <linux/module.h>
+#include <linux/tty.h>
+#include <linux/console.h>
+#include <linux/string.h>
+#include <linux/config.h>
+#include <linux/fb.h>
+
+#include "fbcon.h"
+
+
+#ifndef __mc68000__
+#error No support for non-m68k yet
+#endif
+
+
+ /*
+ * Prototypes
+ */
+
+static int open_iplan2p4(struct display *p);
+static void release_iplan2p4(void);
+static void bmove_iplan2p4(struct display *p, int sy, int sx, int dy, int dx,
+ int height, int width);
+static void clear_iplan2p4(struct vc_data *conp, struct display *p, int sy,
+ int sx, int height, int width);
+static void putc_iplan2p4(struct vc_data *conp, struct display *p, int c,
+ int yy, int xx);
+static void putcs_iplan2p4(struct vc_data *conp, struct display *p,
+ const char *s, int count, int yy, int xx);
+static void rev_char_iplan2p4(struct display *p, int xx, int yy);
+
+
+ /*
+ * `switch' for the low level operations
+ */
+
+static struct display_switch dispsw_iplan2p4 = {
+ open_iplan2p4, release_iplan2p4, bmove_iplan2p4, clear_iplan2p4,
+ putc_iplan2p4, putcs_iplan2p4, rev_char_iplan2p4
+};
+
+
+ /*
+ * Interleaved bitplanes à la Atari (4 planes, 2 bytes interleave)
+ */
+
+/* Increment/decrement 4 plane addresses */
+
+#define INC_4P(p) do { if (!((long)(++(p)) & 1)) (p) += 6; } while(0)
+#define DEC_4P(p) do { if ((long)(--(p)) & 1) (p) -= 6; } while(0)
+
+
+/* Sets the bytes in the visible column at d, height h, to the value
+ * val for a 4 plane screen. The the bis of the color in 'color' are
+ * moved (8 times) to the respective bytes. This means:
+ *
+ * for(h times; d += bpr)
+ * *d = (color & 1) ? 0xff : 0;
+ * *(d+2) = (color & 2) ? 0xff : 0;
+ * *(d+4) = (color & 4) ? 0xff : 0;
+ * *(d+6) = (color & 8) ? 0xff : 0;
+ */
+
+static __inline__ void memclear_4p_col(void *d, size_t h, u_long val, int bpr)
+{
+#ifdef __mc68000__
+ __asm__ __volatile__ ("1: movepl %4,%0@(0)\n\t"
+ "addal %5,%0\n\t"
+ "dbra %1,1b"
+ : "=a" (d), "=d" (h)
+ : "0" (d), "1" (h - 1), "d" (val), "r" (bpr));
+#endif /* !m68k */
+}
+
+/* Sets a 4 plane region from 'd', length 'count' bytes, to the color
+ * in val1/val2. 'd' has to be an even address and count must be divisible
+ * by 8, because only whole words and all planes are accessed. I.e.:
+ *
+ * for(count/8 times)
+ * *d = *(d+1) = (color & 1) ? 0xff : 0;
+ * *(d+2) = *(d+3) = (color & 2) ? 0xff : 0;
+ * *(d+4) = *(d+5) = (color & 4) ? 0xff : 0;
+ * *(d+6) = *(d+7) = (color & 8) ? 0xff : 0;
+ */
+
+static __inline__ void memset_even_4p(void *d, size_t count, u_long val1,
+ u_long val2)
+{
+ u_long *dd = d;
+
+ count /= 8;
+ while (count--) {
+ *dd++ = val1;
+ *dd++ = val2;
+ }
+}
+
+/* Copies a 4 plane column from 's', height 'h', to 'd'. */
+
+static __inline__ void memmove_4p_col (void *d, void *s, int h, int bpr)
+{
+ u_char *dd = d, *ss = s;
+
+ while (h--) {
+ dd[0] = ss[0];
+ dd[2] = ss[2];
+ dd[4] = ss[4];
+ dd[6] = ss[6];
+ dd += bpr;
+ ss += bpr;
+ }
+}
+
+
+/* This expands a 4 bit color into a long for movepl (4 plane) operations. */
+
+static __inline__ u_long expand4l(u_char c)
+{
+ u_long rv;
+
+#ifdef __mc68000__
+ __asm__ __volatile__ ("lsrb #1,%2\n\t"
+ "scs %0\n\t"
+ "lsll #8,%0\n\t"
+ "lsrb #1,%2\n\t"
+ "scs %0\n\t"
+ "lsll #8,%0\n\t"
+ "lsrb #1,%2\n\t"
+ "scs %0\n\t"
+ "lsll #8,%0\n\t"
+ "lsrb #1,%2\n\t"
+ "scs %0\n\t"
+ : "=&d" (rv), "=d" (c)
+ : "1" (c));
+#endif /* !m68k */
+ return(rv);
+}
+
+/* This expands a 4 bit color into two longs for two movel operations
+ * (4 planes).
+ */
+
+static __inline__ void expand4dl(u_char c, u_long *ret1, u_long *ret2)
+{
+ u_long rv1, rv2;
+
+#ifdef __mc68000__
+ __asm__ __volatile__ ("lsrb #1,%3\n\t"
+ "scs %0\n\t"
+ "extw %0\n\t"
+ "swap %0\n\t"
+ "lsrb #1,%3\n\t"
+ "scs %0\n\t"
+ "extw %0\n\t"
+ "lsrb #1,%3\n\t"
+ "scs %1\n\t"
+ "extw %1\n\t"
+ "swap %1\n\t"
+ "lsrb #1,%3\n\t"
+ "scs %1\n\t"
+ "extw %1"
+ : "=&d" (rv1), "=&d" (rv2), "=d" (c)
+ : "2" (c));
+#endif /* !m68k */
+ *ret1 = rv1;
+ *ret2 = rv2;
+}
+
+
+/* This duplicates a byte 4 times into a long. */
+
+static __inline__ u_long dup4l(u_char c)
+{
+ ushort tmp;
+ ulong rv;
+
+#ifdef __mc68000__
+ __asm__ __volatile__ ("moveb %2,%0\n\t"
+ "lslw #8,%0\n\t"
+ "moveb %2,%0\n\t"
+ "movew %0,%1\n\t"
+ "swap %0\n\t"
+ "movew %1,%0"
+ : "=&d" (rv), "=d" (tmp)
+ : "d" (c));
+#endif /* !m68k */
+ return(rv);
+}
+
+
+static int open_iplan2p4(struct display *p)
+{
+ if (p->type != FB_TYPE_INTERLEAVED_PLANES || p->type_aux != 2 ||
+ p->var.bits_per_pixel != 4)
+ return -EINVAL;
+
+ p->next_line = p->var.xres_virtual>>1;
+ p->next_plane = 2;
+ MOD_INC_USE_COUNT;
+ return 0;
+}
+
+static void release_iplan2p4(void)
+{
+ MOD_DEC_USE_COUNT;
+}
+
+static void bmove_iplan2p4(struct display *p, int sy, int sx, int dy, int dx,
+ int height, int width)
+{
+ /* bmove() has to distinguish two major cases: If both, source and
+ * destination, start at even addresses or both are at odd
+ * addresses, just the first odd and last even column (if present)
+ * require special treatment (memmove_col()). The rest between
+ * then can be copied by normal operations, because all adjacent
+ * bytes are affected and are to be stored in the same order.
+ * The pathological case is when the move should go from an odd
+ * address to an even or vice versa. Since the bytes in the plane
+ * words must be assembled in new order, it seems wisest to make
+ * all movements by memmove_col().
+ */
+
+ if (sx == 0 && dx == 0 && width == p->next_line/4) {
+ /* Special (but often used) case: Moving whole lines can be
+ *done with memmove()
+ */
+ mymemmove(p->screen_base + dy * p->next_line * p->fontheight,
+ p->screen_base + sy * p->next_line * p->fontheight,
+ p->next_line * height * p->fontheight);
+ } else {
+ int rows, cols;
+ u_char *src;
+ u_char *dst;
+ int bytes = p->next_line;
+ int linesize = bytes * p->fontheight;
+ u_int colsize = height * p->fontheight;
+ u_int upwards = (dy < sy) || (dy == sy && dx < sx);
+
+ if ((sx & 1) == (dx & 1)) {
+ /* odd->odd or even->even */
+
+ if (upwards) {
+ src = p->screen_base + sy * linesize + (sx>>1)*8 + (sx & 1);
+ dst = p->screen_base + dy * linesize + (dx>>1)*8 + (dx & 1);
+ if (sx & 1) {
+ memmove_4p_col(dst, src, colsize, bytes);
+ src += 7;
+ dst += 7;
+ --width;
+ }
+ if (width > 1) {
+ for(rows = colsize; rows > 0; --rows) {
+ mymemmove(dst, src, (width>>1)*8);
+ src += bytes;
+ dst += bytes;
+ }
+ }
+ if (width & 1) {
+ src -= colsize * bytes;
+ dst -= colsize * bytes;
+ memmove_4p_col(dst + (width>>1)*8, src + (width>>1)*8,
+ colsize, bytes);
+ }
+ } else {
+ if (!((sx+width-1) & 1)) {
+ src = p->screen_base + sy * linesize + ((sx+width-1)>>1)*8;
+ dst = p->screen_base + dy * linesize + ((dx+width-1)>>1)*8;
+ memmove_4p_col(dst, src, colsize, bytes);
+ --width;
+ }
+ src = p->screen_base + sy * linesize + (sx>>1)*8 + (sx & 1);
+ dst = p->screen_base + dy * linesize + (dx>>1)*8 + (dx & 1);
+ if (width > 1) {
+ src += colsize * bytes + (sx & 1)*7;
+ dst += colsize * bytes + (sx & 1)*7;
+ for(rows = colsize; rows > 0; --rows) {
+ src -= bytes;
+ dst -= bytes;
+ mymemmove(dst, src, (width>>1)*8);
+ }
+ }
+ if (width & 1) {
+ memmove_4p_col(dst-7, src-7, colsize, bytes);
+ }
+ }
+ } else {
+ /* odd->even or even->odd */
+
+ if (upwards) {
+ src = p->screen_base + sy * linesize + (sx>>1)*8 + (sx & 1);
+ dst = p->screen_base + dy * linesize + (dx>>1)*8 + (dx & 1);
+ for(cols = width; cols > 0; --cols) {
+ memmove_4p_col(dst, src, colsize, bytes);
+ INC_4P(src);
+ INC_4P(dst);
+ }
+ } else {
+ sx += width-1;
+ dx += width-1;
+ src = p->screen_base + sy * linesize + (sx>>1)*8 + (sx & 1);
+ dst = p->screen_base + dy * linesize + (dx>>1)*8 + (dx & 1);
+ for(cols = width; cols > 0; --cols) {
+ memmove_4p_col(dst, src, colsize, bytes);
+ DEC_4P(src);
+ DEC_4P(dst);
+ }
+ }
+ }
+ }
+}
+
+static void clear_iplan2p4(struct vc_data *conp, struct display *p, int sy,
+ int sx, int height, int width)
+{
+ ulong offset;
+ u_char *start;
+ int rows;
+ int bytes = p->next_line;
+ int lines = height * p->fontheight;
+ ulong size;
+ u_long cval1, cval2, pcval;
+
+ expand4dl(attr_bgcol_ec(p,conp), &cval1, &cval2);
+
+ if (sx == 0 && width == bytes/4) {
+ offset = sy * bytes * p->fontheight;
+ size = lines * bytes;
+ memset_even_4p(p->screen_base+offset, size, cval1, cval2);
+ } else {
+ offset = (sy * bytes * p->fontheight) + (sx>>1)*8 + (sx & 1);
+ start = p->screen_base + offset;
+ pcval = expand4l(attr_bgcol_ec(p,conp));
+
+ /* Clears are split if the region starts at an odd column or
+ * end at an even column. These extra columns are spread
+ * across the interleaved planes. All in between can be
+ * cleared by normal mymemclear_small(), because both bytes of
+ * the single plane words are affected.
+ */
+
+ if (sx & 1) {
+ memclear_4p_col(start, lines, pcval, bytes);
+ start += 7;
+ width--;
+ }
+ if (width & 1) {
+ memclear_4p_col(start + (width>>1)*8, lines, pcval, bytes);
+ width--;
+ }
+ if (width) {
+ for(rows = lines; rows-- ; start += bytes)
+ memset_even_4p(start, width*4, cval1, cval2);
+ }
+ }
+}
+
+static void putc_iplan2p4(struct vc_data *conp, struct display *p, int c,
+ int yy, int xx)
+{
+ u_char *dest;
+ u_char *cdat;
+ int rows;
+ int bytes = p->next_line;
+ ulong eorx, fgx, bgx, fdx;
+
+ c &= 0xff;
+
+ dest = p->screen_base + yy * p->fontheight * bytes + (xx>>1)*8 + (xx & 1);
+ cdat = p->fontdata + (c * p->fontheight);
+
+ fgx = expand4l(attr_fgcol(p,conp));
+ bgx = expand4l(attr_bgcol(p,conp));
+ eorx = fgx ^ bgx;
+
+ for(rows = p->fontheight ; rows-- ; dest += bytes) {
+ fdx = dup4l(*cdat++);
+#ifdef __mc68000__
+ __asm__ __volatile__ ("movepl %1,%0@(0)"
+ : /* no outputs */
+ : "a" (dest), "d" ((fdx & eorx) ^ bgx));
+#endif /* !m68k */
+ }
+}
+
+static void putcs_iplan2p4(struct vc_data *conp, struct display *p,
+ const char *s, int count, int yy, int xx)
+{
+ u_char *dest, *dest0;
+ u_char *cdat, c;
+ int rows;
+ int bytes;
+ ulong eorx, fgx, bgx, fdx;
+
+ bytes = p->next_line;
+ dest0 = p->screen_base + yy * p->fontheight * bytes + (xx>>1)*8 + (xx & 1);
+ fgx = expand4l(attr_fgcol(p,conp));
+ bgx = expand4l(attr_bgcol(p,conp));
+ eorx = fgx ^ bgx;
+
+ while (count--) {
+ /* I think, unrolling the loops like in the 1 plane case isn't
+ * practicable here, because the body is much longer for 4
+ * planes (mostly the dup4l()). I guess, unrolling this would
+ * need more than 256 bytes and so exceed the instruction
+ * cache :-(
+ */
+
+ c = *s++;
+ cdat = p->fontdata + (c * p->fontheight);
+
+ for(rows = p->fontheight, dest = dest0; rows-- ; dest += bytes) {
+ fdx = dup4l(*cdat++);
+#ifdef __mc68000__
+ __asm__ __volatile__ ("movepl %1,%0@(0)"
+ : /* no outputs */
+ : "a" (dest), "d" ((fdx & eorx) ^ bgx));
+#endif /* !m68k */
+ }
+ INC_4P(dest0);
+ }
+}
+
+static void rev_char_iplan2p4(struct display *p, int xx, int yy)
+{
+ u_char *dest;
+ int j;
+ int bytes;
+
+ dest = p->screen_base + yy * p->fontheight * p->next_line + (xx>>1)*8 +
+ (xx & 1);
+ j = p->fontheight;
+ bytes = p->next_line;
+
+ while (j--) {
+ /* This should really obey the individual character's
+ * background and foreground colors instead of simply
+ * inverting.
+ */
+ dest[0] = ~dest[0];
+ dest[2] = ~dest[2];
+ dest[4] = ~dest[4];
+ dest[6] = ~dest[6];
+ dest += bytes;
+ }
+}
+
+
+#ifdef MODULE
+int init_module(void)
+#else
+int fbcon_init_iplan2p4(void)
+#endif
+{
+ return(fbcon_register_driver(&dispsw_iplan2p4, 0));
+}
+
+#ifdef MODULE
+void cleanup_module(void)
+{
+ fbcon_unregister_driver(&dispsw_iplan2p4);
+}
+#endif /* MODULE */