1 /* cg14.c: CGFOURTEEN frame buffer driver
3 * Copyright (C) 2003 David S. Miller (davem@redhat.com)
4 * Copyright (C) 1996,1998 Jakub Jelinek (jj@ultra.linux.cz)
5 * Copyright (C) 1995 Miguel de Icaza (miguel@nuclecu.unam.mx)
7 * Driver layout based loosely on tgafb.c, see that file for credits.
10 #include <linux/module.h>
11 #include <linux/kernel.h>
12 #include <linux/errno.h>
13 #include <linux/string.h>
14 #include <linux/slab.h>
15 #include <linux/delay.h>
16 #include <linux/init.h>
22 #include <asm/oplib.h>
31 static int cg14_setcolreg(unsigned, unsigned, unsigned, unsigned,
32 unsigned, struct fb_info
*);
34 static int cg14_mmap(struct fb_info
*, struct file
*, struct vm_area_struct
*);
35 static int cg14_ioctl(struct inode
*, struct file
*, unsigned int,
36 unsigned long, struct fb_info
*);
37 static int cg14_pan_display(struct fb_var_screeninfo
*, struct fb_info
*);
40 * Frame buffer operations
43 static struct fb_ops cg14_ops
= {
45 .fb_setcolreg
= cg14_setcolreg
,
46 .fb_pan_display
= cg14_pan_display
,
47 .fb_fillrect
= cfb_fillrect
,
48 .fb_copyarea
= cfb_copyarea
,
49 .fb_imageblit
= cfb_imageblit
,
51 .fb_ioctl
= cg14_ioctl
,
52 .fb_cursor
= soft_cursor
,
55 #define CG14_MCR_INTENABLE_SHIFT 7
56 #define CG14_MCR_INTENABLE_MASK 0x80
57 #define CG14_MCR_VIDENABLE_SHIFT 6
58 #define CG14_MCR_VIDENABLE_MASK 0x40
59 #define CG14_MCR_PIXMODE_SHIFT 4
60 #define CG14_MCR_PIXMODE_MASK 0x30
61 #define CG14_MCR_TMR_SHIFT 2
62 #define CG14_MCR_TMR_MASK 0x0c
63 #define CG14_MCR_TMENABLE_SHIFT 1
64 #define CG14_MCR_TMENABLE_MASK 0x02
65 #define CG14_MCR_RESET_SHIFT 0
66 #define CG14_MCR_RESET_MASK 0x01
67 #define CG14_REV_REVISION_SHIFT 4
68 #define CG14_REV_REVISION_MASK 0xf0
69 #define CG14_REV_IMPL_SHIFT 0
70 #define CG14_REV_IMPL_MASK 0x0f
71 #define CG14_VBR_FRAMEBASE_SHIFT 12
72 #define CG14_VBR_FRAMEBASE_MASK 0x00fff000
73 #define CG14_VMCR1_SETUP_SHIFT 0
74 #define CG14_VMCR1_SETUP_MASK 0x000001ff
75 #define CG14_VMCR1_VCONFIG_SHIFT 9
76 #define CG14_VMCR1_VCONFIG_MASK 0x00000e00
77 #define CG14_VMCR2_REFRESH_SHIFT 0
78 #define CG14_VMCR2_REFRESH_MASK 0x00000001
79 #define CG14_VMCR2_TESTROWCNT_SHIFT 1
80 #define CG14_VMCR2_TESTROWCNT_MASK 0x00000002
81 #define CG14_VMCR2_FBCONFIG_SHIFT 2
82 #define CG14_VMCR2_FBCONFIG_MASK 0x0000000c
83 #define CG14_VCR_REFRESHREQ_SHIFT 0
84 #define CG14_VCR_REFRESHREQ_MASK 0x000003ff
85 #define CG14_VCR1_REFRESHENA_SHIFT 10
86 #define CG14_VCR1_REFRESHENA_MASK 0x00000400
87 #define CG14_VCA_CAD_SHIFT 0
88 #define CG14_VCA_CAD_MASK 0x000003ff
89 #define CG14_VCA_VERS_SHIFT 10
90 #define CG14_VCA_VERS_MASK 0x00000c00
91 #define CG14_VCA_RAMSPEED_SHIFT 12
92 #define CG14_VCA_RAMSPEED_MASK 0x00001000
93 #define CG14_VCA_8MB_SHIFT 13
94 #define CG14_VCA_8MB_MASK 0x00002000
96 #define CG14_MCR_PIXMODE_8 0
97 #define CG14_MCR_PIXMODE_16 2
98 #define CG14_MCR_PIXMODE_32 3
101 volatile u8 mcr
; /* Master Control Reg */
102 volatile u8 ppr
; /* Packed Pixel Reg */
103 volatile u8 tms
[2]; /* Test Mode Status Regs */
104 volatile u8 msr
; /* Master Status Reg */
105 volatile u8 fsr
; /* Fault Status Reg */
106 volatile u8 rev
; /* Revision & Impl */
107 volatile u8 ccr
; /* Clock Control Reg */
108 volatile u32 tmr
; /* Test Mode Read Back */
109 volatile u8 mod
; /* Monitor Operation Data Reg */
110 volatile u8 acr
; /* Aux Control */
112 volatile u16 hct
; /* Hor Counter */
113 volatile u16 vct
; /* Vert Counter */
114 volatile u16 hbs
; /* Hor Blank Start */
115 volatile u16 hbc
; /* Hor Blank Clear */
116 volatile u16 hss
; /* Hor Sync Start */
117 volatile u16 hsc
; /* Hor Sync Clear */
118 volatile u16 csc
; /* Composite Sync Clear */
119 volatile u16 vbs
; /* Vert Blank Start */
120 volatile u16 vbc
; /* Vert Blank Clear */
121 volatile u16 vss
; /* Vert Sync Start */
122 volatile u16 vsc
; /* Vert Sync Clear */
125 volatile u16 fsa
; /* Fault Status Address */
126 volatile u16 adr
; /* Address Registers */
128 volatile u8 pcg
[0x100]; /* Pixel Clock Generator */
129 volatile u32 vbr
; /* Frame Base Row */
130 volatile u32 vmcr
; /* VBC Master Control */
131 volatile u32 vcr
; /* VBC refresh */
132 volatile u32 vca
; /* VBC Config */
135 #define CG14_CCR_ENABLE 0x04
136 #define CG14_CCR_SELECT 0x02 /* HW/Full screen */
139 volatile u32 cpl0
[32]; /* Enable plane 0 */
140 volatile u32 cpl1
[32]; /* Color selection plane */
141 volatile u8 ccr
; /* Cursor Control Reg */
143 volatile u16 cursx
; /* Cursor x,y position */
144 volatile u16 cursy
; /* Cursor x,y position */
148 volatile u32 cpl0i
[32]; /* Enable plane 0 autoinc */
149 volatile u32 cpl1i
[32]; /* Color selection autoinc */
153 volatile u8 addr
; /* Address Register */
155 volatile u8 glut
; /* Gamma table */
157 volatile u8 select
; /* Register Select */
159 volatile u8 mode
; /* Mode Register */
163 volatile u8 x_xlut
[256];
164 volatile u8 x_xlutd
[256];
166 volatile u8 x_xlut_inc
[256];
167 volatile u8 x_xlutd_inc
[256];
170 /* Color look up table (clut) */
171 /* Each one of these arrays hold the color lookup table (for 256
172 * colors) for each MDI page (I assume then there should be 4 MDI
173 * pages, I still wonder what they are. I have seen NeXTStep split
174 * the screen in four parts, while operating in 24 bits mode. Each
175 * integer holds 4 values: alpha value (transparency channel, thanks
176 * go to John Stone (johns@umr.edu) from OpenBSD), red, green and blue
178 * I currently use the clut instead of the Xlut
182 u32 c_clutd
[256]; /* i wonder what the 'd' is for */
183 u32 c_clut_inc
[256];
184 u32 c_clutd_inc
[256];
187 #define CG14_MMAP_ENTRIES 16
191 struct cg14_regs
*regs
;
192 struct cg14_clut
*clut
;
193 struct cg14_cursor
*cursor
;
196 #define CG14_FLAG_BLANKED 0x00000001
198 unsigned long physbase
;
199 unsigned long iospace
;
200 unsigned long fbsize
;
202 struct sbus_mmap_map mmap_map
[CG14_MMAP_ENTRIES
];
206 struct sbus_dev
*sdev
;
207 struct list_head list
;
210 static void __cg14_reset(struct cg14_par
*par
)
212 struct cg14_regs
*regs
= par
->regs
;
215 val
= sbus_readb(®s
->mcr
);
216 val
&= ~(CG14_MCR_PIXMODE_MASK
);
217 sbus_writeb(val
, ®s
->mcr
);
220 static int cg14_pan_display(struct fb_var_screeninfo
*var
, struct fb_info
*info
)
222 struct cg14_par
*par
= (struct cg14_par
*) info
->par
;
225 /* We just use this to catch switches out of
228 spin_lock_irqsave(&par
->lock
, flags
);
230 spin_unlock_irqrestore(&par
->lock
, flags
);
232 if (var
->xoffset
|| var
->yoffset
|| var
->vmode
)
238 * cg14_setcolreg - Optional function. Sets a color register.
239 * @regno: boolean, 0 copy local, 1 get_user() function
240 * @red: frame buffer colormap structure
241 * @green: The green value which can be up to 16 bits wide
242 * @blue: The blue value which can be up to 16 bits wide.
243 * @transp: If supported the alpha value which can be up to 16 bits wide.
244 * @info: frame buffer info structure
246 static int cg14_setcolreg(unsigned regno
,
247 unsigned red
, unsigned green
, unsigned blue
,
248 unsigned transp
, struct fb_info
*info
)
250 struct cg14_par
*par
= (struct cg14_par
*) info
->par
;
251 struct cg14_clut
*clut
= par
->clut
;
258 val
= (red
| (green
<< 8) | (blue
<< 16));
260 spin_lock_irqsave(&par
->lock
, flags
);
261 sbus_writel(val
, &clut
->c_clut
[regno
]);
262 spin_unlock_irqrestore(&par
->lock
, flags
);
267 static int cg14_mmap(struct fb_info
*info
, struct file
*file
, struct vm_area_struct
*vma
)
269 struct cg14_par
*par
= (struct cg14_par
*) info
->par
;
271 return sbusfb_mmap_helper(par
->mmap_map
,
272 par
->physbase
, par
->fbsize
,
276 static int cg14_ioctl(struct inode
*inode
, struct file
*file
, unsigned int cmd
,
277 unsigned long arg
, struct fb_info
*info
)
279 struct cg14_par
*par
= (struct cg14_par
*) info
->par
;
280 struct cg14_regs
*regs
= par
->regs
;
281 struct mdi_cfginfo kmdi
, __user
*mdii
;
283 int cur_mode
, mode
, ret
= 0;
287 spin_lock_irqsave(&par
->lock
, flags
);
289 spin_unlock_irqrestore(&par
->lock
, flags
);
292 case MDI_GET_CFGINFO
:
293 memset(&kmdi
, 0, sizeof(kmdi
));
295 spin_lock_irqsave(&par
->lock
, flags
);
296 kmdi
.mdi_type
= FBTYPE_MDICOLOR
;
297 kmdi
.mdi_height
= info
->var
.yres
;
298 kmdi
.mdi_width
= info
->var
.xres
;
299 kmdi
.mdi_mode
= par
->mode
;
300 kmdi
.mdi_pixfreq
= 72; /* FIXME */
301 kmdi
.mdi_size
= par
->ramsize
;
302 spin_unlock_irqrestore(&par
->lock
, flags
);
304 mdii
= (struct mdi_cfginfo __user
*) arg
;
305 if (copy_to_user(mdii
, &kmdi
, sizeof(kmdi
)))
309 case MDI_SET_PIXELMODE
:
310 if (get_user(mode
, (int __user
*) arg
)) {
315 spin_lock_irqsave(&par
->lock
, flags
);
316 cur_mode
= sbus_readb(®s
->mcr
);
317 cur_mode
&= ~CG14_MCR_PIXMODE_MASK
;
320 cur_mode
|= (CG14_MCR_PIXMODE_32
<<
321 CG14_MCR_PIXMODE_SHIFT
);
336 sbus_writeb(cur_mode
, ®s
->mcr
);
339 spin_unlock_irqrestore(&par
->lock
, flags
);
343 ret
= sbusfb_ioctl_helper(cmd
, arg
, info
,
344 FBTYPE_MDICOLOR
, 24, par
->fbsize
);
355 static void cg14_init_fix(struct fb_info
*info
, int linebytes
)
357 struct cg14_par
*par
= (struct cg14_par
*)info
->par
;
359 strlcpy(info
->fix
.id
, par
->sdev
->prom_name
, sizeof(info
->fix
.id
));
361 info
->fix
.type
= FB_TYPE_PACKED_PIXELS
;
362 info
->fix
.visual
= FB_VISUAL_TRUECOLOR
;
364 info
->fix
.line_length
= linebytes
;
366 info
->fix
.accel
= FB_ACCEL_SUN_CG14
;
369 static struct sbus_mmap_map __cg14_mmap_map
[CG14_MMAP_ENTRIES
] __initdata
= {
396 .voff
= CG3_MMAP_OFFSET
- 0x7000,
401 .voff
= CG3_MMAP_OFFSET
,
403 .size
= SBUS_MMAP_FBSIZE(1)
406 .voff
= MDI_CURSOR_MAP
,
411 .voff
= MDI_CHUNKY_BGR_MAP
,
416 .voff
= MDI_PLANAR_X16_MAP
,
421 .voff
= MDI_PLANAR_C16_MAP
,
426 .voff
= MDI_PLANAR_X32_MAP
,
431 .voff
= MDI_PLANAR_B32_MAP
,
436 .voff
= MDI_PLANAR_G32_MAP
,
441 .voff
= MDI_PLANAR_R32_MAP
,
451 struct list_head list
;
453 static LIST_HEAD(cg14_list
);
455 static void cg14_init_one(struct sbus_dev
*sdev
, int node
, int parent_node
)
457 struct all_info
*all
;
458 unsigned long phys
, rphys
;
460 int is_8mb
, linebytes
, i
;
463 prom_getproperty(node
, "address",
464 (char *) &bases
[0], sizeof(bases
));
466 printk(KERN_ERR
"cg14: Device is not mapped.\n");
469 if (__get_iospace(bases
[0]) != __get_iospace(bases
[1])) {
470 printk(KERN_ERR
"cg14: I/O spaces don't match.\n");
475 all
= kmalloc(sizeof(*all
), GFP_KERNEL
);
477 printk(KERN_ERR
"cg14: Cannot allocate memory.\n");
480 memset(all
, 0, sizeof(*all
));
482 INIT_LIST_HEAD(&all
->list
);
484 spin_lock_init(&all
->par
.lock
);
486 sbusfb_fill_var(&all
->info
.var
, node
, 8);
488 linebytes
= prom_getintdefault(sdev
->prom_node
, "linebytes",
490 all
->par
.fbsize
= PAGE_ALIGN(linebytes
* all
->info
.var
.yres
);
492 all
->par
.sdev
= sdev
;
494 rphys
= sdev
->reg_addrs
[0].phys_addr
;
495 all
->par
.physbase
= phys
= sdev
->reg_addrs
[1].phys_addr
;
496 all
->par
.iospace
= sdev
->reg_addrs
[0].which_io
;
498 all
->par
.regs
= (struct cg14_regs
*)
499 sbus_ioremap(&sdev
->resource
[0], 0,
500 sizeof(struct cg14_regs
),
502 all
->par
.clut
= (struct cg14_clut
*)
503 sbus_ioremap(&sdev
->resource
[0], CG14_CLUT1
,
504 sizeof(struct cg14_clut
),
506 all
->par
.cursor
= (struct cg14_cursor
*)
507 sbus_ioremap(&sdev
->resource
[0], CG14_CURSORREGS
,
508 sizeof(struct cg14_cursor
),
510 all
->info
.screen_base
= (char *)
511 sbus_ioremap(&sdev
->resource
[1], 0,
512 all
->par
.fbsize
, "cg14 ram");
514 rphys
= __get_phys(bases
[0]);
515 all
->par
.physbase
= phys
= __get_phys(bases
[1]);
516 all
->par
.iospace
= __get_iospace(bases
[0]);
517 all
->par
.regs
= (struct cg14_regs
*)(unsigned long)bases
[0];
518 all
->par
.clut
= (struct cg14_clut
*)((unsigned long)bases
[0] +
521 (struct cg14_cursor
*)((unsigned long)bases
[0] +
524 all
->info
.screen_base
= (char *)(unsigned long)bases
[1];
527 prom_getproperty(node
, "reg", (char *) &bases
[0], sizeof(bases
));
528 is_8mb
= (bases
[5] == 0x800000);
530 if (sizeof(all
->par
.mmap_map
) != sizeof(__cg14_mmap_map
)) {
531 extern void __cg14_mmap_sized_wrongly(void);
533 __cg14_mmap_sized_wrongly();
536 memcpy(&all
->par
.mmap_map
, &__cg14_mmap_map
, sizeof(all
->par
.mmap_map
));
537 for (i
= 0; i
< CG14_MMAP_ENTRIES
; i
++) {
538 struct sbus_mmap_map
*map
= &all
->par
.mmap_map
[i
];
542 if (map
->poff
& 0x80000000)
543 map
->poff
= (map
->poff
& 0x7fffffff) + rphys
- phys
;
545 map
->size
>= 0x100000 &&
546 map
->size
<= 0x400000)
550 all
->par
.mode
= MDI_8_PIX
;
551 all
->par
.ramsize
= (is_8mb
? 0x800000 : 0x400000);
553 all
->info
.flags
= FBINFO_DEFAULT
| FBINFO_HWACCEL_YPAN
;
554 all
->info
.fbops
= &cg14_ops
;
555 all
->info
.currcon
= -1;
556 all
->info
.par
= &all
->par
;
558 __cg14_reset(&all
->par
);
560 if (fb_alloc_cmap(&all
->info
.cmap
, 256, 0)) {
561 printk(KERN_ERR
"cg14: Could not allocate color map.\n");
566 cg14_init_fix(&all
->info
, linebytes
);
568 if (register_framebuffer(&all
->info
) < 0) {
569 printk(KERN_ERR
"cg14: Could not register framebuffer.\n");
570 fb_dealloc_cmap(&all
->info
.cmap
);
575 list_add(&all
->list
, &cg14_list
);
577 printk("cg14: cgfourteen at %lx:%lx\n",
578 all
->par
.physbase
, all
->par
.iospace
);
582 int __init
cg14_init(void)
584 struct sbus_bus
*sbus
;
585 struct sbus_dev
*sdev
;
587 if (fb_get_options("cg14fb", NULL
))
590 #ifdef CONFIG_SPARC32
594 root
= prom_getchild(prom_root_node
);
595 root
= prom_searchsiblings(root
, "obio");
597 node
= prom_searchsiblings(prom_getchild(root
),
600 cg14_init_one(NULL
, node
, root
);
604 for_all_sbusdev(sdev
, sbus
) {
605 if (!strcmp(sdev
->prom_name
, "cgfourteen"))
606 cg14_init_one(sdev
, sdev
->prom_node
, sbus
->prom_node
);
612 void __exit
cg14_exit(void)
614 struct list_head
*pos
, *tmp
;
616 list_for_each_safe(pos
, tmp
, &cg14_list
) {
617 struct all_info
*all
= list_entry(pos
, typeof(*all
), list
);
619 unregister_framebuffer(&all
->info
);
620 fb_dealloc_cmap(&all
->info
.cmap
);
626 cg14_setup(char *arg
)
628 /* No cmdline options yet... */
632 module_init(cg14_init
);
635 module_exit(cg14_exit
);
638 MODULE_DESCRIPTION("framebuffer driver for CGfourteen chipsets");
639 MODULE_AUTHOR("David S. Miller <davem@redhat.com>");
640 MODULE_LICENSE("GPL");