maxinefb.c 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. /*
  2. * linux/drivers/video/maxinefb.c
  3. *
  4. * DECstation 5000/xx onboard framebuffer support ... derived from:
  5. * "HP300 Topcat framebuffer support (derived from macfb of all things)
  6. * Phil Blundell <philb@gnu.org> 1998", the original code can be
  7. * found in the file hpfb.c in the same directory.
  8. *
  9. * DECstation related code Copyright (C) 1999,2000,2001 by
  10. * Michael Engel <engel@unix-ag.org> and
  11. * Karsten Merker <merker@linuxtag.org>.
  12. * This file is subject to the terms and conditions of the GNU General
  13. * Public License. See the file COPYING in the main directory of this
  14. * archive for more details.
  15. *
  16. */
  17. /*
  18. * Changes:
  19. * 2001/01/27 removed debugging and testing code, fixed fb_ops
  20. * initialization which had caused a crash before,
  21. * general cleanup, first official release (KM)
  22. *
  23. */
  24. #include <linux/module.h>
  25. #include <linux/kernel.h>
  26. #include <linux/sched.h>
  27. #include <linux/errno.h>
  28. #include <linux/string.h>
  29. #include <linux/mm.h>
  30. #include <linux/tty.h>
  31. #include <linux/slab.h>
  32. #include <linux/delay.h>
  33. #include <linux/init.h>
  34. #include <linux/fb.h>
  35. #include <video/maxinefb.h>
  36. /* bootinfo.h defines the machine type values, needed when checking */
  37. /* whether are really running on a maxine, KM */
  38. #include <asm/bootinfo.h>
  39. static struct fb_info fb_info;
  40. static struct fb_var_screeninfo maxinefb_defined = {
  41. .xres = 1024,
  42. .yres = 768,
  43. .xres_virtual = 1024,
  44. .yres_virtual = 768,
  45. .bits_per_pixel =8,
  46. .activate = FB_ACTIVATE_NOW,
  47. .height = -1,
  48. .width = -1,
  49. .vmode = FB_VMODE_NONINTERLACED,
  50. };
  51. static struct fb_fix_screeninfo maxinefb_fix = {
  52. .id = "Maxine onboard graphics 1024x768x8",
  53. .smem_len = (1024*768),
  54. .type = FB_TYPE_PACKED_PIXELS,
  55. .visual = FB_VISUAL_PSEUDOCOLOR,
  56. .line_length = 1024,
  57. };
  58. /* Handle the funny Inmos RamDAC/video controller ... */
  59. void maxinefb_ims332_write_register(int regno, register unsigned int val)
  60. {
  61. register unsigned char *regs = (char *) MAXINEFB_IMS332_ADDRESS;
  62. unsigned char *wptr;
  63. wptr = regs + 0xa0000 + (regno << 4);
  64. *((volatile unsigned int *) (regs)) = (val >> 8) & 0xff00;
  65. *((volatile unsigned short *) (wptr)) = val;
  66. }
  67. unsigned int maxinefb_ims332_read_register(int regno)
  68. {
  69. register unsigned char *regs = (char *) MAXINEFB_IMS332_ADDRESS;
  70. unsigned char *rptr;
  71. register unsigned int j, k;
  72. rptr = regs + 0x80000 + (regno << 4);
  73. j = *((volatile unsigned short *) rptr);
  74. k = *((volatile unsigned short *) regs);
  75. return (j & 0xffff) | ((k & 0xff00) << 8);
  76. }
  77. /* Set the palette */
  78. static int maxinefb_setcolreg(unsigned regno, unsigned red, unsigned green,
  79. unsigned blue, unsigned transp, struct fb_info *info)
  80. {
  81. /* value to be written into the palette reg. */
  82. unsigned long hw_colorvalue = 0;
  83. red >>= 8; /* The cmap fields are 16 bits */
  84. green >>= 8; /* wide, but the harware colormap */
  85. blue >>= 8; /* registers are only 8 bits wide */
  86. hw_colorvalue = (blue << 16) + (green << 8) + (red);
  87. maxinefb_ims332_write_register(IMS332_REG_COLOR_PALETTE + regno,
  88. hw_colorvalue);
  89. return 0;
  90. }
  91. static struct fb_ops maxinefb_ops = {
  92. .owner = THIS_MODULE,
  93. .fb_get_fix = gen_get_fix,
  94. .fb_get_var = gen_get_var,
  95. .fb_setcolreg = maxinefb_setcolreg,
  96. .fb_fillrect = cfb_fillrect,
  97. .fb_copyarea = cfb_copyarea,
  98. .fb_imageblit = cfb_imageblit,
  99. };
  100. int __init maxinefb_init(void)
  101. {
  102. unsigned long fboff;
  103. unsigned long fb_start;
  104. int i;
  105. if (fb_get_options("maxinefb", NULL))
  106. return -ENODEV;
  107. /* Validate we're on the proper machine type */
  108. if (mips_machtype != MACH_DS5000_XX) {
  109. return -EINVAL;
  110. }
  111. printk(KERN_INFO "Maxinefb: Personal DECstation detected\n");
  112. printk(KERN_INFO "Maxinefb: initializing onboard framebuffer\n");
  113. /* Framebuffer display memory base address */
  114. fb_start = DS5000_xx_ONBOARD_FBMEM_START;
  115. /* Clear screen */
  116. for (fboff = fb_start; fboff < fb_start + 0x1ffff; fboff++)
  117. *(volatile unsigned char *)fboff = 0x0;
  118. maxinefb_fix.smem_start = fb_start;
  119. /* erase hardware cursor */
  120. for (i = 0; i < 512; i++) {
  121. maxinefb_ims332_write_register(IMS332_REG_CURSOR_RAM + i,
  122. 0);
  123. /*
  124. if (i&0x8 == 0)
  125. maxinefb_ims332_write_register (IMS332_REG_CURSOR_RAM + i, 0x0f);
  126. else
  127. maxinefb_ims332_write_register (IMS332_REG_CURSOR_RAM + i, 0xf0);
  128. */
  129. }
  130. fb_info.fbops = &maxinefb_ops;
  131. fb_info.screen_base = (char *)maxinefb_fix.smem_start;
  132. fb_info.var = maxinefb_defined;
  133. fb_info.fix = maxinefb_fix;
  134. fb_info.flags = FBINFO_DEFAULT;
  135. fb_alloc_cmap(&fb_info.cmap, 256, 0);
  136. if (register_framebuffer(&fb_info) < 0)
  137. return 1;
  138. return 0;
  139. }
  140. static void __exit maxinefb_exit(void)
  141. {
  142. unregister_framebuffer(&fb_info);
  143. }
  144. #ifdef MODULE
  145. MODULE_LICENSE("GPL");
  146. #endif
  147. module_init(maxinefb_init);
  148. module_exit(maxinefb_exit);