2 * linux/drivers/video/vga16.c -- VGA 16-color framebuffer driver
4 * Copyright 1999 Ben Pfaff <pfaffben@debian.org> and Petr Vandrovec <VANDROVE@vc.cvut.cz>
5 * Based on VGA info at http://www.goodnet.com/~tinara/FreeVGA/home.htm
6 * Based on VESA framebuffer (c) 1998 Gerd Knorr <kraxel@goldbach.in-berlin.de>
8 * This file is subject to the terms and conditions of the GNU General
9 * Public License. See the file COPYING in the main directory of this
10 * archive for more details.
13 #include <linux/module.h>
14 #include <linux/kernel.h>
15 #include <linux/errno.h>
16 #include <linux/string.h>
18 #include <linux/slab.h>
19 #include <linux/delay.h>
21 #include <linux/ioport.h>
22 #include <linux/init.h>
23 #include <linux/platform_device.h>
24 #include <linux/screen_info.h>
27 #include <video/vga.h>
29 #define VGA_FB_PHYS 0xA0000
30 #define VGA_FB_PHYS_LEN 65536
37 /* --------------------------------------------------------------------- */
44 /* structure holding original VGA register settings when the
47 unsigned char SeqCtrlIndex
; /* Sequencer Index reg. */
48 unsigned char CrtCtrlIndex
; /* CRT-Contr. Index reg. */
49 unsigned char CrtMiscIO
; /* Miscellaneous register */
50 unsigned char HorizontalTotal
; /* CRT-Controller:00h */
51 unsigned char HorizDisplayEnd
; /* CRT-Controller:01h */
52 unsigned char StartHorizRetrace
;/* CRT-Controller:04h */
53 unsigned char EndHorizRetrace
; /* CRT-Controller:05h */
54 unsigned char Overflow
; /* CRT-Controller:07h */
55 unsigned char StartVertRetrace
; /* CRT-Controller:10h */
56 unsigned char EndVertRetrace
; /* CRT-Controller:11h */
57 unsigned char ModeControl
; /* CRT-Controller:17h */
58 unsigned char ClockingMode
; /* Seq-Controller:01h */
60 struct vgastate state
;
61 unsigned int ref_count
;
62 int palette_blanked
, vesa_blanked
, mode
, isVGA
;
63 u8 misc
, pel_msk
, vss
, clkdiv
;
67 /* --------------------------------------------------------------------- */
69 static struct fb_var_screeninfo vga16fb_defined __initdata
= {
75 .activate
= FB_ACTIVATE_TEST
,
85 .vmode
= FB_VMODE_NONINTERLACED
,
88 /* name should not depend on EGA/VGA */
89 static struct fb_fix_screeninfo vga16fb_fix __initdata
= {
91 .smem_start
= VGA_FB_PHYS
,
92 .smem_len
= VGA_FB_PHYS_LEN
,
93 .type
= FB_TYPE_VGA_PLANES
,
94 .type_aux
= FB_AUX_VGA_PLANES_VGA4
,
95 .visual
= FB_VISUAL_PSEUDOCOLOR
,
98 .line_length
= 640 / 8,
99 .accel
= FB_ACCEL_NONE
102 /* The VGA's weird architecture often requires that we read a byte and
103 write a byte to the same location. It doesn't matter *what* byte
104 we write, however. This is because all the action goes on behind
105 the scenes in the VGA's 32-bit latch register, and reading and writing
106 video memory just invokes latch behavior.
108 To avoid race conditions (is this necessary?), reading and writing
109 the memory byte should be done with a single instruction. One
110 suitable instruction is the x86 bitwise OR. The following
111 read-modify-write routine should optimize to one such bitwise
113 static inline void rmw(volatile char __iomem
*p
)
119 /* Set the Graphics Mode Register, and return its previous value.
120 Bits 0-1 are write mode, bit 3 is read mode. */
121 static inline int setmode(int mode
)
125 oldmode
= vga_io_rgfx(VGA_GFX_MODE
);
126 vga_io_w(VGA_GFX_D
, mode
);
130 /* Select the Bit Mask Register and return its value. */
131 static inline int selectmask(void)
133 return vga_io_rgfx(VGA_GFX_BIT_MASK
);
136 /* Set the value of the Bit Mask Register. It must already have been
137 selected with selectmask(). */
138 static inline void setmask(int mask
)
140 vga_io_w(VGA_GFX_D
, mask
);
143 /* Set the Data Rotate Register and return its old value.
144 Bits 0-2 are rotate count, bits 3-4 are logical operation
145 (0=NOP, 1=AND, 2=OR, 3=XOR). */
146 static inline int setop(int op
)
150 oldop
= vga_io_rgfx(VGA_GFX_DATA_ROTATE
);
151 vga_io_w(VGA_GFX_D
, op
);
155 /* Set the Enable Set/Reset Register and return its old value.
156 The code here always uses value 0xf for thsi register. */
157 static inline int setsr(int sr
)
161 oldsr
= vga_io_rgfx(VGA_GFX_SR_ENABLE
);
162 vga_io_w(VGA_GFX_D
, sr
);
166 /* Set the Set/Reset Register and return its old value. */
167 static inline int setcolor(int color
)
171 oldcolor
= vga_io_rgfx(VGA_GFX_SR_VALUE
);
172 vga_io_w(VGA_GFX_D
, color
);
176 /* Return the value in the Graphics Address Register. */
177 static inline int getindex(void)
179 return vga_io_r(VGA_GFX_I
);
182 /* Set the value in the Graphics Address Register. */
183 static inline void setindex(int index
)
185 vga_io_w(VGA_GFX_I
, index
);
188 static void vga16fb_pan_var(struct fb_info
*info
,
189 struct fb_var_screeninfo
*var
)
191 struct vga16fb_par
*par
= info
->par
;
194 xoffset
= var
->xoffset
;
195 if (info
->var
.bits_per_pixel
== 8) {
196 pos
= (info
->var
.xres_virtual
* var
->yoffset
+ xoffset
) >> 2;
197 } else if (par
->mode
& MODE_TEXT
) {
198 int fh
= 16; // FIXME !!! font height. Fugde for now.
199 pos
= (info
->var
.xres_virtual
* (var
->yoffset
/ fh
) + xoffset
) >> 3;
201 if (info
->var
.nonstd
)
203 pos
= (info
->var
.xres_virtual
* var
->yoffset
+ xoffset
) >> 3;
205 vga_io_wcrt(VGA_CRTC_START_HI
, pos
>> 8);
206 vga_io_wcrt(VGA_CRTC_START_LO
, pos
& 0xFF);
207 /* if we support CFB4, then we must! support xoffset with pixel
208 * granularity if someone supports xoffset in bit resolution */
209 vga_io_r(VGA_IS1_RC
); /* reset flip-flop */
210 vga_io_w(VGA_ATT_IW
, VGA_ATC_PEL
);
211 if (var
->bits_per_pixel
== 8)
212 vga_io_w(VGA_ATT_IW
, (xoffset
& 3) << 1);
214 vga_io_w(VGA_ATT_IW
, xoffset
& 7);
215 vga_io_r(VGA_IS1_RC
);
216 vga_io_w(VGA_ATT_IW
, 0x20);
219 static void vga16fb_update_fix(struct fb_info
*info
)
221 if (info
->var
.bits_per_pixel
== 4) {
222 if (info
->var
.nonstd
) {
223 info
->fix
.type
= FB_TYPE_PACKED_PIXELS
;
224 info
->fix
.line_length
= info
->var
.xres_virtual
/ 2;
226 info
->fix
.type
= FB_TYPE_VGA_PLANES
;
227 info
->fix
.type_aux
= FB_AUX_VGA_PLANES_VGA4
;
228 info
->fix
.line_length
= info
->var
.xres_virtual
/ 8;
230 } else if (info
->var
.bits_per_pixel
== 0) {
231 info
->fix
.type
= FB_TYPE_TEXT
;
232 info
->fix
.type_aux
= FB_AUX_TEXT_CGA
;
233 info
->fix
.line_length
= info
->var
.xres_virtual
/ 4;
235 if (info
->var
.nonstd
) {
236 info
->fix
.type
= FB_TYPE_VGA_PLANES
;
237 info
->fix
.type_aux
= FB_AUX_VGA_PLANES_CFB8
;
238 info
->fix
.line_length
= info
->var
.xres_virtual
/ 4;
240 info
->fix
.type
= FB_TYPE_PACKED_PIXELS
;
241 info
->fix
.line_length
= info
->var
.xres_virtual
;
246 static void vga16fb_clock_chip(struct vga16fb_par
*par
,
247 unsigned int pixclock
,
248 const struct fb_info
*info
,
251 static const struct {
255 } *ptr
, *best
, vgaclocks
[] = {
256 { 79442 /* 12.587 */, 0x00, 0x08},
257 { 70616 /* 14.161 */, 0x04, 0x08},
258 { 39721 /* 25.175 */, 0x00, 0x00},
259 { 35308 /* 28.322 */, 0x04, 0x00},
260 { 0 /* bad */, 0x00, 0x00}};
263 pixclock
= (pixclock
* mul
) / div
;
265 err
= pixclock
- best
->pixclock
;
266 if (err
< 0) err
= -err
;
267 for (ptr
= vgaclocks
+ 1; ptr
->pixclock
; ptr
++) {
270 tmp
= pixclock
- ptr
->pixclock
;
271 if (tmp
< 0) tmp
= -tmp
;
277 par
->misc
|= best
->misc
;
278 par
->clkdiv
= best
->seq_clock_mode
;
279 pixclock
= (best
->pixclock
* div
) / mul
;
282 #define FAIL(X) return -EINVAL
284 static int vga16fb_open(struct fb_info
*info
, int user
)
286 struct vga16fb_par
*par
= info
->par
;
288 if (!par
->ref_count
) {
289 memset(&par
->state
, 0, sizeof(struct vgastate
));
290 par
->state
.flags
= VGA_SAVE_FONTS
| VGA_SAVE_MODE
|
292 save_vga(&par
->state
);
299 static int vga16fb_release(struct fb_info
*info
, int user
)
301 struct vga16fb_par
*par
= info
->par
;
306 if (par
->ref_count
== 1)
307 restore_vga(&par
->state
);
313 static int vga16fb_check_var(struct fb_var_screeninfo
*var
,
314 struct fb_info
*info
)
316 struct vga16fb_par
*par
= info
->par
;
317 u32 xres
, right
, hslen
, left
, xtotal
;
318 u32 yres
, lower
, vslen
, upper
, ytotal
;
319 u32 vxres
, xoffset
, vyres
, yoffset
;
328 if (var
->bits_per_pixel
== 4) {
333 mode
= MODE_SKIP4
| MODE_CFB
;
341 } else if (var
->bits_per_pixel
== 8) {
343 return -EINVAL
; /* no support on EGA */
346 mode
= MODE_8BPP
| MODE_CFB
;
349 mode
= MODE_SKIP4
| MODE_8BPP
| MODE_CFB
;
355 xres
= (var
->xres
+ 7) & ~7;
356 vxres
= (var
->xres_virtual
+ 0xF) & ~0xF;
357 xoffset
= (var
->xoffset
+ 7) & ~7;
358 left
= (var
->left_margin
+ 7) & ~7;
359 right
= (var
->right_margin
+ 7) & ~7;
360 hslen
= (var
->hsync_len
+ 7) & ~7;
364 if (xres
+ xoffset
> vxres
)
365 xoffset
= vxres
- xres
;
368 var
->right_margin
= right
;
369 var
->hsync_len
= hslen
;
370 var
->left_margin
= left
;
371 var
->xres_virtual
= vxres
;
372 var
->xoffset
= xoffset
;
379 xtotal
= xres
+ right
+ hslen
+ left
;
381 FAIL("xtotal too big");
383 FAIL("hslen too big");
384 if (right
+ hslen
+ left
> 64)
385 FAIL("hblank too big");
386 par
->crtc
[VGA_CRTC_H_TOTAL
] = xtotal
- 5;
387 par
->crtc
[VGA_CRTC_H_BLANK_START
] = xres
- 1;
388 par
->crtc
[VGA_CRTC_H_DISP
] = xres
- 1;
390 par
->crtc
[VGA_CRTC_H_SYNC_START
] = pos
;
392 par
->crtc
[VGA_CRTC_H_SYNC_END
] = pos
& 0x1F;
393 pos
+= left
- 2; /* blank_end + 2 <= total + 5 */
394 par
->crtc
[VGA_CRTC_H_BLANK_END
] = (pos
& 0x1F) | 0x80;
396 par
->crtc
[VGA_CRTC_H_SYNC_END
] |= 0x80;
399 lower
= var
->lower_margin
;
400 vslen
= var
->vsync_len
;
401 upper
= var
->upper_margin
;
402 vyres
= var
->yres_virtual
;
403 yoffset
= var
->yoffset
;
407 if (vxres
* vyres
> maxmem
) {
408 vyres
= maxmem
/ vxres
;
412 if (yoffset
+ yres
> vyres
)
413 yoffset
= vyres
- yres
;
415 var
->lower_margin
= lower
;
416 var
->vsync_len
= vslen
;
417 var
->upper_margin
= upper
;
418 var
->yres_virtual
= vyres
;
419 var
->yoffset
= yoffset
;
421 if (var
->vmode
& FB_VMODE_DOUBLE
) {
427 ytotal
= yres
+ lower
+ vslen
+ upper
;
438 FAIL("ytotal too big");
440 FAIL("vslen too big");
441 par
->crtc
[VGA_CRTC_V_TOTAL
] = ytotal
- 2;
442 r7
= 0x10; /* disable linecompare */
443 if (ytotal
& 0x100) r7
|= 0x01;
444 if (ytotal
& 0x200) r7
|= 0x20;
445 par
->crtc
[VGA_CRTC_PRESET_ROW
] = 0;
446 par
->crtc
[VGA_CRTC_MAX_SCAN
] = 0x40; /* 1 scanline, no linecmp */
447 if (var
->vmode
& FB_VMODE_DOUBLE
)
448 par
->crtc
[VGA_CRTC_MAX_SCAN
] |= 0x80;
449 par
->crtc
[VGA_CRTC_CURSOR_START
] = 0x20;
450 par
->crtc
[VGA_CRTC_CURSOR_END
] = 0x00;
451 if ((mode
& (MODE_CFB
| MODE_8BPP
)) == MODE_CFB
)
453 pos
= yoffset
* vxres
+ (xoffset
>> shift
);
454 par
->crtc
[VGA_CRTC_START_HI
] = pos
>> 8;
455 par
->crtc
[VGA_CRTC_START_LO
] = pos
& 0xFF;
456 par
->crtc
[VGA_CRTC_CURSOR_HI
] = 0x00;
457 par
->crtc
[VGA_CRTC_CURSOR_LO
] = 0x00;
459 par
->crtc
[VGA_CRTC_V_DISP_END
] = pos
& 0xFF;
460 par
->crtc
[VGA_CRTC_V_BLANK_START
] = pos
& 0xFF;
462 r7
|= 0x0A; /* 0x02 -> DISP_END, 0x08 -> BLANK_START */
464 r7
|= 0x40; /* 0x40 -> DISP_END */
465 par
->crtc
[VGA_CRTC_MAX_SCAN
] |= 0x20; /* BLANK_START */
468 par
->crtc
[VGA_CRTC_V_SYNC_START
] = pos
& 0xFF;
474 par
->crtc
[VGA_CRTC_V_SYNC_END
] = (pos
& 0x0F) & ~0x10; /* disabled IRQ */
475 pos
+= upper
- 1; /* blank_end + 1 <= ytotal + 2 */
476 par
->crtc
[VGA_CRTC_V_BLANK_END
] = pos
& 0xFF; /* 0x7F for original VGA,
477 but some SVGA chips requires all 8 bits to set */
479 FAIL("vxres too long");
480 par
->crtc
[VGA_CRTC_OFFSET
] = vxres
>> 1;
481 if (mode
& MODE_SKIP4
)
482 par
->crtc
[VGA_CRTC_UNDERLINE
] = 0x5F; /* 256, cfb8 */
484 par
->crtc
[VGA_CRTC_UNDERLINE
] = 0x1F; /* 16, vgap */
485 par
->crtc
[VGA_CRTC_MODE
] = rMode
| ((mode
& MODE_TEXT
) ? 0xA3 : 0xE3);
486 par
->crtc
[VGA_CRTC_LINE_COMPARE
] = 0xFF;
487 par
->crtc
[VGA_CRTC_OVERFLOW
] = r7
;
489 par
->vss
= 0x00; /* 3DA */
491 par
->misc
= 0xE3; /* enable CPU, ports 0x3Dx, positive sync */
492 if (var
->sync
& FB_SYNC_HOR_HIGH_ACT
)
494 if (var
->sync
& FB_SYNC_VERT_HIGH_ACT
)
499 if (mode
& MODE_8BPP
)
500 /* pixel clock == vga clock / 2 */
501 vga16fb_clock_chip(par
, var
->pixclock
, info
, 1, 2);
503 /* pixel clock == vga clock */
504 vga16fb_clock_chip(par
, var
->pixclock
, info
, 1, 1);
506 var
->red
.offset
= var
->green
.offset
= var
->blue
.offset
=
507 var
->transp
.offset
= 0;
508 var
->red
.length
= var
->green
.length
= var
->blue
.length
=
509 (par
->isVGA
) ? 6 : 2;
510 var
->transp
.length
= 0;
511 var
->activate
= FB_ACTIVATE_NOW
;
514 var
->accel_flags
= 0;
519 static int vga16fb_set_par(struct fb_info
*info
)
521 struct vga16fb_par
*par
= info
->par
;
527 seq
[VGA_SEQ_CLOCK_MODE
] = 0x01 | par
->clkdiv
;
528 if (par
->mode
& MODE_TEXT
)
529 seq
[VGA_SEQ_PLANE_WRITE
] = 0x03;
531 seq
[VGA_SEQ_PLANE_WRITE
] = 0x0F;
532 seq
[VGA_SEQ_CHARACTER_MAP
] = 0x00;
533 if (par
->mode
& MODE_TEXT
)
534 seq
[VGA_SEQ_MEMORY_MODE
] = 0x03;
535 else if (par
->mode
& MODE_SKIP4
)
536 seq
[VGA_SEQ_MEMORY_MODE
] = 0x0E;
538 seq
[VGA_SEQ_MEMORY_MODE
] = 0x06;
540 gdc
[VGA_GFX_SR_VALUE
] = 0x00;
541 gdc
[VGA_GFX_SR_ENABLE
] = 0x00;
542 gdc
[VGA_GFX_COMPARE_VALUE
] = 0x00;
543 gdc
[VGA_GFX_DATA_ROTATE
] = 0x00;
544 gdc
[VGA_GFX_PLANE_READ
] = 0;
545 if (par
->mode
& MODE_TEXT
) {
546 gdc
[VGA_GFX_MODE
] = 0x10;
547 gdc
[VGA_GFX_MISC
] = 0x06;
549 if (par
->mode
& MODE_CFB
)
550 gdc
[VGA_GFX_MODE
] = 0x40;
552 gdc
[VGA_GFX_MODE
] = 0x00;
553 gdc
[VGA_GFX_MISC
] = 0x05;
555 gdc
[VGA_GFX_COMPARE_MASK
] = 0x0F;
556 gdc
[VGA_GFX_BIT_MASK
] = 0xFF;
558 for (i
= 0x00; i
< 0x10; i
++)
560 if (par
->mode
& MODE_TEXT
)
561 atc
[VGA_ATC_MODE
] = 0x04;
562 else if (par
->mode
& MODE_8BPP
)
563 atc
[VGA_ATC_MODE
] = 0x41;
565 atc
[VGA_ATC_MODE
] = 0x81;
566 atc
[VGA_ATC_OVERSCAN
] = 0x00; /* 0 for EGA, 0xFF for VGA */
567 atc
[VGA_ATC_PLANE_ENABLE
] = 0x0F;
568 if (par
->mode
& MODE_8BPP
)
569 atc
[VGA_ATC_PEL
] = (info
->var
.xoffset
& 3) << 1;
571 atc
[VGA_ATC_PEL
] = info
->var
.xoffset
& 7;
572 atc
[VGA_ATC_COLOR_PAGE
] = 0x00;
574 if (par
->mode
& MODE_TEXT
) {
575 fh
= 16; // FIXME !!! Fudge font height.
576 par
->crtc
[VGA_CRTC_MAX_SCAN
] = (par
->crtc
[VGA_CRTC_MAX_SCAN
]
580 vga_io_w(VGA_MIS_W
, vga_io_r(VGA_MIS_R
) | 0x01);
582 /* Enable graphics register modification */
584 vga_io_w(EGA_GFX_E0
, 0x00);
585 vga_io_w(EGA_GFX_E1
, 0x01);
588 /* update misc output register */
589 vga_io_w(VGA_MIS_W
, par
->misc
);
591 /* synchronous reset on */
592 vga_io_wseq(0x00, 0x01);
595 vga_io_w(VGA_PEL_MSK
, par
->pel_msk
);
597 /* write sequencer registers */
598 vga_io_wseq(VGA_SEQ_CLOCK_MODE
, seq
[VGA_SEQ_CLOCK_MODE
] | 0x20);
599 for (i
= 2; i
< VGA_SEQ_C
; i
++) {
600 vga_io_wseq(i
, seq
[i
]);
603 /* synchronous reset off */
604 vga_io_wseq(0x00, 0x03);
606 /* deprotect CRT registers 0-7 */
607 vga_io_wcrt(VGA_CRTC_V_SYNC_END
, par
->crtc
[VGA_CRTC_V_SYNC_END
]);
609 /* write CRT registers */
610 for (i
= 0; i
< VGA_CRTC_REGS
; i
++) {
611 vga_io_wcrt(i
, par
->crtc
[i
]);
614 /* write graphics controller registers */
615 for (i
= 0; i
< VGA_GFX_C
; i
++) {
616 vga_io_wgfx(i
, gdc
[i
]);
619 /* write attribute controller registers */
620 for (i
= 0; i
< VGA_ATT_C
; i
++) {
621 vga_io_r(VGA_IS1_RC
); /* reset flip-flop */
622 vga_io_wattr(i
, atc
[i
]);
625 /* Wait for screen to stabilize. */
628 vga_io_wseq(VGA_SEQ_CLOCK_MODE
, seq
[VGA_SEQ_CLOCK_MODE
]);
630 vga_io_r(VGA_IS1_RC
);
631 vga_io_w(VGA_ATT_IW
, 0x20);
633 vga16fb_update_fix(info
);
637 static void ega16_setpalette(int regno
, unsigned red
, unsigned green
, unsigned blue
)
639 static const unsigned char map
[] = { 000, 001, 010, 011 };
644 val
= map
[red
>>14] | ((map
[green
>>14]) << 1) | ((map
[blue
>>14]) << 2);
645 vga_io_r(VGA_IS1_RC
); /* ! 0x3BA */
646 vga_io_wattr(regno
, val
);
647 vga_io_r(VGA_IS1_RC
); /* some clones need it */
648 vga_io_w(VGA_ATT_IW
, 0x20); /* unblank screen */
651 static void vga16_setpalette(int regno
, unsigned red
, unsigned green
, unsigned blue
)
653 outb(regno
, VGA_PEL_IW
);
654 outb(red
>> 10, VGA_PEL_D
);
655 outb(green
>> 10, VGA_PEL_D
);
656 outb(blue
>> 10, VGA_PEL_D
);
659 static int vga16fb_setcolreg(unsigned regno
, unsigned red
, unsigned green
,
660 unsigned blue
, unsigned transp
,
661 struct fb_info
*info
)
663 struct vga16fb_par
*par
= info
->par
;
667 * Set a single color register. The values supplied are
668 * already rounded down to the hardware's capabilities
669 * (according to the entries in the `var' structure). Return
670 * != 0 for invalid regno.
676 gray
= info
->var
.grayscale
;
679 /* gray = 0.30*R + 0.59*G + 0.11*B */
680 red
= green
= blue
= (red
* 77 + green
* 151 + blue
* 28) >> 8;
683 vga16_setpalette(regno
,red
,green
,blue
);
685 ega16_setpalette(regno
,red
,green
,blue
);
689 static int vga16fb_pan_display(struct fb_var_screeninfo
*var
,
690 struct fb_info
*info
)
692 vga16fb_pan_var(info
, var
);
696 /* The following VESA blanking code is taken from vgacon.c. The VGA
697 blanking code was originally by Huang shi chao, and modified by
698 Christoph Rimek (chrimek@toppoint.de) and todd j. derr
699 (tjd@barefoot.org) for Linux. */
701 static void vga_vesa_blank(struct vga16fb_par
*par
, int mode
)
703 unsigned char SeqCtrlIndex
= vga_io_r(VGA_SEQ_I
);
704 unsigned char CrtCtrlIndex
= vga_io_r(VGA_CRT_IC
);
706 /* save original values of VGA controller registers */
707 if(!par
->vesa_blanked
) {
708 par
->vga_state
.CrtMiscIO
= vga_io_r(VGA_MIS_R
);
711 par
->vga_state
.HorizontalTotal
= vga_io_rcrt(0x00); /* HorizontalTotal */
712 par
->vga_state
.HorizDisplayEnd
= vga_io_rcrt(0x01); /* HorizDisplayEnd */
713 par
->vga_state
.StartHorizRetrace
= vga_io_rcrt(0x04); /* StartHorizRetrace */
714 par
->vga_state
.EndHorizRetrace
= vga_io_rcrt(0x05); /* EndHorizRetrace */
715 par
->vga_state
.Overflow
= vga_io_rcrt(0x07); /* Overflow */
716 par
->vga_state
.StartVertRetrace
= vga_io_rcrt(0x10); /* StartVertRetrace */
717 par
->vga_state
.EndVertRetrace
= vga_io_rcrt(0x11); /* EndVertRetrace */
718 par
->vga_state
.ModeControl
= vga_io_rcrt(0x17); /* ModeControl */
719 par
->vga_state
.ClockingMode
= vga_io_rseq(0x01); /* ClockingMode */
722 /* assure that video is enabled */
723 /* "0x20" is VIDEO_ENABLE_bit in register 01 of sequencer */
724 vga_io_wseq(0x01, par
->vga_state
.ClockingMode
| 0x20);
726 /* test for vertical retrace in process.... */
727 if ((par
->vga_state
.CrtMiscIO
& 0x80) == 0x80)
728 vga_io_w(VGA_MIS_W
, par
->vga_state
.CrtMiscIO
& 0xef);
731 * Set <End of vertical retrace> to minimum (0) and
732 * <Start of vertical Retrace> to maximum (incl. overflow)
733 * Result: turn off vertical sync (VSync) pulse.
735 if (mode
& FB_BLANK_VSYNC_SUSPEND
) {
736 vga_io_wcrt(VGA_CRTC_V_SYNC_START
, 0xff);
737 vga_io_wcrt(VGA_CRTC_V_SYNC_END
, 0x40);
738 /* bits 9,10 of vert. retrace */
739 vga_io_wcrt(VGA_CRTC_OVERFLOW
, par
->vga_state
.Overflow
| 0x84);
742 if (mode
& FB_BLANK_HSYNC_SUSPEND
) {
744 * Set <End of horizontal retrace> to minimum (0) and
745 * <Start of horizontal Retrace> to maximum
746 * Result: turn off horizontal sync (HSync) pulse.
748 vga_io_wcrt(VGA_CRTC_H_SYNC_START
, 0xff);
749 vga_io_wcrt(VGA_CRTC_H_SYNC_END
, 0x00);
752 /* restore both index registers */
753 outb_p(SeqCtrlIndex
, VGA_SEQ_I
);
754 outb_p(CrtCtrlIndex
, VGA_CRT_IC
);
757 static void vga_vesa_unblank(struct vga16fb_par
*par
)
759 unsigned char SeqCtrlIndex
= vga_io_r(VGA_SEQ_I
);
760 unsigned char CrtCtrlIndex
= vga_io_r(VGA_CRT_IC
);
762 /* restore original values of VGA controller registers */
763 vga_io_w(VGA_MIS_W
, par
->vga_state
.CrtMiscIO
);
765 /* HorizontalTotal */
766 vga_io_wcrt(0x00, par
->vga_state
.HorizontalTotal
);
767 /* HorizDisplayEnd */
768 vga_io_wcrt(0x01, par
->vga_state
.HorizDisplayEnd
);
769 /* StartHorizRetrace */
770 vga_io_wcrt(0x04, par
->vga_state
.StartHorizRetrace
);
771 /* EndHorizRetrace */
772 vga_io_wcrt(0x05, par
->vga_state
.EndHorizRetrace
);
774 vga_io_wcrt(0x07, par
->vga_state
.Overflow
);
775 /* StartVertRetrace */
776 vga_io_wcrt(0x10, par
->vga_state
.StartVertRetrace
);
778 vga_io_wcrt(0x11, par
->vga_state
.EndVertRetrace
);
780 vga_io_wcrt(0x17, par
->vga_state
.ModeControl
);
782 vga_io_wseq(0x01, par
->vga_state
.ClockingMode
);
784 /* restore index/control registers */
785 vga_io_w(VGA_SEQ_I
, SeqCtrlIndex
);
786 vga_io_w(VGA_CRT_IC
, CrtCtrlIndex
);
789 static void vga_pal_blank(void)
793 for (i
=0; i
<16; i
++) {
794 outb_p(i
, VGA_PEL_IW
);
795 outb_p(0, VGA_PEL_D
);
796 outb_p(0, VGA_PEL_D
);
797 outb_p(0, VGA_PEL_D
);
801 /* 0 unblank, 1 blank, 2 no vsync, 3 no hsync, 4 off */
802 static int vga16fb_blank(int blank
, struct fb_info
*info
)
804 struct vga16fb_par
*par
= info
->par
;
807 case FB_BLANK_UNBLANK
: /* Unblank */
808 if (par
->vesa_blanked
) {
809 vga_vesa_unblank(par
);
810 par
->vesa_blanked
= 0;
812 if (par
->palette_blanked
) {
813 par
->palette_blanked
= 0;
816 case FB_BLANK_NORMAL
: /* blank */
818 par
->palette_blanked
= 1;
820 default: /* VESA blanking */
821 vga_vesa_blank(par
, blank
);
822 par
->vesa_blanked
= 1;
828 static void vga_8planes_fillrect(struct fb_info
*info
, const struct fb_fillrect
*rect
)
830 u32 dx
= rect
->dx
, width
= rect
->width
;
831 char oldindex
= getindex();
832 char oldmode
= setmode(0x40);
833 char oldmask
= selectmask();
834 int line_ofs
, height
;
839 where
= info
->screen_base
+ dx
+ rect
->dy
* info
->fix
.line_length
;
841 if (rect
->rop
== ROP_COPY
) {
846 line_ofs
= info
->fix
.line_length
- width
;
849 height
= rect
->height
;
854 /* we can do memset... */
855 for (x
= width
; x
> 0; --x
) {
856 writeb(rect
->color
, where
);
862 char oldcolor
= setcolor(0xf);
868 for (y
= 0; y
< rect
->height
; y
++) {
871 where
+= info
->fix
.line_length
;
882 static void vga16fb_fillrect(struct fb_info
*info
, const struct fb_fillrect
*rect
)
884 int x
, x2
, y2
, vxres
, vyres
, width
, height
, line_ofs
;
887 vxres
= info
->var
.xres_virtual
;
888 vyres
= info
->var
.yres_virtual
;
890 if (!rect
->width
|| !rect
->height
|| rect
->dx
> vxres
|| rect
->dy
> vyres
)
893 /* We could use hardware clipping but on many cards you get around
894 * hardware clipping by writing to framebuffer directly. */
896 x2
= rect
->dx
+ rect
->width
;
897 y2
= rect
->dy
+ rect
->height
;
898 x2
= x2
< vxres
? x2
: vxres
;
899 y2
= y2
< vyres
? y2
: vyres
;
900 width
= x2
- rect
->dx
;
902 switch (info
->fix
.type
) {
903 case FB_TYPE_VGA_PLANES
:
904 if (info
->fix
.type_aux
== FB_AUX_VGA_PLANES_VGA4
) {
906 height
= y2
- rect
->dy
;
907 width
= rect
->width
/8;
909 line_ofs
= info
->fix
.line_length
- width
;
910 dst
= info
->screen_base
+ (rect
->dx
/8) + rect
->dy
* info
->fix
.line_length
;
917 setcolor(rect
->color
);
923 for (x
= 0; x
< width
; x
++) {
939 for (x
= 0; x
< width
; x
++) {
948 vga_8planes_fillrect(info
, rect
);
950 case FB_TYPE_PACKED_PIXELS
:
952 cfb_fillrect(info
, rect
);
957 static void vga_8planes_copyarea(struct fb_info
*info
, const struct fb_copyarea
*area
)
959 char oldindex
= getindex();
960 char oldmode
= setmode(0x41);
961 char oldop
= setop(0);
962 char oldsr
= setsr(0xf);
963 int height
, line_ofs
, x
;
968 height
= area
->height
;
972 width
= area
->width
/ 4;
974 if (area
->dy
< area
->sy
|| (area
->dy
== area
->sy
&& dx
< sx
)) {
975 line_ofs
= info
->fix
.line_length
- width
;
976 dest
= info
->screen_base
+ dx
+ area
->dy
* info
->fix
.line_length
;
977 src
= info
->screen_base
+ sx
+ area
->sy
* info
->fix
.line_length
;
979 for (x
= 0; x
< width
; x
++) {
989 line_ofs
= info
->fix
.line_length
- width
;
990 dest
= info
->screen_base
+ dx
+ width
+
991 (area
->dy
+ height
- 1) * info
->fix
.line_length
;
992 src
= info
->screen_base
+ sx
+ width
+
993 (area
->sy
+ height
- 1) * info
->fix
.line_length
;
995 for (x
= 0; x
< width
; x
++) {
1012 static void vga16fb_copyarea(struct fb_info
*info
, const struct fb_copyarea
*area
)
1014 u32 dx
= area
->dx
, dy
= area
->dy
, sx
= area
->sx
, sy
= area
->sy
;
1015 int x
, x2
, y2
, old_dx
, old_dy
, vxres
, vyres
;
1016 int height
, width
, line_ofs
;
1017 char __iomem
*dst
= NULL
;
1018 char __iomem
*src
= NULL
;
1020 vxres
= info
->var
.xres_virtual
;
1021 vyres
= info
->var
.yres_virtual
;
1023 if (area
->dx
> vxres
|| area
->sx
> vxres
|| area
->dy
> vyres
||
1027 /* clip the destination */
1032 * We could use hardware clipping but on many cards you get around
1033 * hardware clipping by writing to framebuffer directly.
1035 x2
= area
->dx
+ area
->width
;
1036 y2
= area
->dy
+ area
->height
;
1037 dx
= area
->dx
> 0 ? area
->dx
: 0;
1038 dy
= area
->dy
> 0 ? area
->dy
: 0;
1039 x2
= x2
< vxres
? x2
: vxres
;
1040 y2
= y2
< vyres
? y2
: vyres
;
1044 if (sx
+ dx
< old_dx
|| sy
+ dy
< old_dy
)
1047 /* update sx1,sy1 */
1048 sx
+= (dx
- old_dx
);
1049 sy
+= (dy
- old_dy
);
1051 /* the source must be completely inside the virtual screen */
1052 if (sx
+ width
> vxres
|| sy
+ height
> vyres
)
1055 switch (info
->fix
.type
) {
1056 case FB_TYPE_VGA_PLANES
:
1057 if (info
->fix
.type_aux
== FB_AUX_VGA_PLANES_VGA4
) {
1060 line_ofs
= info
->fix
.line_length
- width
;
1066 if (dy
< sy
|| (dy
== sy
&& dx
< sx
)) {
1067 dst
= info
->screen_base
+ (dx
/8) + dy
* info
->fix
.line_length
;
1068 src
= info
->screen_base
+ (sx
/8) + sy
* info
->fix
.line_length
;
1070 for (x
= 0; x
< width
; x
++) {
1080 dst
= info
->screen_base
+ (dx
/8) + width
+
1081 (dy
+ height
- 1) * info
->fix
.line_length
;
1082 src
= info
->screen_base
+ (sx
/8) + width
+
1083 (sy
+ height
- 1) * info
->fix
.line_length
;
1085 for (x
= 0; x
< width
; x
++) {
1096 vga_8planes_copyarea(info
, area
);
1098 case FB_TYPE_PACKED_PIXELS
:
1100 cfb_copyarea(info
, area
);
1105 #define TRANS_MASK_LOW {0x0,0x8,0x4,0xC,0x2,0xA,0x6,0xE,0x1,0x9,0x5,0xD,0x3,0xB,0x7,0xF}
1106 #define TRANS_MASK_HIGH {0x000, 0x800, 0x400, 0xC00, 0x200, 0xA00, 0x600, 0xE00, \
1107 0x100, 0x900, 0x500, 0xD00, 0x300, 0xB00, 0x700, 0xF00}
1109 #if defined(__LITTLE_ENDIAN)
1110 static const u16 transl_l
[] = TRANS_MASK_LOW
;
1111 static const u16 transl_h
[] = TRANS_MASK_HIGH
;
1112 #elif defined(__BIG_ENDIAN)
1113 static const u16 transl_l
[] = TRANS_MASK_HIGH
;
1114 static const u16 transl_h
[] = TRANS_MASK_LOW
;
1116 #error "Only __BIG_ENDIAN and __LITTLE_ENDIAN are supported in vga-planes"
1119 static void vga_8planes_imageblit(struct fb_info
*info
, const struct fb_image
*image
)
1121 char oldindex
= getindex();
1122 char oldmode
= setmode(0x40);
1123 char oldop
= setop(0);
1124 char oldsr
= setsr(0);
1125 char oldmask
= selectmask();
1126 const char *cdat
= image
->data
;
1128 char __iomem
*where
;
1132 where
= info
->screen_base
+ dx
+ image
->dy
* info
->fix
.line_length
;
1135 writeb(image
->bg_color
, where
);
1138 setmask(image
->fg_color
^ image
->bg_color
);
1141 for (y
= 0; y
< image
->height
; y
++, where
+= info
->fix
.line_length
)
1142 writew(transl_h
[cdat
[y
]&0xF] | transl_l
[cdat
[y
] >> 4], where
);
1150 static void vga_imageblit_expand(struct fb_info
*info
, const struct fb_image
*image
)
1152 char __iomem
*where
= info
->screen_base
+ (image
->dx
/8) +
1153 image
->dy
* info
->fix
.line_length
;
1154 struct vga16fb_par
*par
= info
->par
;
1155 char *cdat
= (char *) image
->data
;
1159 switch (info
->fix
.type
) {
1160 case FB_TYPE_VGA_PLANES
:
1161 if (info
->fix
.type_aux
== FB_AUX_VGA_PLANES_VGA4
) {
1166 setcolor(image
->fg_color
);
1170 writeb(image
->bg_color
, where
);
1172 readb(where
); /* fill latches */
1175 for (y
= 0; y
< image
->height
; y
++) {
1177 for (x
= image
->width
/8; x
--;)
1178 writeb(*cdat
++, dst
++);
1179 where
+= info
->fix
.line_length
;
1186 setcolor(image
->bg_color
);
1190 for (y
= 0; y
< image
->height
; y
++) {
1192 for (x
=image
->width
/8; x
--;){
1194 setcolor(image
->fg_color
);
1201 where
+= info
->fix
.line_length
;
1205 vga_8planes_imageblit(info
, image
);
1207 case FB_TYPE_PACKED_PIXELS
:
1209 cfb_imageblit(info
, image
);
1214 static void vga_imageblit_color(struct fb_info
*info
, const struct fb_image
*image
)
1219 struct vga16fb_par
*par
= info
->par
;
1220 char __iomem
*where
=
1221 info
->screen_base
+ image
->dy
* info
->fix
.line_length
+
1223 const char *cdat
= image
->data
;
1227 switch (info
->fix
.type
) {
1228 case FB_TYPE_VGA_PLANES
:
1229 if (info
->fix
.type_aux
== FB_AUX_VGA_PLANES_VGA4
&&
1235 for (y
= 0; y
< image
->height
; y
++) {
1236 for (x
= 0; x
< image
->width
; x
++) {
1241 setmask(1 << (7 - (x
% 8)));
1247 where
+= info
->fix
.line_length
;
1251 case FB_TYPE_PACKED_PIXELS
:
1252 cfb_imageblit(info
, image
);
1259 static void vga16fb_imageblit(struct fb_info
*info
, const struct fb_image
*image
)
1261 if (image
->depth
== 1)
1262 vga_imageblit_expand(info
, image
);
1264 vga_imageblit_color(info
, image
);
1267 static struct fb_ops vga16fb_ops
= {
1268 .owner
= THIS_MODULE
,
1269 .fb_open
= vga16fb_open
,
1270 .fb_release
= vga16fb_release
,
1271 .fb_check_var
= vga16fb_check_var
,
1272 .fb_set_par
= vga16fb_set_par
,
1273 .fb_setcolreg
= vga16fb_setcolreg
,
1274 .fb_pan_display
= vga16fb_pan_display
,
1275 .fb_blank
= vga16fb_blank
,
1276 .fb_fillrect
= vga16fb_fillrect
,
1277 .fb_copyarea
= vga16fb_copyarea
,
1278 .fb_imageblit
= vga16fb_imageblit
,
1282 static int vga16fb_setup(char *options
)
1286 if (!options
|| !*options
)
1289 while ((this_opt
= strsep(&options
, ",")) != NULL
) {
1290 if (!*this_opt
) continue;
1296 static int __devinit
vga16fb_probe(struct platform_device
*dev
)
1298 struct fb_info
*info
;
1299 struct vga16fb_par
*par
;
1303 printk(KERN_DEBUG
"vga16fb: initializing\n");
1304 info
= framebuffer_alloc(sizeof(struct vga16fb_par
), &dev
->dev
);
1311 /* XXX share VGA_FB_PHYS and I/O region with vgacon and others */
1312 info
->screen_base
= (void __iomem
*)VGA_MAP_MEM(VGA_FB_PHYS
, 0);
1314 if (!info
->screen_base
) {
1315 printk(KERN_ERR
"vga16fb: unable to map device\n");
1320 printk(KERN_INFO
"vga16fb: mapped to 0x%p\n", info
->screen_base
);
1323 par
->isVGA
= screen_info
.orig_video_isVGA
;
1324 par
->palette_blanked
= 0;
1325 par
->vesa_blanked
= 0;
1327 i
= par
->isVGA
? 6 : 2;
1329 vga16fb_defined
.red
.length
= i
;
1330 vga16fb_defined
.green
.length
= i
;
1331 vga16fb_defined
.blue
.length
= i
;
1333 /* name should not depend on EGA/VGA */
1334 info
->fbops
= &vga16fb_ops
;
1335 info
->var
= vga16fb_defined
;
1336 info
->fix
= vga16fb_fix
;
1337 /* supports rectangles with widths of multiples of 8 */
1338 info
->pixmap
.blit_x
= 1 << 7 | 1 << 15 | 1 << 23 | 1 << 31;
1339 info
->flags
= FBINFO_FLAG_DEFAULT
|
1340 FBINFO_HWACCEL_YPAN
;
1342 i
= (info
->var
.bits_per_pixel
== 8) ? 256 : 16;
1343 ret
= fb_alloc_cmap(&info
->cmap
, i
, 0);
1345 printk(KERN_ERR
"vga16fb: unable to allocate colormap\n");
1347 goto err_alloc_cmap
;
1350 if (vga16fb_check_var(&info
->var
, info
)) {
1351 printk(KERN_ERR
"vga16fb: unable to validate variable\n");
1356 vga16fb_update_fix(info
);
1358 if (register_framebuffer(info
) < 0) {
1359 printk(KERN_ERR
"vga16fb: unable to register framebuffer\n");
1364 printk(KERN_INFO
"fb%d: %s frame buffer device\n",
1365 info
->node
, info
->fix
.id
);
1366 platform_set_drvdata(dev
, info
);
1371 fb_dealloc_cmap(&info
->cmap
);
1373 iounmap(info
->screen_base
);
1375 framebuffer_release(info
);
1380 static int vga16fb_remove(struct platform_device
*dev
)
1382 struct fb_info
*info
= platform_get_drvdata(dev
);
1385 unregister_framebuffer(info
);
1386 iounmap(info
->screen_base
);
1387 fb_dealloc_cmap(&info
->cmap
);
1388 /* XXX unshare VGA regions */
1389 framebuffer_release(info
);
1395 static struct platform_driver vga16fb_driver
= {
1396 .probe
= vga16fb_probe
,
1397 .remove
= vga16fb_remove
,
1403 static struct platform_device
*vga16fb_device
;
1405 static int __init
vga16fb_init(void)
1409 char *option
= NULL
;
1411 if (fb_get_options("vga16fb", &option
))
1414 vga16fb_setup(option
);
1416 ret
= platform_driver_register(&vga16fb_driver
);
1419 vga16fb_device
= platform_device_alloc("vga16fb", 0);
1422 ret
= platform_device_add(vga16fb_device
);
1427 platform_device_put(vga16fb_device
);
1428 platform_driver_unregister(&vga16fb_driver
);
1435 static void __exit
vga16fb_exit(void)
1437 platform_device_unregister(vga16fb_device
);
1438 platform_driver_unregister(&vga16fb_driver
);
1441 MODULE_DESCRIPTION("Legacy VGA framebuffer device driver");
1442 MODULE_LICENSE("GPL");
1443 module_init(vga16fb_init
);
1444 module_exit(vga16fb_exit
);
1448 * Overrides for Emacs so that we follow Linus's tabbing style.
1449 * ---------------------------------------------------------------------------