dnfb.c 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305
  1. #include <linux/kernel.h>
  2. #include <linux/errno.h>
  3. #include <linux/string.h>
  4. #include <linux/mm.h>
  5. #include <linux/delay.h>
  6. #include <linux/interrupt.h>
  7. #include <linux/platform_device.h>
  8. #include <asm/setup.h>
  9. #include <asm/system.h>
  10. #include <asm/irq.h>
  11. #include <asm/amigahw.h>
  12. #include <asm/amigaints.h>
  13. #include <asm/apollohw.h>
  14. #include <linux/fb.h>
  15. #include <linux/module.h>
  16. /* apollo video HW definitions */
  17. /*
  18. * Control Registers. IOBASE + $x
  19. *
  20. * Note: these are the Memory/IO BASE definitions for a mono card set to the
  21. * alternate address
  22. *
  23. * Control 3A and 3B serve identical functions except that 3A
  24. * deals with control 1 and 3b deals with Color LUT reg.
  25. */
  26. #define AP_IOBASE 0x3b0 /* Base address of 1 plane board. */
  27. #define AP_STATUS isaIO2mem(AP_IOBASE+0) /* Status register. Read */
  28. #define AP_WRITE_ENABLE isaIO2mem(AP_IOBASE+0) /* Write Enable Register Write */
  29. #define AP_DEVICE_ID isaIO2mem(AP_IOBASE+1) /* Device ID Register. Read */
  30. #define AP_ROP_1 isaIO2mem(AP_IOBASE+2) /* Raster Operation reg. Write Word */
  31. #define AP_DIAG_MEM_REQ isaIO2mem(AP_IOBASE+4) /* Diagnostic Memory Request. Write Word */
  32. #define AP_CONTROL_0 isaIO2mem(AP_IOBASE+8) /* Control Register 0. Read/Write */
  33. #define AP_CONTROL_1 isaIO2mem(AP_IOBASE+0xa) /* Control Register 1. Read/Write */
  34. #define AP_CONTROL_3A isaIO2mem(AP_IOBASE+0xe) /* Control Register 3a. Read/Write */
  35. #define AP_CONTROL_2 isaIO2mem(AP_IOBASE+0xc) /* Control Register 2. Read/Write */
  36. #define FRAME_BUFFER_START 0x0FA0000
  37. #define FRAME_BUFFER_LEN 0x40000
  38. /* CREG 0 */
  39. #define VECTOR_MODE 0x40 /* 010x.xxxx */
  40. #define DBLT_MODE 0x80 /* 100x.xxxx */
  41. #define NORMAL_MODE 0xE0 /* 111x.xxxx */
  42. #define SHIFT_BITS 0x1F /* xxx1.1111 */
  43. /* other bits are Shift value */
  44. /* CREG 1 */
  45. #define AD_BLT 0x80 /* 1xxx.xxxx */
  46. #define NORMAL 0x80 /* 1xxx.xxxx */ /* What is happening here ?? */
  47. #define INVERSE 0x00 /* 0xxx.xxxx */ /* Clearing this reverses the screen */
  48. #define PIX_BLT 0x00 /* 0xxx.xxxx */
  49. #define AD_HIBIT 0x40 /* xIxx.xxxx */
  50. #define ROP_EN 0x10 /* xxx1.xxxx */
  51. #define DST_EQ_SRC 0x00 /* xxx0.xxxx */
  52. #define nRESET_SYNC 0x08 /* xxxx.1xxx */
  53. #define SYNC_ENAB 0x02 /* xxxx.xx1x */
  54. #define BLANK_DISP 0x00 /* xxxx.xxx0 */
  55. #define ENAB_DISP 0x01 /* xxxx.xxx1 */
  56. #define NORM_CREG1 (nRESET_SYNC | SYNC_ENAB | ENAB_DISP) /* no reset sync */
  57. /* CREG 2 */
  58. /*
  59. * Following 3 defines are common to 1, 4 and 8 plane.
  60. */
  61. #define S_DATA_1s 0x00 /* 00xx.xxxx */ /* set source to all 1's -- vector drawing */
  62. #define S_DATA_PIX 0x40 /* 01xx.xxxx */ /* takes source from ls-bits and replicates over 16 bits */
  63. #define S_DATA_PLN 0xC0 /* 11xx.xxxx */ /* normal, each data access =16-bits in
  64. one plane of image mem */
  65. /* CREG 3A/CREG 3B */
  66. # define RESET_CREG 0x80 /* 1000.0000 */
  67. /* ROP REG - all one nibble */
  68. /* ********* NOTE : this is used r0,r1,r2,r3 *********** */
  69. #define ROP(r2,r3,r0,r1) ( (U_SHORT)((r0)|((r1)<<4)|((r2)<<8)|((r3)<<12)) )
  70. #define DEST_ZERO 0x0
  71. #define SRC_AND_DEST 0x1
  72. #define SRC_AND_nDEST 0x2
  73. #define SRC 0x3
  74. #define nSRC_AND_DEST 0x4
  75. #define DEST 0x5
  76. #define SRC_XOR_DEST 0x6
  77. #define SRC_OR_DEST 0x7
  78. #define SRC_NOR_DEST 0x8
  79. #define SRC_XNOR_DEST 0x9
  80. #define nDEST 0xA
  81. #define SRC_OR_nDEST 0xB
  82. #define nSRC 0xC
  83. #define nSRC_OR_DEST 0xD
  84. #define SRC_NAND_DEST 0xE
  85. #define DEST_ONE 0xF
  86. #define SWAP(A) ((A>>8) | ((A&0xff) <<8))
  87. /* frame buffer operations */
  88. static int dnfb_blank(int blank, struct fb_info *info);
  89. static void dnfb_copyarea(struct fb_info *info, const struct fb_copyarea *area);
  90. static struct fb_ops dn_fb_ops = {
  91. .owner = THIS_MODULE,
  92. .fb_blank = dnfb_blank,
  93. .fb_fillrect = cfb_fillrect,
  94. .fb_copyarea = dnfb_copyarea,
  95. .fb_imageblit = cfb_imageblit,
  96. };
  97. struct fb_var_screeninfo dnfb_var __devinitdata = {
  98. .xres = 1280,
  99. .yres = 1024,
  100. .xres_virtual = 2048,
  101. .yres_virtual = 1024,
  102. .bits_per_pixel = 1,
  103. .height = -1,
  104. .width = -1,
  105. .vmode = FB_VMODE_NONINTERLACED,
  106. };
  107. static struct fb_fix_screeninfo dnfb_fix __devinitdata = {
  108. .id = "Apollo Mono",
  109. .smem_start = (FRAME_BUFFER_START + IO_BASE),
  110. .smem_len = FRAME_BUFFER_LEN,
  111. .type = FB_TYPE_PACKED_PIXELS,
  112. .visual = FB_VISUAL_MONO10,
  113. .line_length = 256,
  114. };
  115. static int dnfb_blank(int blank, struct fb_info *info)
  116. {
  117. if (blank)
  118. out_8(AP_CONTROL_3A, 0x0);
  119. else
  120. out_8(AP_CONTROL_3A, 0x1);
  121. return 0;
  122. }
  123. static
  124. void dnfb_copyarea(struct fb_info *info, const struct fb_copyarea *area)
  125. {
  126. int incr, y_delta, pre_read = 0, x_end, x_word_count;
  127. uint start_mask, end_mask, dest;
  128. ushort *src, dummy;
  129. short i, j;
  130. incr = (area->dy <= area->sy) ? 1 : -1;
  131. src = (ushort *)(info->screen_base + area->sy * info->fix.line_length +
  132. (area->sx >> 4));
  133. dest = area->dy * (info->fix.line_length >> 1) + (area->dx >> 4);
  134. if (incr > 0) {
  135. y_delta = (info->fix.line_length * 8) - area->sx - area->width;
  136. x_end = area->dx + area->width - 1;
  137. x_word_count = (x_end >> 4) - (area->dx >> 4) + 1;
  138. start_mask = 0xffff0000 >> (area->dx & 0xf);
  139. end_mask = 0x7ffff >> (x_end & 0xf);
  140. out_8(AP_CONTROL_0,
  141. (((area->dx & 0xf) - (area->sx & 0xf)) % 16) | (0x4 << 5));
  142. if ((area->dx & 0xf) < (area->sx & 0xf))
  143. pre_read = 1;
  144. } else {
  145. y_delta = -((info->fix.line_length * 8) - area->sx - area->width);
  146. x_end = area->dx - area->width + 1;
  147. x_word_count = (area->dx >> 4) - (x_end >> 4) + 1;
  148. start_mask = 0x7ffff >> (area->dx & 0xf);
  149. end_mask = 0xffff0000 >> (x_end & 0xf);
  150. out_8(AP_CONTROL_0,
  151. ((-((area->sx & 0xf) - (area->dx & 0xf))) % 16) |
  152. (0x4 << 5));
  153. if ((area->dx & 0xf) > (area->sx & 0xf))
  154. pre_read = 1;
  155. }
  156. for (i = 0; i < area->height; i++) {
  157. out_8(AP_CONTROL_3A, 0xc | (dest >> 16));
  158. if (pre_read) {
  159. dummy = *src;
  160. src += incr;
  161. }
  162. if (x_word_count) {
  163. out_8(AP_WRITE_ENABLE, start_mask);
  164. *src = dest;
  165. src += incr;
  166. dest += incr;
  167. out_8(AP_WRITE_ENABLE, 0);
  168. for (j = 1; j < (x_word_count - 1); j++) {
  169. *src = dest;
  170. src += incr;
  171. dest += incr;
  172. }
  173. out_8(AP_WRITE_ENABLE, start_mask);
  174. *src = dest;
  175. dest += incr;
  176. src += incr;
  177. } else {
  178. out_8(AP_WRITE_ENABLE, start_mask | end_mask);
  179. *src = dest;
  180. dest += incr;
  181. src += incr;
  182. }
  183. src += (y_delta / 16);
  184. dest += (y_delta / 16);
  185. }
  186. out_8(AP_CONTROL_0, NORMAL_MODE);
  187. }
  188. /*
  189. * Initialization
  190. */
  191. static int __devinit dnfb_probe(struct platform_device *dev)
  192. {
  193. struct fb_info *info;
  194. int err = 0;
  195. info = framebuffer_alloc(0, &dev->dev);
  196. if (!info)
  197. return -ENOMEM;
  198. info->fbops = &dn_fb_ops;
  199. info->fix = dnfb_fix;
  200. info->var = dnfb_var;
  201. info->var.red.length = 1;
  202. info->var.red.offset = 0;
  203. info->var.green = info->var.blue = info->var.red;
  204. info->screen_base = (u_char *) info->fix.smem_start;
  205. err = fb_alloc_cmap(&info->cmap, 2, 0);
  206. if (err < 0) {
  207. framebuffer_release(info);
  208. return err;
  209. }
  210. err = register_framebuffer(info);
  211. if (err < 0) {
  212. fb_dealloc_cmap(&info->cmap);
  213. framebuffer_release(info);
  214. return err;
  215. }
  216. platform_set_drvdata(dev, info);
  217. /* now we have registered we can safely setup the hardware */
  218. out_8(AP_CONTROL_3A, RESET_CREG);
  219. out_be16(AP_WRITE_ENABLE, 0x0);
  220. out_8(AP_CONTROL_0, NORMAL_MODE);
  221. out_8(AP_CONTROL_1, (AD_BLT | DST_EQ_SRC | NORM_CREG1));
  222. out_8(AP_CONTROL_2, S_DATA_PLN);
  223. out_be16(AP_ROP_1, SWAP(0x3));
  224. printk("apollo frame buffer alive and kicking !\n");
  225. return err;
  226. }
  227. static struct platform_driver dnfb_driver = {
  228. .probe = dnfb_probe,
  229. .driver = {
  230. .name = "dnfb",
  231. },
  232. };
  233. static struct platform_device dnfb_device = {
  234. .name = "dnfb",
  235. };
  236. int __init dnfb_init(void)
  237. {
  238. int ret;
  239. if (!MACH_IS_APOLLO)
  240. return -ENODEV;
  241. if (fb_get_options("dnfb", NULL))
  242. return -ENODEV;
  243. ret = platform_driver_register(&dnfb_driver);
  244. if (!ret) {
  245. ret = platform_device_register(&dnfb_device);
  246. if (ret)
  247. platform_driver_unregister(&dnfb_driver);
  248. }
  249. return ret;
  250. }
  251. module_init(dnfb_init);
  252. MODULE_LICENSE("GPL");