root/drivers/video/fbdev/cirrusfb.c

/* [<][>][^][v][top][bottom][index][help] */

DEFINITIONS

This source file includes following definitions.
  1. is_laguna
  2. cirrusfb_open
  3. cirrusfb_release
  4. cirrusfb_check_mclk
  5. cirrusfb_check_pixclock
  6. cirrusfb_check_var
  7. cirrusfb_set_mclk_as_source
  8. cirrusfb_set_par_foo
  9. cirrusfb_set_par
  10. cirrusfb_setcolreg
  11. cirrusfb_pan_display
  12. cirrusfb_blank
  13. init_vgachip
  14. switch_monitor
  15. cirrusfb_sync
  16. cirrusfb_fillrect
  17. cirrusfb_copyarea
  18. cirrusfb_imageblit
  19. cirrusfb_get_memsize
  20. get_pci_addrs
  21. cirrusfb_pci_unmap
  22. cirrusfb_zorro_unmap
  23. cirrusfb_set_fbinfo
  24. cirrusfb_register
  25. cirrusfb_cleanup
  26. cirrusfb_pci_register
  27. cirrusfb_pci_unregister
  28. cirrusfb_zorro_register
  29. cirrusfb_zorro_unregister
  30. cirrusfb_setup
  31. cirrusfb_init
  32. cirrusfb_exit
  33. WGen
  34. RGen
  35. AttrOn
  36. WHDR
  37. WSFR
  38. WSFR2
  39. WClut
  40. RClut
  41. cirrusfb_WaitBLT
  42. cirrusfb_set_blitter
  43. cirrusfb_BitBLT
  44. cirrusfb_RectFill
  45. bestclock
  46. cirrusfb_dbg_print_regs
  47. cirrusfb_dbg_reg_dump

   1 /*
   2  * drivers/video/cirrusfb.c - driver for Cirrus Logic chipsets
   3  *
   4  * Copyright 1999-2001 Jeff Garzik <jgarzik@pobox.com>
   5  *
   6  * Contributors (thanks, all!)
   7  *
   8  *      David Eger:
   9  *      Overhaul for Linux 2.6
  10  *
  11  *      Jeff Rugen:
  12  *      Major contributions;  Motorola PowerStack (PPC and PCI) support,
  13  *      GD54xx, 1280x1024 mode support, change MCLK based on VCLK.
  14  *
  15  *      Geert Uytterhoeven:
  16  *      Excellent code review.
  17  *
  18  *      Lars Hecking:
  19  *      Amiga updates and testing.
  20  *
  21  * Original cirrusfb author:  Frank Neumann
  22  *
  23  * Based on retz3fb.c and cirrusfb.c:
  24  *      Copyright (C) 1997 Jes Sorensen
  25  *      Copyright (C) 1996 Frank Neumann
  26  *
  27  ***************************************************************
  28  *
  29  * Format this code with GNU indent '-kr -i8 -pcs' options.
  30  *
  31  * This file is subject to the terms and conditions of the GNU General Public
  32  * License.  See the file COPYING in the main directory of this archive
  33  * for more details.
  34  *
  35  */
  36 
  37 #include <linux/module.h>
  38 #include <linux/kernel.h>
  39 #include <linux/errno.h>
  40 #include <linux/string.h>
  41 #include <linux/mm.h>
  42 #include <linux/delay.h>
  43 #include <linux/fb.h>
  44 #include <linux/init.h>
  45 #include <asm/pgtable.h>
  46 
  47 #ifdef CONFIG_ZORRO
  48 #include <linux/zorro.h>
  49 #endif
  50 #ifdef CONFIG_PCI
  51 #include <linux/pci.h>
  52 #endif
  53 #ifdef CONFIG_AMIGA
  54 #include <asm/amigahw.h>
  55 #endif
  56 
  57 #include <video/vga.h>
  58 #include <video/cirrus.h>
  59 
  60 /*****************************************************************
  61  *
  62  * debugging and utility macros
  63  *
  64  */
  65 
  66 /* disable runtime assertions? */
  67 /* #define CIRRUSFB_NDEBUG */
  68 
  69 /* debugging assertions */
  70 #ifndef CIRRUSFB_NDEBUG
  71 #define assert(expr) \
  72         if (!(expr)) { \
  73                 printk("Assertion failed! %s,%s,%s,line=%d\n", \
  74                 #expr, __FILE__, __func__, __LINE__); \
  75         }
  76 #else
  77 #define assert(expr)
  78 #endif
  79 
  80 #define MB_ (1024 * 1024)
  81 
  82 /*****************************************************************
  83  *
  84  * chipset information
  85  *
  86  */
  87 
  88 /* board types */
  89 enum cirrus_board {
  90         BT_NONE = 0,
  91         BT_SD64,        /* GD5434 */
  92         BT_PICCOLO,     /* GD5426 */
  93         BT_PICASSO,     /* GD5426 or GD5428 */
  94         BT_SPECTRUM,    /* GD5426 or GD5428 */
  95         BT_PICASSO4,    /* GD5446 */
  96         BT_ALPINE,      /* GD543x/4x */
  97         BT_GD5480,
  98         BT_LAGUNA,      /* GD5462/64 */
  99         BT_LAGUNAB,     /* GD5465 */
 100 };
 101 
 102 /*
 103  * per-board-type information, used for enumerating and abstracting
 104  * chip-specific information
 105  * NOTE: MUST be in the same order as enum cirrus_board in order to
 106  * use direct indexing on this array
 107  * NOTE: '__initdata' cannot be used as some of this info
 108  * is required at runtime.  Maybe separate into an init-only and
 109  * a run-time table?
 110  */
 111 static const struct cirrusfb_board_info_rec {
 112         char *name;             /* ASCII name of chipset */
 113         long maxclock[5];               /* maximum video clock */
 114         /* for  1/4bpp, 8bpp 15/16bpp, 24bpp, 32bpp - numbers from xorg code */
 115         bool init_sr07 : 1; /* init SR07 during init_vgachip() */
 116         bool init_sr1f : 1; /* write SR1F during init_vgachip() */
 117         /* construct bit 19 of screen start address */
 118         bool scrn_start_bit19 : 1;
 119 
 120         /* initial SR07 value, then for each mode */
 121         unsigned char sr07;
 122         unsigned char sr07_1bpp;
 123         unsigned char sr07_1bpp_mux;
 124         unsigned char sr07_8bpp;
 125         unsigned char sr07_8bpp_mux;
 126 
 127         unsigned char sr1f;     /* SR1F VGA initial register value */
 128 } cirrusfb_board_info[] = {
 129         [BT_SD64] = {
 130                 .name                   = "CL SD64",
 131                 .maxclock               = {
 132                         /* guess */
 133                         /* the SD64/P4 have a higher max. videoclock */
 134                         135100, 135100, 85500, 85500, 0
 135                 },
 136                 .init_sr07              = true,
 137                 .init_sr1f              = true,
 138                 .scrn_start_bit19       = true,
 139                 .sr07                   = 0xF0,
 140                 .sr07_1bpp              = 0xF0,
 141                 .sr07_1bpp_mux          = 0xF6,
 142                 .sr07_8bpp              = 0xF1,
 143                 .sr07_8bpp_mux          = 0xF7,
 144                 .sr1f                   = 0x1E
 145         },
 146         [BT_PICCOLO] = {
 147                 .name                   = "CL Piccolo",
 148                 .maxclock               = {
 149                         /* guess */
 150                         90000, 90000, 90000, 90000, 90000
 151                 },
 152                 .init_sr07              = true,
 153                 .init_sr1f              = true,
 154                 .scrn_start_bit19       = false,
 155                 .sr07                   = 0x80,
 156                 .sr07_1bpp              = 0x80,
 157                 .sr07_8bpp              = 0x81,
 158                 .sr1f                   = 0x22
 159         },
 160         [BT_PICASSO] = {
 161                 .name                   = "CL Picasso",
 162                 .maxclock               = {
 163                         /* guess */
 164                         90000, 90000, 90000, 90000, 90000
 165                 },
 166                 .init_sr07              = true,
 167                 .init_sr1f              = true,
 168                 .scrn_start_bit19       = false,
 169                 .sr07                   = 0x20,
 170                 .sr07_1bpp              = 0x20,
 171                 .sr07_8bpp              = 0x21,
 172                 .sr1f                   = 0x22
 173         },
 174         [BT_SPECTRUM] = {
 175                 .name                   = "CL Spectrum",
 176                 .maxclock               = {
 177                         /* guess */
 178                         90000, 90000, 90000, 90000, 90000
 179                 },
 180                 .init_sr07              = true,
 181                 .init_sr1f              = true,
 182                 .scrn_start_bit19       = false,
 183                 .sr07                   = 0x80,
 184                 .sr07_1bpp              = 0x80,
 185                 .sr07_8bpp              = 0x81,
 186                 .sr1f                   = 0x22
 187         },
 188         [BT_PICASSO4] = {
 189                 .name                   = "CL Picasso4",
 190                 .maxclock               = {
 191                         135100, 135100, 85500, 85500, 0
 192                 },
 193                 .init_sr07              = true,
 194                 .init_sr1f              = false,
 195                 .scrn_start_bit19       = true,
 196                 .sr07                   = 0xA0,
 197                 .sr07_1bpp              = 0xA0,
 198                 .sr07_1bpp_mux          = 0xA6,
 199                 .sr07_8bpp              = 0xA1,
 200                 .sr07_8bpp_mux          = 0xA7,
 201                 .sr1f                   = 0
 202         },
 203         [BT_ALPINE] = {
 204                 .name                   = "CL Alpine",
 205                 .maxclock               = {
 206                         /* for the GD5430.  GD5446 can do more... */
 207                         85500, 85500, 50000, 28500, 0
 208                 },
 209                 .init_sr07              = true,
 210                 .init_sr1f              = true,
 211                 .scrn_start_bit19       = true,
 212                 .sr07                   = 0xA0,
 213                 .sr07_1bpp              = 0xA0,
 214                 .sr07_1bpp_mux          = 0xA6,
 215                 .sr07_8bpp              = 0xA1,
 216                 .sr07_8bpp_mux          = 0xA7,
 217                 .sr1f                   = 0x1C
 218         },
 219         [BT_GD5480] = {
 220                 .name                   = "CL GD5480",
 221                 .maxclock               = {
 222                         135100, 200000, 200000, 135100, 135100
 223                 },
 224                 .init_sr07              = true,
 225                 .init_sr1f              = true,
 226                 .scrn_start_bit19       = true,
 227                 .sr07                   = 0x10,
 228                 .sr07_1bpp              = 0x11,
 229                 .sr07_8bpp              = 0x11,
 230                 .sr1f                   = 0x1C
 231         },
 232         [BT_LAGUNA] = {
 233                 .name                   = "CL Laguna",
 234                 .maxclock               = {
 235                         /* taken from X11 code */
 236                         170000, 170000, 170000, 170000, 135100,
 237                 },
 238                 .init_sr07              = false,
 239                 .init_sr1f              = false,
 240                 .scrn_start_bit19       = true,
 241         },
 242         [BT_LAGUNAB] = {
 243                 .name                   = "CL Laguna AGP",
 244                 .maxclock               = {
 245                         /* taken from X11 code */
 246                         170000, 250000, 170000, 170000, 135100,
 247                 },
 248                 .init_sr07              = false,
 249                 .init_sr1f              = false,
 250                 .scrn_start_bit19       = true,
 251         }
 252 };
 253 
 254 #ifdef CONFIG_PCI
 255 #define CHIP(id, btype) \
 256         { PCI_VENDOR_ID_CIRRUS, id, PCI_ANY_ID, PCI_ANY_ID, 0, 0, (btype) }
 257 
 258 static struct pci_device_id cirrusfb_pci_table[] = {
 259         CHIP(PCI_DEVICE_ID_CIRRUS_5436, BT_ALPINE),
 260         CHIP(PCI_DEVICE_ID_CIRRUS_5434_8, BT_SD64),
 261         CHIP(PCI_DEVICE_ID_CIRRUS_5434_4, BT_SD64),
 262         CHIP(PCI_DEVICE_ID_CIRRUS_5430, BT_ALPINE), /* GD-5440 is same id */
 263         CHIP(PCI_DEVICE_ID_CIRRUS_7543, BT_ALPINE),
 264         CHIP(PCI_DEVICE_ID_CIRRUS_7548, BT_ALPINE),
 265         CHIP(PCI_DEVICE_ID_CIRRUS_5480, BT_GD5480), /* MacPicasso likely */
 266         CHIP(PCI_DEVICE_ID_CIRRUS_5446, BT_PICASSO4), /* Picasso 4 is 5446 */
 267         CHIP(PCI_DEVICE_ID_CIRRUS_5462, BT_LAGUNA), /* CL Laguna */
 268         CHIP(PCI_DEVICE_ID_CIRRUS_5464, BT_LAGUNA), /* CL Laguna 3D */
 269         CHIP(PCI_DEVICE_ID_CIRRUS_5465, BT_LAGUNAB), /* CL Laguna 3DA*/
 270         { 0, }
 271 };
 272 MODULE_DEVICE_TABLE(pci, cirrusfb_pci_table);
 273 #undef CHIP
 274 #endif /* CONFIG_PCI */
 275 
 276 #ifdef CONFIG_ZORRO
 277 struct zorrocl {
 278         enum cirrus_board type; /* Board type */
 279         u32 regoffset;          /* Offset of registers in first Zorro device */
 280         u32 ramsize;            /* Size of video RAM in first Zorro device */
 281                                 /* If zero, use autoprobe on RAM device */
 282         u32 ramoffset;          /* Offset of video RAM in first Zorro device */
 283         zorro_id ramid;         /* Zorro ID of RAM device */
 284         zorro_id ramid2;        /* Zorro ID of optional second RAM device */
 285 };
 286 
 287 static const struct zorrocl zcl_sd64 = {
 288         .type           = BT_SD64,
 289         .ramid          = ZORRO_PROD_HELFRICH_SD64_RAM,
 290 };
 291 
 292 static const struct zorrocl zcl_piccolo = {
 293         .type           = BT_PICCOLO,
 294         .ramid          = ZORRO_PROD_HELFRICH_PICCOLO_RAM,
 295 };
 296 
 297 static const struct zorrocl zcl_picasso = {
 298         .type           = BT_PICASSO,
 299         .ramid          = ZORRO_PROD_VILLAGE_TRONIC_PICASSO_II_II_PLUS_RAM,
 300 };
 301 
 302 static const struct zorrocl zcl_spectrum = {
 303         .type           = BT_SPECTRUM,
 304         .ramid          = ZORRO_PROD_GVP_EGS_28_24_SPECTRUM_RAM,
 305 };
 306 
 307 static const struct zorrocl zcl_picasso4_z3 = {
 308         .type           = BT_PICASSO4,
 309         .regoffset      = 0x00600000,
 310         .ramsize        = 4 * MB_,
 311         .ramoffset      = 0x01000000,   /* 0x02000000 for 64 MiB boards */
 312 };
 313 
 314 static const struct zorrocl zcl_picasso4_z2 = {
 315         .type           = BT_PICASSO4,
 316         .regoffset      = 0x10000,
 317         .ramid          = ZORRO_PROD_VILLAGE_TRONIC_PICASSO_IV_Z2_RAM1,
 318         .ramid2         = ZORRO_PROD_VILLAGE_TRONIC_PICASSO_IV_Z2_RAM2,
 319 };
 320 
 321 
 322 static const struct zorro_device_id cirrusfb_zorro_table[] = {
 323         {
 324                 .id             = ZORRO_PROD_HELFRICH_SD64_REG,
 325                 .driver_data    = (unsigned long)&zcl_sd64,
 326         }, {
 327                 .id             = ZORRO_PROD_HELFRICH_PICCOLO_REG,
 328                 .driver_data    = (unsigned long)&zcl_piccolo,
 329         }, {
 330                 .id     = ZORRO_PROD_VILLAGE_TRONIC_PICASSO_II_II_PLUS_REG,
 331                 .driver_data    = (unsigned long)&zcl_picasso,
 332         }, {
 333                 .id             = ZORRO_PROD_GVP_EGS_28_24_SPECTRUM_REG,
 334                 .driver_data    = (unsigned long)&zcl_spectrum,
 335         }, {
 336                 .id             = ZORRO_PROD_VILLAGE_TRONIC_PICASSO_IV_Z3,
 337                 .driver_data    = (unsigned long)&zcl_picasso4_z3,
 338         }, {
 339                 .id             = ZORRO_PROD_VILLAGE_TRONIC_PICASSO_IV_Z2_REG,
 340                 .driver_data    = (unsigned long)&zcl_picasso4_z2,
 341         },
 342         { 0 }
 343 };
 344 MODULE_DEVICE_TABLE(zorro, cirrusfb_zorro_table);
 345 #endif /* CONFIG_ZORRO */
 346 
 347 #ifdef CIRRUSFB_DEBUG
 348 enum cirrusfb_dbg_reg_class {
 349         CRT,
 350         SEQ
 351 };
 352 #endif          /* CIRRUSFB_DEBUG */
 353 
 354 /* info about board */
 355 struct cirrusfb_info {
 356         u8 __iomem *regbase;
 357         u8 __iomem *laguna_mmio;
 358         enum cirrus_board btype;
 359         unsigned char SFR;      /* Shadow of special function register */
 360 
 361         int multiplexing;
 362         int doubleVCLK;
 363         int blank_mode;
 364         u32 pseudo_palette[16];
 365 
 366         void (*unmap)(struct fb_info *info);
 367 };
 368 
 369 static bool noaccel;
 370 static char *mode_option = "640x480@60";
 371 
 372 /****************************************************************************/
 373 /**** BEGIN PROTOTYPES ******************************************************/
 374 
 375 /*--- Interface used by the world ------------------------------------------*/
 376 static int cirrusfb_pan_display(struct fb_var_screeninfo *var,
 377                                 struct fb_info *info);
 378 
 379 /*--- Internal routines ----------------------------------------------------*/
 380 static void init_vgachip(struct fb_info *info);
 381 static void switch_monitor(struct cirrusfb_info *cinfo, int on);
 382 static void WGen(const struct cirrusfb_info *cinfo,
 383                  int regnum, unsigned char val);
 384 static unsigned char RGen(const struct cirrusfb_info *cinfo, int regnum);
 385 static void AttrOn(const struct cirrusfb_info *cinfo);
 386 static void WHDR(const struct cirrusfb_info *cinfo, unsigned char val);
 387 static void WSFR(struct cirrusfb_info *cinfo, unsigned char val);
 388 static void WSFR2(struct cirrusfb_info *cinfo, unsigned char val);
 389 static void WClut(struct cirrusfb_info *cinfo, unsigned char regnum,
 390                   unsigned char red, unsigned char green, unsigned char blue);
 391 #if 0
 392 static void RClut(struct cirrusfb_info *cinfo, unsigned char regnum,
 393                   unsigned char *red, unsigned char *green,
 394                   unsigned char *blue);
 395 #endif
 396 static void cirrusfb_WaitBLT(u8 __iomem *regbase);
 397 static void cirrusfb_BitBLT(u8 __iomem *regbase, int bits_per_pixel,
 398                             u_short curx, u_short cury,
 399                             u_short destx, u_short desty,
 400                             u_short width, u_short height,
 401                             u_short line_length);
 402 static void cirrusfb_RectFill(u8 __iomem *regbase, int bits_per_pixel,
 403                               u_short x, u_short y,
 404                               u_short width, u_short height,
 405                               u32 fg_color, u32 bg_color,
 406                               u_short line_length, u_char blitmode);
 407 
 408 static void bestclock(long freq, int *nom, int *den, int *div);
 409 
 410 #ifdef CIRRUSFB_DEBUG
 411 static void cirrusfb_dbg_reg_dump(struct fb_info *info, caddr_t regbase);
 412 static void cirrusfb_dbg_print_regs(struct fb_info *info,
 413                                     caddr_t regbase,
 414                                     enum cirrusfb_dbg_reg_class reg_class, ...);
 415 #endif /* CIRRUSFB_DEBUG */
 416 
 417 /*** END   PROTOTYPES ********************************************************/
 418 /*****************************************************************************/
 419 /*** BEGIN Interface Used by the World ***************************************/
 420 
 421 static inline int is_laguna(const struct cirrusfb_info *cinfo)
 422 {
 423         return cinfo->btype == BT_LAGUNA || cinfo->btype == BT_LAGUNAB;
 424 }
 425 
 426 static int opencount;
 427 
 428 /*--- Open /dev/fbx ---------------------------------------------------------*/
 429 static int cirrusfb_open(struct fb_info *info, int user)
 430 {
 431         if (opencount++ == 0)
 432                 switch_monitor(info->par, 1);
 433         return 0;
 434 }
 435 
 436 /*--- Close /dev/fbx --------------------------------------------------------*/
 437 static int cirrusfb_release(struct fb_info *info, int user)
 438 {
 439         if (--opencount == 0)
 440                 switch_monitor(info->par, 0);
 441         return 0;
 442 }
 443 
 444 /**** END   Interface used by the World *************************************/
 445 /****************************************************************************/
 446 /**** BEGIN Hardware specific Routines **************************************/
 447 
 448 /* Check if the MCLK is not a better clock source */
 449 static int cirrusfb_check_mclk(struct fb_info *info, long freq)
 450 {
 451         struct cirrusfb_info *cinfo = info->par;
 452         long mclk = vga_rseq(cinfo->regbase, CL_SEQR1F) & 0x3f;
 453 
 454         /* Read MCLK value */
 455         mclk = (14318 * mclk) >> 3;
 456         dev_dbg(info->device, "Read MCLK of %ld kHz\n", mclk);
 457 
 458         /* Determine if we should use MCLK instead of VCLK, and if so, what we
 459          * should divide it by to get VCLK
 460          */
 461 
 462         if (abs(freq - mclk) < 250) {
 463                 dev_dbg(info->device, "Using VCLK = MCLK\n");
 464                 return 1;
 465         } else if (abs(freq - (mclk / 2)) < 250) {
 466                 dev_dbg(info->device, "Using VCLK = MCLK/2\n");
 467                 return 2;
 468         }
 469 
 470         return 0;
 471 }
 472 
 473 static int cirrusfb_check_pixclock(const struct fb_var_screeninfo *var,
 474                                    struct fb_info *info)
 475 {
 476         long freq;
 477         long maxclock;
 478         struct cirrusfb_info *cinfo = info->par;
 479         unsigned maxclockidx = var->bits_per_pixel >> 3;
 480 
 481         /* convert from ps to kHz */
 482         freq = PICOS2KHZ(var->pixclock);
 483 
 484         dev_dbg(info->device, "desired pixclock: %ld kHz\n", freq);
 485 
 486         maxclock = cirrusfb_board_info[cinfo->btype].maxclock[maxclockidx];
 487         cinfo->multiplexing = 0;
 488 
 489         /* If the frequency is greater than we can support, we might be able
 490          * to use multiplexing for the video mode */
 491         if (freq > maxclock) {
 492                 dev_err(info->device,
 493                         "Frequency greater than maxclock (%ld kHz)\n",
 494                         maxclock);
 495                 return -EINVAL;
 496         }
 497         /*
 498          * Additional constraint: 8bpp uses DAC clock doubling to allow maximum
 499          * pixel clock
 500          */
 501         if (var->bits_per_pixel == 8) {
 502                 switch (cinfo->btype) {
 503                 case BT_ALPINE:
 504                 case BT_SD64:
 505                 case BT_PICASSO4:
 506                         if (freq > 85500)
 507                                 cinfo->multiplexing = 1;
 508                         break;
 509                 case BT_GD5480:
 510                         if (freq > 135100)
 511                                 cinfo->multiplexing = 1;
 512                         break;
 513 
 514                 default:
 515                         break;
 516                 }
 517         }
 518 
 519         /* If we have a 1MB 5434, we need to put ourselves in a mode where
 520          * the VCLK is double the pixel clock. */
 521         cinfo->doubleVCLK = 0;
 522         if (cinfo->btype == BT_SD64 && info->fix.smem_len <= MB_ &&
 523             var->bits_per_pixel == 16) {
 524                 cinfo->doubleVCLK = 1;
 525         }
 526 
 527         return 0;
 528 }
 529 
 530 static int cirrusfb_check_var(struct fb_var_screeninfo *var,
 531                               struct fb_info *info)
 532 {
 533         int yres;
 534         /* memory size in pixels */
 535         unsigned pixels = info->screen_size * 8 / var->bits_per_pixel;
 536         struct cirrusfb_info *cinfo = info->par;
 537 
 538         switch (var->bits_per_pixel) {
 539         case 1:
 540                 var->red.offset = 0;
 541                 var->red.length = 1;
 542                 var->green = var->red;
 543                 var->blue = var->red;
 544                 break;
 545 
 546         case 8:
 547                 var->red.offset = 0;
 548                 var->red.length = 8;
 549                 var->green = var->red;
 550                 var->blue = var->red;
 551                 break;
 552 
 553         case 16:
 554                 var->red.offset = 11;
 555                 var->green.offset = 5;
 556                 var->blue.offset = 0;
 557                 var->red.length = 5;
 558                 var->green.length = 6;
 559                 var->blue.length = 5;
 560                 break;
 561 
 562         case 24:
 563                 var->red.offset = 16;
 564                 var->green.offset = 8;
 565                 var->blue.offset = 0;
 566                 var->red.length = 8;
 567                 var->green.length = 8;
 568                 var->blue.length = 8;
 569                 break;
 570 
 571         default:
 572                 dev_dbg(info->device,
 573                         "Unsupported bpp size: %d\n", var->bits_per_pixel);
 574                 return -EINVAL;
 575         }
 576 
 577         if (var->xres_virtual < var->xres)
 578                 var->xres_virtual = var->xres;
 579         /* use highest possible virtual resolution */
 580         if (var->yres_virtual == -1) {
 581                 var->yres_virtual = pixels / var->xres_virtual;
 582 
 583                 dev_info(info->device,
 584                          "virtual resolution set to maximum of %dx%d\n",
 585                          var->xres_virtual, var->yres_virtual);
 586         }
 587         if (var->yres_virtual < var->yres)
 588                 var->yres_virtual = var->yres;
 589 
 590         if (var->xres_virtual * var->yres_virtual > pixels) {
 591                 dev_err(info->device, "mode %dx%dx%d rejected... "
 592                       "virtual resolution too high to fit into video memory!\n",
 593                         var->xres_virtual, var->yres_virtual,
 594                         var->bits_per_pixel);
 595                 return -EINVAL;
 596         }
 597 
 598         /* truncate xoffset and yoffset to maximum if too high */
 599         if (var->xoffset > var->xres_virtual - var->xres)
 600                 var->xoffset = var->xres_virtual - var->xres - 1;
 601         if (var->yoffset > var->yres_virtual - var->yres)
 602                 var->yoffset = var->yres_virtual - var->yres - 1;
 603 
 604         var->red.msb_right =
 605             var->green.msb_right =
 606             var->blue.msb_right =
 607             var->transp.offset =
 608             var->transp.length =
 609             var->transp.msb_right = 0;
 610 
 611         yres = var->yres;
 612         if (var->vmode & FB_VMODE_DOUBLE)
 613                 yres *= 2;
 614         else if (var->vmode & FB_VMODE_INTERLACED)
 615                 yres = (yres + 1) / 2;
 616 
 617         if (yres >= 1280) {
 618                 dev_err(info->device, "ERROR: VerticalTotal >= 1280; "
 619                         "special treatment required! (TODO)\n");
 620                 return -EINVAL;
 621         }
 622 
 623         if (cirrusfb_check_pixclock(var, info))
 624                 return -EINVAL;
 625 
 626         if (!is_laguna(cinfo))
 627                 var->accel_flags = FB_ACCELF_TEXT;
 628 
 629         return 0;
 630 }
 631 
 632 static void cirrusfb_set_mclk_as_source(const struct fb_info *info, int div)
 633 {
 634         struct cirrusfb_info *cinfo = info->par;
 635         unsigned char old1f, old1e;
 636 
 637         assert(cinfo != NULL);
 638         old1f = vga_rseq(cinfo->regbase, CL_SEQR1F) & ~0x40;
 639 
 640         if (div) {
 641                 dev_dbg(info->device, "Set %s as pixclock source.\n",
 642                         (div == 2) ? "MCLK/2" : "MCLK");
 643                 old1f |= 0x40;
 644                 old1e = vga_rseq(cinfo->regbase, CL_SEQR1E) & ~0x1;
 645                 if (div == 2)
 646                         old1e |= 1;
 647 
 648                 vga_wseq(cinfo->regbase, CL_SEQR1E, old1e);
 649         }
 650         vga_wseq(cinfo->regbase, CL_SEQR1F, old1f);
 651 }
 652 
 653 /*************************************************************************
 654         cirrusfb_set_par_foo()
 655 
 656         actually writes the values for a new video mode into the hardware,
 657 **************************************************************************/
 658 static int cirrusfb_set_par_foo(struct fb_info *info)
 659 {
 660         struct cirrusfb_info *cinfo = info->par;
 661         struct fb_var_screeninfo *var = &info->var;
 662         u8 __iomem *regbase = cinfo->regbase;
 663         unsigned char tmp;
 664         int pitch;
 665         const struct cirrusfb_board_info_rec *bi;
 666         int hdispend, hsyncstart, hsyncend, htotal;
 667         int yres, vdispend, vsyncstart, vsyncend, vtotal;
 668         long freq;
 669         int nom, den, div;
 670         unsigned int control = 0, format = 0, threshold = 0;
 671 
 672         dev_dbg(info->device, "Requested mode: %dx%dx%d\n",
 673                var->xres, var->yres, var->bits_per_pixel);
 674 
 675         switch (var->bits_per_pixel) {
 676         case 1:
 677                 info->fix.line_length = var->xres_virtual / 8;
 678                 info->fix.visual = FB_VISUAL_MONO10;
 679                 break;
 680 
 681         case 8:
 682                 info->fix.line_length = var->xres_virtual;
 683                 info->fix.visual = FB_VISUAL_PSEUDOCOLOR;
 684                 break;
 685 
 686         case 16:
 687         case 24:
 688                 info->fix.line_length = var->xres_virtual *
 689                                         var->bits_per_pixel >> 3;
 690                 info->fix.visual = FB_VISUAL_TRUECOLOR;
 691                 break;
 692         }
 693         info->fix.type = FB_TYPE_PACKED_PIXELS;
 694 
 695         init_vgachip(info);
 696 
 697         bi = &cirrusfb_board_info[cinfo->btype];
 698 
 699         hsyncstart = var->xres + var->right_margin;
 700         hsyncend = hsyncstart + var->hsync_len;
 701         htotal = (hsyncend + var->left_margin) / 8;
 702         hdispend = var->xres / 8;
 703         hsyncstart = hsyncstart / 8;
 704         hsyncend = hsyncend / 8;
 705 
 706         vdispend = var->yres;
 707         vsyncstart = vdispend + var->lower_margin;
 708         vsyncend = vsyncstart + var->vsync_len;
 709         vtotal = vsyncend + var->upper_margin;
 710 
 711         if (var->vmode & FB_VMODE_DOUBLE) {
 712                 vdispend *= 2;
 713                 vsyncstart *= 2;
 714                 vsyncend *= 2;
 715                 vtotal *= 2;
 716         } else if (var->vmode & FB_VMODE_INTERLACED) {
 717                 vdispend = (vdispend + 1) / 2;
 718                 vsyncstart = (vsyncstart + 1) / 2;
 719                 vsyncend = (vsyncend + 1) / 2;
 720                 vtotal = (vtotal + 1) / 2;
 721         }
 722         yres = vdispend;
 723         if (yres >= 1024) {
 724                 vtotal /= 2;
 725                 vsyncstart /= 2;
 726                 vsyncend /= 2;
 727                 vdispend /= 2;
 728         }
 729 
 730         vdispend -= 1;
 731         vsyncstart -= 1;
 732         vsyncend -= 1;
 733         vtotal -= 2;
 734 
 735         if (cinfo->multiplexing) {
 736                 htotal /= 2;
 737                 hsyncstart /= 2;
 738                 hsyncend /= 2;
 739                 hdispend /= 2;
 740         }
 741 
 742         htotal -= 5;
 743         hdispend -= 1;
 744         hsyncstart += 1;
 745         hsyncend += 1;
 746 
 747         /* unlock register VGA_CRTC_H_TOTAL..CRT7 */
 748         vga_wcrt(regbase, VGA_CRTC_V_SYNC_END, 0x20);   /* previously: 0x00) */
 749 
 750         /* if debugging is enabled, all parameters get output before writing */
 751         dev_dbg(info->device, "CRT0: %d\n", htotal);
 752         vga_wcrt(regbase, VGA_CRTC_H_TOTAL, htotal);
 753 
 754         dev_dbg(info->device, "CRT1: %d\n", hdispend);
 755         vga_wcrt(regbase, VGA_CRTC_H_DISP, hdispend);
 756 
 757         dev_dbg(info->device, "CRT2: %d\n", var->xres / 8);
 758         vga_wcrt(regbase, VGA_CRTC_H_BLANK_START, var->xres / 8);
 759 
 760         /*  + 128: Compatible read */
 761         dev_dbg(info->device, "CRT3: 128+%d\n", (htotal + 5) % 32);
 762         vga_wcrt(regbase, VGA_CRTC_H_BLANK_END,
 763                  128 + ((htotal + 5) % 32));
 764 
 765         dev_dbg(info->device, "CRT4: %d\n", hsyncstart);
 766         vga_wcrt(regbase, VGA_CRTC_H_SYNC_START, hsyncstart);
 767 
 768         tmp = hsyncend % 32;
 769         if ((htotal + 5) & 32)
 770                 tmp += 128;
 771         dev_dbg(info->device, "CRT5: %d\n", tmp);
 772         vga_wcrt(regbase, VGA_CRTC_H_SYNC_END, tmp);
 773 
 774         dev_dbg(info->device, "CRT6: %d\n", vtotal & 0xff);
 775         vga_wcrt(regbase, VGA_CRTC_V_TOTAL, vtotal & 0xff);
 776 
 777         tmp = 16;               /* LineCompare bit #9 */
 778         if (vtotal & 256)
 779                 tmp |= 1;
 780         if (vdispend & 256)
 781                 tmp |= 2;
 782         if (vsyncstart & 256)
 783                 tmp |= 4;
 784         if ((vdispend + 1) & 256)
 785                 tmp |= 8;
 786         if (vtotal & 512)
 787                 tmp |= 32;
 788         if (vdispend & 512)
 789                 tmp |= 64;
 790         if (vsyncstart & 512)
 791                 tmp |= 128;
 792         dev_dbg(info->device, "CRT7: %d\n", tmp);
 793         vga_wcrt(regbase, VGA_CRTC_OVERFLOW, tmp);
 794 
 795         tmp = 0x40;             /* LineCompare bit #8 */
 796         if ((vdispend + 1) & 512)
 797                 tmp |= 0x20;
 798         if (var->vmode & FB_VMODE_DOUBLE)
 799                 tmp |= 0x80;
 800         dev_dbg(info->device, "CRT9: %d\n", tmp);
 801         vga_wcrt(regbase, VGA_CRTC_MAX_SCAN, tmp);
 802 
 803         dev_dbg(info->device, "CRT10: %d\n", vsyncstart & 0xff);
 804         vga_wcrt(regbase, VGA_CRTC_V_SYNC_START, vsyncstart & 0xff);
 805 
 806         dev_dbg(info->device, "CRT11: 64+32+%d\n", vsyncend % 16);
 807         vga_wcrt(regbase, VGA_CRTC_V_SYNC_END, vsyncend % 16 + 64 + 32);
 808 
 809         dev_dbg(info->device, "CRT12: %d\n", vdispend & 0xff);
 810         vga_wcrt(regbase, VGA_CRTC_V_DISP_END, vdispend & 0xff);
 811 
 812         dev_dbg(info->device, "CRT15: %d\n", (vdispend + 1) & 0xff);
 813         vga_wcrt(regbase, VGA_CRTC_V_BLANK_START, (vdispend + 1) & 0xff);
 814 
 815         dev_dbg(info->device, "CRT16: %d\n", vtotal & 0xff);
 816         vga_wcrt(regbase, VGA_CRTC_V_BLANK_END, vtotal & 0xff);
 817 
 818         dev_dbg(info->device, "CRT18: 0xff\n");
 819         vga_wcrt(regbase, VGA_CRTC_LINE_COMPARE, 0xff);
 820 
 821         tmp = 0;
 822         if (var->vmode & FB_VMODE_INTERLACED)
 823                 tmp |= 1;
 824         if ((htotal + 5) & 64)
 825                 tmp |= 16;
 826         if ((htotal + 5) & 128)
 827                 tmp |= 32;
 828         if (vtotal & 256)
 829                 tmp |= 64;
 830         if (vtotal & 512)
 831                 tmp |= 128;
 832 
 833         dev_dbg(info->device, "CRT1a: %d\n", tmp);
 834         vga_wcrt(regbase, CL_CRT1A, tmp);
 835 
 836         freq = PICOS2KHZ(var->pixclock);
 837         if (var->bits_per_pixel == 24)
 838                 if (cinfo->btype == BT_ALPINE || cinfo->btype == BT_SD64)
 839                         freq *= 3;
 840         if (cinfo->multiplexing)
 841                 freq /= 2;
 842         if (cinfo->doubleVCLK)
 843                 freq *= 2;
 844 
 845         bestclock(freq, &nom, &den, &div);
 846 
 847         dev_dbg(info->device, "VCLK freq: %ld kHz  nom: %d  den: %d  div: %d\n",
 848                 freq, nom, den, div);
 849 
 850         /* set VCLK0 */
 851         /* hardware RefClock: 14.31818 MHz */
 852         /* formula: VClk = (OSC * N) / (D * (1+P)) */
 853         /* Example: VClk = (14.31818 * 91) / (23 * (1+1)) = 28.325 MHz */
 854 
 855         if (cinfo->btype == BT_ALPINE || cinfo->btype == BT_PICASSO4 ||
 856             cinfo->btype == BT_SD64) {
 857                 /* if freq is close to mclk or mclk/2 select mclk
 858                  * as clock source
 859                  */
 860                 int divMCLK = cirrusfb_check_mclk(info, freq);
 861                 if (divMCLK)
 862                         nom = 0;
 863                 cirrusfb_set_mclk_as_source(info, divMCLK);
 864         }
 865         if (is_laguna(cinfo)) {
 866                 long pcifc = fb_readl(cinfo->laguna_mmio + 0x3fc);
 867                 unsigned char tile = fb_readb(cinfo->laguna_mmio + 0x407);
 868                 unsigned short tile_control;
 869 
 870                 if (cinfo->btype == BT_LAGUNAB) {
 871                         tile_control = fb_readw(cinfo->laguna_mmio + 0x2c4);
 872                         tile_control &= ~0x80;
 873                         fb_writew(tile_control, cinfo->laguna_mmio + 0x2c4);
 874                 }
 875 
 876                 fb_writel(pcifc | 0x10000000l, cinfo->laguna_mmio + 0x3fc);
 877                 fb_writeb(tile & 0x3f, cinfo->laguna_mmio + 0x407);
 878                 control = fb_readw(cinfo->laguna_mmio + 0x402);
 879                 threshold = fb_readw(cinfo->laguna_mmio + 0xea);
 880                 control &= ~0x6800;
 881                 format = 0;
 882                 threshold &= 0xffc0 & 0x3fbf;
 883         }
 884         if (nom) {
 885                 tmp = den << 1;
 886                 if (div != 0)
 887                         tmp |= 1;
 888                 /* 6 bit denom; ONLY 5434!!! (bugged me 10 days) */
 889                 if ((cinfo->btype == BT_SD64) ||
 890                     (cinfo->btype == BT_ALPINE) ||
 891                     (cinfo->btype == BT_GD5480))
 892                         tmp |= 0x80;
 893 
 894                 /* Laguna chipset has reversed clock registers */
 895                 if (is_laguna(cinfo)) {
 896                         vga_wseq(regbase, CL_SEQRE, tmp);
 897                         vga_wseq(regbase, CL_SEQR1E, nom);
 898                 } else {
 899                         vga_wseq(regbase, CL_SEQRE, nom);
 900                         vga_wseq(regbase, CL_SEQR1E, tmp);
 901                 }
 902         }
 903 
 904         if (yres >= 1024)
 905                 /* 1280x1024 */
 906                 vga_wcrt(regbase, VGA_CRTC_MODE, 0xc7);
 907         else
 908                 /* mode control: VGA_CRTC_START_HI enable, ROTATE(?), 16bit
 909                  * address wrap, no compat. */
 910                 vga_wcrt(regbase, VGA_CRTC_MODE, 0xc3);
 911 
 912         /* don't know if it would hurt to also program this if no interlaced */
 913         /* mode is used, but I feel better this way.. :-) */
 914         if (var->vmode & FB_VMODE_INTERLACED)
 915                 vga_wcrt(regbase, VGA_CRTC_REGS, htotal / 2);
 916         else
 917                 vga_wcrt(regbase, VGA_CRTC_REGS, 0x00); /* interlace control */
 918 
 919         /* adjust horizontal/vertical sync type (low/high), use VCLK3 */
 920         /* enable display memory & CRTC I/O address for color mode */
 921         tmp = 0x03 | 0xc;
 922         if (var->sync & FB_SYNC_HOR_HIGH_ACT)
 923                 tmp |= 0x40;
 924         if (var->sync & FB_SYNC_VERT_HIGH_ACT)
 925                 tmp |= 0x80;
 926         WGen(cinfo, VGA_MIS_W, tmp);
 927 
 928         /* text cursor on and start line */
 929         vga_wcrt(regbase, VGA_CRTC_CURSOR_START, 0);
 930         /* text cursor end line */
 931         vga_wcrt(regbase, VGA_CRTC_CURSOR_END, 31);
 932 
 933         /******************************************************
 934          *
 935          * 1 bpp
 936          *
 937          */
 938 
 939         /* programming for different color depths */
 940         if (var->bits_per_pixel == 1) {
 941                 dev_dbg(info->device, "preparing for 1 bit deep display\n");
 942                 vga_wgfx(regbase, VGA_GFX_MODE, 0);     /* mode register */
 943 
 944                 /* SR07 */
 945                 switch (cinfo->btype) {
 946                 case BT_SD64:
 947                 case BT_PICCOLO:
 948                 case BT_PICASSO:
 949                 case BT_SPECTRUM:
 950                 case BT_PICASSO4:
 951                 case BT_ALPINE:
 952                 case BT_GD5480:
 953                         vga_wseq(regbase, CL_SEQR7,
 954                                  cinfo->multiplexing ?
 955                                         bi->sr07_1bpp_mux : bi->sr07_1bpp);
 956                         break;
 957 
 958                 case BT_LAGUNA:
 959                 case BT_LAGUNAB:
 960                         vga_wseq(regbase, CL_SEQR7,
 961                                 vga_rseq(regbase, CL_SEQR7) & ~0x01);
 962                         break;
 963 
 964                 default:
 965                         dev_warn(info->device, "unknown Board\n");
 966                         break;
 967                 }
 968 
 969                 /* Extended Sequencer Mode */
 970                 switch (cinfo->btype) {
 971 
 972                 case BT_PICCOLO:
 973                 case BT_SPECTRUM:
 974                         /* evtl d0 bei 1 bit? avoid FIFO underruns..? */
 975                         vga_wseq(regbase, CL_SEQRF, 0xb0);
 976                         break;
 977 
 978                 case BT_PICASSO:
 979                         /* ## vorher d0 avoid FIFO underruns..? */
 980                         vga_wseq(regbase, CL_SEQRF, 0xd0);
 981                         break;
 982 
 983                 case BT_SD64:
 984                 case BT_PICASSO4:
 985                 case BT_ALPINE:
 986                 case BT_GD5480:
 987                 case BT_LAGUNA:
 988                 case BT_LAGUNAB:
 989                         /* do nothing */
 990                         break;
 991 
 992                 default:
 993                         dev_warn(info->device, "unknown Board\n");
 994                         break;
 995                 }
 996 
 997                 /* pixel mask: pass-through for first plane */
 998                 WGen(cinfo, VGA_PEL_MSK, 0x01);
 999                 if (cinfo->multiplexing)
1000                         /* hidden dac reg: 1280x1024 */
1001                         WHDR(cinfo, 0x4a);
1002                 else
1003                         /* hidden dac: nothing */
1004                         WHDR(cinfo, 0);
1005                 /* memory mode: odd/even, ext. memory */
1006                 vga_wseq(regbase, VGA_SEQ_MEMORY_MODE, 0x06);
1007                 /* plane mask: only write to first plane */
1008                 vga_wseq(regbase, VGA_SEQ_PLANE_WRITE, 0x01);
1009         }
1010 
1011         /******************************************************
1012          *
1013          * 8 bpp
1014          *
1015          */
1016 
1017         else if (var->bits_per_pixel == 8) {
1018                 dev_dbg(info->device, "preparing for 8 bit deep display\n");
1019                 switch (cinfo->btype) {
1020                 case BT_SD64:
1021                 case BT_PICCOLO:
1022                 case BT_PICASSO:
1023                 case BT_SPECTRUM:
1024                 case BT_PICASSO4:
1025                 case BT_ALPINE:
1026                 case BT_GD5480:
1027                         vga_wseq(regbase, CL_SEQR7,
1028                                   cinfo->multiplexing ?
1029                                         bi->sr07_8bpp_mux : bi->sr07_8bpp);
1030                         break;
1031 
1032                 case BT_LAGUNA:
1033                 case BT_LAGUNAB:
1034                         vga_wseq(regbase, CL_SEQR7,
1035                                 vga_rseq(regbase, CL_SEQR7) | 0x01);
1036                         threshold |= 0x10;
1037                         break;
1038 
1039                 default:
1040                         dev_warn(info->device, "unknown Board\n");
1041                         break;
1042                 }
1043 
1044                 switch (cinfo->btype) {
1045                 case BT_PICCOLO:
1046                 case BT_PICASSO:
1047                 case BT_SPECTRUM:
1048                         /* Fast Page-Mode writes */
1049                         vga_wseq(regbase, CL_SEQRF, 0xb0);
1050                         break;
1051 
1052                 case BT_PICASSO4:
1053 #ifdef CONFIG_ZORRO
1054                         /* ### INCOMPLETE!! */
1055                         vga_wseq(regbase, CL_SEQRF, 0xb8);
1056 #endif
1057                 case BT_ALPINE:
1058                 case BT_SD64:
1059                 case BT_GD5480:
1060                 case BT_LAGUNA:
1061                 case BT_LAGUNAB:
1062                         /* do nothing */
1063                         break;
1064 
1065                 default:
1066                         dev_warn(info->device, "unknown board\n");
1067                         break;
1068                 }
1069 
1070                 /* mode register: 256 color mode */
1071                 vga_wgfx(regbase, VGA_GFX_MODE, 64);
1072                 if (cinfo->multiplexing)
1073                         /* hidden dac reg: 1280x1024 */
1074                         WHDR(cinfo, 0x4a);
1075                 else
1076                         /* hidden dac: nothing */
1077                         WHDR(cinfo, 0);
1078         }
1079 
1080         /******************************************************
1081          *
1082          * 16 bpp
1083          *
1084          */
1085 
1086         else if (var->bits_per_pixel == 16) {
1087                 dev_dbg(info->device, "preparing for 16 bit deep display\n");
1088                 switch (cinfo->btype) {
1089                 case BT_PICCOLO:
1090                 case BT_SPECTRUM:
1091                         vga_wseq(regbase, CL_SEQR7, 0x87);
1092                         /* Fast Page-Mode writes */
1093                         vga_wseq(regbase, CL_SEQRF, 0xb0);
1094                         break;
1095 
1096                 case BT_PICASSO:
1097                         vga_wseq(regbase, CL_SEQR7, 0x27);
1098                         /* Fast Page-Mode writes */
1099                         vga_wseq(regbase, CL_SEQRF, 0xb0);
1100                         break;
1101 
1102                 case BT_SD64:
1103                 case BT_PICASSO4:
1104                 case BT_ALPINE:
1105                         /* Extended Sequencer Mode: 256c col. mode */
1106                         vga_wseq(regbase, CL_SEQR7,
1107                                         cinfo->doubleVCLK ? 0xa3 : 0xa7);
1108                         break;
1109 
1110                 case BT_GD5480:
1111                         vga_wseq(regbase, CL_SEQR7, 0x17);
1112                         /* We already set SRF and SR1F */
1113                         break;
1114 
1115                 case BT_LAGUNA:
1116                 case BT_LAGUNAB:
1117                         vga_wseq(regbase, CL_SEQR7,
1118                                 vga_rseq(regbase, CL_SEQR7) & ~0x01);
1119                         control |= 0x2000;
1120                         format |= 0x1400;
1121                         threshold |= 0x10;
1122                         break;
1123 
1124                 default:
1125                         dev_warn(info->device, "unknown Board\n");
1126                         break;
1127                 }
1128 
1129                 /* mode register: 256 color mode */
1130                 vga_wgfx(regbase, VGA_GFX_MODE, 64);
1131 #ifdef CONFIG_PCI
1132                 WHDR(cinfo, cinfo->doubleVCLK ? 0xe1 : 0xc1);
1133 #elif defined(CONFIG_ZORRO)
1134                 /* FIXME: CONFIG_PCI and CONFIG_ZORRO may be defined both */
1135                 WHDR(cinfo, 0xa0);      /* hidden dac reg: nothing special */
1136 #endif
1137         }
1138 
1139         /******************************************************
1140          *
1141          * 24 bpp
1142          *
1143          */
1144 
1145         else if (var->bits_per_pixel == 24) {
1146                 dev_dbg(info->device, "preparing for 24 bit deep display\n");
1147                 switch (cinfo->btype) {
1148                 case BT_PICCOLO:
1149                 case BT_SPECTRUM:
1150                         vga_wseq(regbase, CL_SEQR7, 0x85);
1151                         /* Fast Page-Mode writes */
1152                         vga_wseq(regbase, CL_SEQRF, 0xb0);
1153                         break;
1154 
1155                 case BT_PICASSO:
1156                         vga_wseq(regbase, CL_SEQR7, 0x25);
1157                         /* Fast Page-Mode writes */
1158                         vga_wseq(regbase, CL_SEQRF, 0xb0);
1159                         break;
1160 
1161                 case BT_SD64:
1162                 case BT_PICASSO4:
1163                 case BT_ALPINE:
1164                         /* Extended Sequencer Mode: 256c col. mode */
1165                         vga_wseq(regbase, CL_SEQR7, 0xa5);
1166                         break;
1167 
1168                 case BT_GD5480:
1169                         vga_wseq(regbase, CL_SEQR7, 0x15);
1170                         /* We already set SRF and SR1F */
1171                         break;
1172 
1173                 case BT_LAGUNA:
1174                 case BT_LAGUNAB:
1175                         vga_wseq(regbase, CL_SEQR7,
1176                                 vga_rseq(regbase, CL_SEQR7) & ~0x01);
1177                         control |= 0x4000;
1178                         format |= 0x2400;
1179                         threshold |= 0x20;
1180                         break;
1181 
1182                 default:
1183                         dev_warn(info->device, "unknown Board\n");
1184                         break;
1185                 }
1186 
1187                 /* mode register: 256 color mode */
1188                 vga_wgfx(regbase, VGA_GFX_MODE, 64);
1189                 /* hidden dac reg: 8-8-8 mode (24 or 32) */
1190                 WHDR(cinfo, 0xc5);
1191         }
1192 
1193         /******************************************************
1194          *
1195          * unknown/unsupported bpp
1196          *
1197          */
1198 
1199         else
1200                 dev_err(info->device,
1201                         "What's this? requested color depth == %d.\n",
1202                         var->bits_per_pixel);
1203 
1204         pitch = info->fix.line_length >> 3;
1205         vga_wcrt(regbase, VGA_CRTC_OFFSET, pitch & 0xff);
1206         tmp = 0x22;
1207         if (pitch & 0x100)
1208                 tmp |= 0x10;    /* offset overflow bit */
1209 
1210         /* screen start addr #16-18, fastpagemode cycles */
1211         vga_wcrt(regbase, CL_CRT1B, tmp);
1212 
1213         /* screen start address bit 19 */
1214         if (cirrusfb_board_info[cinfo->btype].scrn_start_bit19)
1215                 vga_wcrt(regbase, CL_CRT1D, (pitch >> 9) & 1);
1216 
1217         if (is_laguna(cinfo)) {
1218                 tmp = 0;
1219                 if ((htotal + 5) & 256)
1220                         tmp |= 128;
1221                 if (hdispend & 256)
1222                         tmp |= 64;
1223                 if (hsyncstart & 256)
1224                         tmp |= 48;
1225                 if (vtotal & 1024)
1226                         tmp |= 8;
1227                 if (vdispend & 1024)
1228                         tmp |= 4;
1229                 if (vsyncstart & 1024)
1230                         tmp |= 3;
1231 
1232                 vga_wcrt(regbase, CL_CRT1E, tmp);
1233                 dev_dbg(info->device, "CRT1e: %d\n", tmp);
1234         }
1235 
1236         /* pixel panning */
1237         vga_wattr(regbase, CL_AR33, 0);
1238 
1239         /* [ EGS: SetOffset(); ] */
1240         /* From SetOffset(): Turn on VideoEnable bit in Attribute controller */
1241         AttrOn(cinfo);
1242 
1243         if (is_laguna(cinfo)) {
1244                 /* no tiles */
1245                 fb_writew(control | 0x1000, cinfo->laguna_mmio + 0x402);
1246                 fb_writew(format, cinfo->laguna_mmio + 0xc0);
1247                 fb_writew(threshold, cinfo->laguna_mmio + 0xea);
1248         }
1249         /* finally, turn on everything - turn off "FullBandwidth" bit */
1250         /* also, set "DotClock%2" bit where requested */
1251         tmp = 0x01;
1252 
1253 /*** FB_VMODE_CLOCK_HALVE in linux/fb.h not defined anymore ?
1254     if (var->vmode & FB_VMODE_CLOCK_HALVE)
1255         tmp |= 0x08;
1256 */
1257 
1258         vga_wseq(regbase, VGA_SEQ_CLOCK_MODE, tmp);
1259         dev_dbg(info->device, "CL_SEQR1: %d\n", tmp);
1260 
1261 #ifdef CIRRUSFB_DEBUG
1262         cirrusfb_dbg_reg_dump(info, NULL);
1263 #endif
1264 
1265         return 0;
1266 }
1267 
1268 /* for some reason incomprehensible to me, cirrusfb requires that you write
1269  * the registers twice for the settings to take..grr. -dte */
1270 static int cirrusfb_set_par(struct fb_info *info)
1271 {
1272         cirrusfb_set_par_foo(info);
1273         return cirrusfb_set_par_foo(info);
1274 }
1275 
1276 static int cirrusfb_setcolreg(unsigned regno, unsigned red, unsigned green,
1277                               unsigned blue, unsigned transp,
1278                               struct fb_info *info)
1279 {
1280         struct cirrusfb_info *cinfo = info->par;
1281 
1282         if (regno > 255)
1283                 return -EINVAL;
1284 
1285         if (info->fix.visual == FB_VISUAL_TRUECOLOR) {
1286                 u32 v;
1287                 red >>= (16 - info->var.red.length);
1288                 green >>= (16 - info->var.green.length);
1289                 blue >>= (16 - info->var.blue.length);
1290 
1291                 if (regno >= 16)
1292                         return 1;
1293                 v = (red << info->var.red.offset) |
1294                     (green << info->var.green.offset) |
1295                     (blue << info->var.blue.offset);
1296 
1297                 cinfo->pseudo_palette[regno] = v;
1298                 return 0;
1299         }
1300 
1301         if (info->var.bits_per_pixel == 8)
1302                 WClut(cinfo, regno, red >> 10, green >> 10, blue >> 10);
1303 
1304         return 0;
1305 
1306 }
1307 
1308 /*************************************************************************
1309         cirrusfb_pan_display()
1310 
1311         performs display panning - provided hardware permits this
1312 **************************************************************************/
1313 static int cirrusfb_pan_display(struct fb_var_screeninfo *var,
1314                                 struct fb_info *info)
1315 {
1316         int xoffset;
1317         unsigned long base;
1318         unsigned char tmp, xpix;
1319         struct cirrusfb_info *cinfo = info->par;
1320 
1321         /* no range checks for xoffset and yoffset,   */
1322         /* as fb_pan_display has already done this */
1323         if (var->vmode & FB_VMODE_YWRAP)
1324                 return -EINVAL;
1325 
1326         xoffset = var->xoffset * info->var.bits_per_pixel / 8;
1327 
1328         base = var->yoffset * info->fix.line_length + xoffset;
1329 
1330         if (info->var.bits_per_pixel == 1) {
1331                 /* base is already correct */
1332                 xpix = (unsigned char) (var->xoffset % 8);
1333         } else {
1334                 base /= 4;
1335                 xpix = (unsigned char) ((xoffset % 4) * 2);
1336         }
1337 
1338         if (!is_laguna(cinfo))
1339                 cirrusfb_WaitBLT(cinfo->regbase);
1340 
1341         /* lower 8 + 8 bits of screen start address */
1342         vga_wcrt(cinfo->regbase, VGA_CRTC_START_LO, base & 0xff);
1343         vga_wcrt(cinfo->regbase, VGA_CRTC_START_HI, (base >> 8) & 0xff);
1344 
1345         /* 0xf2 is %11110010, exclude tmp bits */
1346         tmp = vga_rcrt(cinfo->regbase, CL_CRT1B) & 0xf2;
1347         /* construct bits 16, 17 and 18 of screen start address */
1348         if (base & 0x10000)
1349                 tmp |= 0x01;
1350         if (base & 0x20000)
1351                 tmp |= 0x04;
1352         if (base & 0x40000)
1353                 tmp |= 0x08;
1354 
1355         vga_wcrt(cinfo->regbase, CL_CRT1B, tmp);
1356 
1357         /* construct bit 19 of screen start address */
1358         if (cirrusfb_board_info[cinfo->btype].scrn_start_bit19) {
1359                 tmp = vga_rcrt(cinfo->regbase, CL_CRT1D);
1360                 if (is_laguna(cinfo))
1361                         tmp = (tmp & ~0x18) | ((base >> 16) & 0x18);
1362                 else
1363                         tmp = (tmp & ~0x80) | ((base >> 12) & 0x80);
1364                 vga_wcrt(cinfo->regbase, CL_CRT1D, tmp);
1365         }
1366 
1367         /* write pixel panning value to AR33; this does not quite work in 8bpp
1368          *
1369          * ### Piccolo..? Will this work?
1370          */
1371         if (info->var.bits_per_pixel == 1)
1372                 vga_wattr(cinfo->regbase, CL_AR33, xpix);
1373 
1374         return 0;
1375 }
1376 
1377 static int cirrusfb_blank(int blank_mode, struct fb_info *info)
1378 {
1379         /*
1380          * Blank the screen if blank_mode != 0, else unblank. If blank == NULL
1381          * then the caller blanks by setting the CLUT (Color Look Up Table)
1382          * to all black. Return 0 if blanking succeeded, != 0 if un-/blanking
1383          * failed due to e.g. a video mode which doesn't support it.
1384          * Implements VESA suspend and powerdown modes on hardware that
1385          * supports disabling hsync/vsync:
1386          *   blank_mode == 2: suspend vsync
1387          *   blank_mode == 3: suspend hsync
1388          *   blank_mode == 4: powerdown
1389          */
1390         unsigned char val;
1391         struct cirrusfb_info *cinfo = info->par;
1392         int current_mode = cinfo->blank_mode;
1393 
1394         dev_dbg(info->device, "ENTER, blank mode = %d\n", blank_mode);
1395 
1396         if (info->state != FBINFO_STATE_RUNNING ||
1397             current_mode == blank_mode) {
1398                 dev_dbg(info->device, "EXIT, returning 0\n");
1399                 return 0;
1400         }
1401 
1402         /* Undo current */
1403         if (current_mode == FB_BLANK_NORMAL ||
1404             current_mode == FB_BLANK_UNBLANK)
1405                 /* clear "FullBandwidth" bit */
1406                 val = 0;
1407         else
1408                 /* set "FullBandwidth" bit */
1409                 val = 0x20;
1410 
1411         val |= vga_rseq(cinfo->regbase, VGA_SEQ_CLOCK_MODE) & 0xdf;
1412         vga_wseq(cinfo->regbase, VGA_SEQ_CLOCK_MODE, val);
1413 
1414         switch (blank_mode) {
1415         case FB_BLANK_UNBLANK:
1416         case FB_BLANK_NORMAL:
1417                 val = 0x00;
1418                 break;
1419         case FB_BLANK_VSYNC_SUSPEND:
1420                 val = 0x04;
1421                 break;
1422         case FB_BLANK_HSYNC_SUSPEND:
1423                 val = 0x02;
1424                 break;
1425         case FB_BLANK_POWERDOWN:
1426                 val = 0x06;
1427                 break;
1428         default:
1429                 dev_dbg(info->device, "EXIT, returning 1\n");
1430                 return 1;
1431         }
1432 
1433         vga_wgfx(cinfo->regbase, CL_GRE, val);
1434 
1435         cinfo->blank_mode = blank_mode;
1436         dev_dbg(info->device, "EXIT, returning 0\n");
1437 
1438         /* Let fbcon do a soft blank for us */
1439         return (blank_mode == FB_BLANK_NORMAL) ? 1 : 0;
1440 }
1441 
1442 /**** END   Hardware specific Routines **************************************/
1443 /****************************************************************************/
1444 /**** BEGIN Internal Routines ***********************************************/
1445 
1446 static void init_vgachip(struct fb_info *info)
1447 {
1448         struct cirrusfb_info *cinfo = info->par;
1449         const struct cirrusfb_board_info_rec *bi;
1450 
1451         assert(cinfo != NULL);
1452 
1453         bi = &cirrusfb_board_info[cinfo->btype];
1454 
1455         /* reset board globally */
1456         switch (cinfo->btype) {
1457         case BT_PICCOLO:
1458                 WSFR(cinfo, 0x01);
1459                 udelay(500);
1460                 WSFR(cinfo, 0x51);
1461                 udelay(500);
1462                 break;
1463         case BT_PICASSO:
1464                 WSFR2(cinfo, 0xff);
1465                 udelay(500);
1466                 break;
1467         case BT_SD64:
1468         case BT_SPECTRUM:
1469                 WSFR(cinfo, 0x1f);
1470                 udelay(500);
1471                 WSFR(cinfo, 0x4f);
1472                 udelay(500);
1473                 break;
1474         case BT_PICASSO4:
1475                 /* disable flickerfixer */
1476                 vga_wcrt(cinfo->regbase, CL_CRT51, 0x00);
1477                 mdelay(100);
1478                 /* mode */
1479                 vga_wgfx(cinfo->regbase, CL_GR31, 0x00);
1480                 /* fall through */
1481         case BT_GD5480:
1482                 /* from Klaus' NetBSD driver: */
1483                 vga_wgfx(cinfo->regbase, CL_GR2F, 0x00);
1484                 /* fall through */
1485         case BT_ALPINE:
1486                 /* put blitter into 542x compat */
1487                 vga_wgfx(cinfo->regbase, CL_GR33, 0x00);
1488                 break;
1489 
1490         case BT_LAGUNA:
1491         case BT_LAGUNAB:
1492                 /* Nothing to do to reset the board. */
1493                 break;
1494 
1495         default:
1496                 dev_err(info->device, "Warning: Unknown board type\n");
1497                 break;
1498         }
1499 
1500         /* make sure RAM size set by this point */
1501         assert(info->screen_size > 0);
1502 
1503         /* the P4 is not fully initialized here; I rely on it having been */
1504         /* inited under AmigaOS already, which seems to work just fine    */
1505         /* (Klaus advised to do it this way)                          */
1506 
1507         if (cinfo->btype != BT_PICASSO4) {
1508                 WGen(cinfo, CL_VSSM, 0x10);     /* EGS: 0x16 */
1509                 WGen(cinfo, CL_POS102, 0x01);
1510                 WGen(cinfo, CL_VSSM, 0x08);     /* EGS: 0x0e */
1511 
1512                 if (cinfo->btype != BT_SD64)
1513                         WGen(cinfo, CL_VSSM2, 0x01);
1514 
1515                 /* reset sequencer logic */
1516                 vga_wseq(cinfo->regbase, VGA_SEQ_RESET, 0x03);
1517 
1518                 /* FullBandwidth (video off) and 8/9 dot clock */
1519                 vga_wseq(cinfo->regbase, VGA_SEQ_CLOCK_MODE, 0x21);
1520 
1521                 /* "magic cookie" - doesn't make any sense to me.. */
1522 /*      vga_wgfx(cinfo->regbase, CL_GRA, 0xce);   */
1523                 /* unlock all extension registers */
1524                 vga_wseq(cinfo->regbase, CL_SEQR6, 0x12);
1525 
1526                 switch (cinfo->btype) {
1527                 case BT_GD5480:
1528                         vga_wseq(cinfo->regbase, CL_SEQRF, 0x98);
1529                         break;
1530                 case BT_ALPINE:
1531                 case BT_LAGUNA:
1532                 case BT_LAGUNAB:
1533                         break;
1534                 case BT_SD64:
1535 #ifdef CONFIG_ZORRO
1536                         vga_wseq(cinfo->regbase, CL_SEQRF, 0xb8);
1537 #endif
1538                         break;
1539                 default:
1540                         vga_wseq(cinfo->regbase, CL_SEQR16, 0x0f);
1541                         vga_wseq(cinfo->regbase, CL_SEQRF, 0xb0);
1542                         break;
1543                 }
1544         }
1545         /* plane mask: nothing */
1546         vga_wseq(cinfo->regbase, VGA_SEQ_PLANE_WRITE, 0xff);
1547         /* character map select: doesn't even matter in gx mode */
1548         vga_wseq(cinfo->regbase, VGA_SEQ_CHARACTER_MAP, 0x00);
1549         /* memory mode: chain4, ext. memory */
1550         vga_wseq(cinfo->regbase, VGA_SEQ_MEMORY_MODE, 0x0a);
1551 
1552         /* controller-internal base address of video memory */
1553         if (bi->init_sr07)
1554                 vga_wseq(cinfo->regbase, CL_SEQR7, bi->sr07);
1555 
1556         /*  vga_wseq(cinfo->regbase, CL_SEQR8, 0x00); */
1557         /* EEPROM control: shouldn't be necessary to write to this at all.. */
1558 
1559         /* graphics cursor X position (incomplete; position gives rem. 3 bits */
1560         vga_wseq(cinfo->regbase, CL_SEQR10, 0x00);
1561         /* graphics cursor Y position (..."... ) */
1562         vga_wseq(cinfo->regbase, CL_SEQR11, 0x00);
1563         /* graphics cursor attributes */
1564         vga_wseq(cinfo->regbase, CL_SEQR12, 0x00);
1565         /* graphics cursor pattern address */
1566         vga_wseq(cinfo->regbase, CL_SEQR13, 0x00);
1567 
1568         /* writing these on a P4 might give problems..  */
1569         if (cinfo->btype != BT_PICASSO4) {
1570                 /* configuration readback and ext. color */
1571                 vga_wseq(cinfo->regbase, CL_SEQR17, 0x00);
1572                 /* signature generator */
1573                 vga_wseq(cinfo->regbase, CL_SEQR18, 0x02);
1574         }
1575 
1576         /* Screen A preset row scan: none */
1577         vga_wcrt(cinfo->regbase, VGA_CRTC_PRESET_ROW, 0x00);
1578         /* Text cursor start: disable text cursor */
1579         vga_wcrt(cinfo->regbase, VGA_CRTC_CURSOR_START, 0x20);
1580         /* Text cursor end: - */
1581         vga_wcrt(cinfo->regbase, VGA_CRTC_CURSOR_END, 0x00);
1582         /* text cursor location high: 0 */
1583         vga_wcrt(cinfo->regbase, VGA_CRTC_CURSOR_HI, 0x00);
1584         /* text cursor location low: 0 */
1585         vga_wcrt(cinfo->regbase, VGA_CRTC_CURSOR_LO, 0x00);
1586 
1587         /* Underline Row scanline: - */
1588         vga_wcrt(cinfo->regbase, VGA_CRTC_UNDERLINE, 0x00);
1589         /* ### add 0x40 for text modes with > 30 MHz pixclock */
1590         /* ext. display controls: ext.adr. wrap */
1591         vga_wcrt(cinfo->regbase, CL_CRT1B, 0x02);
1592 
1593         /* Set/Reset registers: - */
1594         vga_wgfx(cinfo->regbase, VGA_GFX_SR_VALUE, 0x00);
1595         /* Set/Reset enable: - */
1596         vga_wgfx(cinfo->regbase, VGA_GFX_SR_ENABLE, 0x00);
1597         /* Color Compare: - */
1598         vga_wgfx(cinfo->regbase, VGA_GFX_COMPARE_VALUE, 0x00);
1599         /* Data Rotate: - */
1600         vga_wgfx(cinfo->regbase, VGA_GFX_DATA_ROTATE, 0x00);
1601         /* Read Map Select: - */
1602         vga_wgfx(cinfo->regbase, VGA_GFX_PLANE_READ, 0x00);
1603         /* Mode: conf. for 16/4/2 color mode, no odd/even, read/write mode 0 */
1604         vga_wgfx(cinfo->regbase, VGA_GFX_MODE, 0x00);
1605         /* Miscellaneous: memory map base address, graphics mode */
1606         vga_wgfx(cinfo->regbase, VGA_GFX_MISC, 0x01);
1607         /* Color Don't care: involve all planes */
1608         vga_wgfx(cinfo->regbase, VGA_GFX_COMPARE_MASK, 0x0f);
1609         /* Bit Mask: no mask at all */
1610         vga_wgfx(cinfo->regbase, VGA_GFX_BIT_MASK, 0xff);
1611 
1612         if (cinfo->btype == BT_ALPINE || cinfo->btype == BT_SD64 ||
1613             is_laguna(cinfo))
1614                 /* (5434 can't have bit 3 set for bitblt) */
1615                 vga_wgfx(cinfo->regbase, CL_GRB, 0x20);
1616         else
1617         /* Graphics controller mode extensions: finer granularity,
1618          * 8byte data latches
1619          */
1620                 vga_wgfx(cinfo->regbase, CL_GRB, 0x28);
1621 
1622         vga_wgfx(cinfo->regbase, CL_GRC, 0xff); /* Color Key compare: - */
1623         vga_wgfx(cinfo->regbase, CL_GRD, 0x00); /* Color Key compare mask: - */
1624         vga_wgfx(cinfo->regbase, CL_GRE, 0x00); /* Miscellaneous control: - */
1625         /* Background color byte 1: - */
1626         /*  vga_wgfx (cinfo->regbase, CL_GR10, 0x00); */
1627         /*  vga_wgfx (cinfo->regbase, CL_GR11, 0x00); */
1628 
1629         /* Attribute Controller palette registers: "identity mapping" */
1630         vga_wattr(cinfo->regbase, VGA_ATC_PALETTE0, 0x00);
1631         vga_wattr(cinfo->regbase, VGA_ATC_PALETTE1, 0x01);
1632         vga_wattr(cinfo->regbase, VGA_ATC_PALETTE2, 0x02);
1633         vga_wattr(cinfo->regbase, VGA_ATC_PALETTE3, 0x03);
1634         vga_wattr(cinfo->regbase, VGA_ATC_PALETTE4, 0x04);
1635         vga_wattr(cinfo->regbase, VGA_ATC_PALETTE5, 0x05);
1636         vga_wattr(cinfo->regbase, VGA_ATC_PALETTE6, 0x06);
1637         vga_wattr(cinfo->regbase, VGA_ATC_PALETTE7, 0x07);
1638         vga_wattr(cinfo->regbase, VGA_ATC_PALETTE8, 0x08);
1639         vga_wattr(cinfo->regbase, VGA_ATC_PALETTE9, 0x09);
1640         vga_wattr(cinfo->regbase, VGA_ATC_PALETTEA, 0x0a);
1641         vga_wattr(cinfo->regbase, VGA_ATC_PALETTEB, 0x0b);
1642         vga_wattr(cinfo->regbase, VGA_ATC_PALETTEC, 0x0c);
1643         vga_wattr(cinfo->regbase, VGA_ATC_PALETTED, 0x0d);
1644         vga_wattr(cinfo->regbase, VGA_ATC_PALETTEE, 0x0e);
1645         vga_wattr(cinfo->regbase, VGA_ATC_PALETTEF, 0x0f);
1646 
1647         /* Attribute Controller mode: graphics mode */
1648         vga_wattr(cinfo->regbase, VGA_ATC_MODE, 0x01);
1649         /* Overscan color reg.: reg. 0 */
1650         vga_wattr(cinfo->regbase, VGA_ATC_OVERSCAN, 0x00);
1651         /* Color Plane enable: Enable all 4 planes */
1652         vga_wattr(cinfo->regbase, VGA_ATC_PLANE_ENABLE, 0x0f);
1653         /* Color Select: - */
1654         vga_wattr(cinfo->regbase, VGA_ATC_COLOR_PAGE, 0x00);
1655 
1656         WGen(cinfo, VGA_PEL_MSK, 0xff); /* Pixel mask: no mask */
1657 
1658         /* BLT Start/status: Blitter reset */
1659         vga_wgfx(cinfo->regbase, CL_GR31, 0x04);
1660         /* - " -           : "end-of-reset" */
1661         vga_wgfx(cinfo->regbase, CL_GR31, 0x00);
1662 
1663         /* misc... */
1664         WHDR(cinfo, 0); /* Hidden DAC register: - */
1665         return;
1666 }
1667 
1668 static void switch_monitor(struct cirrusfb_info *cinfo, int on)
1669 {
1670 #ifdef CONFIG_ZORRO /* only works on Zorro boards */
1671         static int IsOn = 0;    /* XXX not ok for multiple boards */
1672 
1673         if (cinfo->btype == BT_PICASSO4)
1674                 return;         /* nothing to switch */
1675         if (cinfo->btype == BT_ALPINE)
1676                 return;         /* nothing to switch */
1677         if (cinfo->btype == BT_GD5480)
1678                 return;         /* nothing to switch */
1679         if (cinfo->btype == BT_PICASSO) {
1680                 if ((on && !IsOn) || (!on && IsOn))
1681                         WSFR(cinfo, 0xff);
1682                 return;
1683         }
1684         if (on) {
1685                 switch (cinfo->btype) {
1686                 case BT_SD64:
1687                         WSFR(cinfo, cinfo->SFR | 0x21);
1688                         break;
1689                 case BT_PICCOLO:
1690                         WSFR(cinfo, cinfo->SFR | 0x28);
1691                         break;
1692                 case BT_SPECTRUM:
1693                         WSFR(cinfo, 0x6f);
1694                         break;
1695                 default: /* do nothing */ break;
1696                 }
1697         } else {
1698                 switch (cinfo->btype) {
1699                 case BT_SD64:
1700                         WSFR(cinfo, cinfo->SFR & 0xde);
1701                         break;
1702                 case BT_PICCOLO:
1703                         WSFR(cinfo, cinfo->SFR & 0xd7);
1704                         break;
1705                 case BT_SPECTRUM:
1706                         WSFR(cinfo, 0x4f);
1707                         break;
1708                 default: /* do nothing */
1709                         break;
1710                 }
1711         }
1712 #endif /* CONFIG_ZORRO */
1713 }
1714 
1715 /******************************************/
1716 /* Linux 2.6-style  accelerated functions */
1717 /******************************************/
1718 
1719 static int cirrusfb_sync(struct fb_info *info)
1720 {
1721         struct cirrusfb_info *cinfo = info->par;
1722 
1723         if (!is_laguna(cinfo)) {
1724                 while (vga_rgfx(cinfo->regbase, CL_GR31) & 0x03)
1725                         cpu_relax();
1726         }
1727         return 0;
1728 }
1729 
1730 static void cirrusfb_fillrect(struct fb_info *info,
1731                               const struct fb_fillrect *region)
1732 {
1733         struct fb_fillrect modded;
1734         int vxres, vyres;
1735         struct cirrusfb_info *cinfo = info->par;
1736         int m = info->var.bits_per_pixel;
1737         u32 color = (info->fix.visual == FB_VISUAL_TRUECOLOR) ?
1738                 cinfo->pseudo_palette[region->color] : region->color;
1739 
1740         if (info->state != FBINFO_STATE_RUNNING)
1741                 return;
1742         if (info->flags & FBINFO_HWACCEL_DISABLED) {
1743                 cfb_fillrect(info, region);
1744                 return;
1745         }
1746 
1747         vxres = info->var.xres_virtual;
1748         vyres = info->var.yres_virtual;
1749 
1750         memcpy(&modded, region, sizeof(struct fb_fillrect));
1751 
1752         if (!modded.width || !modded.height ||
1753            modded.dx >= vxres || modded.dy >= vyres)
1754                 return;
1755 
1756         if (modded.dx + modded.width  > vxres)
1757                 modded.width  = vxres - modded.dx;
1758         if (modded.dy + modded.height > vyres)
1759                 modded.height = vyres - modded.dy;
1760 
1761         cirrusfb_RectFill(cinfo->regbase,
1762                           info->var.bits_per_pixel,
1763                           (region->dx * m) / 8, region->dy,
1764                           (region->width * m) / 8, region->height,
1765                           color, color,
1766                           info->fix.line_length, 0x40);
1767 }
1768 
1769 static void cirrusfb_copyarea(struct fb_info *info,
1770                               const struct fb_copyarea *area)
1771 {
1772         struct fb_copyarea modded;
1773         u32 vxres, vyres;
1774         struct cirrusfb_info *cinfo = info->par;
1775         int m = info->var.bits_per_pixel;
1776 
1777         if (info->state != FBINFO_STATE_RUNNING)
1778                 return;
1779         if (info->flags & FBINFO_HWACCEL_DISABLED) {
1780                 cfb_copyarea(info, area);
1781                 return;
1782         }
1783 
1784         vxres = info->var.xres_virtual;
1785         vyres = info->var.yres_virtual;
1786         memcpy(&modded, area, sizeof(struct fb_copyarea));
1787 
1788         if (!modded.width || !modded.height ||
1789            modded.sx >= vxres || modded.sy >= vyres ||
1790            modded.dx >= vxres || modded.dy >= vyres)
1791                 return;
1792 
1793         if (modded.sx + modded.width > vxres)
1794                 modded.width = vxres - modded.sx;
1795         if (modded.dx + modded.width > vxres)
1796                 modded.width = vxres - modded.dx;
1797         if (modded.sy + modded.height > vyres)
1798                 modded.height = vyres - modded.sy;
1799         if (modded.dy + modded.height > vyres)
1800                 modded.height = vyres - modded.dy;
1801 
1802         cirrusfb_BitBLT(cinfo->regbase, info->var.bits_per_pixel,
1803                         (area->sx * m) / 8, area->sy,
1804                         (area->dx * m) / 8, area->dy,
1805                         (area->width * m) / 8, area->height,
1806                         info->fix.line_length);
1807 
1808 }
1809 
1810 static void cirrusfb_imageblit(struct fb_info *info,
1811                                const struct fb_image *image)
1812 {
1813         struct cirrusfb_info *cinfo = info->par;
1814         unsigned char op = (info->var.bits_per_pixel == 24) ? 0xc : 0x4;
1815 
1816         if (info->state != FBINFO_STATE_RUNNING)
1817                 return;
1818         /* Alpine/SD64 does not work at 24bpp ??? */
1819         if (info->flags & FBINFO_HWACCEL_DISABLED || image->depth != 1)
1820                 cfb_imageblit(info, image);
1821         else if ((cinfo->btype == BT_ALPINE || cinfo->btype == BT_SD64) &&
1822                   op == 0xc)
1823                 cfb_imageblit(info, image);
1824         else {
1825                 unsigned size = ((image->width + 7) >> 3) * image->height;
1826                 int m = info->var.bits_per_pixel;
1827                 u32 fg, bg;
1828 
1829                 if (info->var.bits_per_pixel == 8) {
1830                         fg = image->fg_color;
1831                         bg = image->bg_color;
1832                 } else {
1833                         fg = ((u32 *)(info->pseudo_palette))[image->fg_color];
1834                         bg = ((u32 *)(info->pseudo_palette))[image->bg_color];
1835                 }
1836                 if (info->var.bits_per_pixel == 24) {
1837                         /* clear background first */
1838                         cirrusfb_RectFill(cinfo->regbase,
1839                                           info->var.bits_per_pixel,
1840                                           (image->dx * m) / 8, image->dy,
1841                                           (image->width * m) / 8,
1842                                           image->height,
1843                                           bg, bg,
1844                                           info->fix.line_length, 0x40);
1845                 }
1846                 cirrusfb_RectFill(cinfo->regbase,
1847                                   info->var.bits_per_pixel,
1848                                   (image->dx * m) / 8, image->dy,
1849                                   (image->width * m) / 8, image->height,
1850                                   fg, bg,
1851                                   info->fix.line_length, op);
1852                 memcpy(info->screen_base, image->data, size);
1853         }
1854 }
1855 
1856 #ifdef CONFIG_PCI
1857 static int release_io_ports;
1858 
1859 /* Pulled the logic from XFree86 Cirrus driver to get the memory size,
1860  * based on the DRAM bandwidth bit and DRAM bank switching bit.  This
1861  * works with 1MB, 2MB and 4MB configurations (which the Motorola boards
1862  * seem to have. */
1863 static unsigned int cirrusfb_get_memsize(struct fb_info *info,
1864                                          u8 __iomem *regbase)
1865 {
1866         unsigned long mem;
1867         struct cirrusfb_info *cinfo = info->par;
1868 
1869         if (is_laguna(cinfo)) {
1870                 unsigned char SR14 = vga_rseq(regbase, CL_SEQR14);
1871 
1872                 mem = ((SR14 & 7) + 1) << 20;
1873         } else {
1874                 unsigned char SRF = vga_rseq(regbase, CL_SEQRF);
1875                 switch ((SRF & 0x18)) {
1876                 case 0x08:
1877                         mem = 512 * 1024;
1878                         break;
1879                 case 0x10:
1880                         mem = 1024 * 1024;
1881                         break;
1882                 /* 64-bit DRAM data bus width; assume 2MB.
1883                  * Also indicates 2MB memory on the 5430.
1884                  */
1885                 case 0x18:
1886                         mem = 2048 * 1024;
1887                         break;
1888                 default:
1889                         dev_warn(info->device, "Unknown memory size!\n");
1890                         mem = 1024 * 1024;
1891                 }
1892                 /* If DRAM bank switching is enabled, there must be
1893                  * twice as much memory installed. (4MB on the 5434)
1894                  */
1895                 if (cinfo->btype != BT_ALPINE && (SRF & 0x80) != 0)
1896                         mem *= 2;
1897         }
1898 
1899         /* TODO: Handling of GD5446/5480 (see XF86 sources ...) */
1900         return mem;
1901 }
1902 
1903 static void get_pci_addrs(const struct pci_dev *pdev,
1904                           unsigned long *display, unsigned long *registers)
1905 {
1906         assert(pdev != NULL);
1907         assert(display != NULL);
1908         assert(registers != NULL);
1909 
1910         *display = 0;
1911         *registers = 0;
1912 
1913         /* This is a best-guess for now */
1914 
1915         if (pci_resource_flags(pdev, 0) & IORESOURCE_IO) {
1916                 *display = pci_resource_start(pdev, 1);
1917                 *registers = pci_resource_start(pdev, 0);
1918         } else {
1919                 *display = pci_resource_start(pdev, 0);
1920                 *registers = pci_resource_start(pdev, 1);
1921         }
1922 
1923         assert(*display != 0);
1924 }
1925 
1926 static void cirrusfb_pci_unmap(struct fb_info *info)
1927 {
1928         struct pci_dev *pdev = to_pci_dev(info->device);
1929         struct cirrusfb_info *cinfo = info->par;
1930 
1931         if (cinfo->laguna_mmio == NULL)
1932                 iounmap(cinfo->laguna_mmio);
1933         iounmap(info->screen_base);
1934 #if 0 /* if system didn't claim this region, we would... */
1935         release_mem_region(0xA0000, 65535);
1936 #endif
1937         if (release_io_ports)
1938                 release_region(0x3C0, 32);
1939         pci_release_regions(pdev);
1940 }
1941 #endif /* CONFIG_PCI */
1942 
1943 #ifdef CONFIG_ZORRO
1944 static void cirrusfb_zorro_unmap(struct fb_info *info)
1945 {
1946         struct cirrusfb_info *cinfo = info->par;
1947         struct zorro_dev *zdev = to_zorro_dev(info->device);
1948 
1949         if (info->fix.smem_start > 16 * MB_)
1950                 iounmap(info->screen_base);
1951         if (info->fix.mmio_start > 16 * MB_)
1952                 iounmap(cinfo->regbase);
1953 
1954         zorro_release_device(zdev);
1955 }
1956 #endif /* CONFIG_ZORRO */
1957 
1958 /* function table of the above functions */
1959 static struct fb_ops cirrusfb_ops = {
1960         .owner          = THIS_MODULE,
1961         .fb_open        = cirrusfb_open,
1962         .fb_release     = cirrusfb_release,
1963         .fb_setcolreg   = cirrusfb_setcolreg,
1964         .fb_check_var   = cirrusfb_check_var,
1965         .fb_set_par     = cirrusfb_set_par,
1966         .fb_pan_display = cirrusfb_pan_display,
1967         .fb_blank       = cirrusfb_blank,
1968         .fb_fillrect    = cirrusfb_fillrect,
1969         .fb_copyarea    = cirrusfb_copyarea,
1970         .fb_sync        = cirrusfb_sync,
1971         .fb_imageblit   = cirrusfb_imageblit,
1972 };
1973 
1974 static int cirrusfb_set_fbinfo(struct fb_info *info)
1975 {
1976         struct cirrusfb_info *cinfo = info->par;
1977         struct fb_var_screeninfo *var = &info->var;
1978 
1979         info->pseudo_palette = cinfo->pseudo_palette;
1980         info->flags = FBINFO_DEFAULT
1981                     | FBINFO_HWACCEL_XPAN
1982                     | FBINFO_HWACCEL_YPAN
1983                     | FBINFO_HWACCEL_FILLRECT
1984                     | FBINFO_HWACCEL_IMAGEBLIT
1985                     | FBINFO_HWACCEL_COPYAREA;
1986         if (noaccel || is_laguna(cinfo)) {
1987                 info->flags |= FBINFO_HWACCEL_DISABLED;
1988                 info->fix.accel = FB_ACCEL_NONE;
1989         } else
1990                 info->fix.accel = FB_ACCEL_CIRRUS_ALPINE;
1991 
1992         info->fbops = &cirrusfb_ops;
1993 
1994         if (cinfo->btype == BT_GD5480) {
1995                 if (var->bits_per_pixel == 16)
1996                         info->screen_base += 1 * MB_;
1997                 if (var->bits_per_pixel == 32)
1998                         info->screen_base += 2 * MB_;
1999         }
2000 
2001         /* Fill fix common fields */
2002         strlcpy(info->fix.id, cirrusfb_board_info[cinfo->btype].name,
2003                 sizeof(info->fix.id));
2004 
2005         /* monochrome: only 1 memory plane */
2006         /* 8 bit and above: Use whole memory area */
2007         info->fix.smem_len   = info->screen_size;
2008         if (var->bits_per_pixel == 1)
2009                 info->fix.smem_len /= 4;
2010         info->fix.type_aux   = 0;
2011         info->fix.xpanstep   = 1;
2012         info->fix.ypanstep   = 1;
2013         info->fix.ywrapstep  = 0;
2014 
2015         /* FIXME: map region at 0xB8000 if available, fill in here */
2016         info->fix.mmio_len   = 0;
2017 
2018         fb_alloc_cmap(&info->cmap, 256, 0);
2019 
2020         return 0;
2021 }
2022 
2023 static int cirrusfb_register(struct fb_info *info)
2024 {
2025         struct cirrusfb_info *cinfo = info->par;
2026         int err;
2027 
2028         /* sanity checks */
2029         assert(cinfo->btype != BT_NONE);
2030 
2031         /* set all the vital stuff */
2032         cirrusfb_set_fbinfo(info);
2033 
2034         dev_dbg(info->device, "(RAM start set to: 0x%p)\n", info->screen_base);
2035 
2036         err = fb_find_mode(&info->var, info, mode_option, NULL, 0, NULL, 8);
2037         if (!err) {
2038                 dev_dbg(info->device, "wrong initial video mode\n");
2039                 err = -EINVAL;
2040                 goto err_dealloc_cmap;
2041         }
2042 
2043         info->var.activate = FB_ACTIVATE_NOW;
2044 
2045         err = cirrusfb_check_var(&info->var, info);
2046         if (err < 0) {
2047                 /* should never happen */
2048                 dev_dbg(info->device,
2049                         "choking on default var... umm, no good.\n");
2050                 goto err_dealloc_cmap;
2051         }
2052 
2053         err = register_framebuffer(info);
2054         if (err < 0) {
2055                 dev_err(info->device,
2056                         "could not register fb device; err = %d!\n", err);
2057                 goto err_dealloc_cmap;
2058         }
2059 
2060         return 0;
2061 
2062 err_dealloc_cmap:
2063         fb_dealloc_cmap(&info->cmap);
2064         return err;
2065 }
2066 
2067 static void cirrusfb_cleanup(struct fb_info *info)
2068 {
2069         struct cirrusfb_info *cinfo = info->par;
2070 
2071         switch_monitor(cinfo, 0);
2072         unregister_framebuffer(info);
2073         fb_dealloc_cmap(&info->cmap);
2074         dev_dbg(info->device, "Framebuffer unregistered\n");
2075         cinfo->unmap(info);
2076         framebuffer_release(info);
2077 }
2078 
2079 #ifdef CONFIG_PCI
2080 static int cirrusfb_pci_register(struct pci_dev *pdev,
2081                                  const struct pci_device_id *ent)
2082 {
2083         struct cirrusfb_info *cinfo;
2084         struct fb_info *info;
2085         unsigned long board_addr, board_size;
2086         int ret;
2087 
2088         ret = pci_enable_device(pdev);
2089         if (ret < 0) {
2090                 printk(KERN_ERR "cirrusfb: Cannot enable PCI device\n");
2091                 goto err_out;
2092         }
2093 
2094         info = framebuffer_alloc(sizeof(struct cirrusfb_info), &pdev->dev);
2095         if (!info) {
2096                 ret = -ENOMEM;
2097                 goto err_out;
2098         }
2099 
2100         cinfo = info->par;
2101         cinfo->btype = (enum cirrus_board) ent->driver_data;
2102 
2103         dev_dbg(info->device,
2104                 " Found PCI device, base address 0 is 0x%Lx, btype set to %d\n",
2105                 (unsigned long long)pdev->resource[0].start,  cinfo->btype);
2106         dev_dbg(info->device, " base address 1 is 0x%Lx\n",
2107                 (unsigned long long)pdev->resource[1].start);
2108 
2109         dev_dbg(info->device,
2110                 "Attempt to get PCI info for Cirrus Graphics Card\n");
2111         get_pci_addrs(pdev, &board_addr, &info->fix.mmio_start);
2112         /* FIXME: this forces VGA.  alternatives? */
2113         cinfo->regbase = NULL;
2114         cinfo->laguna_mmio = ioremap(info->fix.mmio_start, 0x1000);
2115 
2116         dev_dbg(info->device, "Board address: 0x%lx, register address: 0x%lx\n",
2117                 board_addr, info->fix.mmio_start);
2118 
2119         board_size = (cinfo->btype == BT_GD5480) ?
2120                 32 * MB_ : cirrusfb_get_memsize(info, cinfo->regbase);
2121 
2122         ret = pci_request_regions(pdev, "cirrusfb");
2123         if (ret < 0) {
2124                 dev_err(info->device, "cannot reserve region 0x%lx, abort\n",
2125                         board_addr);
2126                 goto err_release_fb;
2127         }
2128 #if 0 /* if the system didn't claim this region, we would... */
2129         if (!request_mem_region(0xA0000, 65535, "cirrusfb")) {
2130                 dev_err(info->device, "cannot reserve region 0x%lx, abort\n",
2131                         0xA0000L);
2132                 ret = -EBUSY;
2133                 goto err_release_regions;
2134         }
2135 #endif
2136         if (request_region(0x3C0, 32, "cirrusfb"))
2137                 release_io_ports = 1;
2138 
2139         info->screen_base = ioremap(board_addr, board_size);
2140         if (!info->screen_base) {
2141                 ret = -EIO;
2142                 goto err_release_legacy;
2143         }
2144 
2145         info->fix.smem_start = board_addr;
2146         info->screen_size = board_size;
2147         cinfo->unmap = cirrusfb_pci_unmap;
2148 
2149         dev_info(info->device,
2150                  "Cirrus Logic chipset on PCI bus, RAM (%lu kB) at 0x%lx\n",
2151                  info->screen_size >> 10, board_addr);
2152         pci_set_drvdata(pdev, info);
2153 
2154         ret = cirrusfb_register(info);
2155         if (!ret)
2156                 return 0;
2157 
2158         iounmap(info->screen_base);
2159 err_release_legacy:
2160         if (release_io_ports)
2161                 release_region(0x3C0, 32);
2162 #if 0
2163         release_mem_region(0xA0000, 65535);
2164 err_release_regions:
2165 #endif
2166         pci_release_regions(pdev);
2167 err_release_fb:
2168         if (cinfo->laguna_mmio != NULL)
2169                 iounmap(cinfo->laguna_mmio);
2170         framebuffer_release(info);
2171 err_out:
2172         return ret;
2173 }
2174 
2175 static void cirrusfb_pci_unregister(struct pci_dev *pdev)
2176 {
2177         struct fb_info *info = pci_get_drvdata(pdev);
2178 
2179         cirrusfb_cleanup(info);
2180 }
2181 
2182 static struct pci_driver cirrusfb_pci_driver = {
2183         .name           = "cirrusfb",
2184         .id_table       = cirrusfb_pci_table,
2185         .probe          = cirrusfb_pci_register,
2186         .remove         = cirrusfb_pci_unregister,
2187 #ifdef CONFIG_PM
2188 #if 0
2189         .suspend        = cirrusfb_pci_suspend,
2190         .resume         = cirrusfb_pci_resume,
2191 #endif
2192 #endif
2193 };
2194 #endif /* CONFIG_PCI */
2195 
2196 #ifdef CONFIG_ZORRO
2197 static int cirrusfb_zorro_register(struct zorro_dev *z,
2198                                    const struct zorro_device_id *ent)
2199 {
2200         struct fb_info *info;
2201         int error;
2202         const struct zorrocl *zcl;
2203         enum cirrus_board btype;
2204         unsigned long regbase, ramsize, rambase;
2205         struct cirrusfb_info *cinfo;
2206 
2207         info = framebuffer_alloc(sizeof(struct cirrusfb_info), &z->dev);
2208         if (!info)
2209                 return -ENOMEM;
2210 
2211         zcl = (const struct zorrocl *)ent->driver_data;
2212         btype = zcl->type;
2213         regbase = zorro_resource_start(z) + zcl->regoffset;
2214         ramsize = zcl->ramsize;
2215         if (ramsize) {
2216                 rambase = zorro_resource_start(z) + zcl->ramoffset;
2217                 if (zorro_resource_len(z) == 64 * MB_) {
2218                         /* Quirk for 64 MiB Picasso IV */
2219                         rambase += zcl->ramoffset;
2220                 }
2221         } else {
2222                 struct zorro_dev *ram = zorro_find_device(zcl->ramid, NULL);
2223                 if (!ram || !zorro_resource_len(ram)) {
2224                         dev_err(info->device, "No video RAM found\n");
2225                         error = -ENODEV;
2226                         goto err_release_fb;
2227                 }
2228                 rambase = zorro_resource_start(ram);
2229                 ramsize = zorro_resource_len(ram);
2230                 if (zcl->ramid2 &&
2231                     (ram = zorro_find_device(zcl->ramid2, NULL))) {
2232                         if (zorro_resource_start(ram) != rambase + ramsize) {
2233                                 dev_warn(info->device,
2234                                          "Skipping non-contiguous RAM at %pR\n",
2235                                          &ram->resource);
2236                         } else {
2237                                 ramsize += zorro_resource_len(ram);
2238                         }
2239                 }
2240         }
2241 
2242         dev_info(info->device,
2243                  "%s board detected, REG at 0x%lx, %lu MiB RAM at 0x%lx\n",
2244                  cirrusfb_board_info[btype].name, regbase, ramsize / MB_,
2245                  rambase);
2246 
2247         if (!zorro_request_device(z, "cirrusfb")) {
2248                 dev_err(info->device, "Cannot reserve %pR\n", &z->resource);
2249                 error = -EBUSY;
2250                 goto err_release_fb;
2251         }
2252 
2253         cinfo = info->par;
2254         cinfo->btype = btype;
2255 
2256         info->fix.mmio_start = regbase;
2257         cinfo->regbase = regbase > 16 * MB_ ? ioremap(regbase, 64 * 1024)
2258                                             : ZTWO_VADDR(regbase);
2259         if (!cinfo->regbase) {
2260                 dev_err(info->device, "Cannot map registers\n");
2261                 error = -EIO;
2262                 goto err_release_dev;
2263         }
2264 
2265         info->fix.smem_start = rambase;
2266         info->screen_size = ramsize;
2267         info->screen_base = rambase > 16 * MB_ ? ioremap(rambase, ramsize)
2268                                                : ZTWO_VADDR(rambase);
2269         if (!info->screen_base) {
2270                 dev_err(info->device, "Cannot map video RAM\n");
2271                 error = -EIO;
2272                 goto err_unmap_reg;
2273         }
2274 
2275         cinfo->unmap = cirrusfb_zorro_unmap;
2276 
2277         dev_info(info->device,
2278                  "Cirrus Logic chipset on Zorro bus, RAM (%lu MiB) at 0x%lx\n",
2279                  ramsize / MB_, rambase);
2280 
2281         /* MCLK select etc. */
2282         if (cirrusfb_board_info[btype].init_sr1f)
2283                 vga_wseq(cinfo->regbase, CL_SEQR1F,
2284                          cirrusfb_board_info[btype].sr1f);
2285 
2286         error = cirrusfb_register(info);
2287         if (error) {
2288                 dev_err(info->device, "Failed to register device, error %d\n",
2289                         error);
2290                 goto err_unmap_ram;
2291         }
2292 
2293         zorro_set_drvdata(z, info);
2294         return 0;
2295 
2296 err_unmap_ram:
2297         if (rambase > 16 * MB_)
2298                 iounmap(info->screen_base);
2299 
2300 err_unmap_reg:
2301         if (regbase > 16 * MB_)
2302                 iounmap(cinfo->regbase);
2303 err_release_dev:
2304         zorro_release_device(z);
2305 err_release_fb:
2306         framebuffer_release(info);
2307         return error;
2308 }
2309 
2310 void cirrusfb_zorro_unregister(struct zorro_dev *z)
2311 {
2312         struct fb_info *info = zorro_get_drvdata(z);
2313 
2314         cirrusfb_cleanup(info);
2315         zorro_set_drvdata(z, NULL);
2316 }
2317 
2318 static struct zorro_driver cirrusfb_zorro_driver = {
2319         .name           = "cirrusfb",
2320         .id_table       = cirrusfb_zorro_table,
2321         .probe          = cirrusfb_zorro_register,
2322         .remove         = cirrusfb_zorro_unregister,
2323 };
2324 #endif /* CONFIG_ZORRO */
2325 
2326 #ifndef MODULE
2327 static int __init cirrusfb_setup(char *options)
2328 {
2329         char *this_opt;
2330 
2331         if (!options || !*options)
2332                 return 0;
2333 
2334         while ((this_opt = strsep(&options, ",")) != NULL) {
2335                 if (!*this_opt)
2336                         continue;
2337 
2338                 if (!strcmp(this_opt, "noaccel"))
2339                         noaccel = 1;
2340                 else if (!strncmp(this_opt, "mode:", 5))
2341                         mode_option = this_opt + 5;
2342                 else
2343                         mode_option = this_opt;
2344         }
2345         return 0;
2346 }
2347 #endif
2348 
2349     /*
2350      *  Modularization
2351      */
2352 
2353 MODULE_AUTHOR("Copyright 1999,2000 Jeff Garzik <jgarzik@pobox.com>");
2354 MODULE_DESCRIPTION("Accelerated FBDev driver for Cirrus Logic chips");
2355 MODULE_LICENSE("GPL");
2356 
2357 static int __init cirrusfb_init(void)
2358 {
2359         int error = 0;
2360 
2361 #ifndef MODULE
2362         char *option = NULL;
2363 
2364         if (fb_get_options("cirrusfb", &option))
2365                 return -ENODEV;
2366         cirrusfb_setup(option);
2367 #endif
2368 
2369 #ifdef CONFIG_ZORRO
2370         error |= zorro_register_driver(&cirrusfb_zorro_driver);
2371 #endif
2372 #ifdef CONFIG_PCI
2373         error |= pci_register_driver(&cirrusfb_pci_driver);
2374 #endif
2375         return error;
2376 }
2377 
2378 static void __exit cirrusfb_exit(void)
2379 {
2380 #ifdef CONFIG_PCI
2381         pci_unregister_driver(&cirrusfb_pci_driver);
2382 #endif
2383 #ifdef CONFIG_ZORRO
2384         zorro_unregister_driver(&cirrusfb_zorro_driver);
2385 #endif
2386 }
2387 
2388 module_init(cirrusfb_init);
2389 
2390 module_param(mode_option, charp, 0);
2391 MODULE_PARM_DESC(mode_option, "Initial video mode e.g. '648x480-8@60'");
2392 module_param(noaccel, bool, 0);
2393 MODULE_PARM_DESC(noaccel, "Disable acceleration");
2394 
2395 #ifdef MODULE
2396 module_exit(cirrusfb_exit);
2397 #endif
2398 
2399 /**********************************************************************/
2400 /* about the following functions - I have used the same names for the */
2401 /* functions as Markus Wild did in his Retina driver for NetBSD as    */
2402 /* they just made sense for this purpose. Apart from that, I wrote    */
2403 /* these functions myself.                                          */
2404 /**********************************************************************/
2405 
2406 /*** WGen() - write into one of the external/general registers ***/
2407 static void WGen(const struct cirrusfb_info *cinfo,
2408                   int regnum, unsigned char val)
2409 {
2410         unsigned long regofs = 0;
2411 
2412         if (cinfo->btype == BT_PICASSO) {
2413                 /* Picasso II specific hack */
2414 /*            if (regnum == VGA_PEL_IR || regnum == VGA_PEL_D ||
2415                   regnum == CL_VSSM2) */
2416                 if (regnum == VGA_PEL_IR || regnum == VGA_PEL_D)
2417                         regofs = 0xfff;
2418         }
2419 
2420         vga_w(cinfo->regbase, regofs + regnum, val);
2421 }
2422 
2423 /*** RGen() - read out one of the external/general registers ***/
2424 static unsigned char RGen(const struct cirrusfb_info *cinfo, int regnum)
2425 {
2426         unsigned long regofs = 0;
2427 
2428         if (cinfo->btype == BT_PICASSO) {
2429                 /* Picasso II specific hack */
2430 /*            if (regnum == VGA_PEL_IR || regnum == VGA_PEL_D ||
2431                   regnum == CL_VSSM2) */
2432                 if (regnum == VGA_PEL_IR || regnum == VGA_PEL_D)
2433                         regofs = 0xfff;
2434         }
2435 
2436         return vga_r(cinfo->regbase, regofs + regnum);
2437 }
2438 
2439 /*** AttrOn() - turn on VideoEnable for Attribute controller ***/
2440 static void AttrOn(const struct cirrusfb_info *cinfo)
2441 {
2442         assert(cinfo != NULL);
2443 
2444         if (vga_rcrt(cinfo->regbase, CL_CRT24) & 0x80) {
2445                 /* if we're just in "write value" mode, write back the */
2446                 /* same value as before to not modify anything */
2447                 vga_w(cinfo->regbase, VGA_ATT_IW,
2448                       vga_r(cinfo->regbase, VGA_ATT_R));
2449         }
2450         /* turn on video bit */
2451 /*      vga_w(cinfo->regbase, VGA_ATT_IW, 0x20); */
2452         vga_w(cinfo->regbase, VGA_ATT_IW, 0x33);
2453 
2454         /* dummy write on Reg0 to be on "write index" mode next time */
2455         vga_w(cinfo->regbase, VGA_ATT_IW, 0x00);
2456 }
2457 
2458 /*** WHDR() - write into the Hidden DAC register ***/
2459 /* as the HDR is the only extension register that requires special treatment
2460  * (the other extension registers are accessible just like the "ordinary"
2461  * registers of their functional group) here is a specialized routine for
2462  * accessing the HDR
2463  */
2464 static void WHDR(const struct cirrusfb_info *cinfo, unsigned char val)
2465 {
2466         unsigned char dummy;
2467 
2468         if (is_laguna(cinfo))
2469                 return;
2470         if (cinfo->btype == BT_PICASSO) {
2471                 /* Klaus' hint for correct access to HDR on some boards */
2472                 /* first write 0 to pixel mask (3c6) */
2473                 WGen(cinfo, VGA_PEL_MSK, 0x00);
2474                 udelay(200);
2475                 /* next read dummy from pixel address (3c8) */
2476                 dummy = RGen(cinfo, VGA_PEL_IW);
2477                 udelay(200);
2478         }
2479         /* now do the usual stuff to access the HDR */
2480 
2481         dummy = RGen(cinfo, VGA_PEL_MSK);
2482         udelay(200);
2483         dummy = RGen(cinfo, VGA_PEL_MSK);
2484         udelay(200);
2485         dummy = RGen(cinfo, VGA_PEL_MSK);
2486         udelay(200);
2487         dummy = RGen(cinfo, VGA_PEL_MSK);
2488         udelay(200);
2489 
2490         WGen(cinfo, VGA_PEL_MSK, val);
2491         udelay(200);
2492 
2493         if (cinfo->btype == BT_PICASSO) {
2494                 /* now first reset HDR access counter */
2495                 dummy = RGen(cinfo, VGA_PEL_IW);
2496                 udelay(200);
2497 
2498                 /* and at the end, restore the mask value */
2499                 /* ## is this mask always 0xff? */
2500                 WGen(cinfo, VGA_PEL_MSK, 0xff);
2501                 udelay(200);
2502         }
2503 }
2504 
2505 /*** WSFR() - write to the "special function register" (SFR) ***/
2506 static void WSFR(struct cirrusfb_info *cinfo, unsigned char val)
2507 {
2508 #ifdef CONFIG_ZORRO
2509         assert(cinfo->regbase != NULL);
2510         cinfo->SFR = val;
2511         z_writeb(val, cinfo->regbase + 0x8000);
2512 #endif
2513 }
2514 
2515 /* The Picasso has a second register for switching the monitor bit */
2516 static void WSFR2(struct cirrusfb_info *cinfo, unsigned char val)
2517 {
2518 #ifdef CONFIG_ZORRO
2519         /* writing an arbitrary value to this one causes the monitor switcher */
2520         /* to flip to Amiga display */
2521         assert(cinfo->regbase != NULL);
2522         cinfo->SFR = val;
2523         z_writeb(val, cinfo->regbase + 0x9000);
2524 #endif
2525 }
2526 
2527 /*** WClut - set CLUT entry (range: 0..63) ***/
2528 static void WClut(struct cirrusfb_info *cinfo, unsigned char regnum, unsigned char red,
2529             unsigned char green, unsigned char blue)
2530 {
2531         unsigned int data = VGA_PEL_D;
2532 
2533         /* address write mode register is not translated.. */
2534         vga_w(cinfo->regbase, VGA_PEL_IW, regnum);
2535 
2536         if (cinfo->btype == BT_PICASSO || cinfo->btype == BT_PICASSO4 ||
2537             cinfo->btype == BT_ALPINE || cinfo->btype == BT_GD5480 ||
2538             cinfo->btype == BT_SD64 || is_laguna(cinfo)) {
2539                 /* but DAC data register IS, at least for Picasso II */
2540                 if (cinfo->btype == BT_PICASSO)
2541                         data += 0xfff;
2542                 vga_w(cinfo->regbase, data, red);
2543                 vga_w(cinfo->regbase, data, green);
2544                 vga_w(cinfo->regbase, data, blue);
2545         } else {
2546                 vga_w(cinfo->regbase, data, blue);
2547                 vga_w(cinfo->regbase, data, green);
2548                 vga_w(cinfo->regbase, data, red);
2549         }
2550 }
2551 
2552 #if 0
2553 /*** RClut - read CLUT entry (range 0..63) ***/
2554 static void RClut(struct cirrusfb_info *cinfo, unsigned char regnum, unsigned char *red,
2555             unsigned char *green, unsigned char *blue)
2556 {
2557         unsigned int data = VGA_PEL_D;
2558 
2559         vga_w(cinfo->regbase, VGA_PEL_IR, regnum);
2560 
2561         if (cinfo->btype == BT_PICASSO || cinfo->btype == BT_PICASSO4 ||
2562             cinfo->btype == BT_ALPINE || cinfo->btype == BT_GD5480) {
2563                 if (cinfo->btype == BT_PICASSO)
2564                         data += 0xfff;
2565                 *red = vga_r(cinfo->regbase, data);
2566                 *green = vga_r(cinfo->regbase, data);
2567                 *blue = vga_r(cinfo->regbase, data);
2568         } else {
2569                 *blue = vga_r(cinfo->regbase, data);
2570                 *green = vga_r(cinfo->regbase, data);
2571                 *red = vga_r(cinfo->regbase, data);
2572         }
2573 }
2574 #endif
2575 
2576 /*******************************************************************
2577         cirrusfb_WaitBLT()
2578 
2579         Wait for the BitBLT engine to complete a possible earlier job
2580 *********************************************************************/
2581 
2582 /* FIXME: use interrupts instead */
2583 static void cirrusfb_WaitBLT(u8 __iomem *regbase)
2584 {
2585         while (vga_rgfx(regbase, CL_GR31) & 0x08)
2586                 cpu_relax();
2587 }
2588 
2589 /*******************************************************************
2590         cirrusfb_BitBLT()
2591 
2592         perform accelerated "scrolling"
2593 ********************************************************************/
2594 
2595 static void cirrusfb_set_blitter(u8 __iomem *regbase,
2596                             u_short nwidth, u_short nheight,
2597                             u_long nsrc, u_long ndest,
2598                             u_short bltmode, u_short line_length)
2599 
2600 {
2601         /* pitch: set to line_length */
2602         /* dest pitch low */
2603         vga_wgfx(regbase, CL_GR24, line_length & 0xff);
2604         /* dest pitch hi */
2605         vga_wgfx(regbase, CL_GR25, line_length >> 8);
2606         /* source pitch low */
2607         vga_wgfx(regbase, CL_GR26, line_length & 0xff);
2608         /* source pitch hi */
2609         vga_wgfx(regbase, CL_GR27, line_length >> 8);
2610 
2611         /* BLT width: actual number of pixels - 1 */
2612         /* BLT width low */
2613         vga_wgfx(regbase, CL_GR20, nwidth & 0xff);
2614         /* BLT width hi */
2615         vga_wgfx(regbase, CL_GR21, nwidth >> 8);
2616 
2617         /* BLT height: actual number of lines -1 */
2618         /* BLT height low */
2619         vga_wgfx(regbase, CL_GR22, nheight & 0xff);
2620         /* BLT width hi */
2621         vga_wgfx(regbase, CL_GR23, nheight >> 8);
2622 
2623         /* BLT destination */
2624         /* BLT dest low */
2625         vga_wgfx(regbase, CL_GR28, (u_char) (ndest & 0xff));
2626         /* BLT dest mid */
2627         vga_wgfx(regbase, CL_GR29, (u_char) (ndest >> 8));
2628         /* BLT dest hi */
2629         vga_wgfx(regbase, CL_GR2A, (u_char) (ndest >> 16));
2630 
2631         /* BLT source */
2632         /* BLT src low */
2633         vga_wgfx(regbase, CL_GR2C, (u_char) (nsrc & 0xff));
2634         /* BLT src mid */
2635         vga_wgfx(regbase, CL_GR2D, (u_char) (nsrc >> 8));
2636         /* BLT src hi */
2637         vga_wgfx(regbase, CL_GR2E, (u_char) (nsrc >> 16));
2638 
2639         /* BLT mode */
2640         vga_wgfx(regbase, CL_GR30, bltmode);    /* BLT mode */
2641 
2642         /* BLT ROP: SrcCopy */
2643         vga_wgfx(regbase, CL_GR32, 0x0d);       /* BLT ROP */
2644 
2645         /* and finally: GO! */
2646         vga_wgfx(regbase, CL_GR31, 0x02);       /* BLT Start/status */
2647 }
2648 
2649 /*******************************************************************
2650         cirrusfb_BitBLT()
2651 
2652         perform accelerated "scrolling"
2653 ********************************************************************/
2654 
2655 static void cirrusfb_BitBLT(u8 __iomem *regbase, int bits_per_pixel,
2656                             u_short curx, u_short cury,
2657                             u_short destx, u_short desty,
2658                             u_short width, u_short height,
2659                             u_short line_length)
2660 {
2661         u_short nwidth = width - 1;
2662         u_short nheight = height - 1;
2663         u_long nsrc, ndest;
2664         u_char bltmode;
2665 
2666         bltmode = 0x00;
2667         /* if source adr < dest addr, do the Blt backwards */
2668         if (cury <= desty) {
2669                 if (cury == desty) {
2670                         /* if src and dest are on the same line, check x */
2671                         if (curx < destx)
2672                                 bltmode |= 0x01;
2673                 } else
2674                         bltmode |= 0x01;
2675         }
2676         /* standard case: forward blitting */
2677         nsrc = (cury * line_length) + curx;
2678         ndest = (desty * line_length) + destx;
2679         if (bltmode) {
2680                 /* this means start addresses are at the end,
2681                  * counting backwards
2682                  */
2683                 nsrc += nheight * line_length + nwidth;
2684                 ndest += nheight * line_length + nwidth;
2685         }
2686 
2687         cirrusfb_WaitBLT(regbase);
2688 
2689         cirrusfb_set_blitter(regbase, nwidth, nheight,
2690                             nsrc, ndest, bltmode, line_length);
2691 }
2692 
2693 /*******************************************************************
2694         cirrusfb_RectFill()
2695 
2696         perform accelerated rectangle fill
2697 ********************************************************************/
2698 
2699 static void cirrusfb_RectFill(u8 __iomem *regbase, int bits_per_pixel,
2700                      u_short x, u_short y, u_short width, u_short height,
2701                      u32 fg_color, u32 bg_color, u_short line_length,
2702                      u_char blitmode)
2703 {
2704         u_long ndest = (y * line_length) + x;
2705         u_char op;
2706 
2707         cirrusfb_WaitBLT(regbase);
2708 
2709         /* This is a ColorExpand Blt, using the */
2710         /* same color for foreground and background */
2711         vga_wgfx(regbase, VGA_GFX_SR_VALUE, bg_color);
2712         vga_wgfx(regbase, VGA_GFX_SR_ENABLE, fg_color);
2713 
2714         op = 0x80;
2715         if (bits_per_pixel >= 16) {
2716                 vga_wgfx(regbase, CL_GR10, bg_color >> 8);
2717                 vga_wgfx(regbase, CL_GR11, fg_color >> 8);
2718                 op = 0x90;
2719         }
2720         if (bits_per_pixel >= 24) {
2721                 vga_wgfx(regbase, CL_GR12, bg_color >> 16);
2722                 vga_wgfx(regbase, CL_GR13, fg_color >> 16);
2723                 op = 0xa0;
2724         }
2725         if (bits_per_pixel == 32) {
2726                 vga_wgfx(regbase, CL_GR14, bg_color >> 24);
2727                 vga_wgfx(regbase, CL_GR15, fg_color >> 24);
2728                 op = 0xb0;
2729         }
2730         cirrusfb_set_blitter(regbase, width - 1, height - 1,
2731                             0, ndest, op | blitmode, line_length);
2732 }
2733 
2734 /**************************************************************************
2735  * bestclock() - determine closest possible clock lower(?) than the
2736  * desired pixel clock
2737  **************************************************************************/
2738 static void bestclock(long freq, int *nom, int *den, int *div)
2739 {
2740         int n, d;
2741         long h, diff;
2742 
2743         assert(nom != NULL);
2744         assert(den != NULL);
2745         assert(div != NULL);
2746 
2747         *nom = 0;
2748         *den = 0;
2749         *div = 0;
2750 
2751         if (freq < 8000)
2752                 freq = 8000;
2753 
2754         diff = freq;
2755 
2756         for (n = 32; n < 128; n++) {
2757                 int s = 0;
2758 
2759                 d = (14318 * n) / freq;
2760                 if ((d >= 7) && (d <= 63)) {
2761                         int temp = d;
2762 
2763                         if (temp > 31) {
2764                                 s = 1;
2765                                 temp >>= 1;
2766                         }
2767                         h = ((14318 * n) / temp) >> s;
2768                         h = h > freq ? h - freq : freq - h;
2769                         if (h < diff) {
2770                                 diff = h;
2771                                 *nom = n;
2772                                 *den = temp;
2773                                 *div = s;
2774                         }
2775                 }
2776                 d++;
2777                 if ((d >= 7) && (d <= 63)) {
2778                         if (d > 31) {
2779                                 s = 1;
2780                                 d >>= 1;
2781                         }
2782                         h = ((14318 * n) / d) >> s;
2783                         h = h > freq ? h - freq : freq - h;
2784                         if (h < diff) {
2785                                 diff = h;
2786                                 *nom = n;
2787                                 *den = d;
2788                                 *div = s;
2789                         }
2790                 }
2791         }
2792 }
2793 
2794 /* -------------------------------------------------------------------------
2795  *
2796  * debugging functions
2797  *
2798  * -------------------------------------------------------------------------
2799  */
2800 
2801 #ifdef CIRRUSFB_DEBUG
2802 
2803 /**
2804  * cirrusfb_dbg_print_regs
2805  * @base: If using newmmio, the newmmio base address, otherwise %NULL
2806  * @reg_class: type of registers to read: %CRT, or %SEQ
2807  *
2808  * DESCRIPTION:
2809  * Dumps the given list of VGA CRTC registers.  If @base is %NULL,
2810  * old-style I/O ports are queried for information, otherwise MMIO is
2811  * used at the given @base address to query the information.
2812  */
2813 
2814 static void cirrusfb_dbg_print_regs(struct fb_info *info,
2815                                     caddr_t regbase,
2816                                     enum cirrusfb_dbg_reg_class reg_class, ...)
2817 {
2818         va_list list;
2819         unsigned char val = 0;
2820         unsigned reg;
2821         char *name;
2822 
2823         va_start(list, reg_class);
2824 
2825         name = va_arg(list, char *);
2826         while (name != NULL) {
2827                 reg = va_arg(list, int);
2828 
2829                 switch (reg_class) {
2830                 case CRT:
2831                         val = vga_rcrt(regbase, (unsigned char) reg);
2832                         break;
2833                 case SEQ:
2834                         val = vga_rseq(regbase, (unsigned char) reg);
2835                         break;
2836                 default:
2837                         /* should never occur */
2838                         assert(false);
2839                         break;
2840                 }
2841 
2842                 dev_dbg(info->device, "%8s = 0x%02X\n", name, val);
2843 
2844                 name = va_arg(list, char *);
2845         }
2846 
2847         va_end(list);
2848 }
2849 
2850 /**
2851  * cirrusfb_dbg_reg_dump
2852  * @base: If using newmmio, the newmmio base address, otherwise %NULL
2853  *
2854  * DESCRIPTION:
2855  * Dumps a list of interesting VGA and CIRRUSFB registers.  If @base is %NULL,
2856  * old-style I/O ports are queried for information, otherwise MMIO is
2857  * used at the given @base address to query the information.
2858  */
2859 
2860 static void cirrusfb_dbg_reg_dump(struct fb_info *info, caddr_t regbase)
2861 {
2862         dev_dbg(info->device, "VGA CRTC register dump:\n");
2863 
2864         cirrusfb_dbg_print_regs(info, regbase, CRT,
2865                            "CR00", 0x00,
2866                            "CR01", 0x01,
2867                            "CR02", 0x02,
2868                            "CR03", 0x03,
2869                            "CR04", 0x04,
2870                            "CR05", 0x05,
2871                            "CR06", 0x06,
2872                            "CR07", 0x07,
2873                            "CR08", 0x08,
2874                            "CR09", 0x09,
2875                            "CR0A", 0x0A,
2876                            "CR0B", 0x0B,
2877                            "CR0C", 0x0C,
2878                            "CR0D", 0x0D,
2879                            "CR0E", 0x0E,
2880                            "CR0F", 0x0F,
2881                            "CR10", 0x10,
2882                            "CR11", 0x11,
2883                            "CR12", 0x12,
2884                            "CR13", 0x13,
2885                            "CR14", 0x14,
2886                            "CR15", 0x15,
2887                            "CR16", 0x16,
2888                            "CR17", 0x17,
2889                            "CR18", 0x18,
2890                            "CR22", 0x22,
2891                            "CR24", 0x24,
2892                            "CR26", 0x26,
2893                            "CR2D", 0x2D,
2894                            "CR2E", 0x2E,
2895                            "CR2F", 0x2F,
2896                            "CR30", 0x30,
2897                            "CR31", 0x31,
2898                            "CR32", 0x32,
2899                            "CR33", 0x33,
2900                            "CR34", 0x34,
2901                            "CR35", 0x35,
2902                            "CR36", 0x36,
2903                            "CR37", 0x37,
2904                            "CR38", 0x38,
2905                            "CR39", 0x39,
2906                            "CR3A", 0x3A,
2907                            "CR3B", 0x3B,
2908                            "CR3C", 0x3C,
2909                            "CR3D", 0x3D,
2910                            "CR3E", 0x3E,
2911                            "CR3F", 0x3F,
2912                            NULL);
2913 
2914         dev_dbg(info->device, "\n");
2915 
2916         dev_dbg(info->device, "VGA SEQ register dump:\n");
2917 
2918         cirrusfb_dbg_print_regs(info, regbase, SEQ,
2919                            "SR00", 0x00,
2920                            "SR01", 0x01,
2921                            "SR02", 0x02,
2922                            "SR03", 0x03,
2923                            "SR04", 0x04,
2924                            "SR08", 0x08,
2925                            "SR09", 0x09,
2926                            "SR0A", 0x0A,
2927                            "SR0B", 0x0B,
2928                            "SR0D", 0x0D,
2929                            "SR10", 0x10,
2930                            "SR11", 0x11,
2931                            "SR12", 0x12,
2932                            "SR13", 0x13,
2933                            "SR14", 0x14,
2934                            "SR15", 0x15,
2935                            "SR16", 0x16,
2936                            "SR17", 0x17,
2937                            "SR18", 0x18,
2938                            "SR19", 0x19,
2939                            "SR1A", 0x1A,
2940                            "SR1B", 0x1B,
2941                            "SR1C", 0x1C,
2942                            "SR1D", 0x1D,
2943                            "SR1E", 0x1E,
2944                            "SR1F", 0x1F,
2945                            NULL);
2946 
2947         dev_dbg(info->device, "\n");
2948 }
2949 
2950 #endif                          /* CIRRUSFB_DEBUG */
2951 

/* [<][>][^][v][top][bottom][index][help] */