dnfb.c 7.9 KB

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