1f7018c21STomi Valkeinen /*
2f7018c21STomi Valkeinen  *  linux/drivers/video/iplan2p2.c -- Low level frame buffer operations for
3f7018c21STomi Valkeinen  *				      interleaved bitplanes à la Atari (2
4f7018c21STomi Valkeinen  *				      planes, 2 bytes interleave)
5f7018c21STomi Valkeinen  *
6f7018c21STomi Valkeinen  *	Created 5 Apr 1997 by Geert Uytterhoeven
7f7018c21STomi Valkeinen  *
8f7018c21STomi Valkeinen  *  This file is subject to the terms and conditions of the GNU General Public
9f7018c21STomi Valkeinen  *  License.  See the file COPYING in the main directory of this archive for
10f7018c21STomi Valkeinen  *  more details.
11f7018c21STomi Valkeinen  */
12f7018c21STomi Valkeinen 
13f7018c21STomi Valkeinen #include <linux/string.h>
14f7018c21STomi Valkeinen #include <linux/fb.h>
15f7018c21STomi Valkeinen 
16f7018c21STomi Valkeinen #include <asm/setup.h>
17f7018c21STomi Valkeinen 
18f7018c21STomi Valkeinen #include "atafb.h"
19f7018c21STomi Valkeinen 
20f7018c21STomi Valkeinen #define BPL	2
21f7018c21STomi Valkeinen #include "atafb_utils.h"
22f7018c21STomi Valkeinen 
atafb_iplan2p2_copyarea(struct fb_info * info,u_long next_line,int sy,int sx,int dy,int dx,int height,int width)23f7018c21STomi Valkeinen void atafb_iplan2p2_copyarea(struct fb_info *info, u_long next_line,
24f7018c21STomi Valkeinen 			     int sy, int sx, int dy, int dx,
25f7018c21STomi Valkeinen 			     int height, int width)
26f7018c21STomi Valkeinen {
27f7018c21STomi Valkeinen 	/*  bmove() has to distinguish two major cases: If both, source and
28f7018c21STomi Valkeinen 	 *  destination, start at even addresses or both are at odd
29f7018c21STomi Valkeinen 	 *  addresses, just the first odd and last even column (if present)
30f7018c21STomi Valkeinen 	 *  require special treatment (memmove_col()). The rest between
31f7018c21STomi Valkeinen 	 *  then can be copied by normal operations, because all adjacent
32f7018c21STomi Valkeinen 	 *  bytes are affected and are to be stored in the same order.
33f7018c21STomi Valkeinen 	 *    The pathological case is when the move should go from an odd
34f7018c21STomi Valkeinen 	 *  address to an even or vice versa. Since the bytes in the plane
35f7018c21STomi Valkeinen 	 *  words must be assembled in new order, it seems wisest to make
36f7018c21STomi Valkeinen 	 *  all movements by memmove_col().
37f7018c21STomi Valkeinen 	 */
38f7018c21STomi Valkeinen 
39f7018c21STomi Valkeinen 	u8 *src, *dst;
40f7018c21STomi Valkeinen 	u32 *s, *d;
41f7018c21STomi Valkeinen 	int w, l , i, j;
42f7018c21STomi Valkeinen 	u_int colsize;
43f7018c21STomi Valkeinen 	u_int upwards = (dy < sy) || (dy == sy && dx < sx);
44f7018c21STomi Valkeinen 
45f7018c21STomi Valkeinen 	colsize = height;
46f7018c21STomi Valkeinen 	if (!((sx ^ dx) & 15)) {
47f7018c21STomi Valkeinen 		/* odd->odd or even->even */
48f7018c21STomi Valkeinen 
49f7018c21STomi Valkeinen 		if (upwards) {
50f7018c21STomi Valkeinen 			src = (u8 *)info->screen_base + sy * next_line + (sx & ~15) / (8 / BPL);
51f7018c21STomi Valkeinen 			dst = (u8 *)info->screen_base + dy * next_line + (dx & ~15) / (8 / BPL);
52f7018c21STomi Valkeinen 			if (sx & 15) {
53f7018c21STomi Valkeinen 				memmove32_col(dst, src, 0xff00ff, height, next_line - BPL * 2);
54f7018c21STomi Valkeinen 				src += BPL * 2;
55f7018c21STomi Valkeinen 				dst += BPL * 2;
56f7018c21STomi Valkeinen 				width -= 8;
57f7018c21STomi Valkeinen 			}
58f7018c21STomi Valkeinen 			w = width >> 4;
59f7018c21STomi Valkeinen 			if (w) {
60f7018c21STomi Valkeinen 				s = (u32 *)src;
61f7018c21STomi Valkeinen 				d = (u32 *)dst;
62f7018c21STomi Valkeinen 				w *= BPL / 2;
63f7018c21STomi Valkeinen 				l = next_line - w * 4;
64f7018c21STomi Valkeinen 				for (j = height; j > 0; j--) {
65f7018c21STomi Valkeinen 					for (i = w; i > 0; i--)
66f7018c21STomi Valkeinen 						*d++ = *s++;
67f7018c21STomi Valkeinen 					s = (u32 *)((u8 *)s + l);
68f7018c21STomi Valkeinen 					d = (u32 *)((u8 *)d + l);
69f7018c21STomi Valkeinen 				}
70f7018c21STomi Valkeinen 			}
71f7018c21STomi Valkeinen 			if (width & 15)
72f7018c21STomi Valkeinen 				memmove32_col(dst + width / (8 / BPL), src + width / (8 / BPL),
73f7018c21STomi Valkeinen 					      0xff00ff00, height, next_line - BPL * 2);
74f7018c21STomi Valkeinen 		} else {
75f7018c21STomi Valkeinen 			src = (u8 *)info->screen_base + (sy - 1) * next_line + ((sx + width + 8) & ~15) / (8 / BPL);
76f7018c21STomi Valkeinen 			dst = (u8 *)info->screen_base + (dy - 1) * next_line + ((dx + width + 8) & ~15) / (8 / BPL);
77f7018c21STomi Valkeinen 
78f7018c21STomi Valkeinen 			if ((sx + width) & 15) {
79f7018c21STomi Valkeinen 				src -= BPL * 2;
80f7018c21STomi Valkeinen 				dst -= BPL * 2;
81f7018c21STomi Valkeinen 				memmove32_col(dst, src, 0xff00ff00, colsize, -next_line - BPL * 2);
82f7018c21STomi Valkeinen 				width -= 8;
83f7018c21STomi Valkeinen 			}
84f7018c21STomi Valkeinen 			w = width >> 4;
85f7018c21STomi Valkeinen 			if (w) {
86f7018c21STomi Valkeinen 				s = (u32 *)src;
87f7018c21STomi Valkeinen 				d = (u32 *)dst;
88f7018c21STomi Valkeinen 				w *= BPL / 2;
89f7018c21STomi Valkeinen 				l = next_line - w * 4;
90f7018c21STomi Valkeinen 				for (j = height; j > 0; j--) {
91f7018c21STomi Valkeinen 					for (i = w; i > 0; i--)
92f7018c21STomi Valkeinen 						*--d = *--s;
93f7018c21STomi Valkeinen 					s = (u32 *)((u8 *)s - l);
94f7018c21STomi Valkeinen 					d = (u32 *)((u8 *)d - l);
95f7018c21STomi Valkeinen 				}
96f7018c21STomi Valkeinen 			}
97f7018c21STomi Valkeinen 			if (sx & 15)
98f7018c21STomi Valkeinen 				memmove32_col(dst - (width - 16) / (8 / BPL),
99f7018c21STomi Valkeinen 					      src - (width - 16) / (8 / BPL),
100f7018c21STomi Valkeinen 					      0xff00ff, colsize, -next_line - BPL * 2);
101f7018c21STomi Valkeinen 		}
102f7018c21STomi Valkeinen 	} else {
103f7018c21STomi Valkeinen 		/* odd->even or even->odd */
104f7018c21STomi Valkeinen 		if (upwards) {
105f7018c21STomi Valkeinen 			u32 *src32, *dst32;
106f7018c21STomi Valkeinen 			u32 pval[4], v, v1, mask;
107f7018c21STomi Valkeinen 			int i, j, w, f;
108f7018c21STomi Valkeinen 
109f7018c21STomi Valkeinen 			src = (u8 *)info->screen_base + sy * next_line + (sx & ~15) / (8 / BPL);
110f7018c21STomi Valkeinen 			dst = (u8 *)info->screen_base + dy * next_line + (dx & ~15) / (8 / BPL);
111f7018c21STomi Valkeinen 
112f7018c21STomi Valkeinen 			mask = 0xff00ff00;
113f7018c21STomi Valkeinen 			f = 0;
114f7018c21STomi Valkeinen 			w = width;
115f7018c21STomi Valkeinen 			if (sx & 15) {
116f7018c21STomi Valkeinen 				f = 1;
117f7018c21STomi Valkeinen 				w += 8;
118f7018c21STomi Valkeinen 			}
119f7018c21STomi Valkeinen 			if ((sx + width) & 15)
120f7018c21STomi Valkeinen 				f |= 2;
121f7018c21STomi Valkeinen 			w >>= 4;
122f7018c21STomi Valkeinen 			for (i = height; i; i--) {
123f7018c21STomi Valkeinen 				src32 = (u32 *)src;
124f7018c21STomi Valkeinen 				dst32 = (u32 *)dst;
125f7018c21STomi Valkeinen 
126f7018c21STomi Valkeinen 				if (f & 1) {
127f7018c21STomi Valkeinen 					pval[0] = (*src32++ << 8) & mask;
128f7018c21STomi Valkeinen 				} else {
129f7018c21STomi Valkeinen 					pval[0] = dst32[0] & mask;
130f7018c21STomi Valkeinen 				}
131f7018c21STomi Valkeinen 
132f7018c21STomi Valkeinen 				for (j = w; j > 0; j--) {
133f7018c21STomi Valkeinen 					v = *src32++;
134f7018c21STomi Valkeinen 					v1 = v & mask;
135f7018c21STomi Valkeinen 					*dst32++ = pval[0] | (v1 >> 8);
136f7018c21STomi Valkeinen 					pval[0] = (v ^ v1) << 8;
137f7018c21STomi Valkeinen 				}
138f7018c21STomi Valkeinen 
139f7018c21STomi Valkeinen 				if (f & 2) {
140f7018c21STomi Valkeinen 					dst32[0] = (dst32[0] & mask) | pval[0];
141f7018c21STomi Valkeinen 				}
142f7018c21STomi Valkeinen 
143f7018c21STomi Valkeinen 				src += next_line;
144f7018c21STomi Valkeinen 				dst += next_line;
145f7018c21STomi Valkeinen 			}
146f7018c21STomi Valkeinen 		} else {
147f7018c21STomi Valkeinen 			u32 *src32, *dst32;
148f7018c21STomi Valkeinen 			u32 pval[4], v, v1, mask;
149f7018c21STomi Valkeinen 			int i, j, w, f;
150f7018c21STomi Valkeinen 
151f7018c21STomi Valkeinen 			src = (u8 *)info->screen_base + (sy - 1) * next_line + ((sx + width + 8) & ~15) / (8 / BPL);
152f7018c21STomi Valkeinen 			dst = (u8 *)info->screen_base + (dy - 1) * next_line + ((dx + width + 8) & ~15) / (8 / BPL);
153f7018c21STomi Valkeinen 
154f7018c21STomi Valkeinen 			mask = 0xff00ff;
155f7018c21STomi Valkeinen 			f = 0;
156f7018c21STomi Valkeinen 			w = width;
157f7018c21STomi Valkeinen 			if ((dx + width) & 15)
158f7018c21STomi Valkeinen 				f = 1;
159f7018c21STomi Valkeinen 			if (sx & 15) {
160f7018c21STomi Valkeinen 				f |= 2;
161f7018c21STomi Valkeinen 				w += 8;
162f7018c21STomi Valkeinen 			}
163f7018c21STomi Valkeinen 			w >>= 4;
164f7018c21STomi Valkeinen 			for (i = height; i; i--) {
165f7018c21STomi Valkeinen 				src32 = (u32 *)src;
166f7018c21STomi Valkeinen 				dst32 = (u32 *)dst;
167f7018c21STomi Valkeinen 
168f7018c21STomi Valkeinen 				if (f & 1) {
169f7018c21STomi Valkeinen 					pval[0] = dst32[-1] & mask;
170f7018c21STomi Valkeinen 				} else {
171f7018c21STomi Valkeinen 					pval[0] = (*--src32 >> 8) & mask;
172f7018c21STomi Valkeinen 				}
173f7018c21STomi Valkeinen 
174f7018c21STomi Valkeinen 				for (j = w; j > 0; j--) {
175f7018c21STomi Valkeinen 					v = *--src32;
176f7018c21STomi Valkeinen 					v1 = v & mask;
177f7018c21STomi Valkeinen 					*--dst32 = pval[0] | (v1 << 8);
178f7018c21STomi Valkeinen 					pval[0] = (v ^ v1) >> 8;
179f7018c21STomi Valkeinen 				}
180f7018c21STomi Valkeinen 
181f7018c21STomi Valkeinen 				if (!(f & 2)) {
182f7018c21STomi Valkeinen 					dst32[-1] = (dst32[-1] & mask) | pval[0];
183f7018c21STomi Valkeinen 				}
184f7018c21STomi Valkeinen 
185f7018c21STomi Valkeinen 				src -= next_line;
186f7018c21STomi Valkeinen 				dst -= next_line;
187f7018c21STomi Valkeinen 			}
188f7018c21STomi Valkeinen 		}
189f7018c21STomi Valkeinen 	}
190f7018c21STomi Valkeinen }
191f7018c21STomi Valkeinen 
atafb_iplan2p2_fillrect(struct fb_info * info,u_long next_line,u32 color,int sy,int sx,int height,int width)192f7018c21STomi Valkeinen void atafb_iplan2p2_fillrect(struct fb_info *info, u_long next_line, u32 color,
193f7018c21STomi Valkeinen                              int sy, int sx, int height, int width)
194f7018c21STomi Valkeinen {
195f7018c21STomi Valkeinen 	u32 *dest;
196f7018c21STomi Valkeinen 	int rows, i;
197f7018c21STomi Valkeinen 	u32 cval[4];
198f7018c21STomi Valkeinen 
199f7018c21STomi Valkeinen 	dest = (u32 *)(info->screen_base + sy * next_line + (sx & ~15) / (8 / BPL));
200f7018c21STomi Valkeinen 	if (sx & 15) {
201f7018c21STomi Valkeinen 		u8 *dest8 = (u8 *)dest + 1;
202f7018c21STomi Valkeinen 
203f7018c21STomi Valkeinen 		expand8_col2mask(color, cval);
204f7018c21STomi Valkeinen 
205f7018c21STomi Valkeinen 		for (i = height; i; i--) {
206f7018c21STomi Valkeinen 			fill8_col(dest8, cval);
207f7018c21STomi Valkeinen 			dest8 += next_line;
208f7018c21STomi Valkeinen 		}
209f7018c21STomi Valkeinen 		dest += BPL / 2;
210f7018c21STomi Valkeinen 		width -= 8;
211f7018c21STomi Valkeinen 	}
212f7018c21STomi Valkeinen 
213f7018c21STomi Valkeinen 	expand16_col2mask(color, cval);
214f7018c21STomi Valkeinen 	rows = width >> 4;
215f7018c21STomi Valkeinen 	if (rows) {
216f7018c21STomi Valkeinen 		u32 *d = dest;
217f7018c21STomi Valkeinen 		u32 off = next_line - rows * BPL * 2;
218f7018c21STomi Valkeinen 		for (i = height; i; i--) {
219f7018c21STomi Valkeinen 			d = fill16_col(d, rows, cval);
220f7018c21STomi Valkeinen 			d = (u32 *)((long)d + off);
221f7018c21STomi Valkeinen 		}
222f7018c21STomi Valkeinen 		dest += rows * BPL / 2;
223f7018c21STomi Valkeinen 		width &= 15;
224f7018c21STomi Valkeinen 	}
225f7018c21STomi Valkeinen 
226f7018c21STomi Valkeinen 	if (width) {
227f7018c21STomi Valkeinen 		u8 *dest8 = (u8 *)dest;
228f7018c21STomi Valkeinen 
229f7018c21STomi Valkeinen 		expand8_col2mask(color, cval);
230f7018c21STomi Valkeinen 
231f7018c21STomi Valkeinen 		for (i = height; i; i--) {
232f7018c21STomi Valkeinen 			fill8_col(dest8, cval);
233f7018c21STomi Valkeinen 			dest8 += next_line;
234f7018c21STomi Valkeinen 		}
235f7018c21STomi Valkeinen 	}
236f7018c21STomi Valkeinen }
237f7018c21STomi Valkeinen 
atafb_iplan2p2_linefill(struct fb_info * info,u_long next_line,int dy,int dx,u32 width,const u8 * data,u32 bgcolor,u32 fgcolor)238f7018c21STomi Valkeinen void atafb_iplan2p2_linefill(struct fb_info *info, u_long next_line,
239f7018c21STomi Valkeinen                              int dy, int dx, u32 width,
240f7018c21STomi Valkeinen                              const u8 *data, u32 bgcolor, u32 fgcolor)
241f7018c21STomi Valkeinen {
242f7018c21STomi Valkeinen 	u32 *dest;
243f7018c21STomi Valkeinen 	const u16 *data16;
244f7018c21STomi Valkeinen 	int rows;
245f7018c21STomi Valkeinen 	u32 fgm[4], bgm[4], m;
246f7018c21STomi Valkeinen 
247f7018c21STomi Valkeinen 	dest = (u32 *)(info->screen_base + dy * next_line + (dx & ~15) / (8 / BPL));
248f7018c21STomi Valkeinen 	if (dx & 15) {
249f7018c21STomi Valkeinen 		fill8_2col((u8 *)dest + 1, fgcolor, bgcolor, *data++);
250f7018c21STomi Valkeinen 		dest += BPL / 2;
251f7018c21STomi Valkeinen 		width -= 8;
252f7018c21STomi Valkeinen 	}
253f7018c21STomi Valkeinen 
254f7018c21STomi Valkeinen 	if (width >= 16) {
255f7018c21STomi Valkeinen 		data16 = (const u16 *)data;
256f7018c21STomi Valkeinen 		expand16_2col2mask(fgcolor, bgcolor, fgm, bgm);
257f7018c21STomi Valkeinen 
258f7018c21STomi Valkeinen 		for (rows = width / 16; rows; rows--) {
259f7018c21STomi Valkeinen 			u16 d = *data16++;
260f7018c21STomi Valkeinen 			m = d | ((u32)d << 16);
261f7018c21STomi Valkeinen 			*dest++ = (m & fgm[0]) ^ bgm[0];
262f7018c21STomi Valkeinen 		}
263f7018c21STomi Valkeinen 
264f7018c21STomi Valkeinen 		data = (const u8 *)data16;
265f7018c21STomi Valkeinen 		width &= 15;
266f7018c21STomi Valkeinen 	}
267f7018c21STomi Valkeinen 
268f7018c21STomi Valkeinen 	if (width)
269f7018c21STomi Valkeinen 		fill8_2col((u8 *)dest, fgcolor, bgcolor, *data);
270f7018c21STomi Valkeinen }
271