123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490 |
- /*
- * linux/drivers/video/vgastate.c -- VGA state save/restore
- *
- * Copyright 2002 James Simmons
- *
- * Copyright history from vga16fb.c:
- * Copyright 1999 Ben Pfaff and Petr Vandrovec
- * Based on VGA info at http://www.goodnet.com/~tinara/FreeVGA/home.htm
- * Based on VESA framebuffer (c) 1998 Gerd Knorr
- *
- * 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/slab.h>
- #include <linux/fb.h>
- #include <linux/vmalloc.h>
- #include <video/vga.h>
- struct regstate {
- __u8 *vga_font0;
- __u8 *vga_font1;
- __u8 *vga_text;
- __u8 *vga_cmap;
- __u8 *attr;
- __u8 *crtc;
- __u8 *gfx;
- __u8 *seq;
- __u8 misc;
- };
- static inline unsigned char vga_rcrtcs(void __iomem *regbase, unsigned short iobase,
- unsigned char reg)
- {
- vga_w(regbase, iobase + 0x4, reg);
- return vga_r(regbase, iobase + 0x5);
- }
- static inline void vga_wcrtcs(void __iomem *regbase, unsigned short iobase,
- unsigned char reg, unsigned char val)
- {
- vga_w(regbase, iobase + 0x4, reg);
- vga_w(regbase, iobase + 0x5, val);
- }
- static void save_vga_text(struct vgastate *state, void __iomem *fbbase)
- {
- struct regstate *saved = (struct regstate *) state->vidstate;
- int i;
- u8 misc, attr10, gr4, gr5, gr6, seq1, seq2, seq4;
- unsigned short iobase;
- /* if in graphics mode, no need to save */
- misc = vga_r(state->vgabase, VGA_MIS_R);
- iobase = (misc & 1) ? 0x3d0 : 0x3b0;
- vga_r(state->vgabase, iobase + 0xa);
- vga_w(state->vgabase, VGA_ATT_W, 0x00);
- attr10 = vga_rattr(state->vgabase, 0x10);
- vga_r(state->vgabase, iobase + 0xa);
- vga_w(state->vgabase, VGA_ATT_W, 0x20);
- if (attr10 & 1)
- return;
- /* save regs */
- gr4 = vga_rgfx(state->vgabase, VGA_GFX_PLANE_READ);
- gr5 = vga_rgfx(state->vgabase, VGA_GFX_MODE);
- gr6 = vga_rgfx(state->vgabase, VGA_GFX_MISC);
- seq2 = vga_rseq(state->vgabase, VGA_SEQ_PLANE_WRITE);
- seq4 = vga_rseq(state->vgabase, VGA_SEQ_MEMORY_MODE);
- /* blank screen */
- seq1 = vga_rseq(state->vgabase, VGA_SEQ_CLOCK_MODE);
- vga_wseq(state->vgabase, VGA_SEQ_RESET, 0x1);
- vga_wseq(state->vgabase, VGA_SEQ_CLOCK_MODE, seq1 | 1 << 5);
- vga_wseq(state->vgabase, VGA_SEQ_RESET, 0x3);
- /* save font at plane 2 */
- if (state->flags & VGA_SAVE_FONT0) {
- vga_wseq(state->vgabase, VGA_SEQ_PLANE_WRITE, 0x4);
- vga_wseq(state->vgabase, VGA_SEQ_MEMORY_MODE, 0x6);
- vga_wgfx(state->vgabase, VGA_GFX_PLANE_READ, 0x2);
- vga_wgfx(state->vgabase, VGA_GFX_MODE, 0x0);
- vga_wgfx(state->vgabase, VGA_GFX_MISC, 0x5);
- for (i = 0; i < 4 * 8192; i++)
- saved->vga_font0[i] = vga_r(fbbase, i);
- }
- /* save font at plane 3 */
- if (state->flags & VGA_SAVE_FONT1) {
- vga_wseq(state->vgabase, VGA_SEQ_PLANE_WRITE, 0x8);
- vga_wseq(state->vgabase, VGA_SEQ_MEMORY_MODE, 0x6);
- vga_wgfx(state->vgabase, VGA_GFX_PLANE_READ, 0x3);
- vga_wgfx(state->vgabase, VGA_GFX_MODE, 0x0);
- vga_wgfx(state->vgabase, VGA_GFX_MISC, 0x5);
- for (i = 0; i < state->memsize; i++)
- saved->vga_font1[i] = vga_r(fbbase, i);
- }
- /* save font at plane 0/1 */
- if (state->flags & VGA_SAVE_TEXT) {
- vga_wseq(state->vgabase, VGA_SEQ_PLANE_WRITE, 0x1);
- vga_wseq(state->vgabase, VGA_SEQ_MEMORY_MODE, 0x6);
- vga_wgfx(state->vgabase, VGA_GFX_PLANE_READ, 0x0);
- vga_wgfx(state->vgabase, VGA_GFX_MODE, 0x0);
- vga_wgfx(state->vgabase, VGA_GFX_MISC, 0x5);
- for (i = 0; i < 8192; i++)
- saved->vga_text[i] = vga_r(fbbase, i);
- vga_wseq(state->vgabase, VGA_SEQ_PLANE_WRITE, 0x2);
- vga_wseq(state->vgabase, VGA_SEQ_MEMORY_MODE, 0x6);
- vga_wgfx(state->vgabase, VGA_GFX_PLANE_READ, 0x1);
- vga_wgfx(state->vgabase, VGA_GFX_MODE, 0x0);
- vga_wgfx(state->vgabase, VGA_GFX_MISC, 0x5);
- for (i = 0; i < 8192; i++)
- saved->vga_text[8192+i] = vga_r(fbbase + 2 * 8192, i);
- }
- /* restore regs */
- vga_wseq(state->vgabase, VGA_SEQ_PLANE_WRITE, seq2);
- vga_wseq(state->vgabase, VGA_SEQ_MEMORY_MODE, seq4);
- vga_wgfx(state->vgabase, VGA_GFX_PLANE_READ, gr4);
- vga_wgfx(state->vgabase, VGA_GFX_MODE, gr5);
- vga_wgfx(state->vgabase, VGA_GFX_MISC, gr6);
- /* unblank screen */
- vga_wseq(state->vgabase, VGA_SEQ_RESET, 0x1);
- vga_wseq(state->vgabase, VGA_SEQ_CLOCK_MODE, seq1 & ~(1 << 5));
- vga_wseq(state->vgabase, VGA_SEQ_RESET, 0x3);
- vga_wseq(state->vgabase, VGA_SEQ_CLOCK_MODE, seq1);
- }
- static void restore_vga_text(struct vgastate *state, void __iomem *fbbase)
- {
- struct regstate *saved = (struct regstate *) state->vidstate;
- int i;
- u8 gr1, gr3, gr4, gr5, gr6, gr8;
- u8 seq1, seq2, seq4;
- /* save regs */
- gr1 = vga_rgfx(state->vgabase, VGA_GFX_SR_ENABLE);
- gr3 = vga_rgfx(state->vgabase, VGA_GFX_DATA_ROTATE);
- gr4 = vga_rgfx(state->vgabase, VGA_GFX_PLANE_READ);
- gr5 = vga_rgfx(state->vgabase, VGA_GFX_MODE);
- gr6 = vga_rgfx(state->vgabase, VGA_GFX_MISC);
- gr8 = vga_rgfx(state->vgabase, VGA_GFX_BIT_MASK);
- seq2 = vga_rseq(state->vgabase, VGA_SEQ_PLANE_WRITE);
- seq4 = vga_rseq(state->vgabase, VGA_SEQ_MEMORY_MODE);
- /* blank screen */
- seq1 = vga_rseq(state->vgabase, VGA_SEQ_CLOCK_MODE);
- vga_wseq(state->vgabase, VGA_SEQ_RESET, 0x1);
- vga_wseq(state->vgabase, VGA_SEQ_CLOCK_MODE, seq1 | 1 << 5);
- vga_wseq(state->vgabase, VGA_SEQ_RESET, 0x3);
- if (state->depth == 4) {
- vga_wgfx(state->vgabase, VGA_GFX_DATA_ROTATE, 0x0);
- vga_wgfx(state->vgabase, VGA_GFX_BIT_MASK, 0xff);
- vga_wgfx(state->vgabase, VGA_GFX_SR_ENABLE, 0x00);
- }
- /* restore font at plane 2 */
- if (state->flags & VGA_SAVE_FONT0) {
- vga_wseq(state->vgabase, VGA_SEQ_PLANE_WRITE, 0x4);
- vga_wseq(state->vgabase, VGA_SEQ_MEMORY_MODE, 0x6);
- vga_wgfx(state->vgabase, VGA_GFX_PLANE_READ, 0x2);
- vga_wgfx(state->vgabase, VGA_GFX_MODE, 0x0);
- vga_wgfx(state->vgabase, VGA_GFX_MISC, 0x5);
- for (i = 0; i < 4 * 8192; i++)
- vga_w(fbbase, i, saved->vga_font0[i]);
- }
- /* restore font at plane 3 */
- if (state->flags & VGA_SAVE_FONT1) {
- vga_wseq(state->vgabase, VGA_SEQ_PLANE_WRITE, 0x8);
- vga_wseq(state->vgabase, VGA_SEQ_MEMORY_MODE, 0x6);
- vga_wgfx(state->vgabase, VGA_GFX_PLANE_READ, 0x3);
- vga_wgfx(state->vgabase, VGA_GFX_MODE, 0x0);
- vga_wgfx(state->vgabase, VGA_GFX_MISC, 0x5);
- for (i = 0; i < state->memsize; i++)
- vga_w(fbbase, i, saved->vga_font1[i]);
- }
- /* restore font at plane 0/1 */
- if (state->flags & VGA_SAVE_TEXT) {
- vga_wseq(state->vgabase, VGA_SEQ_PLANE_WRITE, 0x1);
- vga_wseq(state->vgabase, VGA_SEQ_MEMORY_MODE, 0x6);
- vga_wgfx(state->vgabase, VGA_GFX_PLANE_READ, 0x0);
- vga_wgfx(state->vgabase, VGA_GFX_MODE, 0x0);
- vga_wgfx(state->vgabase, VGA_GFX_MISC, 0x5);
- for (i = 0; i < 8192; i++)
- vga_w(fbbase, i, saved->vga_text[i]);
- vga_wseq(state->vgabase, VGA_SEQ_PLANE_WRITE, 0x2);
- vga_wseq(state->vgabase, VGA_SEQ_MEMORY_MODE, 0x6);
- vga_wgfx(state->vgabase, VGA_GFX_PLANE_READ, 0x1);
- vga_wgfx(state->vgabase, VGA_GFX_MODE, 0x0);
- vga_wgfx(state->vgabase, VGA_GFX_MISC, 0x5);
- for (i = 0; i < 8192; i++)
- vga_w(fbbase, i, saved->vga_text[8192+i]);
- }
- /* unblank screen */
- vga_wseq(state->vgabase, VGA_SEQ_RESET, 0x1);
- vga_wseq(state->vgabase, VGA_SEQ_CLOCK_MODE, seq1 & ~(1 << 5));
- vga_wseq(state->vgabase, VGA_SEQ_RESET, 0x3);
- /* restore regs */
- vga_wgfx(state->vgabase, VGA_GFX_SR_ENABLE, gr1);
- vga_wgfx(state->vgabase, VGA_GFX_DATA_ROTATE, gr3);
- vga_wgfx(state->vgabase, VGA_GFX_PLANE_READ, gr4);
- vga_wgfx(state->vgabase, VGA_GFX_MODE, gr5);
- vga_wgfx(state->vgabase, VGA_GFX_MISC, gr6);
- vga_wgfx(state->vgabase, VGA_GFX_BIT_MASK, gr8);
- vga_wseq(state->vgabase, VGA_SEQ_CLOCK_MODE, seq1);
- vga_wseq(state->vgabase, VGA_SEQ_PLANE_WRITE, seq2);
- vga_wseq(state->vgabase, VGA_SEQ_MEMORY_MODE, seq4);
- }
- static void save_vga_mode(struct vgastate *state)
- {
- struct regstate *saved = (struct regstate *) state->vidstate;
- unsigned short iobase;
- int i;
- saved->misc = vga_r(state->vgabase, VGA_MIS_R);
- if (saved->misc & 1)
- iobase = 0x3d0;
- else
- iobase = 0x3b0;
- for (i = 0; i < state->num_crtc; i++)
- saved->crtc[i] = vga_rcrtcs(state->vgabase, iobase, i);
- vga_r(state->vgabase, iobase + 0xa);
- vga_w(state->vgabase, VGA_ATT_W, 0x00);
- for (i = 0; i < state->num_attr; i++) {
- vga_r(state->vgabase, iobase + 0xa);
- saved->attr[i] = vga_rattr(state->vgabase, i);
- }
- vga_r(state->vgabase, iobase + 0xa);
- vga_w(state->vgabase, VGA_ATT_W, 0x20);
- for (i = 0; i < state->num_gfx; i++)
- saved->gfx[i] = vga_rgfx(state->vgabase, i);
- for (i = 0; i < state->num_seq; i++)
- saved->seq[i] = vga_rseq(state->vgabase, i);
- }
- static void restore_vga_mode(struct vgastate *state)
- {
- struct regstate *saved = (struct regstate *) state->vidstate;
- unsigned short iobase;
- int i;
- vga_w(state->vgabase, VGA_MIS_W, saved->misc);
- if (saved->misc & 1)
- iobase = 0x3d0;
- else
- iobase = 0x3b0;
- /* turn off display */
- vga_wseq(state->vgabase, VGA_SEQ_CLOCK_MODE,
- saved->seq[VGA_SEQ_CLOCK_MODE] | 0x20);
- /* disable sequencer */
- vga_wseq(state->vgabase, VGA_SEQ_RESET, 0x01);
- /* enable palette addressing */
- vga_r(state->vgabase, iobase + 0xa);
- vga_w(state->vgabase, VGA_ATT_W, 0x00);
- for (i = 2; i < state->num_seq; i++)
- vga_wseq(state->vgabase, i, saved->seq[i]);
- /* unprotect vga regs */
- vga_wcrtcs(state->vgabase, iobase, 17, saved->crtc[17] & ~0x80);
- for (i = 0; i < state->num_crtc; i++)
- vga_wcrtcs(state->vgabase, iobase, i, saved->crtc[i]);
- for (i = 0; i < state->num_gfx; i++)
- vga_wgfx(state->vgabase, i, saved->gfx[i]);
- for (i = 0; i < state->num_attr; i++) {
- vga_r(state->vgabase, iobase + 0xa);
- vga_wattr(state->vgabase, i, saved->attr[i]);
- }
- /* reenable sequencer */
- vga_wseq(state->vgabase, VGA_SEQ_RESET, 0x03);
- /* turn display on */
- vga_wseq(state->vgabase, VGA_SEQ_CLOCK_MODE,
- saved->seq[VGA_SEQ_CLOCK_MODE] & ~(1 << 5));
- /* disable video/palette source */
- vga_r(state->vgabase, iobase + 0xa);
- vga_w(state->vgabase, VGA_ATT_W, 0x20);
- }
- static void save_vga_cmap(struct vgastate *state)
- {
- struct regstate *saved = (struct regstate *) state->vidstate;
- int i;
- vga_w(state->vgabase, VGA_PEL_MSK, 0xff);
- /* assumes DAC is readable and writable */
- vga_w(state->vgabase, VGA_PEL_IR, 0x00);
- for (i = 0; i < 768; i++)
- saved->vga_cmap[i] = vga_r(state->vgabase, VGA_PEL_D);
- }
- static void restore_vga_cmap(struct vgastate *state)
- {
- struct regstate *saved = (struct regstate *) state->vidstate;
- int i;
- vga_w(state->vgabase, VGA_PEL_MSK, 0xff);
- /* assumes DAC is readable and writable */
- vga_w(state->vgabase, VGA_PEL_IW, 0x00);
- for (i = 0; i < 768; i++)
- vga_w(state->vgabase, VGA_PEL_D, saved->vga_cmap[i]);
- }
- static void vga_cleanup(struct vgastate *state)
- {
- if (state->vidstate != NULL) {
- struct regstate *saved = (struct regstate *) state->vidstate;
- vfree(saved->vga_font0);
- vfree(saved->vga_font1);
- vfree(saved->vga_text);
- vfree(saved->vga_cmap);
- vfree(saved->attr);
- kfree(saved);
- state->vidstate = NULL;
- }
- }
- int save_vga(struct vgastate *state)
- {
- struct regstate *saved;
- saved = kzalloc(sizeof(struct regstate), GFP_KERNEL);
- if (saved == NULL)
- return 1;
- state->vidstate = (void *)saved;
- if (state->flags & VGA_SAVE_CMAP) {
- saved->vga_cmap = vmalloc(768);
- if (!saved->vga_cmap) {
- vga_cleanup(state);
- return 1;
- }
- save_vga_cmap(state);
- }
- if (state->flags & VGA_SAVE_MODE) {
- int total;
- if (state->num_attr < 21)
- state->num_attr = 21;
- if (state->num_crtc < 25)
- state->num_crtc = 25;
- if (state->num_gfx < 9)
- state->num_gfx = 9;
- if (state->num_seq < 5)
- state->num_seq = 5;
- total = state->num_attr + state->num_crtc +
- state->num_gfx + state->num_seq;
- saved->attr = vmalloc(total);
- if (!saved->attr) {
- vga_cleanup(state);
- return 1;
- }
- saved->crtc = saved->attr + state->num_attr;
- saved->gfx = saved->crtc + state->num_crtc;
- saved->seq = saved->gfx + state->num_gfx;
- save_vga_mode(state);
- }
- if (state->flags & VGA_SAVE_FONTS) {
- void __iomem *fbbase;
- /* exit if window is less than 32K */
- if (state->memsize && state->memsize < 4 * 8192) {
- vga_cleanup(state);
- return 1;
- }
- if (!state->memsize)
- state->memsize = 8 * 8192;
- if (!state->membase)
- state->membase = 0xA0000;
- fbbase = ioremap(state->membase, state->memsize);
- if (!fbbase) {
- vga_cleanup(state);
- return 1;
- }
- /*
- * save only first 32K used by vgacon
- */
- if (state->flags & VGA_SAVE_FONT0) {
- saved->vga_font0 = vmalloc(4 * 8192);
- if (!saved->vga_font0) {
- iounmap(fbbase);
- vga_cleanup(state);
- return 1;
- }
- }
- /*
- * largely unused, but if required by the caller
- * we'll just save everything.
- */
- if (state->flags & VGA_SAVE_FONT1) {
- saved->vga_font1 = vmalloc(state->memsize);
- if (!saved->vga_font1) {
- iounmap(fbbase);
- vga_cleanup(state);
- return 1;
- }
- }
- /*
- * Save 8K at plane0[0], and 8K at plane1[16K]
- */
- if (state->flags & VGA_SAVE_TEXT) {
- saved->vga_text = vmalloc(8192 * 2);
- if (!saved->vga_text) {
- iounmap(fbbase);
- vga_cleanup(state);
- return 1;
- }
- }
- save_vga_text(state, fbbase);
- iounmap(fbbase);
- }
- return 0;
- }
- int restore_vga (struct vgastate *state)
- {
- if (state->vidstate == NULL)
- return 1;
- if (state->flags & VGA_SAVE_MODE)
- restore_vga_mode(state);
- if (state->flags & VGA_SAVE_FONTS) {
- void __iomem *fbbase = ioremap(state->membase, state->memsize);
- if (!fbbase) {
- vga_cleanup(state);
- return 1;
- }
- restore_vga_text(state, fbbase);
- iounmap(fbbase);
- }
- if (state->flags & VGA_SAVE_CMAP)
- restore_vga_cmap(state);
- vga_cleanup(state);
- return 0;
- }
- EXPORT_SYMBOL(save_vga);
- EXPORT_SYMBOL(restore_vga);
- MODULE_AUTHOR("James Simmons <jsimmons@users.sf.net>");
- MODULE_DESCRIPTION("VGA State Save/Restore");
- MODULE_LICENSE("GPL");
|