p9100fb.c
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:5k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. /* $id: p9100fb.c,v 1.4 1999/08/18 10:55:01 shadow Exp $
  2.  * p9100fb.c: P9100 frame buffer driver
  3.  *
  4.  * Copyright 1999 Derrick J Brashear (shadow@dementia.org)
  5.  */
  6. #include <linux/module.h>
  7. #include <linux/sched.h>
  8. #include <linux/kernel.h>
  9. #include <linux/errno.h>
  10. #include <linux/string.h>
  11. #include <linux/mm.h>
  12. #include <linux/tty.h>
  13. #include <linux/slab.h>
  14. #include <linux/vmalloc.h>
  15. #include <linux/delay.h>
  16. #include <linux/interrupt.h>
  17. #include <linux/fb.h>
  18. #include <linux/init.h>
  19. #include <linux/selection.h>
  20. #include <video/sbusfb.h>
  21. #include <asm/io.h>
  22. #include <video/fbcon-cfb8.h>
  23. #include "p9100.h"
  24. static struct sbus_mmap_map p9100_mmap_map[] = {
  25. #if 0 /* For now, play we're a dumb color fb */
  26.   { P9100_CTL_OFF, 0x38000000, 0x2000 },
  27.   { P9100_CMD_OFF, 0x38002000, 0x2000 },
  28.   { P9100_FB_OFF, 0x38800000, 0x200000 },
  29.   { CG3_MMAP_OFFSET, 0x38800000, SBUS_MMAP_FBSIZE(1) },
  30. #else
  31.   { CG3_MMAP_OFFSET, 0x0, SBUS_MMAP_FBSIZE(1) },
  32. #endif
  33.   { 0, 0, 0 }
  34. };
  35. #define _READCTL(member, out) 
  36.   struct p9100_ctrl *actual; 
  37.   actual = (struct p9100_ctrl *)fb->s.p9100.ctrl; 
  38.   out = sbus_readl(&actual-> ## member ); 
  39. }
  40. #define READCTL(member, out) 
  41.   struct p9100_ctrl *enab, *actual; 
  42.   actual = (struct p9100_ctrl *)fb->s.p9100.ctrl; 
  43.   enab = (struct p9100_ctrl *)fb->s.p9100.fbmem; 
  44.   out = sbus_readl(&enab-> ## member ); 
  45.   out = sbus_readl(&actual-> ## member ); 
  46. }
  47. #define WRITECTL(member, val) 
  48.   u32 __writetmp; 
  49.   struct p9100_ctrl *enab, *actual; 
  50.   actual = (struct p9100_ctrl *)fb->s.p9100.ctrl; 
  51.   enab = (struct p9100_ctrl *)fb->s.p9100.fbmem; 
  52.   __writetmp = sbus_readl(&enab-> ## member ); 
  53.   sbus_writel(val, &actual-> ## member ); 
  54. }
  55. static void p9100_loadcmap (struct fb_info_sbusfb *fb, struct display *p, int index, int count)
  56. {
  57. unsigned long flags;
  58. u32 tmp;
  59. int i;
  60. spin_lock_irqsave(&fb->lock, flags);
  61. _READCTL(pwrup_cfg, tmp);
  62. WRITECTL(ramdac_cmap_wridx, (index << 16));
  63. for (i = index; count--; i++){
  64. _READCTL(pwrup_cfg, tmp);
  65. WRITECTL(ramdac_palette_data, (fb->color_map CM(i,0) << 16));
  66. _READCTL(pwrup_cfg, tmp);
  67. WRITECTL(ramdac_palette_data, (fb->color_map CM(i,1) << 16));
  68. _READCTL(pwrup_cfg, tmp);
  69. WRITECTL(ramdac_palette_data, (fb->color_map CM(i,2) << 16));
  70. }
  71. spin_unlock_irqrestore(&fb->lock, flags);
  72. }
  73. static void p9100_blank (struct fb_info_sbusfb *fb)
  74. {
  75. unsigned long flags;
  76. u32 val;
  77. spin_lock_irqsave(&fb->lock, flags);
  78. READCTL(vid_screenpaint_timectl1, val);
  79. val &= ~ SCREENPAINT_TIMECTL1_ENABLE_VIDEO;
  80. WRITECTL(vid_screenpaint_timectl1, val);
  81. spin_unlock_irqrestore(&fb->lock, flags);
  82. }
  83. static void p9100_unblank (struct fb_info_sbusfb *fb)
  84. {
  85. unsigned long flags;
  86. u32 val;
  87. spin_lock_irqsave(&fb->lock, flags);
  88. READCTL(vid_screenpaint_timectl1, val);
  89. val |= SCREENPAINT_TIMECTL1_ENABLE_VIDEO;
  90. WRITECTL(vid_screenpaint_timectl1, val);
  91. spin_unlock_irqrestore(&fb->lock, flags);
  92. }
  93. static void p9100_margins (struct fb_info_sbusfb *fb, struct display *p, int x_margin, int y_margin)
  94. {
  95.   p->screen_base += (y_margin - fb->y_margin) * p->line_length + 
  96.     (x_margin - fb->x_margin);
  97. }
  98. static char idstring[60] __initdata = { 0 };
  99. char * __init p9100fb_init(struct fb_info_sbusfb *fb)
  100. {
  101. struct fb_fix_screeninfo *fix = &fb->fix;
  102. struct display *disp = &fb->disp;
  103. struct fbtype *type = &fb->type;
  104. struct sbus_dev *sdev = fb->sbdp;
  105. unsigned long phys = sdev->reg_addrs[2].phys_addr;
  106. int tmp;
  107. #ifndef FBCON_HAS_CFB8
  108. return NULL;
  109. #endif
  110. /* Control regs: fb->sbdp->reg_addrs[0].phys_addr 
  111.  * Command regs: fb->sbdp->reg_addrs[1].phys_addr 
  112.  * Frame buffer: fb->sbdp->reg_addrs[2].phys_addr 
  113.          */
  114. if (!fb->s.p9100.ctrl) {
  115. fb->s.p9100.ctrl = (struct p9100_ctrl *)
  116. sbus_ioremap(&sdev->resource[0], 0,
  117.      sdev->reg_addrs[0].reg_size, "p9100 ctrl");
  118. }
  119. strcpy(fb->info.modename, "p9100");
  120. strcpy(fix->id, "p9100");
  121. fix->accel = FB_ACCEL_SUN_CGTHREE;
  122. fix->line_length = fb->var.xres_virtual;
  123. disp->scrollmode = SCROLL_YREDRAW;
  124. if (!disp->screen_base)
  125. disp->screen_base = (char *)
  126. sbus_ioremap(&sdev->resource[2], 0,
  127.      type->fb_size, "p9100 ram");
  128. fb->s.p9100.fbmem = (volatile u32 *)disp->screen_base;
  129. disp->screen_base += fix->line_length * fb->y_margin + fb->x_margin;
  130. READCTL(sys_config, tmp);
  131.         switch ((tmp >> SYS_CONFIG_PIXELSIZE_SHIFT) & 7) {
  132. case 7: 
  133.   type->fb_depth = 24; 
  134.   break;
  135. case 5: 
  136.   type->fb_depth = 32;
  137.   break;
  138. case 3: 
  139.   type->fb_depth = 16; 
  140.   break;
  141. case 2: 
  142.   type->fb_depth = 8; 
  143.   break;
  144. default: 
  145.   printk("p9100: screen depth unknown: 0x%x", tmp);
  146.   return NULL;
  147.         }
  148. fb->dispsw = fbcon_cfb8;
  149. fb->margins = p9100_margins;
  150. fb->loadcmap = p9100_loadcmap;
  151. fb->blank = p9100_blank;
  152. fb->unblank = p9100_unblank;
  153. fb->physbase = phys;
  154. fb->mmap_map = p9100_mmap_map;
  155. sprintf(idstring, "%s at 0x%x", "p9100", 
  156. (unsigned int)disp->screen_base);
  157. return idstring;
  158. }