vga16fb.c
上传用户:lgb322
上传日期:2013-02-24
资源大小:30529k
文件大小:26k
源码类别:

嵌入式Linux

开发平台:

Unix_Linux

  1. /*
  2.  * linux/drivers/video/vga16.c -- VGA 16-color framebuffer driver
  3.  * 
  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>
  7.  *
  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.  */
  11. #include <linux/module.h>
  12. #include <linux/kernel.h>
  13. #include <linux/errno.h>
  14. #include <linux/string.h>
  15. #include <linux/mm.h>
  16. #include <linux/tty.h>
  17. #include <linux/slab.h>
  18. #include <linux/delay.h>
  19. #include <linux/fb.h>
  20. #include <linux/console.h>
  21. #include <linux/selection.h>
  22. #include <linux/ioport.h>
  23. #include <linux/init.h>
  24. #include <asm/io.h>
  25. #include <video/fbcon.h>
  26. #include <video/fbcon-vga-planes.h>
  27. #include "vga.h"
  28. #define dac_reg (0x3c8)
  29. #define dac_val (0x3c9)
  30. #define VGA_FB_PHYS 0xA0000
  31. #define VGA_FB_PHYS_LEN 65536
  32. /* --------------------------------------------------------------------- */
  33. /*
  34.  * card parameters
  35.  */
  36. static struct vga16fb_info {
  37. struct fb_info  fb_info;
  38. char *video_vbase; /* 0xa0000 map address */
  39. int isVGA;
  40. /* structure holding original VGA register settings when the
  41.            screen is blanked */
  42. struct {
  43. unsigned char SeqCtrlIndex; /* Sequencer Index reg.   */
  44. unsigned char CrtCtrlIndex; /* CRT-Contr. Index reg.  */
  45. unsigned char CrtMiscIO; /* Miscellaneous register */
  46. unsigned char HorizontalTotal; /* CRT-Controller:00h */
  47. unsigned char HorizDisplayEnd; /* CRT-Controller:01h */
  48. unsigned char StartHorizRetrace; /* CRT-Controller:04h */
  49. unsigned char EndHorizRetrace; /* CRT-Controller:05h */
  50. unsigned char Overflow; /* CRT-Controller:07h */
  51. unsigned char StartVertRetrace; /* CRT-Controller:10h */
  52. unsigned char EndVertRetrace; /* CRT-Controller:11h */
  53. unsigned char ModeControl; /* CRT-Controller:17h */
  54. unsigned char ClockingMode; /* Seq-Controller:01h */
  55. } vga_state;
  56. int palette_blanked;
  57. int vesa_blanked;
  58. } vga16fb;
  59. struct vga16fb_par {
  60. u8 crtc[VGA_CRT_C];
  61. u8 atc[VGA_ATT_C];
  62. u8 gdc[VGA_GFX_C];
  63. u8 seq[VGA_SEQ_C];
  64. u8 misc;
  65. u8 vss;
  66. struct fb_var_screeninfo var;
  67. };
  68. /* --------------------------------------------------------------------- */
  69. static struct fb_var_screeninfo vga16fb_defined = {
  70. 640,480,640,480,/* W,H, W, H (virtual) load xres,xres_virtual*/
  71. 0,0, /* virtual -> visible no offset */
  72. 4, /* depth -> load bits_per_pixel */
  73. 0, /* greyscale ? */
  74. {0,0,0}, /* R */
  75. {0,0,0}, /* G */
  76. {0,0,0}, /* B */
  77. {0,0,0}, /* transparency */
  78. 0, /* standard pixel format */
  79. FB_ACTIVATE_NOW,
  80. -1,-1,
  81. 0,
  82. 39721, 48, 16, 39, 8,
  83. 96, 2, 0, /* No sync info */
  84. FB_VMODE_NONINTERLACED,
  85. {0,0,0,0,0,0}
  86. };
  87. static struct display disp;
  88. static struct { u_short blue, green, red, pad; } palette[256];
  89. static int             currcon   = 0;
  90. /* --------------------------------------------------------------------- */
  91. static void vga16fb_pan_var(struct fb_info *info, struct fb_var_screeninfo *var)
  92. {
  93. u32 pos = (var->xres_virtual * var->yoffset + var->xoffset) >> 3;
  94. outb(VGA_CRTC_START_HI, VGA_CRT_IC);
  95. outb(pos >> 8, VGA_CRT_DC);
  96. outb(VGA_CRTC_START_LO, VGA_CRT_IC);
  97. outb(pos & 0xFF, VGA_CRT_DC);
  98. #if 0
  99. /* if someone supports xoffset in bit resolution */
  100. inb(VGA_IS1_RC); /* reset flip-flop */
  101. outb(VGA_ATC_PEL, VGA_ATT_IW);
  102. outb(xoffset & 7, VGA_ATT_IW);
  103. inb(VGA_IS1_RC);
  104. outb(0x20, VGA_ATT_IW);
  105. #endif
  106. }
  107. static int vga16fb_update_var(int con, struct fb_info *info)
  108. {
  109. vga16fb_pan_var(info, &fb_display[con].var);
  110. return 0;
  111. }
  112. static int vga16fb_get_fix(struct fb_fix_screeninfo *fix, int con,
  113.    struct fb_info *info)
  114. {
  115. struct display *p;
  116. if (con < 0)
  117. p = &disp;
  118. else
  119. p = fb_display + con;
  120. memset(fix, 0, sizeof(struct fb_fix_screeninfo));
  121. strcpy(fix->id,"VGA16 VGA");
  122. fix->smem_start = VGA_FB_PHYS;
  123. fix->smem_len = VGA_FB_PHYS_LEN;
  124. fix->type = FB_TYPE_VGA_PLANES;
  125. fix->visual = FB_VISUAL_PSEUDOCOLOR;
  126. fix->xpanstep  = 8;
  127. fix->ypanstep  = 1;
  128. fix->ywrapstep = 0;
  129. fix->line_length = p->var.xres_virtual / 8;
  130. return 0;
  131. }
  132. static int vga16fb_get_var(struct fb_var_screeninfo *var, int con,
  133.  struct fb_info *info)
  134. {
  135. if(con==-1)
  136. memcpy(var, &vga16fb_defined, sizeof(struct fb_var_screeninfo));
  137. else
  138. *var=fb_display[con].var;
  139. return 0;
  140. }
  141. static void vga16fb_set_disp(int con, struct vga16fb_info *info)
  142. {
  143. struct fb_fix_screeninfo fix;
  144. struct display *display;
  145. if (con < 0)
  146. display = &disp;
  147. else
  148. display = fb_display + con;
  149. vga16fb_get_fix(&fix, con, &info->fb_info);
  150. display->screen_base = info->video_vbase;
  151. display->visual = fix.visual;
  152. display->type = fix.type;
  153. display->type_aux = fix.type_aux;
  154. display->ypanstep = fix.ypanstep;
  155. display->ywrapstep = fix.ywrapstep;
  156. display->line_length = fix.line_length;
  157. display->next_line = fix.line_length;
  158. display->can_soft_blank = 1;
  159. display->inverse = 0;
  160. if (info->isVGA)
  161. display->dispsw = &fbcon_vga_planes;
  162. else
  163. display->dispsw = &fbcon_ega_planes;
  164. display->scrollmode = SCROLL_YREDRAW;
  165. }
  166. static void vga16fb_encode_var(struct fb_var_screeninfo *var,
  167.        const struct vga16fb_par *par,
  168.        const struct vga16fb_info *info)
  169. {
  170. *var = par->var;
  171. }
  172. static void vga16fb_clock_chip(struct vga16fb_par *par,
  173.        unsigned int pixclock,
  174.        const struct vga16fb_info *info)
  175. {
  176. static struct {
  177. u32 pixclock;
  178. u8  misc;
  179. u8  seq_clock_mode;
  180. } *ptr, *best, vgaclocks[] = {
  181. { 79442 /* 12.587 */, 0x00, 0x08},
  182. { 70616 /* 14.161 */, 0x04, 0x08},
  183. { 39721 /* 25.175 */, 0x00, 0x00},
  184. { 35308 /* 28.322 */, 0x04, 0x00},
  185. {     0 /* bad */,    0x00, 0x00}};
  186. int err;
  187. best = vgaclocks;
  188. err = pixclock - best->pixclock;
  189. if (err < 0) err = -err;
  190. for (ptr = vgaclocks + 1; ptr->pixclock; ptr++) {
  191. int tmp;
  192. tmp = pixclock - ptr->pixclock;
  193. if (tmp < 0) tmp = -tmp;
  194. if (tmp < err) {
  195. err = tmp;
  196. best = ptr;
  197. }
  198. }
  199. par->misc |= best->misc;
  200. par->seq[VGA_SEQ_CLOCK_MODE] |= best->seq_clock_mode;
  201. par->var.pixclock = best->pixclock;
  202. }
  203.        
  204. #define FAIL(X) return -EINVAL
  205. static int vga16fb_decode_var(const struct fb_var_screeninfo *var,
  206.       struct vga16fb_par *par,
  207.       const struct vga16fb_info *info)
  208. {
  209. u32 xres, right, hslen, left, xtotal;
  210. u32 yres, lower, vslen, upper, ytotal;
  211. u32 vxres, xoffset, vyres, yoffset;
  212. u32 pos;
  213. u8 r7, rMode;
  214. int i;
  215. if (var->bits_per_pixel != 4)
  216. return -EINVAL;
  217. xres = (var->xres + 7) & ~7;
  218. vxres = (var->xres_virtual + 0xF) & ~0xF;
  219. xoffset = (var->xoffset + 7) & ~7;
  220. left = (var->left_margin + 7) & ~7;
  221. right = (var->right_margin + 7) & ~7;
  222. hslen = (var->hsync_len + 7) & ~7;
  223. if (vxres < xres)
  224. vxres = xres;
  225. if (xres + xoffset > vxres)
  226. xoffset = vxres - xres;
  227. par->var.xres = xres;
  228. par->var.right_margin = right;
  229. par->var.hsync_len = hslen;
  230. par->var.left_margin = left;
  231. par->var.xres_virtual = vxres;
  232. par->var.xoffset = xoffset;
  233. xres >>= 3;
  234. right >>= 3;
  235. hslen >>= 3;
  236. left >>= 3;
  237. vxres >>= 3;
  238. xtotal = xres + right + hslen + left;
  239. if (xtotal >= 256)
  240. FAIL("xtotal too big");
  241. if (hslen > 32)
  242. FAIL("hslen too big");
  243. if (right + hslen + left > 64)
  244. FAIL("hblank too big");
  245. par->crtc[VGA_CRTC_H_TOTAL] = xtotal - 5;
  246. par->crtc[VGA_CRTC_H_BLANK_START] = xres - 1;
  247. par->crtc[VGA_CRTC_H_DISP] = xres - 1;
  248. pos = xres + right;
  249. par->crtc[VGA_CRTC_H_SYNC_START] = pos;
  250. pos += hslen;
  251. par->crtc[VGA_CRTC_H_SYNC_END] = pos & 0x1F;
  252. pos += left - 2; /* blank_end + 2 <= total + 5 */
  253. par->crtc[VGA_CRTC_H_BLANK_END] = (pos & 0x1F) | 0x80;
  254. if (pos & 0x20)
  255. par->crtc[VGA_CRTC_H_SYNC_END] |= 0x80;
  256. yres = var->yres;
  257. lower = var->lower_margin;
  258. vslen = var->vsync_len;
  259. upper = var->upper_margin;
  260. vyres = var->yres_virtual;
  261. yoffset = var->yoffset;
  262. if (yres > vyres)
  263. vyres = yres;
  264. if (vxres * vyres > 65536) {
  265. vyres = 65536 / vxres;
  266. if (vyres < yres)
  267. return -ENOMEM;
  268. }
  269. if (yoffset + yres > vyres)
  270. yoffset = vyres - yres;
  271. par->var.yres = yres;
  272. par->var.lower_margin = lower;
  273. par->var.vsync_len = vslen;
  274. par->var.upper_margin = upper;
  275. par->var.yres_virtual = vyres;
  276. par->var.yoffset = yoffset;
  277. if (var->vmode & FB_VMODE_DOUBLE) {
  278. yres <<= 1;
  279. lower <<= 1;
  280. vslen <<= 1;
  281. upper <<= 1;
  282. }
  283. ytotal = yres + lower + vslen + upper;
  284. if (ytotal > 1024) {
  285. ytotal >>= 1;
  286. yres >>= 1;
  287. lower >>= 1;
  288. vslen >>= 1;
  289. upper >>= 1;
  290. rMode = 0x04;
  291. } else
  292. rMode = 0x00;
  293. if (ytotal > 1024)
  294. FAIL("ytotal too big");
  295. if (vslen > 16)
  296. FAIL("vslen too big");
  297. par->crtc[VGA_CRTC_V_TOTAL] = ytotal - 2;
  298. r7 = 0x10; /* disable linecompare */
  299. if (ytotal & 0x100) r7 |= 0x01;
  300. if (ytotal & 0x200) r7 |= 0x20;
  301. par->crtc[VGA_CRTC_PRESET_ROW] = 0;
  302. par->crtc[VGA_CRTC_MAX_SCAN] = 0x40; /* 1 scanline, no linecmp */
  303. par->var.vmode = var->vmode;
  304. if (var->vmode & FB_VMODE_DOUBLE)
  305. par->crtc[VGA_CRTC_MAX_SCAN] |= 0x80;
  306. par->crtc[VGA_CRTC_CURSOR_START] = 0x20;
  307. par->crtc[VGA_CRTC_CURSOR_END]   = 0x00;
  308. pos = yoffset * vxres + (xoffset >> 3);
  309. par->crtc[VGA_CRTC_START_HI]     = pos >> 8;
  310. par->crtc[VGA_CRTC_START_LO]     = pos & 0xFF;
  311. par->crtc[VGA_CRTC_CURSOR_HI]    = 0x00;
  312. par->crtc[VGA_CRTC_CURSOR_LO]    = 0x00;
  313. pos = yres - 1;
  314. par->crtc[VGA_CRTC_V_DISP_END] = pos & 0xFF;
  315. par->crtc[VGA_CRTC_V_BLANK_START] = pos & 0xFF;
  316. if (pos & 0x100)
  317. r7 |= 0x0A; /* 0x02 -> DISP_END, 0x08 -> BLANK_START */
  318. if (pos & 0x200) {
  319. r7 |= 0x40; /* 0x40 -> DISP_END */
  320. par->crtc[VGA_CRTC_MAX_SCAN] |= 0x20; /* BLANK_START */
  321. }
  322. pos += lower;
  323. par->crtc[VGA_CRTC_V_SYNC_START] = pos & 0xFF;
  324. if (pos & 0x100)
  325. r7 |= 0x04;
  326. if (pos & 0x200)
  327. r7 |= 0x80;
  328. pos += vslen;
  329. par->crtc[VGA_CRTC_V_SYNC_END] = (pos & 0x0F) & ~0x10; /* disabled IRQ */
  330. pos += upper - 1; /* blank_end + 1 <= ytotal + 2 */
  331. par->crtc[VGA_CRTC_V_BLANK_END] = pos & 0xFF; /* 0x7F for original VGA,
  332.                      but some SVGA chips requires all 8 bits to set */
  333. if (vxres >= 512)
  334. FAIL("vxres too long");
  335. par->crtc[VGA_CRTC_OFFSET] = vxres >> 1;
  336. par->crtc[VGA_CRTC_UNDERLINE] = 0x1F;
  337. par->crtc[VGA_CRTC_MODE] = rMode | 0xE3;
  338. par->crtc[VGA_CRTC_LINE_COMPARE] = 0xFF;
  339. par->crtc[VGA_CRTC_OVERFLOW] = r7;
  340. par->vss = 0x00; /* 3DA */
  341. for (i = 0x00; i < 0x10; i++)
  342. par->atc[i] = i;
  343. par->atc[VGA_ATC_MODE] = 0x81;
  344. par->atc[VGA_ATC_OVERSCAN] = 0x00; /* 0 for EGA, 0xFF for VGA */
  345. par->atc[VGA_ATC_PLANE_ENABLE] = 0x0F;
  346. par->atc[VGA_ATC_PEL] = xoffset & 7;
  347. par->atc[VGA_ATC_COLOR_PAGE] = 0x00;
  348. par->misc = 0xC3; /* enable CPU, ports 0x3Dx, positive sync */
  349. par->var.sync = var->sync;
  350. if (var->sync & FB_SYNC_HOR_HIGH_ACT)
  351. par->misc &= ~0x40;
  352. if (var->sync & FB_SYNC_VERT_HIGH_ACT)
  353. par->misc &= ~0x80;
  354. par->seq[VGA_SEQ_CLOCK_MODE] = 0x01;
  355. par->seq[VGA_SEQ_PLANE_WRITE] = 0x0F;
  356. par->seq[VGA_SEQ_CHARACTER_MAP] = 0x00;
  357. par->seq[VGA_SEQ_MEMORY_MODE] = 0x06;
  358. par->gdc[VGA_GFX_SR_VALUE] = 0x00;
  359. par->gdc[VGA_GFX_SR_ENABLE] = 0x0F;
  360. par->gdc[VGA_GFX_COMPARE_VALUE] = 0x00;
  361. par->gdc[VGA_GFX_DATA_ROTATE] = 0x20;
  362. par->gdc[VGA_GFX_PLANE_READ] = 0;
  363. par->gdc[VGA_GFX_MODE] = 0x00;
  364. par->gdc[VGA_GFX_MISC] = 0x05;
  365. par->gdc[VGA_GFX_COMPARE_MASK] = 0x0F;
  366. par->gdc[VGA_GFX_BIT_MASK] = 0xFF;
  367. vga16fb_clock_chip(par, var->pixclock, info);
  368. par->var.bits_per_pixel = 4;
  369. par->var.grayscale = var->grayscale;
  370. par->var.red.offset = par->var.green.offset = par->var.blue.offset = 
  371. par->var.transp.offset = 0;
  372. par->var.red.length = par->var.green.length = par->var.blue.length =
  373. (info->isVGA) ? 6 : 2;
  374. par->var.transp.length = 0;
  375. par->var.nonstd = 0;
  376. par->var.activate = FB_ACTIVATE_NOW;
  377. par->var.height = -1;
  378. par->var.width = -1;
  379. par->var.accel_flags = 0;
  380. return 0;
  381. }
  382. #undef FAIL
  383. static int vga16fb_set_par(const struct vga16fb_par *par,
  384.    struct vga16fb_info *info)
  385. {
  386. int i;
  387. outb(inb(VGA_MIS_R) | 0x01, VGA_MIS_W);
  388. /* Enable graphics register modification */
  389. if (!info->isVGA) {
  390. outb(0x00, EGA_GFX_E0);
  391. outb(0x01, EGA_GFX_E1);
  392. }
  393. /* update misc output register */
  394. outb(par->misc, VGA_MIS_W);
  395. /* synchronous reset on */
  396. outb(0x00, VGA_SEQ_I);
  397. outb(0x01, VGA_SEQ_D);
  398. /* write sequencer registers */
  399. outb(1, VGA_SEQ_I);
  400. outb(par->seq[1] | 0x20, VGA_SEQ_D);
  401. for (i = 2; i < VGA_SEQ_C; i++) {
  402. outb(i, VGA_SEQ_I);
  403. outb(par->seq[i], VGA_SEQ_D);
  404. }
  405. /* synchronous reset off */
  406. outb(0x00, VGA_SEQ_I);
  407. outb(0x03, VGA_SEQ_D);
  408. /* deprotect CRT registers 0-7 */
  409. outb(0x11, VGA_CRT_IC);
  410. outb(par->crtc[0x11], VGA_CRT_DC);
  411. /* write CRT registers */
  412. for (i = 0; i < VGA_CRT_C; i++) {
  413. outb(i, VGA_CRT_IC);
  414. outb(par->crtc[i], VGA_CRT_DC);
  415. }
  416. /* write graphics controller registers */
  417. for (i = 0; i < VGA_GFX_C; i++) {
  418. outb(i, VGA_GFX_I);
  419. outb(par->gdc[i], VGA_GFX_D);
  420. }
  421. /* write attribute controller registers */
  422. for (i = 0; i < VGA_ATT_C; i++) {
  423. inb_p(VGA_IS1_RC); /* reset flip-flop */
  424. outb_p(i, VGA_ATT_IW);
  425. outb_p(par->atc[i], VGA_ATT_IW);
  426. }
  427. /* Wait for screen to stabilize. */
  428. mdelay(50);
  429. outb(0x01, VGA_SEQ_I);
  430. outb(par->seq[1], VGA_SEQ_D);
  431. inb(VGA_IS1_RC);
  432. outb(0x20, VGA_ATT_IW);
  433. return 0;
  434. }
  435. static int vga16fb_set_var(struct fb_var_screeninfo *var, int con,
  436.   struct fb_info *fb)
  437. {
  438. struct vga16fb_info *info = (struct vga16fb_info*)fb;
  439. struct vga16fb_par par;
  440. struct display *display;
  441. int err;
  442. if (con < 0)
  443. display = fb->disp;
  444. else
  445. display = fb_display + con;
  446. if ((err = vga16fb_decode_var(var, &par, info)) != 0)
  447. return err;
  448. vga16fb_encode_var(var, &par, info);
  449. if ((var->activate & FB_ACTIVATE_MASK) == FB_ACTIVATE_TEST)
  450. return 0;
  451. if ((var->activate & FB_ACTIVATE_MASK) == FB_ACTIVATE_NOW) {
  452. u32 oldxres, oldyres, oldvxres, oldvyres, oldbpp;
  453. oldxres = display->var.xres;
  454. oldyres = display->var.yres;
  455. oldvxres = display->var.xres_virtual;
  456. oldvyres = display->var.yres_virtual;
  457. oldbpp = display->var.bits_per_pixel;
  458. display->var = *var;
  459. if (oldxres != var->xres || oldyres != var->yres ||
  460.     oldvxres != var->xres_virtual || oldvyres != var->yres_virtual ||
  461.     oldbpp != var->bits_per_pixel) {
  462. vga16fb_set_disp(con, info);
  463. if (info->fb_info.changevar)
  464. info->fb_info.changevar(con);
  465. }
  466. if (con == currcon)
  467. vga16fb_set_par(&par, info);
  468. }
  469. return 0;
  470. }
  471. static void ega16_setpalette(int regno, unsigned red, unsigned green, unsigned blue)
  472. {
  473. static unsigned char map[] = { 000, 001, 010, 011 };
  474. int val;
  475. val = map[red>>14] | ((map[green>>14]) << 1) | ((map[blue>>14]) << 2);
  476. inb_p(0x3DA);   /* ! 0x3BA */
  477. outb_p(regno, 0x3C0);
  478. outb_p(val, 0x3C0);
  479. inb_p(0x3DA);   /* some clones need it */
  480. outb_p(0x20, 0x3C0); /* unblank screen */
  481. }
  482. static int vga16_getcolreg(unsigned regno, unsigned *red, unsigned *green,
  483.   unsigned *blue, unsigned *transp,
  484.   struct fb_info *fb_info)
  485. {
  486. /*
  487.  *  Read a single color register and split it into colors/transparent.
  488.  *  Return != 0 for invalid regno.
  489.  */
  490. if (regno >= 16)
  491. return 1;
  492. *red   = palette[regno].red;
  493. *green = palette[regno].green;
  494. *blue  = palette[regno].blue;
  495. *transp = 0;
  496. return 0;
  497. }
  498. static void vga16_setpalette(int regno, unsigned red, unsigned green, unsigned blue)
  499. {
  500. outb(regno,       dac_reg);
  501. outb(red   >> 10, dac_val);
  502. outb(green >> 10, dac_val);
  503. outb(blue  >> 10, dac_val);
  504. }
  505. static int vga16_setcolreg(unsigned regno, unsigned red, unsigned green,
  506.   unsigned blue, unsigned transp,
  507.   struct fb_info *fb_info)
  508. {
  509. int gray;
  510. /*
  511.  *  Set a single color register. The values supplied are
  512.  *  already rounded down to the hardware's capabilities
  513.  *  (according to the entries in the `var' structure). Return
  514.  *  != 0 for invalid regno.
  515.  */
  516. if (regno >= 16)
  517. return 1;
  518. palette[regno].red   = red;
  519. palette[regno].green = green;
  520. palette[regno].blue  = blue;
  521. if (currcon < 0)
  522. gray = disp.var.grayscale;
  523. else
  524. gray = fb_display[currcon].var.grayscale;
  525. if (gray) {
  526. /* gray = 0.30*R + 0.59*G + 0.11*B */
  527. red = green = blue = (red * 77 + green * 151 + blue * 28) >> 8;
  528. }
  529. if (((struct vga16fb_info *) fb_info)->isVGA) 
  530. vga16_setpalette(regno,red,green,blue);
  531. else
  532. ega16_setpalette(regno,red,green,blue);
  533. return 0;
  534. }
  535. static void do_install_cmap(int con, struct fb_info *info)
  536. {
  537. if (con != currcon)
  538. return;
  539. if (fb_display[con].cmap.len)
  540. fb_set_cmap(&fb_display[con].cmap, 1, vga16_setcolreg, info);
  541. else
  542. fb_set_cmap(fb_default_cmap(16), 1, vga16_setcolreg,
  543.     info);
  544. }
  545. static int vga16fb_get_cmap(struct fb_cmap *cmap, int kspc, int con,
  546.    struct fb_info *info)
  547. {
  548. if (con == currcon) /* current console? */
  549. return fb_get_cmap(cmap, kspc, vga16_getcolreg, info);
  550. else if (fb_display[con].cmap.len) /* non default colormap? */
  551. fb_copy_cmap(&fb_display[con].cmap, cmap, kspc ? 0 : 2);
  552. else
  553. fb_copy_cmap(fb_default_cmap(16),
  554.      cmap, kspc ? 0 : 2);
  555. return 0;
  556. }
  557. static int vga16fb_set_cmap(struct fb_cmap *cmap, int kspc, int con,
  558.    struct fb_info *info)
  559. {
  560. int err;
  561. if (!fb_display[con].cmap.len) { /* no colormap allocated? */
  562. err = fb_alloc_cmap(&fb_display[con].cmap,16,0);
  563. if (err)
  564. return err;
  565. }
  566. if (con == currcon) /* current console? */
  567. return fb_set_cmap(cmap, kspc, vga16_setcolreg, info);
  568. else
  569. fb_copy_cmap(cmap, &fb_display[con].cmap, kspc ? 0 : 1);
  570. return 0;
  571. }
  572. static int vga16fb_pan_display(struct fb_var_screeninfo *var, int con,
  573.        struct fb_info *info) 
  574. {
  575. if (var->xoffset + fb_display[con].var.xres > fb_display[con].var.xres_virtual ||
  576.     var->yoffset + fb_display[con].var.yres > fb_display[con].var.yres_virtual)
  577. return -EINVAL;
  578. if (con == currcon)
  579. vga16fb_pan_var(info, var);
  580. fb_display[con].var.xoffset = var->xoffset;
  581. fb_display[con].var.yoffset = var->yoffset;
  582. fb_display[con].var.vmode &= ~FB_VMODE_YWRAP;
  583. return 0;
  584. }
  585. static struct fb_ops vga16fb_ops = {
  586. owner: THIS_MODULE,
  587. fb_get_fix: vga16fb_get_fix,
  588. fb_get_var: vga16fb_get_var,
  589. fb_set_var: vga16fb_set_var,
  590. fb_get_cmap: vga16fb_get_cmap,
  591. fb_set_cmap: vga16fb_set_cmap,
  592. fb_pan_display: vga16fb_pan_display,
  593. };
  594. int vga16fb_setup(char *options)
  595. {
  596. char *this_opt;
  597. vga16fb.fb_info.fontname[0] = '';
  598. if (!options || !*options)
  599. return 0;
  600. while ((this_opt = strsep(&options, ",")) != NULL) {
  601. if (!*this_opt) continue;
  602. if (!strncmp(this_opt, "font:", 5))
  603. strcpy(vga16fb.fb_info.fontname, this_opt+5);
  604. }
  605. return 0;
  606. }
  607. static int vga16fb_switch(int con, struct fb_info *fb)
  608. {
  609. struct vga16fb_par par;
  610. struct vga16fb_info *info = (struct vga16fb_info*)fb;
  611. /* Do we have to save the colormap? */
  612. if (fb_display[currcon].cmap.len)
  613. fb_get_cmap(&fb_display[currcon].cmap, 1, vga16_getcolreg,
  614.     fb);
  615. currcon = con;
  616. vga16fb_decode_var(&fb_display[con].var, &par, info);
  617. vga16fb_set_par(&par, info);
  618. vga16fb_set_disp(con, info);
  619. /* Install new colormap */
  620. do_install_cmap(con, fb);
  621. /* vga16fb_update_var(con, fb); */
  622. return 1;
  623. }
  624. /* The following VESA blanking code is taken from vgacon.c.  The VGA
  625.    blanking code was originally by Huang shi chao, and modified by
  626.    Christoph Rimek (chrimek@toppoint.de) and todd j. derr
  627.    (tjd@barefoot.org) for Linux. */
  628. #define attrib_port 0x3c0
  629. #define seq_port_reg 0x3c4
  630. #define seq_port_val 0x3c5
  631. #define gr_port_reg 0x3ce
  632. #define gr_port_val 0x3cf
  633. #define video_misc_rd 0x3cc
  634. #define video_misc_wr 0x3c2
  635. #define vga_video_port_reg 0x3d4
  636. #define vga_video_port_val 0x3d5
  637. static void vga_vesa_blank(struct vga16fb_info *info, int mode)
  638. {
  639. unsigned char SeqCtrlIndex;
  640. unsigned char CrtCtrlIndex;
  641. cli();
  642. SeqCtrlIndex = inb_p(seq_port_reg);
  643. CrtCtrlIndex = inb_p(vga_video_port_reg);
  644. /* save original values of VGA controller registers */
  645. if(!info->vesa_blanked) {
  646. info->vga_state.CrtMiscIO = inb_p(video_misc_rd);
  647. sti();
  648. outb_p(0x00,vga_video_port_reg); /* HorizontalTotal */
  649. info->vga_state.HorizontalTotal = inb_p(vga_video_port_val);
  650. outb_p(0x01,vga_video_port_reg); /* HorizDisplayEnd */
  651. info->vga_state.HorizDisplayEnd = inb_p(vga_video_port_val);
  652. outb_p(0x04,vga_video_port_reg); /* StartHorizRetrace */
  653. info->vga_state.StartHorizRetrace = inb_p(vga_video_port_val);
  654. outb_p(0x05,vga_video_port_reg); /* EndHorizRetrace */
  655. info->vga_state.EndHorizRetrace = inb_p(vga_video_port_val);
  656. outb_p(0x07,vga_video_port_reg); /* Overflow */
  657. info->vga_state.Overflow = inb_p(vga_video_port_val);
  658. outb_p(0x10,vga_video_port_reg); /* StartVertRetrace */
  659. info->vga_state.StartVertRetrace = inb_p(vga_video_port_val);
  660. outb_p(0x11,vga_video_port_reg); /* EndVertRetrace */
  661. info->vga_state.EndVertRetrace = inb_p(vga_video_port_val);
  662. outb_p(0x17,vga_video_port_reg); /* ModeControl */
  663. info->vga_state.ModeControl = inb_p(vga_video_port_val);
  664. outb_p(0x01,seq_port_reg); /* ClockingMode */
  665. info->vga_state.ClockingMode = inb_p(seq_port_val);
  666. }
  667. /* assure that video is enabled */
  668. /* "0x20" is VIDEO_ENABLE_bit in register 01 of sequencer */
  669. cli();
  670. outb_p(0x01,seq_port_reg);
  671. outb_p(info->vga_state.ClockingMode | 0x20,seq_port_val);
  672. /* test for vertical retrace in process.... */
  673. if ((info->vga_state.CrtMiscIO & 0x80) == 0x80)
  674. outb_p(info->vga_state.CrtMiscIO & 0xef,video_misc_wr);
  675. /*
  676.  * Set <End of vertical retrace> to minimum (0) and
  677.  * <Start of vertical Retrace> to maximum (incl. overflow)
  678.  * Result: turn off vertical sync (VSync) pulse.
  679.  */
  680. if (mode & VESA_VSYNC_SUSPEND) {
  681. outb_p(0x10,vga_video_port_reg); /* StartVertRetrace */
  682. outb_p(0xff,vga_video_port_val);  /* maximum value */
  683. outb_p(0x11,vga_video_port_reg); /* EndVertRetrace */
  684. outb_p(0x40,vga_video_port_val); /* minimum (bits 0..3)  */
  685. outb_p(0x07,vga_video_port_reg); /* Overflow */
  686. outb_p(info->vga_state.Overflow | 0x84,vga_video_port_val); /* bits 9,10 of vert. retrace */
  687. }
  688. if (mode & VESA_HSYNC_SUSPEND) {
  689. /*
  690.  * Set <End of horizontal retrace> to minimum (0) and
  691.  *  <Start of horizontal Retrace> to maximum
  692.  * Result: turn off horizontal sync (HSync) pulse.
  693.  */
  694. outb_p(0x04,vga_video_port_reg); /* StartHorizRetrace */
  695. outb_p(0xff,vga_video_port_val); /* maximum */
  696. outb_p(0x05,vga_video_port_reg); /* EndHorizRetrace */
  697. outb_p(0x00,vga_video_port_val); /* minimum (0) */
  698. }
  699. /* restore both index registers */
  700. outb_p(SeqCtrlIndex,seq_port_reg);
  701. outb_p(CrtCtrlIndex,vga_video_port_reg);
  702. sti();
  703. }
  704. static void vga_vesa_unblank(struct vga16fb_info *info)
  705. {
  706. unsigned char SeqCtrlIndex;
  707. unsigned char CrtCtrlIndex;
  708. cli();
  709. SeqCtrlIndex = inb_p(seq_port_reg);
  710. CrtCtrlIndex = inb_p(vga_video_port_reg);
  711. /* restore original values of VGA controller registers */
  712. outb_p(info->vga_state.CrtMiscIO,video_misc_wr);
  713. outb_p(0x00,vga_video_port_reg); /* HorizontalTotal */
  714. outb_p(info->vga_state.HorizontalTotal,vga_video_port_val);
  715. outb_p(0x01,vga_video_port_reg); /* HorizDisplayEnd */
  716. outb_p(info->vga_state.HorizDisplayEnd,vga_video_port_val);
  717. outb_p(0x04,vga_video_port_reg); /* StartHorizRetrace */
  718. outb_p(info->vga_state.StartHorizRetrace,vga_video_port_val);
  719. outb_p(0x05,vga_video_port_reg); /* EndHorizRetrace */
  720. outb_p(info->vga_state.EndHorizRetrace,vga_video_port_val);
  721. outb_p(0x07,vga_video_port_reg); /* Overflow */
  722. outb_p(info->vga_state.Overflow,vga_video_port_val);
  723. outb_p(0x10,vga_video_port_reg); /* StartVertRetrace */
  724. outb_p(info->vga_state.StartVertRetrace,vga_video_port_val);
  725. outb_p(0x11,vga_video_port_reg); /* EndVertRetrace */
  726. outb_p(info->vga_state.EndVertRetrace,vga_video_port_val);
  727. outb_p(0x17,vga_video_port_reg); /* ModeControl */
  728. outb_p(info->vga_state.ModeControl,vga_video_port_val);
  729. outb_p(0x01,seq_port_reg); /* ClockingMode */
  730. outb_p(info->vga_state.ClockingMode,seq_port_val);
  731. /* restore index/control registers */
  732. outb_p(SeqCtrlIndex,seq_port_reg);
  733. outb_p(CrtCtrlIndex,vga_video_port_reg);
  734. sti();
  735. }
  736. static void vga_pal_blank(void)
  737. {
  738. int i;
  739. for (i=0; i<16; i++) {
  740. outb_p (i, dac_reg) ;
  741. outb_p (0, dac_val) ;
  742. outb_p (0, dac_val) ;
  743. outb_p (0, dac_val) ;
  744. }
  745. }
  746. /* 0 unblank, 1 blank, 2 no vsync, 3 no hsync, 4 off */
  747. static void vga16fb_blank(int blank, struct fb_info *fb_info)
  748. {
  749. struct vga16fb_info *info = (struct vga16fb_info*)fb_info;
  750. switch (blank) {
  751. case 0: /* Unblank */
  752. if (info->vesa_blanked) {
  753. vga_vesa_unblank(info);
  754. info->vesa_blanked = 0;
  755. }
  756. if (info->palette_blanked) {
  757. do_install_cmap(currcon, fb_info);
  758. info->palette_blanked = 0;
  759. }
  760. break;
  761. case 1: /* blank */
  762. vga_pal_blank();
  763. info->palette_blanked = 1;
  764. break;
  765. default: /* VESA blanking */
  766. vga_vesa_blank(info, blank-1);
  767. info->vesa_blanked = 1;
  768. break;
  769. }
  770. }
  771. int __init vga16fb_init(void)
  772. {
  773. int i,j;
  774. printk(KERN_DEBUG "vga16fb: initializingn");
  775. /* XXX share VGA_FB_PHYS region with vgacon */
  776.         vga16fb.video_vbase = ioremap(VGA_FB_PHYS, VGA_FB_PHYS_LEN);
  777. if (!vga16fb.video_vbase) {
  778. printk(KERN_ERR "vga16fb: unable to map devicen");
  779. return -ENOMEM;
  780. }
  781. printk(KERN_INFO "vga16fb: mapped to 0x%pn", vga16fb.video_vbase);
  782. vga16fb.isVGA = ORIG_VIDEO_ISVGA;
  783. vga16fb.palette_blanked = 0;
  784. vga16fb.vesa_blanked = 0;
  785. i = vga16fb.isVGA? 6 : 2;
  786. vga16fb_defined.red.length   = i;
  787. vga16fb_defined.green.length = i;
  788. vga16fb_defined.blue.length  = i;
  789. for(i = 0; i < 16; i++) {
  790. j = color_table[i];
  791. palette[i].red   = default_red[j];
  792. palette[i].green = default_grn[j];
  793. palette[i].blue  = default_blu[j];
  794. }
  795. /* XXX share VGA I/O region with vgacon and others */
  796. disp.var = vga16fb_defined;
  797. /* name should not depend on EGA/VGA */
  798. strcpy(vga16fb.fb_info.modename, "VGA16 VGA");
  799. vga16fb.fb_info.changevar = NULL;
  800. vga16fb.fb_info.node = -1;
  801. vga16fb.fb_info.fbops = &vga16fb_ops;
  802. vga16fb.fb_info.disp=&disp;
  803. vga16fb.fb_info.switch_con=&vga16fb_switch;
  804. vga16fb.fb_info.updatevar=&vga16fb_update_var;
  805. vga16fb.fb_info.blank=&vga16fb_blank;
  806. vga16fb.fb_info.flags=FBINFO_FLAG_DEFAULT;
  807. vga16fb_set_disp(-1, &vga16fb);
  808. if (register_framebuffer(&vga16fb.fb_info)<0) {
  809. iounmap(vga16fb.video_vbase);
  810. return -EINVAL;
  811. }
  812. printk(KERN_INFO "fb%d: %s frame buffer devicen",
  813.        GET_FB_IDX(vga16fb.fb_info.node), vga16fb.fb_info.modename);
  814. return 0;
  815. }
  816. static void __exit vga16fb_exit(void)
  817. {
  818.     unregister_framebuffer(&vga16fb.fb_info);
  819.     iounmap(vga16fb.video_vbase);
  820.     /* XXX unshare VGA regions */
  821. }
  822. #ifdef MODULE
  823. MODULE_LICENSE("GPL");
  824. module_init(vga16fb_init);
  825. #endif
  826. module_exit(vga16fb_exit);
  827. /*
  828.  * Overrides for Emacs so that we follow Linus's tabbing style.
  829.  * ---------------------------------------------------------------------------
  830.  * Local variables:
  831.  * c-basic-offset: 8
  832.  * End:
  833.  */