1 /* cg3.c: CGTHREE 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) 1996 Miguel de Icaza (miguel@nuclecu.unam.mx)
 
   6  * Copyright (C) 1997 Eddie C. Dost (ecd@skynet.be)
 
   8  * Driver layout based loosely on tgafb.c, see that file for credits.
 
  11 #include <linux/module.h>
 
  12 #include <linux/kernel.h>
 
  13 #include <linux/errno.h>
 
  14 #include <linux/string.h>
 
  15 #include <linux/slab.h>
 
  16 #include <linux/delay.h>
 
  17 #include <linux/init.h>
 
  23 #include <asm/oplib.h>
 
  32 static int cg3_setcolreg(unsigned, unsigned, unsigned, unsigned,
 
  33                          unsigned, struct fb_info *);
 
  34 static int cg3_blank(int, struct fb_info *);
 
  36 static int cg3_mmap(struct fb_info *, struct file *, struct vm_area_struct *);
 
  37 static int cg3_ioctl(struct inode *, struct file *, unsigned int,
 
  38                      unsigned long, struct fb_info *);
 
  41  *  Frame buffer operations
 
  44 static struct fb_ops cg3_ops = {
 
  46         .fb_setcolreg           = cg3_setcolreg,
 
  47         .fb_blank               = cg3_blank,
 
  48         .fb_fillrect            = cfb_fillrect,
 
  49         .fb_copyarea            = cfb_copyarea,
 
  50         .fb_imageblit           = cfb_imageblit,
 
  52         .fb_ioctl               = cg3_ioctl,
 
  56 /* Control Register Constants */
 
  57 #define CG3_CR_ENABLE_INTS      0x80
 
  58 #define CG3_CR_ENABLE_VIDEO     0x40
 
  59 #define CG3_CR_ENABLE_TIMING    0x20
 
  60 #define CG3_CR_ENABLE_CURCMP    0x10
 
  61 #define CG3_CR_XTAL_MASK        0x0c
 
  62 #define CG3_CR_DIVISOR_MASK     0x03
 
  64 /* Status Register Constants */
 
  65 #define CG3_SR_PENDING_INT      0x80
 
  66 #define CG3_SR_RES_MASK         0x70
 
  67 #define CG3_SR_1152_900_76_A    0x40
 
  68 #define CG3_SR_1152_900_76_B    0x60
 
  69 #define CG3_SR_ID_MASK          0x0f
 
  70 #define CG3_SR_ID_COLOR         0x01
 
  71 #define CG3_SR_ID_MONO          0x02
 
  72 #define CG3_SR_ID_MONO_ECL      0x03
 
  82         volatile u32 color_map;
 
  91         volatile u8     cursor_start;
 
  92         volatile u8     cursor_end;
 
  93         volatile u8     h_blank_start;
 
  94         volatile u8     h_blank_end;
 
  95         volatile u8     h_sync_start;
 
  96         volatile u8     h_sync_end;
 
  97         volatile u8     comp_sync_end;
 
  98         volatile u8     v_blank_start_high;
 
  99         volatile u8     v_blank_start_low;
 
 100         volatile u8     v_blank_end;
 
 101         volatile u8     v_sync_start;
 
 102         volatile u8     v_sync_end;
 
 103         volatile u8     xfer_holdoff_start;
 
 104         volatile u8     xfer_holdoff_end;
 
 107 /* Offset of interesting structures in the OBIO space */
 
 108 #define CG3_REGS_OFFSET      0x400000UL
 
 109 #define CG3_RAM_OFFSET       0x800000UL
 
 113         struct cg3_regs         __iomem *regs;
 
 114         u32                     sw_cmap[((256 * 3) + 3) / 4];
 
 117 #define CG3_FLAG_BLANKED        0x00000001
 
 118 #define CG3_FLAG_RDI            0x00000002
 
 120         unsigned long           physbase;
 
 121         unsigned long           fbsize;
 
 123         struct sbus_dev         *sdev;
 
 124         struct list_head        list;
 
 128  *      cg3_setcolreg - Optional function. Sets a color register.
 
 129  *      @regno: boolean, 0 copy local, 1 get_user() function
 
 130  *      @red: frame buffer colormap structure
 
 131  *      @green: The green value which can be up to 16 bits wide
 
 132  *      @blue:  The blue value which can be up to 16 bits wide.
 
 133  *      @transp: If supported the alpha value which can be up to 16 bits wide.
 
 134  *      @info: frame buffer info structure
 
 136  * The cg3 palette is loaded with 4 color values at each time
 
 137  * so you end up with: (rgb)(r), (gb)(rg), (b)(rgb), and so on.
 
 138  * We keep a sw copy of the hw cmap to assist us in this esoteric
 
 141 static int cg3_setcolreg(unsigned regno,
 
 142                          unsigned red, unsigned green, unsigned blue,
 
 143                          unsigned transp, struct fb_info *info)
 
 145         struct cg3_par *par = (struct cg3_par *) info->par;
 
 146         struct bt_regs __iomem *bt = &par->regs->cmap;
 
 159         spin_lock_irqsave(&par->lock, flags);
 
 161         p8 = (u8 *)par->sw_cmap + (regno * 3);
 
 166 #define D4M3(x) ((((x)>>2)<<1) + ((x)>>2))      /* (x/4)*3 */
 
 167 #define D4M4(x) ((x)&~0x3)                      /* (x/4)*4 */
 
 170         p32 = &par->sw_cmap[D4M3(regno)];
 
 171         sbus_writel(D4M4(regno), &bt->addr);
 
 173                 sbus_writel(*p32++, &bt->color_map);
 
 178         spin_unlock_irqrestore(&par->lock, flags);
 
 184  *      cg3_blank - Optional function.  Blanks the display.
 
 185  *      @blank_mode: the blank mode we want.
 
 186  *      @info: frame buffer structure that represents a single frame buffer
 
 189 cg3_blank(int blank, struct fb_info *info)
 
 191         struct cg3_par *par = (struct cg3_par *) info->par;
 
 192         struct cg3_regs __iomem *regs = par->regs;
 
 196         spin_lock_irqsave(&par->lock, flags);
 
 199         case FB_BLANK_UNBLANK: /* Unblanking */
 
 200                 val = sbus_readb(®s->control);
 
 201                 val |= CG3_CR_ENABLE_VIDEO;
 
 202                 sbus_writeb(val, ®s->control);
 
 203                 par->flags &= ~CG3_FLAG_BLANKED;
 
 206         case FB_BLANK_NORMAL: /* Normal blanking */
 
 207         case FB_BLANK_VSYNC_SUSPEND: /* VESA blank (vsync off) */
 
 208         case FB_BLANK_HSYNC_SUSPEND: /* VESA blank (hsync off) */
 
 209         case FB_BLANK_POWERDOWN: /* Poweroff */
 
 210                 val = sbus_readb(®s->control);
 
 211                 val &= ~CG3_CR_ENABLE_VIDEO;
 
 212                 sbus_writeb(val, ®s->control);
 
 213                 par->flags |= CG3_FLAG_BLANKED;
 
 217         spin_unlock_irqrestore(&par->lock, flags);
 
 222 static struct sbus_mmap_map cg3_mmap_map[] = {
 
 224                 .voff   = CG3_MMAP_OFFSET,
 
 225                 .poff   = CG3_RAM_OFFSET,
 
 226                 .size   = SBUS_MMAP_FBSIZE(1)
 
 231 static int cg3_mmap(struct fb_info *info, struct file *file, struct vm_area_struct *vma)
 
 233         struct cg3_par *par = (struct cg3_par *)info->par;
 
 235         return sbusfb_mmap_helper(cg3_mmap_map,
 
 236                                   par->physbase, par->fbsize,
 
 237                                   par->sdev->reg_addrs[0].which_io,
 
 241 static int cg3_ioctl(struct inode *inode, struct file *file, unsigned int cmd,
 
 242                      unsigned long arg, struct fb_info *info)
 
 244         struct cg3_par *par = (struct cg3_par *) info->par;
 
 246         return sbusfb_ioctl_helper(cmd, arg, info,
 
 247                                    FBTYPE_SUN3COLOR, 8, par->fbsize);
 
 255 cg3_init_fix(struct fb_info *info, int linebytes)
 
 257         struct cg3_par *par = (struct cg3_par *)info->par;
 
 259         strlcpy(info->fix.id, par->sdev->prom_name, sizeof(info->fix.id));
 
 261         info->fix.type = FB_TYPE_PACKED_PIXELS;
 
 262         info->fix.visual = FB_VISUAL_PSEUDOCOLOR;
 
 264         info->fix.line_length = linebytes;
 
 266         info->fix.accel = FB_ACCEL_SUN_CGTHREE;
 
 269 static void cg3_rdi_maybe_fixup_var(struct fb_var_screeninfo *var,
 
 270                                     struct sbus_dev *sdev)
 
 277         prom_getstring(sdev->prom_node, "params", buffer, sizeof(buffer));
 
 279                 ww = simple_strtoul(buffer, &p, 10);
 
 280                 if (ww && *p == 'x') {
 
 281                         hh = simple_strtoul(p + 1, &p, 10);
 
 282                         if (hh && *p == '-') {
 
 283                                 if (var->xres != ww ||
 
 285                                         var->xres = var->xres_virtual = ww;
 
 286                                         var->yres = var->yres_virtual = hh;
 
 293 static u8 cg3regvals_66hz[] __initdata = {      /* 1152 x 900, 66 Hz */
 
 294         0x14, 0xbb,     0x15, 0x2b,     0x16, 0x04,     0x17, 0x14,
 
 295         0x18, 0xae,     0x19, 0x03,     0x1a, 0xa8,     0x1b, 0x24,
 
 296         0x1c, 0x01,     0x1d, 0x05,     0x1e, 0xff,     0x1f, 0x01,
 
 300 static u8 cg3regvals_76hz[] __initdata = {      /* 1152 x 900, 76 Hz */
 
 301         0x14, 0xb7,     0x15, 0x27,     0x16, 0x03,     0x17, 0x0f,
 
 302         0x18, 0xae,     0x19, 0x03,     0x1a, 0xae,     0x1b, 0x2a,
 
 303         0x1c, 0x01,     0x1d, 0x09,     0x1e, 0xff,     0x1f, 0x01,
 
 307 static u8 cg3regvals_rdi[] __initdata = {       /* 640 x 480, cgRDI */
 
 308         0x14, 0x70,     0x15, 0x20,     0x16, 0x08,     0x17, 0x10,
 
 309         0x18, 0x06,     0x19, 0x02,     0x1a, 0x31,     0x1b, 0x51,
 
 310         0x1c, 0x06,     0x1d, 0x0c,     0x1e, 0xff,     0x1f, 0x01,
 
 314 static u8 *cg3_regvals[] __initdata = {
 
 315         cg3regvals_66hz, cg3regvals_76hz, cg3regvals_rdi
 
 318 static u_char cg3_dacvals[] __initdata = {
 
 319         4, 0xff,        5, 0x00,        6, 0x70,        7, 0x00,        0
 
 322 static void cg3_do_default_mode(struct cg3_par *par)
 
 327         if (par->flags & CG3_FLAG_RDI)
 
 330                 u8 status = sbus_readb(&par->regs->status), mon;
 
 331                 if ((status & CG3_SR_ID_MASK) == CG3_SR_ID_COLOR) {
 
 332                         mon = status & CG3_SR_RES_MASK;
 
 333                         if (mon == CG3_SR_1152_900_76_A ||
 
 334                             mon == CG3_SR_1152_900_76_B)
 
 339                         prom_printf("cgthree: can't handle SR %02x\n",
 
 346         for (p = cg3_regvals[type]; *p; p += 2) {
 
 347                 u8 __iomem *regp = &((u8 __iomem *)par->regs)[p[0]];
 
 348                 sbus_writeb(p[1], regp);
 
 350         for (p = cg3_dacvals; *p; p += 2) {
 
 351                 volatile u8 __iomem *regp;
 
 353                 regp = (volatile u8 __iomem *)&par->regs->cmap.addr;
 
 354                 sbus_writeb(p[0], regp);
 
 355                 regp = (volatile u8 __iomem *)&par->regs->cmap.control;
 
 356                 sbus_writeb(p[1], regp);
 
 363         struct list_head list;
 
 365 static LIST_HEAD(cg3_list);
 
 367 static void cg3_init_one(struct sbus_dev *sdev)
 
 369         struct all_info *all;
 
 372         all = kmalloc(sizeof(*all), GFP_KERNEL);
 
 374                 printk(KERN_ERR "cg3: Cannot allocate memory.\n");
 
 377         memset(all, 0, sizeof(*all));
 
 379         INIT_LIST_HEAD(&all->list);
 
 381         spin_lock_init(&all->par.lock);
 
 382         all->par.sdev = sdev;
 
 384         all->par.physbase = sdev->reg_addrs[0].phys_addr;
 
 386         sbusfb_fill_var(&all->info.var, sdev->prom_node, 8);
 
 387         all->info.var.red.length = 8;
 
 388         all->info.var.green.length = 8;
 
 389         all->info.var.blue.length = 8;
 
 390         if (!strcmp(sdev->prom_name, "cgRDI"))
 
 391                 all->par.flags |= CG3_FLAG_RDI;
 
 392         if (all->par.flags & CG3_FLAG_RDI)
 
 393                 cg3_rdi_maybe_fixup_var(&all->info.var, sdev);
 
 395         linebytes = prom_getintdefault(sdev->prom_node, "linebytes",
 
 397         all->par.fbsize = PAGE_ALIGN(linebytes * all->info.var.yres);
 
 399         all->par.regs = sbus_ioremap(&sdev->resource[0], CG3_REGS_OFFSET,
 
 400                              sizeof(struct cg3_regs), "cg3 regs");
 
 402         all->info.flags = FBINFO_DEFAULT;
 
 403         all->info.fbops = &cg3_ops;
 
 404 #ifdef CONFIG_SPARC32
 
 405         all->info.screen_base = (char __iomem *)
 
 406                 prom_getintdefault(sdev->prom_node, "address", 0);
 
 408         if (!all->info.screen_base)
 
 409                 all->info.screen_base =
 
 410                         sbus_ioremap(&sdev->resource[0], CG3_RAM_OFFSET,
 
 411                                      all->par.fbsize, "cg3 ram");
 
 412         all->info.par = &all->par;
 
 414         cg3_blank(0, &all->info);
 
 416         if (!prom_getbool(sdev->prom_node, "width"))
 
 417                 cg3_do_default_mode(&all->par);
 
 419         if (fb_alloc_cmap(&all->info.cmap, 256, 0)) {
 
 420                 printk(KERN_ERR "cg3: Could not allocate color map.\n");
 
 424         fb_set_cmap(&all->info.cmap, &all->info);
 
 426         cg3_init_fix(&all->info, linebytes);
 
 428         if (register_framebuffer(&all->info) < 0) {
 
 429                 printk(KERN_ERR "cg3: Could not register framebuffer.\n");
 
 430                 fb_dealloc_cmap(&all->info.cmap);
 
 435         list_add(&all->list, &cg3_list);
 
 437         printk("cg3: %s at %lx:%lx\n",
 
 439                (long) sdev->reg_addrs[0].which_io,
 
 440                (long) sdev->reg_addrs[0].phys_addr);
 
 443 int __init cg3_init(void)
 
 445         struct sbus_bus *sbus;
 
 446         struct sbus_dev *sdev;
 
 448         if (fb_get_options("cg3fb", NULL))
 
 451         for_all_sbusdev(sdev, sbus) {
 
 452                 if (!strcmp(sdev->prom_name, "cgthree") ||
 
 453                     !strcmp(sdev->prom_name, "cgRDI"))
 
 460 void __exit cg3_exit(void)
 
 462         struct list_head *pos, *tmp;
 
 464         list_for_each_safe(pos, tmp, &cg3_list) {
 
 465                 struct all_info *all = list_entry(pos, typeof(*all), list);
 
 467                 unregister_framebuffer(&all->info);
 
 468                 fb_dealloc_cmap(&all->info.cmap);
 
 476         /* No cmdline options yet... */
 
 480 module_init(cg3_init);
 
 483 module_exit(cg3_exit);
 
 486 MODULE_DESCRIPTION("framebuffer driver for CGthree chipsets");
 
 487 MODULE_AUTHOR("David S. Miller <davem@redhat.com>");
 
 488 MODULE_LICENSE("GPL");