2  * linux/drivers/video/atafb.c -- Atari builtin chipset frame buffer device
 
   4  *  Copyright (C) 1994 Martin Schaller & Roman Hodek
 
   6  * This file is subject to the terms and conditions of the GNU General Public
 
   7  * License.  See the file COPYING in the main directory of this archive
 
  11  *   - 03 Jan 95: Original version by Martin Schaller: The TT driver and
 
  12  *                all the device independent stuff
 
  13  *   - 09 Jan 95: Roman: I've added the hardware abstraction (hw_switch)
 
  14  *                and wrote the Falcon, ST(E), and External drivers
 
  15  *                based on the original TT driver.
 
  16  *   - 07 May 95: Martin: Added colormap operations for the external driver
 
  17  *   - 21 May 95: Martin: Added support for overscan
 
  18  *                Andreas: some bug fixes for this
 
  19  *   -    Jul 95: Guenther Kelleter <guenther@pool.informatik.rwth-aachen.de>:
 
  20  *                Programmable Falcon video modes
 
  21  *                (thanks to Christian Cartus for documentation
 
  22  *                of VIDEL registers).
 
  23  *   - 27 Dec 95: Guenther: Implemented user definable video modes "user[0-7]"
 
  24  *                on minor 24...31. "user0" may be set on commandline by
 
  25  *                "R<x>;<y>;<depth>". (Makes sense only on Falcon)
 
  26  *                Video mode switch on Falcon now done at next VBL interrupt
 
  27  *                to avoid the annoying right shift of the screen.
 
  28  *   - 23 Sep 97: Juergen: added xres_virtual for cards like ProMST
 
  29  *                The external-part is legacy, therefore hardware-specific
 
  30  *                functions like panning/hardwarescrolling/blanking isn't
 
  32  *   - 29 Sep 97: Juergen: added Romans suggestion for pan_display
 
  33  *                                (var->xoffset was changed even if no set_screen_base avail.)
 
  34  *       - 05 Oct 97: Juergen: extfb (PACKED_PIXEL) is FB_PSEUDOCOLOR 'cause
 
  35  *                                we know how to set the colors
 
  36  *                                ext_*palette: read from ext_colors (former MV300_colors)
 
  37  *                                                          write to ext_colors and RAMDAC
 
  40  *   - For the Falcon it is not possible to set random video modes on
 
  41  *     SM124 and SC/TV, only the bootup resolution is supported.
 
  50 #include <linux/module.h>
 
  51 #include <linux/kernel.h>
 
  52 #include <linux/sched.h>
 
  53 #include <linux/errno.h>
 
  54 #include <linux/string.h>
 
  56 #include <linux/tty.h>
 
  57 #include <linux/slab.h>
 
  58 #include <linux/delay.h>
 
  59 #include <linux/init.h>
 
  60 #include <linux/interrupt.h>
 
  62 #include <asm/setup.h>
 
  63 #include <asm/uaccess.h>
 
  64 #include <asm/pgtable.h>
 
  68 #include <asm/atarihw.h>
 
  69 #include <asm/atariints.h>
 
  70 #include <asm/atari_stram.h>
 
  73 #include <asm/atarikb.h>
 
  75 #include <video/fbcon.h>
 
  76 #include <video/fbcon-cfb8.h>
 
  77 #include <video/fbcon-cfb16.h>
 
  78 #include <video/fbcon-iplan2p2.h>
 
  79 #include <video/fbcon-iplan2p4.h>
 
  80 #include <video/fbcon-iplan2p8.h>
 
  81 #include <video/fbcon-mfb.h>
 
  84 #define SWITCH_ACIA 0x01                /* modes for switch on OverScan */
 
  85 #define SWITCH_SND6 0x40
 
  86 #define SWITCH_SND7 0x80
 
  87 #define SWITCH_NONE 0x00
 
  90 #define up(x, r) (((x) + (r) - 1) & ~((r)-1))
 
  93 static int default_par=0;       /* default resolution (0=none) */
 
  95 static unsigned long default_mem_req=0;
 
  97 static int hwscroll=-1;
 
  99 static int use_hwscroll = 1;
 
 101 static int sttt_xres=640,st_yres=400,tt_yres=480;
 
 102 static int sttt_xres_virtual=640,sttt_yres_virtual=400;
 
 103 static int ovsc_offset=0, ovsc_addlen=0;
 
 105 static struct atafb_par {
 
 108 #if defined ATAFB_TT || defined ATAFB_STE
 
 117                         /* Here are fields for storing a video mode, as direct
 
 118                          * parameters for the hardware.
 
 128                         short hht, hbb, hbe, hdb, hde, hss;
 
 129                         short vft, vbb, vbe, vdb, vde, vss;
 
 130                         /* auxiliary information */
 
 136                 /* Nothing needed for external mode */
 
 140 /* Don't calculate an own resolution, and thus don't change the one found when
 
 141  * booting (currently used for the Falcon to keep settings for internal video
 
 142  * hardware extensions (e.g. ScreenBlaster)  */
 
 143 static int DontCalcRes = 0; 
 
 146 #define HHT hw.falcon.hht
 
 147 #define HBB hw.falcon.hbb
 
 148 #define HBE hw.falcon.hbe
 
 149 #define HDB hw.falcon.hdb
 
 150 #define HDE hw.falcon.hde
 
 151 #define HSS hw.falcon.hss
 
 152 #define VFT hw.falcon.vft
 
 153 #define VBB hw.falcon.vbb
 
 154 #define VBE hw.falcon.vbe
 
 155 #define VDB hw.falcon.vdb
 
 156 #define VDE hw.falcon.vde
 
 157 #define VSS hw.falcon.vss
 
 158 #define VCO_CLOCK25             0x04
 
 159 #define VCO_CSYPOS              0x10
 
 160 #define VCO_VSYPOS              0x20
 
 161 #define VCO_HSYPOS              0x40
 
 162 #define VCO_SHORTOFFS   0x100
 
 163 #define VMO_DOUBLE              0x01
 
 164 #define VMO_INTER               0x02
 
 165 #define VMO_PREMASK             0x0c
 
 168 static struct fb_info fb_info;
 
 170 static void *screen_base;       /* base address of screen */
 
 171 static void *real_screen_base;  /* (only for Overscan) */
 
 173 static int screen_len;
 
 175 static int current_par_valid=0; 
 
 177 static int mono_moni=0;
 
 179 static struct display disp;
 
 183 /* external video handling */
 
 185 static unsigned                 external_xres;
 
 186 static unsigned                 external_xres_virtual;
 
 187 static unsigned                 external_yres;
 
 188 /* not needed - atafb will never support panning/hardwarescroll with external
 
 189  * static unsigned              external_yres_virtual;  
 
 192 static unsigned                 external_depth;
 
 193 static int                              external_pmode;
 
 194 static void *external_addr = 0;
 
 195 static unsigned long    external_len;
 
 196 static unsigned long    external_vgaiobase = 0;
 
 197 static unsigned int             external_bitspercol = 6;
 
 200 JOE <joe@amber.dinoco.de>: 
 
 201 added card type for external driver, is only needed for
 
 205 enum cardtype { IS_VGA, IS_MV300 };
 
 206 static enum cardtype external_card_type = IS_VGA;
 
 209 The MV300 mixes the color registers. So we need an array of munged
 
 210 indices in order to access the correct reg.
 
 212 static int MV300_reg_1bit[2]={0,1};
 
 213 static int MV300_reg_4bit[16]={
 
 214 0, 8, 4, 12, 2, 10, 6, 14, 1, 9, 5, 13, 3, 11, 7, 15 };
 
 215 static int MV300_reg_8bit[256]={
 
 216 0, 128, 64, 192, 32, 160, 96, 224, 16, 144, 80, 208, 48, 176, 112, 240, 
 
 217 8, 136, 72, 200, 40, 168, 104, 232, 24, 152, 88, 216, 56, 184, 120, 248, 
 
 218 4, 132, 68, 196, 36, 164, 100, 228, 20, 148, 84, 212, 52, 180, 116, 244, 
 
 219 12, 140, 76, 204, 44, 172, 108, 236, 28, 156, 92, 220, 60, 188, 124, 252, 
 
 220 2, 130, 66, 194, 34, 162, 98, 226, 18, 146, 82, 210, 50, 178, 114, 242, 
 
 221 10, 138, 74, 202, 42, 170, 106, 234, 26, 154, 90, 218, 58, 186, 122, 250, 
 
 222 6, 134, 70, 198, 38, 166, 102, 230, 22, 150, 86, 214, 54, 182, 118, 246, 
 
 223 14, 142, 78, 206, 46, 174, 110, 238, 30, 158, 94, 222, 62, 190, 126, 254, 
 
 224 1, 129, 65, 193, 33, 161, 97, 225, 17, 145, 81, 209, 49, 177, 113, 241, 
 
 225 9, 137, 73, 201, 41, 169, 105, 233, 25, 153, 89, 217, 57, 185, 121, 249, 
 
 226 5, 133, 69, 197, 37, 165, 101, 229, 21, 149, 85, 213, 53, 181, 117, 245, 
 
 227 13, 141, 77, 205, 45, 173, 109, 237, 29, 157, 93, 221, 61, 189, 125, 253, 
 
 228 3, 131, 67, 195, 35, 163, 99, 227, 19, 147, 83, 211, 51, 179, 115, 243, 
 
 229 11, 139, 75, 203, 43, 171, 107, 235, 27, 155, 91, 219, 59, 187, 123, 251, 
 
 230 7, 135, 71, 199, 39, 167, 103, 231, 23, 151, 87, 215, 55, 183, 119, 247, 
 
 231 15, 143, 79, 207, 47, 175, 111, 239, 31, 159, 95, 223, 63, 191, 127, 255 }; 
 
 233 static int *MV300_reg = MV300_reg_8bit;
 
 236 And on the MV300 it's difficult to read out the hardware palette. So we
 
 237 just keep track of the set colors in our own array here, and use that!
 
 240 static struct { unsigned char red,green,blue,pad; } ext_color[256];
 
 241 #endif /* ATAFB_EXT */
 
 244 static int inverse=0;
 
 246 extern int fontheight_8x8;
 
 247 extern int fontwidth_8x8;
 
 248 extern unsigned char fontdata_8x8[];
 
 250 extern int fontheight_8x16;
 
 251 extern int fontwidth_8x16;
 
 252 extern unsigned char fontdata_8x16[];
 
 254 /* ++roman: This structure abstracts from the underlying hardware (ST(e),
 
 257  * int (*detect)( void )
 
 258  *   This function should detect the current video mode settings and
 
 259  *   store them in atafb_predefined[0] for later reference by the
 
 260  *   user. Return the index+1 of an equivalent predefined mode or 0
 
 261  *   if there is no such.
 
 263  * int (*encode_fix)( struct fb_fix_screeninfo *fix,
 
 264  *                    struct atafb_par *par )
 
 265  *   This function should fill in the 'fix' structure based on the
 
 266  *   values in the 'par' structure.
 
 268  * int (*decode_var)( struct fb_var_screeninfo *var,
 
 269  *                    struct atafb_par *par )
 
 270  *   Get the video params out of 'var'. If a value doesn't fit, round
 
 271  *   it up, if it's too big, return EINVAL.
 
 272  *   Round up in the following order: bits_per_pixel, xres, yres, 
 
 273  *   xres_virtual, yres_virtual, xoffset, yoffset, grayscale, bitfields, 
 
 274  *   horizontal timing, vertical timing.
 
 276  * int (*encode_var)( struct fb_var_screeninfo *var,
 
 277  *                    struct atafb_par *par );
 
 278  *   Fill the 'var' structure based on the values in 'par' and maybe
 
 279  *   other values read out of the hardware.
 
 281  * void (*get_par)( struct atafb_par *par )
 
 282  *   Fill the hardware's 'par' structure.
 
 284  * void (*set_par)( struct atafb_par *par )
 
 285  *   Set the hardware according to 'par'.
 
 287  * int (*getcolreg)( unsigned regno, unsigned *red,
 
 288  *                   unsigned *green, unsigned *blue,
 
 289  *                   unsigned *transp, struct fb_info *info )
 
 290  *   Read a single color register and split it into
 
 291  *   colors/transparent. Return != 0 for invalid regno.
 
 293  * void (*set_screen_base)(void *s_base)
 
 294  *   Set the base address of the displayed frame buffer. Only called
 
 295  *   if yres_virtual > yres or xres_virtual > xres.
 
 297  * int (*blank)( int blank_mode )
 
 298  *   Blank the screen if blank_mode!=0, else unblank. If blank==NULL then
 
 299  *   the caller blanks by setting the CLUT to all black. Return 0 if blanking
 
 300  *   succeeded, !=0 if un-/blanking failed due to e.g. a video mode which
 
 301  *   doesn't support it. Implements VESA suspend and powerdown modes on
 
 302  *   hardware that supports disabling hsync/vsync:
 
 303  *       blank_mode==2: suspend vsync, 3:suspend hsync, 4: powerdown.
 
 306 static struct fb_hwswitch {
 
 307         int  (*detect)( void );
 
 308         int  (*encode_fix)( struct fb_fix_screeninfo *fix,
 
 309                                                 struct atafb_par *par );
 
 310         int  (*decode_var)( struct fb_var_screeninfo *var,
 
 311                                                 struct atafb_par *par );
 
 312         int  (*encode_var)( struct fb_var_screeninfo *var,
 
 313                                                 struct atafb_par *par );
 
 314         void (*get_par)( struct atafb_par *par );
 
 315         void (*set_par)( struct atafb_par *par );
 
 316         int  (*getcolreg)( unsigned regno, unsigned *red,
 
 317                                            unsigned *green, unsigned *blue,
 
 318                                            unsigned *transp, struct fb_info *info );
 
 319         void (*set_screen_base)(void *s_base);
 
 320         int  (*blank)( int blank_mode );
 
 321         int  (*pan_display)( struct fb_var_screeninfo *var,
 
 322                                                  struct atafb_par *par);
 
 325 static char *autodetect_names[] = {"autodetect", NULL};
 
 326 static char *stlow_names[] = {"stlow", NULL};
 
 327 static char *stmid_names[] = {"stmid", "default5", NULL};
 
 328 static char *sthigh_names[] = {"sthigh", "default4", NULL};
 
 329 static char *ttlow_names[] = {"ttlow", NULL};
 
 330 static char *ttmid_names[]= {"ttmid", "default1", NULL};
 
 331 static char *tthigh_names[]= {"tthigh", "default2", NULL};
 
 332 static char *vga2_names[] = {"vga2", NULL};
 
 333 static char *vga4_names[] = {"vga4", NULL};
 
 334 static char *vga16_names[] = {"vga16", "default3", NULL};
 
 335 static char *vga256_names[] = {"vga256", NULL};
 
 336 static char *falh2_names[] = {"falh2", NULL};
 
 337 static char *falh16_names[] = {"falh16", NULL};
 
 339 static char **fb_var_names[] = {
 
 340         /* Writing the name arrays directly in this array (via "(char *[]){...}")
 
 341          * crashes gcc 2.5.8 (sigsegv) if the inner array
 
 342          * contains more than two items. I've also seen that all elements
 
 343          * were identical to the last (my cross-gcc) :-(*/
 
 358         /* ,NULL */ /* this causes a sigsegv on my gcc-2.5.8 */
 
 361 static struct fb_var_screeninfo atafb_predefined[] = {
 
 363          * yres_virtual==0 means use hw-scrolling if possible, else yres
 
 366           0, 0, 0, 0, 0, 0, 0, 0,               /* xres-grayscale */
 
 367           {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0},   /* red green blue tran*/
 
 368           0, 0, -1, -1, 0, 0, 0, 0, 0, 0, 0, 0 },
 
 370           320, 200, 320, 0, 0, 0, 4, 0,
 
 371           {0, 4, 0}, {0, 4, 0}, {0, 4, 0}, {0, 0, 0},
 
 372           0, 0, -1, -1, 0, 0, 0, 0, 0, 0, 0, 0 },
 
 374           640, 200, 640, 0, 0, 0, 2, 0,
 
 375           {0, 4, 0}, {0, 4, 0}, {0, 4, 0}, {0, 0, 0},
 
 376           0, 0, -1, -1, 0, 0, 0, 0, 0, 0, 0, 0 },
 
 378           640, 400, 640, 0, 0, 0, 1, 0,
 
 379           {0, 4, 0}, {0, 4, 0}, {0, 4, 0}, {0, 0, 0},
 
 380           0, 0, -1, -1, 0, 0, 0, 0, 0, 0, 0, 0 },
 
 382           320, 480, 320, 0, 0, 0, 8, 0,
 
 383           {0, 4, 0}, {0, 4, 0}, {0, 4, 0}, {0, 0, 0},
 
 384           0, 0, -1, -1, 0, 0, 0, 0, 0, 0, 0, 0 },
 
 386           640, 480, 640, 0, 0, 0, 4, 0,
 
 387           {0, 4, 0}, {0, 4, 0}, {0, 4, 0}, {0, 0, 0},
 
 388           0, 0, -1, -1, 0, 0, 0, 0, 0, 0, 0, 0 },
 
 390           1280, 960, 1280, 0, 0, 0, 1, 0,
 
 391           {0, 4, 0}, {0, 4, 0}, {0, 4, 0}, {0, 0, 0},
 
 392           0, 0, -1, -1, 0, 0, 0, 0, 0, 0, 0, 0 },
 
 394           640, 480, 640, 0, 0, 0, 1, 0,
 
 395           {0, 6, 0}, {0, 6, 0}, {0, 6, 0}, {0, 0, 0},
 
 396           0, 0, -1, -1, 0, 0, 0, 0, 0, 0, 0, 0 },
 
 398           640, 480, 640, 0, 0, 0, 2, 0,
 
 399           {0, 4, 0}, {0, 4, 0}, {0, 4, 0}, {0, 0, 0},
 
 400           0, 0, -1, -1, 0, 0, 0, 0, 0, 0, 0, 0 },
 
 402           640, 480, 640, 0, 0, 0, 4, 0,
 
 403           {0, 6, 0}, {0, 6, 0}, {0, 6, 0}, {0, 0, 0},
 
 404           0, 0, -1, -1, 0, 0, 0, 0, 0, 0, 0, 0 },
 
 406           640, 480, 640, 0, 0, 0, 8, 0,
 
 407           {0, 6, 0}, {0, 6, 0}, {0, 6, 0}, {0, 0, 0},
 
 408           0, 0, -1, -1, 0, 0, 0, 0, 0, 0, 0, 0 },
 
 410           896, 608, 896, 0, 0, 0, 1, 0,
 
 411           {0, 6, 0}, {0, 6, 0}, {0, 6, 0}, {0, 0, 0},
 
 412           0, 0, -1, -1, 0, 0, 0, 0, 0, 0, 0, 0 },
 
 414           896, 608, 896, 0, 0, 0, 4, 0,
 
 415           {0, 6, 0}, {0, 6, 0}, {0, 6, 0}, {0, 0, 0},
 
 416           0, 0, -1, -1, 0, 0, 0, 0, 0, 0, 0, 0 },
 
 419 static int num_atafb_predefined=ARRAY_SIZE(atafb_predefined);
 
 423 get_video_mode(char *vname)
 
 428     name_list=fb_var_names;
 
 429     for (i = 0 ; i < num_atafb_predefined ; i++) {
 
 431         if (! name || ! *name)
 
 434             if (! strcmp(vname, *name))
 
 444 /* ------------------- TT specific functions ---------------------- */
 
 448 static int tt_encode_fix( struct fb_fix_screeninfo *fix,
 
 449                                                   struct atafb_par *par )
 
 454         strcpy(fix->id,"Atari Builtin");
 
 455         fix->smem_start = (unsigned long)real_screen_base;
 
 456         fix->smem_len = screen_len;
 
 457         fix->type=FB_TYPE_INTERLEAVED_PLANES;
 
 459         fix->visual=FB_VISUAL_PSEUDOCOLOR;
 
 460         mode = par->hw.tt.mode & TT_SHIFTER_MODEMASK;
 
 461         if (mode == TT_SHIFTER_TTHIGH || mode == TT_SHIFTER_STHIGH) {
 
 462                 fix->type=FB_TYPE_PACKED_PIXELS;
 
 464                 if (mode == TT_SHIFTER_TTHIGH)
 
 465                         fix->visual=FB_VISUAL_MONO01;
 
 470         fix->line_length = 0;
 
 471         fix->accel = FB_ACCEL_ATARIBLITT;
 
 476 static int tt_decode_var( struct fb_var_screeninfo *var,
 
 477                                                   struct atafb_par *par )
 
 481         int bpp=var->bits_per_pixel;
 
 483         int yres_virtual = var->yres_virtual;
 
 486                 if (bpp > 1 || xres > sttt_xres*2 || yres >tt_yres*2)
 
 488                 par->hw.tt.mode=TT_SHIFTER_TTHIGH;
 
 493                 if (bpp > 8 || xres > sttt_xres || yres > tt_yres)
 
 496                         if (xres > sttt_xres/2 || yres > tt_yres)
 
 498                         par->hw.tt.mode=TT_SHIFTER_TTLOW;
 
 504                         if (xres > sttt_xres || yres > tt_yres)
 
 506                         if (xres > sttt_xres/2 || yres > st_yres/2) {
 
 507                                 par->hw.tt.mode=TT_SHIFTER_TTMID;
 
 513                                 par->hw.tt.mode=TT_SHIFTER_STLOW;
 
 520                         if (xres > sttt_xres || yres > st_yres/2)
 
 522                         par->hw.tt.mode=TT_SHIFTER_STMID;
 
 527                 else if (var->xres > sttt_xres || var->yres > st_yres) {
 
 531                         par->hw.tt.mode=TT_SHIFTER_STHIGH;
 
 537         if (yres_virtual <= 0)
 
 539         else if (yres_virtual < yres)
 
 541         if (var->sync & FB_SYNC_EXT)
 
 546         if (yres_virtual * linelen > screen_len && screen_len)
 
 548         if (yres * linelen > screen_len && screen_len)
 
 550         if (var->yoffset + yres > yres_virtual && yres_virtual)
 
 552         par->yres_virtual = yres_virtual;
 
 553         par->screen_base = screen_base + var->yoffset * linelen;
 
 557 static int tt_encode_var( struct fb_var_screeninfo *var,
 
 558                                                   struct atafb_par *par )
 
 561         memset(var, 0, sizeof(struct fb_var_screeninfo));
 
 564         var->red.msb_right=0;
 
 568         var->left_margin=120;           /* these may be incorrect       */
 
 569         var->right_margin=100;
 
 571         var->lower_margin=16;
 
 578         if (par->hw.tt.sync & 1)
 
 581                 var->sync=FB_SYNC_EXT;
 
 583         switch (par->hw.tt.mode & TT_SHIFTER_MODEMASK) {
 
 584         case TT_SHIFTER_STLOW:
 
 585                 var->xres=sttt_xres/2;
 
 586                 var->xres_virtual=sttt_xres_virtual/2;
 
 588                 var->bits_per_pixel=4;
 
 590         case TT_SHIFTER_STMID:
 
 592                 var->xres_virtual=sttt_xres_virtual;
 
 594                 var->bits_per_pixel=2;
 
 596         case TT_SHIFTER_STHIGH:
 
 598                 var->xres_virtual=sttt_xres_virtual;
 
 600                 var->bits_per_pixel=1;
 
 602         case TT_SHIFTER_TTLOW:
 
 603                 var->xres=sttt_xres/2;
 
 604                 var->xres_virtual=sttt_xres_virtual/2;
 
 606                 var->bits_per_pixel=8;
 
 608         case TT_SHIFTER_TTMID:
 
 610                 var->xres_virtual=sttt_xres_virtual;
 
 612                 var->bits_per_pixel=4;
 
 614         case TT_SHIFTER_TTHIGH:
 
 616                 var->xres=sttt_xres*2;
 
 617                 var->xres_virtual=sttt_xres_virtual*2;
 
 619                 var->bits_per_pixel=1;
 
 622         var->blue=var->green=var->red;
 
 623         var->transp.offset=0;
 
 624         var->transp.length=0;
 
 625         var->transp.msb_right=0;
 
 626         linelen=var->xres_virtual * var->bits_per_pixel / 8;
 
 628                 var->yres_virtual=var->yres;
 
 629         else if (screen_len) {
 
 630                 if (par->yres_virtual)
 
 631                         var->yres_virtual = par->yres_virtual;
 
 633                         /* yres_virtual==0 means use maximum */
 
 634                         var->yres_virtual = screen_len / linelen;
 
 637                         var->yres_virtual = 2 * var->yres;
 
 639                         var->yres_virtual=var->yres+hwscroll * 16;
 
 643                 var->yoffset=(par->screen_base - screen_base)/linelen;
 
 648         var->vmode=FB_VMODE_NONINTERLACED;
 
 653 static void tt_get_par( struct atafb_par *par )
 
 656         par->hw.tt.mode=shifter_tt.tt_shiftmode;
 
 657         par->hw.tt.sync=shifter.syncmode;
 
 658         addr = ((shifter.bas_hi & 0xff) << 16) |
 
 659                ((shifter.bas_md & 0xff) << 8)  |
 
 660                ((shifter.bas_lo & 0xff));
 
 661         par->screen_base = phys_to_virt(addr);
 
 664 static void tt_set_par( struct atafb_par *par )
 
 666         shifter_tt.tt_shiftmode=par->hw.tt.mode;
 
 667         shifter.syncmode=par->hw.tt.sync;
 
 668         /* only set screen_base if really necessary */
 
 669         if (current_par.screen_base != par->screen_base)
 
 670                 fbhw->set_screen_base(par->screen_base);
 
 674 static int tt_getcolreg(unsigned regno, unsigned *red,
 
 675                         unsigned *green, unsigned *blue,
 
 676                         unsigned *transp, struct fb_info *info)
 
 680         if ((shifter_tt.tt_shiftmode & TT_SHIFTER_MODEMASK) == TT_SHIFTER_STHIGH)
 
 684         t = tt_palette[regno];
 
 702 static int tt_setcolreg(unsigned regno, unsigned red,
 
 703                         unsigned green, unsigned blue,
 
 704                         unsigned transp, struct fb_info *info)
 
 706         if ((shifter_tt.tt_shiftmode & TT_SHIFTER_MODEMASK) == TT_SHIFTER_STHIGH)
 
 710         tt_palette[regno] = (((red >> 12) << 8) | ((green >> 12) << 4) |
 
 712         if ((shifter_tt.tt_shiftmode & TT_SHIFTER_MODEMASK) ==
 
 713                 TT_SHIFTER_STHIGH && regno == 254)
 
 719 static int tt_detect( void )
 
 721 {       struct atafb_par par;
 
 723         /* Determine the connected monitor: The DMA sound must be
 
 724          * disabled before reading the MFP GPIP, because the Sound
 
 725          * Done Signal and the Monochrome Detect are XORed together!
 
 727          * Even on a TT, we should look if there is a DMA sound. It was
 
 728          * announced that the Eagle is TT compatible, but only the PCM is
 
 731         if (ATARIHW_PRESENT(PCM_8BIT)) { 
 
 732                 tt_dmasnd.ctrl = DMASND_CTRL_OFF;
 
 733                 udelay(20);     /* wait a while for things to settle down */
 
 735         mono_moni = (mfp.par_dt_reg & 0x80) == 0;
 
 738         tt_encode_var(&atafb_predefined[0], &par);
 
 743 #endif /* ATAFB_TT */
 
 745 /* ------------------- Falcon specific functions ---------------------- */
 
 749 static int mon_type;            /* Falcon connected monitor */
 
 750 static int f030_bus_width;      /* Falcon ram bus width (for vid_control) */
 
 756 static struct pixel_clock {
 
 757         unsigned long f;        /* f/[Hz] */
 
 758         unsigned long t;        /* t/[ps] (=1/f) */
 
 759         int right, hsync, left; /* standard timing in clock cycles, not pixel */
 
 760                 /* hsync initialized in falcon_detect() */
 
 761         int sync_mask;          /* or-mask for hw.falcon.sync to set this clock */
 
 762         int control_mask;       /* ditto, for hw.falcon.vid_control */
 
 764 f25  = {25175000, 39721, 18, 0, 42, 0x0, VCO_CLOCK25},
 
 765 f32  = {32000000, 31250, 18, 0, 42, 0x0, 0},
 
 766 fext = {       0,     0, 18, 0, 42, 0x1, 0};
 
 768 /* VIDEL-prescale values [mon_type][pixel_length from VCO] */
 
 769 static int vdl_prescale[4][3] = {{4,2,1}, {4,2,1}, {4,2,2}, {4,2,1}};
 
 771 /* Default hsync timing [mon_type] in picoseconds */
 
 772 static long h_syncs[4] = {3000000, 4875000, 4000000, 4875000};
 
 774 #ifdef FBCON_HAS_CFB16
 
 775 static u16 fbcon_cfb16_cmap[16];
 
 778 static inline int hxx_prescale(struct falcon_hw *hw)
 
 780         return hw->ste_mode ? 16 :
 
 781                    vdl_prescale[mon_type][hw->vid_mode >> 2 & 0x3];
 
 784 static int falcon_encode_fix( struct fb_fix_screeninfo *fix,
 
 785                                                           struct atafb_par *par )
 
 787         strcpy(fix->id, "Atari Builtin");
 
 788         fix->smem_start = (unsigned long)real_screen_base;
 
 789         fix->smem_len = screen_len;
 
 790         fix->type = FB_TYPE_INTERLEAVED_PLANES;
 
 792         fix->visual = FB_VISUAL_PSEUDOCOLOR;
 
 796         if (par->hw.falcon.mono) {
 
 797                 fix->type = FB_TYPE_PACKED_PIXELS;
 
 799                 /* no smooth scrolling with longword aligned video mem */
 
 802         else if (par->hw.falcon.f_shift & 0x100) {
 
 803                 fix->type = FB_TYPE_PACKED_PIXELS;
 
 805                 /* Is this ok or should it be DIRECTCOLOR? */
 
 806                 fix->visual = FB_VISUAL_TRUECOLOR;
 
 809         fix->line_length = 0;
 
 810         fix->accel = FB_ACCEL_ATARIBLITT;
 
 815 static int falcon_decode_var( struct fb_var_screeninfo *var,
 
 816                                                           struct atafb_par *par )
 
 818         int bpp = var->bits_per_pixel;
 
 819         int xres = var->xres;
 
 820         int yres = var->yres;
 
 821         int xres_virtual = var->xres_virtual;
 
 822         int yres_virtual = var->yres_virtual;
 
 823         int left_margin, right_margin, hsync_len;
 
 824         int upper_margin, lower_margin, vsync_len;
 
 826         int interlace = 0, doubleline = 0;
 
 827         struct pixel_clock *pclock;
 
 828         int plen; /* width of pixel in clock cycles */
 
 835         Get the video params out of 'var'. If a value doesn't fit, round
 
 836         it up, if it's too big, return EINVAL.
 
 837         Round up in the following order: bits_per_pixel, xres, yres, 
 
 838         xres_virtual, yres_virtual, xoffset, yoffset, grayscale, bitfields, 
 
 839         horizontal timing, vertical timing.
 
 841         There is a maximum of screen resolution determined by pixelclock
 
 842         and minimum frame rate -- (X+hmarg.)*(Y+vmarg.)*vfmin <= pixelclock.
 
 843         In interlace mode this is     "     *    "     *vfmin <= pixelclock.
 
 844         Additional constraints: hfreq.
 
 845         Frequency range for multisync monitors is given via command line.
 
 846         For TV and SM124 both frequencies are fixed.
 
 848         X % 16 == 0 to fit 8x?? font (except 1 bitplane modes must use X%32==0)
 
 849         Y % 16 == 0 to fit 8x16 font
 
 852         Currently interlace and doubleline mode in var are ignored. 
 
 853         On SM124 and TV only the standard resolutions can be used.
 
 856         /* Reject uninitialized mode */
 
 857         if (!xres || !yres || !bpp)
 
 860         if (mon_type == F_MON_SM && bpp != 1) {
 
 865                 par->hw.falcon.f_shift = 0x400;
 
 866                 par->hw.falcon.st_shift = 0x200;
 
 870                 par->hw.falcon.f_shift = 0x000;
 
 871                 par->hw.falcon.st_shift = 0x100;
 
 875                 par->hw.falcon.f_shift = 0x000;
 
 876                 par->hw.falcon.st_shift = 0x000;
 
 880                 par->hw.falcon.f_shift = 0x010;
 
 882         else if (bpp <= 16) {
 
 883                 bpp = 16; /* packed pixel mode */
 
 884                 par->hw.falcon.f_shift = 0x100; /* hicolor, no overlay */
 
 888         par->hw.falcon.bpp = bpp;
 
 890         if (mon_type == F_MON_SM || DontCalcRes) {
 
 891                 /* Skip all calculations. VGA/TV/SC1224 only supported. */
 
 892                 struct fb_var_screeninfo *myvar = &atafb_predefined[0];
 
 894                 if (bpp > myvar->bits_per_pixel ||
 
 895                         var->xres > myvar->xres ||
 
 896                         var->yres > myvar->yres)
 
 898                 fbhw->get_par(par);     /* Current par will be new par */
 
 899                 goto set_screen_base;   /* Don't forget this */
 
 902         /* Only some fixed resolutions < 640x400 */
 
 905         else if (xres <= 640 && bpp != 16)
 
 909         else if (yres <= 240)
 
 911         else if (yres <= 400)
 
 914         /* 2 planes must use STE compatibility mode */
 
 915         par->hw.falcon.ste_mode = bpp==2;
 
 916         par->hw.falcon.mono = bpp==1;
 
 918         /* Total and visible scanline length must be a multiple of one longword,
 
 919          * this and the console fontwidth yields the alignment for xres and
 
 921          * TODO: this way "odd" fontheights are not supported
 
 923          * Special case in STE mode: blank and graphic positions don't align,
 
 924          * avoid trash at right margin
 
 926         if (par->hw.falcon.ste_mode)
 
 927                 xres = (xres + 63) & ~63;
 
 929                 xres = (xres + 31) & ~31;
 
 931                 xres = (xres + 15) & ~15;
 
 933                 yres = (yres + 15) & ~15;
 
 935                 yres = (yres + 7) & ~7;
 
 937         if (xres_virtual < xres)
 
 940                 xres_virtual = (xres_virtual + 31) & ~31;
 
 942                 xres_virtual = (xres_virtual + 15) & ~15;
 
 944         if (yres_virtual <= 0)
 
 946         else if (yres_virtual < yres)
 
 949         /* backward bug-compatibility */
 
 950         if (var->pixclock > 1)
 
 953         par->hw.falcon.line_width = bpp * xres / 16;
 
 954         par->hw.falcon.line_offset = bpp * (xres_virtual - xres) / 16;
 
 956         /* single or double pixel width */
 
 957         xstretch = (xres < 640) ? 2 : 1;
 
 959 #if 0 /* SM124 supports only 640x400, this is rejected above */
 
 960         if (mon_type == F_MON_SM) {
 
 961                 if (xres != 640 && yres != 400)
 
 965                 /* SM124-mode is special */
 
 966                 par->hw.falcon.ste_mode = 1;
 
 967                 par->hw.falcon.f_shift = 0x000;
 
 968                 par->hw.falcon.st_shift = 0x200;
 
 969                 left_margin = hsync_len = 128 / plen;
 
 971                 /* TODO set all margins */
 
 975         if (mon_type == F_MON_SC || mon_type == F_MON_TV) {
 
 977                 if (var->pixclock > f32.t * plen)
 
 982                 if (var->pixclock == 0) {
 
 983                         /* set some minimal margins which center the screen */
 
 986                         hsync_len = pclock->hsync / plen;
 
 989                         vsync_len = interlace ? 3 : 4;
 
 991                         left_margin = var->left_margin;
 
 992                         right_margin = var->right_margin;
 
 993                         hsync_len = var->hsync_len;
 
 994                         upper_margin = var->upper_margin;
 
 995                         lower_margin = var->lower_margin;
 
 996                         vsync_len = var->vsync_len;
 
 997                         if (var->vmode & FB_VMODE_INTERLACED) {
 
 998                                 upper_margin = (upper_margin + 1) / 2;
 
 999                                 lower_margin = (lower_margin + 1) / 2;
 
1000                                 vsync_len = (vsync_len + 1) / 2;
 
1001                         } else if (var->vmode & FB_VMODE_DOUBLE) {
 
1011                         xstretch = 2; /* Double pixel width only for hicolor */
 
1012                 /* Default values are used for vert./hor. timing if no pixelclock given. */
 
1013                 if (var->pixclock == 0) {
 
1016                         /* Choose master pixelclock depending on hor. timing */
 
1017                         plen = 1 * xstretch;
 
1018                         if ((plen * xres + f25.right+f25.hsync+f25.left) *
 
1019                             fb_info.monspecs.hfmin < f25.f)
 
1021                         else if ((plen * xres + f32.right+f32.hsync+f32.left) * 
 
1022                             fb_info.monspecs.hfmin < f32.f)
 
1024                         else if ((plen * xres + fext.right+fext.hsync+fext.left) * 
 
1025                             fb_info.monspecs.hfmin < fext.f
 
1031                         left_margin = pclock->left / plen;
 
1032                         right_margin = pclock->right / plen;
 
1033                         hsync_len = pclock->hsync / plen;
 
1034                         linesize = left_margin + xres + right_margin + hsync_len;
 
1040                         /* Choose largest pixelclock <= wanted clock */
 
1042                         unsigned long pcl = ULONG_MAX;
 
1044                         for (i=1; i <= 4; i *= 2) {
 
1045                                 if (f25.t*i >= var->pixclock && f25.t*i < pcl) {
 
1049                                 if (f32.t*i >= var->pixclock && f32.t*i < pcl) {
 
1053                                 if (fext.t && fext.t*i >= var->pixclock && fext.t*i < pcl) {
 
1060                         plen = pcl / pclock->t;
 
1062                         left_margin = var->left_margin;
 
1063                         right_margin = var->right_margin;
 
1064                         hsync_len = var->hsync_len;
 
1065                         upper_margin = var->upper_margin;
 
1066                         lower_margin = var->lower_margin;
 
1067                         vsync_len = var->vsync_len;
 
1068                         /* Internal unit is [single lines per (half-)frame] */
 
1069                         if (var->vmode & FB_VMODE_INTERLACED) {
 
1070                                 /* # lines in half frame */
 
1071                                 /* External unit is [lines per full frame] */
 
1072                                 upper_margin = (upper_margin + 1) / 2;
 
1073                                 lower_margin = (lower_margin + 1) / 2;
 
1074                                 vsync_len = (vsync_len + 1) / 2;
 
1076                         else if (var->vmode & FB_VMODE_DOUBLE) {
 
1077                                 /* External unit is [double lines per frame] */
 
1083                 if (pclock == &fext)
 
1084                         longoffset = 1; /* VIDEL doesn't synchronize on short offset */
 
1086         /* Is video bus bandwidth (32MB/s) too low for this resolution? */
 
1087         /* this is definitely wrong if bus clock != 32MHz */
 
1088         if (pclock->f / plen / 8 * bpp > 32000000L)
 
1094         /* include sync lengths in right/lower margin for all calculations */
 
1095         right_margin += hsync_len;
 
1096         lower_margin += vsync_len;
 
1098         /* ! In all calculations of margins we use # of lines in half frame
 
1099          * (which is a full frame in non-interlace mode), so we can switch
 
1100          * between interlace and non-interlace without messing around
 
1104         /* Set base_offset 128 and video bus width */
 
1105         par->hw.falcon.vid_control = mon_type | f030_bus_width;
 
1107                 par->hw.falcon.vid_control |= VCO_SHORTOFFS;    /* base_offset 64 */
 
1108         if (var->sync & FB_SYNC_HOR_HIGH_ACT)
 
1109                 par->hw.falcon.vid_control |= VCO_HSYPOS;
 
1110         if (var->sync & FB_SYNC_VERT_HIGH_ACT)
 
1111                 par->hw.falcon.vid_control |= VCO_VSYPOS;
 
1113         par->hw.falcon.vid_control |= pclock->control_mask;
 
1114         /* External or internal clock */
 
1115         par->hw.falcon.sync = pclock->sync_mask | 0x2;
 
1116         /* Pixellength and prescale */
 
1117         par->hw.falcon.vid_mode = (2/plen) << 2;
 
1119                 par->hw.falcon.vid_mode |= VMO_DOUBLE;
 
1121                 par->hw.falcon.vid_mode |= VMO_INTER;
 
1123         /*********************
 
1124         Horizontal timing: unit = [master clock cycles]
 
1125         unit of hxx-registers: [master clock cycles * prescale]
 
1126         Hxx-registers are 9 bit wide
 
1128         1 line = ((hht + 2) * 2 * prescale) clock cycles
 
1130         graphic output = hdb & 0x200 ?
 
1131                ((hht+2)*2 - hdb + hde) * prescale - hdboff + hdeoff:
 
1132                ( hht + 2  - hdb + hde) * prescale - hdboff + hdeoff
 
1133         (this must be a multiple of plen*128/bpp, on VGA pixels
 
1134          to the right may be cut off with a bigger right margin)
 
1136         start of graphics relative to start of 1st halfline = hdb & 0x200 ?
 
1137                (hdb - hht - 2) * prescale + hdboff :
 
1138                hdb * prescale + hdboff
 
1140         end of graphics relative to start of 1st halfline =
 
1141                (hde + hht + 2) * prescale + hdeoff
 
1142         *********************/
 
1143         /* Calculate VIDEL registers */
 
1145         int hdb_off, hde_off, base_off;
 
1146         int gstart, gend1, gend2, align;
 
1148         prescale = hxx_prescale(&par->hw.falcon);
 
1149         base_off = par->hw.falcon.vid_control & VCO_SHORTOFFS ? 64 : 128;
 
1151         /* Offsets depend on video mode */
 
1152         /* Offsets are in clock cycles, divide by prescale to
 
1153          * calculate hd[be]-registers
 
1155         if (par->hw.falcon.f_shift & 0x100) {
 
1158                 hdb_off = (base_off + 16 * plen) + prescale;
 
1162                 hde_off = ((128 / bpp + 2) * plen);
 
1163                 if (par->hw.falcon.ste_mode)
 
1164                         hdb_off = (64 + base_off + (128 / bpp + 2) * plen) + prescale;
 
1166                         hdb_off = (base_off + (128 / bpp + 18) * plen) + prescale;
 
1169         gstart = (prescale/2 + plen * left_margin) / prescale;
 
1170         /* gend1 is for hde (gend-gstart multiple of align), shifter's xres */
 
1171         gend1 = gstart + ((xres + align-1) / align)*align * plen / prescale;
 
1172         /* gend2 is for hbb, visible xres (rest to gend1 is cut off by hblank) */
 
1173         gend2 = gstart + xres * plen / prescale;
 
1174         par->HHT = plen * (left_margin + xres + right_margin) /
 
1176 /*      par->HHT = (gend2 + plen * right_margin / prescale) / 2 - 2;*/
 
1178         par->HDB = gstart - hdb_off/prescale;
 
1180         if (par->HDB < 0) par->HDB += par->HHT + 2 + 0x200;
 
1181         par->HDE = gend1 - par->HHT - 2 - hde_off/prescale;
 
1182         par->HBB = gend2 - par->HHT - 2;
 
1184         /* One more Videl constraint: data fetch of two lines must not overlap */
 
1185         if ((par->HDB & 0x200)  &&  (par->HDB & ~0x200) - par->HDE <= 5) {
 
1186                 /* if this happens increase margins, decrease hfreq. */
 
1189         if (hde_off % prescale)
 
1190                 par->HBB++;             /* compensate for non matching hde and hbb */
 
1191         par->HSS = par->HHT + 2 - plen * hsync_len / prescale;
 
1192         if (par->HSS < par->HBB)
 
1193                 par->HSS = par->HBB;
 
1196         /*  check hor. frequency */
 
1197         hfreq = pclock->f / ((par->HHT+2)*prescale*2);
 
1198         if (hfreq > fb_info.monspecs.hfmax && mon_type!=F_MON_VGA) {
 
1199                 /* ++guenther:   ^^^^^^^^^^^^^^^^^^^ can't remember why I did this */
 
1200                 /* Too high -> enlarge margin */
 
1205         if (hfreq > fb_info.monspecs.hfmax || hfreq < fb_info.monspecs.hfmin)
 
1209         /* All Vxx must be odd in non-interlace, since frame starts in the middle
 
1210          * of the first displayed line!
 
1211          * One frame consists of VFT+1 half lines. VFT+1 must be even in
 
1212          * non-interlace, odd in interlace mode for synchronisation.
 
1213          * Vxx-registers are 11 bit wide
 
1215         par->VBE = (upper_margin * 2 + 1); /* must begin on odd halfline */
 
1216         par->VDB = par->VBE;
 
1218         if (!interlace) par->VDE <<= 1;
 
1219         if (doubleline) par->VDE <<= 1;  /* VDE now half lines per (half-)frame */
 
1220         par->VDE += par->VDB;
 
1221         par->VBB = par->VDE;
 
1222         par->VFT = par->VBB + (lower_margin * 2 - 1) - 1;
 
1223         par->VSS = par->VFT+1 - (vsync_len * 2 - 1);
 
1224         /* vbb,vss,vft must be even in interlace mode */
 
1231         /* V-frequency check, hope I didn't create any loop here. */
 
1232         /* Interlace and doubleline are mutually exclusive. */
 
1233         vfreq = (hfreq * 2) / (par->VFT + 1);
 
1234         if      (vfreq > fb_info.monspecs.vfmax && !doubleline && !interlace) {
 
1235                 /* Too high -> try again with doubleline */
 
1239         else if (vfreq < fb_info.monspecs.vfmin && !interlace && !doubleline) {
 
1240                 /* Too low -> try again with interlace */
 
1244         else if (vfreq < fb_info.monspecs.vfmin && doubleline) {
 
1245                 /* Doubleline too low -> clear doubleline and enlarge margins */
 
1249                      (hfreq*2)/(par->VFT+1+4*lines-2*yres)>fb_info.monspecs.vfmax;
 
1252                 upper_margin += lines;
 
1253                 lower_margin += lines;
 
1256         else if (vfreq > fb_info.monspecs.vfmax && doubleline) {
 
1257                 /* Doubleline too high -> enlarge margins */
 
1260                      (hfreq*2)/(par->VFT+1+4*lines)>fb_info.monspecs.vfmax;
 
1263                 upper_margin += lines;
 
1264                 lower_margin += lines;
 
1267         else if (vfreq > fb_info.monspecs.vfmax && interlace) {
 
1268                 /* Interlace, too high -> enlarge margins */
 
1271                      (hfreq*2)/(par->VFT+1+4*lines)>fb_info.monspecs.vfmax;
 
1274                 upper_margin += lines;
 
1275                 lower_margin += lines;
 
1278         else if (vfreq < fb_info.monspecs.vfmin ||
 
1279                  vfreq > fb_info.monspecs.vfmax)
 
1283         linelen = xres_virtual * bpp / 8;
 
1284         if (yres_virtual * linelen > screen_len && screen_len)
 
1286         if (yres * linelen > screen_len && screen_len)
 
1288         if (var->yoffset + yres > yres_virtual && yres_virtual)
 
1290         par->yres_virtual = yres_virtual;
 
1291         par->screen_base = screen_base + var->yoffset * linelen;
 
1292         par->hw.falcon.xoffset = 0;
 
1297 static int falcon_encode_var( struct fb_var_screeninfo *var,
 
1298                                                           struct atafb_par *par )
 
1300 /* !!! only for VGA !!! */
 
1303         int hdb_off, hde_off, base_off;
 
1304         struct falcon_hw *hw = &par->hw.falcon;
 
1306         memset(var, 0, sizeof(struct fb_var_screeninfo));
 
1307         /* possible frequencies: 25.175 or 32MHz */
 
1308         var->pixclock = hw->sync & 0x1 ? fext.t :
 
1309                         hw->vid_control & VCO_CLOCK25 ? f25.t : f32.t;
 
1315         if (hw->vid_control & VCO_HSYPOS)
 
1316                 var->sync |= FB_SYNC_HOR_HIGH_ACT;
 
1317         if (hw->vid_control & VCO_VSYPOS)
 
1318                 var->sync |= FB_SYNC_VERT_HIGH_ACT;
 
1320         var->vmode = FB_VMODE_NONINTERLACED;
 
1321         if (hw->vid_mode & VMO_INTER)
 
1322                 var->vmode |= FB_VMODE_INTERLACED;
 
1323         if (hw->vid_mode & VMO_DOUBLE)
 
1324                 var->vmode |= FB_VMODE_DOUBLE;
 
1326         /* visible y resolution:
 
1327          * Graphics display starts at line VDB and ends at line
 
1328          * VDE. If interlace mode off unit of VC-registers is
 
1329          * half lines, else lines.
 
1331         var->yres = hw->vde - hw->vdb;
 
1332         if (!(var->vmode & FB_VMODE_INTERLACED))
 
1334         if (var->vmode & FB_VMODE_DOUBLE)
 
1337         /* to get bpp, we must examine f_shift and st_shift.
 
1338          * f_shift is valid if any of bits no. 10, 8 or 4
 
1339          * is set. Priority in f_shift is: 10 ">" 8 ">" 4, i.e.
 
1340          * if bit 10 set then bit 8 and bit 4 don't care...
 
1341          * If all these bits are 0 get display depth from st_shift
 
1342          * (as for ST and STE)
 
1344         if (hw->f_shift & 0x400)                /* 2 colors */
 
1345                 var->bits_per_pixel = 1;
 
1346         else if (hw->f_shift & 0x100)   /* hicolor */
 
1347                 var->bits_per_pixel = 16;
 
1348         else if (hw->f_shift & 0x010)   /* 8 bitplanes */
 
1349                 var->bits_per_pixel = 8;
 
1350         else if (hw->st_shift == 0)
 
1351                 var->bits_per_pixel = 4;
 
1352         else if (hw->st_shift == 0x100)
 
1353                 var->bits_per_pixel = 2;
 
1354         else /* if (hw->st_shift == 0x200) */
 
1355                 var->bits_per_pixel = 1;
 
1357         var->xres = hw->line_width * 16 / var->bits_per_pixel;
 
1358         var->xres_virtual = var->xres + hw->line_offset * 16 / var->bits_per_pixel;
 
1360                 var->xres_virtual += 16;
 
1362         if (var->bits_per_pixel == 16) {
 
1365                 var->red.msb_right=0;
 
1366                 var->green.offset=5;
 
1367                 var->green.length=6;
 
1368                 var->green.msb_right=0;
 
1371                 var->blue.msb_right=0;
 
1375                 var->red.length = hw->ste_mode ? 4 : 6;
 
1376                 var->red.msb_right=0;
 
1378                 var->blue=var->green=var->red;
 
1380         var->transp.offset=0;
 
1381         var->transp.length=0;
 
1382         var->transp.msb_right=0;
 
1384         linelen = var->xres_virtual * var->bits_per_pixel / 8;
 
1386                 if (par->yres_virtual)
 
1387                         var->yres_virtual = par->yres_virtual;
 
1389                         /* yres_virtual==0 means use maximum */
 
1390                         var->yres_virtual = screen_len / linelen;
 
1394                         var->yres_virtual = 2 * var->yres;
 
1396                         var->yres_virtual=var->yres+hwscroll * 16;
 
1398         var->xoffset=0; /* TODO change this */
 
1401         prescale = hxx_prescale(hw);
 
1402         plen = 4 >> (hw->vid_mode >> 2 & 0x3);
 
1403         base_off = hw->vid_control & VCO_SHORTOFFS ? 64 : 128;
 
1404         if (hw->f_shift & 0x100) {
 
1406                 hdb_off = (base_off + 16 * plen) + prescale;
 
1409                 hde_off = ((128 / var->bits_per_pixel + 2) * plen);
 
1411                         hdb_off = (64 + base_off + (128 / var->bits_per_pixel + 2) * plen)
 
1414                         hdb_off = (base_off + (128 / var->bits_per_pixel + 18) * plen)
 
1418         /* Right margin includes hsync */
 
1419         var->left_margin = hdb_off + prescale * ((hw->hdb & 0x1ff) -
 
1420                                            (hw->hdb & 0x200 ? 2+hw->hht : 0));
 
1421         if (hw->ste_mode || mon_type!=F_MON_VGA)
 
1422                 var->right_margin = prescale * (hw->hht + 2 - hw->hde) - hde_off;
 
1424                 /* can't use this in ste_mode, because hbb is +1 off */
 
1425                 var->right_margin = prescale * (hw->hht + 2 - hw->hbb);
 
1426         var->hsync_len = prescale * (hw->hht + 2 - hw->hss);
 
1428         /* Lower margin includes vsync */
 
1429         var->upper_margin = hw->vdb / 2 ;  /* round down to full lines */
 
1430         var->lower_margin = (hw->vft+1 - hw->vde + 1) / 2; /* round up */
 
1431         var->vsync_len    = (hw->vft+1 - hw->vss + 1) / 2; /* round up */
 
1432         if (var->vmode & FB_VMODE_INTERLACED) {
 
1433                 var->upper_margin *= 2;
 
1434                 var->lower_margin *= 2;
 
1435                 var->vsync_len *= 2;
 
1437         else if (var->vmode & FB_VMODE_DOUBLE) {
 
1438                 var->upper_margin = (var->upper_margin + 1) / 2;
 
1439                 var->lower_margin = (var->lower_margin + 1) / 2;
 
1440                 var->vsync_len = (var->vsync_len + 1) / 2;
 
1443         var->pixclock *= plen;
 
1444         var->left_margin /= plen;
 
1445         var->right_margin /= plen;
 
1446         var->hsync_len /= plen;
 
1448         var->right_margin -= var->hsync_len;
 
1449         var->lower_margin -= var->vsync_len;
 
1452                 var->yoffset=(par->screen_base - screen_base)/linelen;
 
1455         var->nonstd=0;  /* what is this for? */
 
1461 static int f_change_mode = 0;
 
1462 static struct falcon_hw f_new_mode;
 
1463 static int f_pan_display = 0;
 
1465 static void falcon_get_par( struct atafb_par *par )
 
1468         struct falcon_hw *hw = &par->hw.falcon;
 
1470         hw->line_width = shifter_f030.scn_width;
 
1471         hw->line_offset = shifter_f030.off_next;
 
1472         hw->st_shift = videl.st_shift & 0x300;
 
1473         hw->f_shift = videl.f_shift;
 
1474         hw->vid_control = videl.control;
 
1475         hw->vid_mode = videl.mode;
 
1476         hw->sync = shifter.syncmode & 0x1;
 
1477         hw->xoffset = videl.xoffset & 0xf;
 
1478         hw->hht = videl.hht;
 
1479         hw->hbb = videl.hbb;
 
1480         hw->hbe = videl.hbe;
 
1481         hw->hdb = videl.hdb;
 
1482         hw->hde = videl.hde;
 
1483         hw->hss = videl.hss;
 
1484         hw->vft = videl.vft;
 
1485         hw->vbb = videl.vbb;
 
1486         hw->vbe = videl.vbe;
 
1487         hw->vdb = videl.vdb;
 
1488         hw->vde = videl.vde;
 
1489         hw->vss = videl.vss;
 
1491         addr = (shifter.bas_hi & 0xff) << 16 |
 
1492                (shifter.bas_md & 0xff) << 8  |
 
1493                (shifter.bas_lo & 0xff);
 
1494         par->screen_base = phys_to_virt(addr);
 
1496         /* derived parameters */
 
1497         hw->ste_mode = (hw->f_shift & 0x510)==0 && hw->st_shift==0x100;
 
1498         hw->mono = (hw->f_shift & 0x400) ||
 
1499                    ((hw->f_shift & 0x510)==0 && hw->st_shift==0x200);
 
1502 static void falcon_set_par( struct atafb_par *par )
 
1506         /* only set screen_base if really necessary */
 
1507         if (current_par.screen_base != par->screen_base)
 
1508                 fbhw->set_screen_base(par->screen_base);
 
1510         /* Don't touch any other registers if we keep the default resolution */
 
1514         /* Tell vbl-handler to change video mode.
 
1515          * We change modes only on next VBL, to avoid desynchronisation
 
1516          * (a shift to the right and wrap around by a random number of pixels
 
1517          * in all monochrome modes).
 
1518          * This seems to work on my Falcon.
 
1520         f_new_mode = par->hw.falcon;
 
1525 static irqreturn_t falcon_vbl_switcher( int irq, void *dummy, struct pt_regs *fp )
 
1527         struct falcon_hw *hw = &f_new_mode;
 
1529         if (f_change_mode) {
 
1532                 if (hw->sync & 0x1) {
 
1533                         /* Enable external pixelclock. This code only for ScreenWonder */
 
1534                         *(volatile unsigned short*)0xffff9202 = 0xffbf;
 
1537                         /* Turn off external clocks. Read sets all output bits to 1. */
 
1538                         *(volatile unsigned short*)0xffff9202;
 
1540                 shifter.syncmode = hw->sync;
 
1542                 videl.hht = hw->hht;
 
1543                 videl.hbb = hw->hbb;
 
1544                 videl.hbe = hw->hbe;
 
1545                 videl.hdb = hw->hdb;
 
1546                 videl.hde = hw->hde;
 
1547                 videl.hss = hw->hss;
 
1548                 videl.vft = hw->vft;
 
1549                 videl.vbb = hw->vbb;
 
1550                 videl.vbe = hw->vbe;
 
1551                 videl.vdb = hw->vdb;
 
1552                 videl.vde = hw->vde;
 
1553                 videl.vss = hw->vss;
 
1555                 videl.f_shift = 0; /* write enables Falcon palette, 0: 4 planes */
 
1557                         videl.st_shift = hw->st_shift; /* write enables STE palette */
 
1561                          * set st_shift 0, so we can tell the screen-depth if f_shift==0.
 
1562                          * Writing 0 to f_shift enables 4 plane Falcon mode but
 
1563                          * doesn't set st_shift. st_shift!=0 (!=4planes) is impossible
 
1564                          * with Falcon palette.
 
1567                         /* now back to Falcon palette mode */
 
1568                         videl.f_shift = hw->f_shift;
 
1570                 /* writing to st_shift changed scn_width and vid_mode */
 
1571                 videl.xoffset = hw->xoffset;
 
1572                 shifter_f030.scn_width = hw->line_width;
 
1573                 shifter_f030.off_next = hw->line_offset;
 
1574                 videl.control = hw->vid_control;
 
1575                 videl.mode = hw->vid_mode;
 
1577         if (f_pan_display) {
 
1579                 videl.xoffset = current_par.hw.falcon.xoffset;
 
1580                 shifter_f030.off_next = current_par.hw.falcon.line_offset;
 
1586 static int falcon_pan_display( struct fb_var_screeninfo *var,
 
1587                                                            struct atafb_par *par )
 
1590         int bpp = fb_display[fb_info.currcon].var.bits_per_pixel;
 
1593                 var->xoffset = up(var->xoffset, 32);
 
1595                 par->hw.falcon.xoffset = var->xoffset & 15;
 
1597                 par->hw.falcon.xoffset = 0;
 
1598                 var->xoffset = up(var->xoffset, 2);
 
1600         par->hw.falcon.line_offset = bpp *
 
1601                 (fb_display[fb_info.currcon].var.xres_virtual - fb_display[fb_info.currcon].var.xres) / 16;
 
1602         if (par->hw.falcon.xoffset)
 
1603                 par->hw.falcon.line_offset -= bpp;
 
1604         xoffset = var->xoffset - par->hw.falcon.xoffset;
 
1606         par->screen_base = screen_base +
 
1607                 (var->yoffset * fb_display[fb_info.currcon].var.xres_virtual + xoffset) * bpp / 8;
 
1608         if (fbhw->set_screen_base)
 
1609                 fbhw->set_screen_base (par->screen_base);
 
1611                 return -EINVAL; /* shouldn't happen */
 
1617 static int falcon_getcolreg( unsigned regno, unsigned *red,
 
1618                                  unsigned *green, unsigned *blue,
 
1619                                  unsigned *transp, struct fb_info *info )
 
1620 {       unsigned long col;
 
1624         /* This works in STE-mode (with 4bit/color) since f030_col-registers
 
1625          * hold up to 6bit/color.
 
1626          * Even with hicolor r/g/b=5/6/5 bit!
 
1628         col = f030_col[regno];
 
1629         *red = (col >> 16) & 0xff00;
 
1630         *green = (col >> 8) & 0xff00;
 
1631         *blue = (col << 8) & 0xff00;
 
1637 static int falcon_setcolreg( unsigned regno, unsigned red,
 
1638                                                          unsigned green, unsigned blue,
 
1639                                                          unsigned transp, struct fb_info *info )
 
1643         f030_col[regno] = (((red & 0xfc00) << 16) |
 
1644                            ((green & 0xfc00) << 8) |
 
1645                            ((blue & 0xfc00) >> 8));
 
1647                 shifter_tt.color_reg[regno] =
 
1648                         (((red & 0xe000) >> 13) | ((red & 0x1000) >> 12) << 8) |
 
1649                         (((green & 0xe000) >> 13) | ((green & 0x1000) >> 12) << 4) |
 
1650                         ((blue & 0xe000) >> 13) | ((blue & 0x1000) >> 12);
 
1651 #ifdef FBCON_HAS_CFB16
 
1652                 fbcon_cfb16_cmap[regno] = ((red & 0xf800) |
 
1653                                            ((green & 0xfc00) >> 5) |
 
1654                                            ((blue & 0xf800) >> 11));
 
1661 static int falcon_blank( int blank_mode )
 
1663 /* ++guenther: we can switch off graphics by changing VDB and VDE,
 
1664  * so VIDEL doesn't hog the bus while saving.
 
1665  * (this may affect usleep()).
 
1667         int vdb, vss, hbe, hss;
 
1669         if (mon_type == F_MON_SM)       /* this doesn't work on SM124 */
 
1672         vdb = current_par.VDB;
 
1673         vss = current_par.VSS;
 
1674         hbe = current_par.HBE;
 
1675         hss = current_par.HSS;
 
1677         if (blank_mode >= 1) {
 
1678                 /* disable graphics output (this speeds up the CPU) ... */
 
1679                 vdb = current_par.VFT + 1;
 
1680                 /* ... and blank all lines */
 
1681                 hbe = current_par.HHT + 2;
 
1683         /* use VESA suspend modes on VGA monitors */
 
1684         if (mon_type == F_MON_VGA) {
 
1685                 if (blank_mode == 2 || blank_mode == 4)
 
1686                         vss = current_par.VFT + 1;
 
1687                 if (blank_mode == 3 || blank_mode == 4)
 
1688                         hss = current_par.HHT + 2;
 
1700 static int falcon_detect( void )
 
1702         struct atafb_par par;
 
1705         /* Determine connected monitor and set monitor parameters */
 
1706         fhw = *(unsigned char*)0xffff8006;
 
1707         mon_type = fhw >> 6 & 0x3;
 
1708         /* bit 1 of fhw: 1=32 bit ram bus, 0=16 bit */
 
1709         f030_bus_width = fhw << 6 & 0x80;
 
1712                 fb_info.monspecs.vfmin = 70;
 
1713                 fb_info.monspecs.vfmax = 72;
 
1714                 fb_info.monspecs.hfmin = 35713;
 
1715                 fb_info.monspecs.hfmax = 35715;
 
1720                 fb_info.monspecs.vfmin = 49; /* not 50, since TOS defaults to 49.9x Hz */
 
1721                 fb_info.monspecs.vfmax = 60;
 
1722                 fb_info.monspecs.hfmin = 15620;
 
1723                 fb_info.monspecs.hfmax = 15755;
 
1726         /* initialize hsync-len */
 
1727         f25.hsync = h_syncs[mon_type] / f25.t;
 
1728         f32.hsync = h_syncs[mon_type] / f32.t;
 
1730                 fext.hsync = h_syncs[mon_type] / fext.t;
 
1732         falcon_get_par(&par);
 
1733         falcon_encode_var(&atafb_predefined[0], &par);
 
1735         /* Detected mode is always the "autodetect" slot */
 
1739 #endif /* ATAFB_FALCON */
 
1741 /* ------------------- ST(E) specific functions ---------------------- */
 
1745 static int stste_encode_fix( struct fb_fix_screeninfo *fix,
 
1746                                                          struct atafb_par *par )
 
1751         strcpy(fix->id,"Atari Builtin");
 
1752         fix->smem_start = (unsigned long)real_screen_base;
 
1753         fix->smem_len = screen_len;
 
1754         fix->type = FB_TYPE_INTERLEAVED_PLANES;
 
1756         fix->visual = FB_VISUAL_PSEUDOCOLOR;
 
1757         mode = par->hw.st.mode & 3;
 
1758         if (mode == ST_HIGH) {
 
1759                 fix->type = FB_TYPE_PACKED_PIXELS;
 
1761                 fix->visual = FB_VISUAL_MONO10;
 
1763         if (ATARIHW_PRESENT(EXTD_SHIFTER)) {
 
1771         fix->line_length = 0;
 
1772         fix->accel = FB_ACCEL_ATARIBLITT;
 
1777 static int stste_decode_var( struct fb_var_screeninfo *var,
 
1778                                                   struct atafb_par *par )
 
1782         int bpp=var->bits_per_pixel;
 
1784         int yres_virtual = var->yres_virtual;
 
1787                 if (bpp > 1 || xres > sttt_xres || yres > st_yres)
 
1789                 par->hw.st.mode=ST_HIGH;
 
1794                 if (bpp > 4 || xres > sttt_xres || yres > st_yres)
 
1797                         if (xres > sttt_xres/2 || yres > st_yres/2)
 
1799                         par->hw.st.mode=ST_LOW;
 
1805                         if (xres > sttt_xres || yres > st_yres/2)
 
1807                         par->hw.st.mode=ST_MID;
 
1815         if (yres_virtual <= 0)
 
1817         else if (yres_virtual < yres)
 
1818                 yres_virtual = yres;
 
1819         if (var->sync & FB_SYNC_EXT)
 
1820                 par->hw.st.sync=(par->hw.st.sync & ~1) | 1;
 
1822                 par->hw.st.sync=(par->hw.st.sync & ~1);
 
1824         if (yres_virtual * linelen > screen_len && screen_len)
 
1826         if (yres * linelen > screen_len && screen_len)
 
1828         if (var->yoffset + yres > yres_virtual && yres_virtual)
 
1830         par->yres_virtual = yres_virtual;
 
1831         par->screen_base=screen_base+ var->yoffset*linelen;
 
1835 static int stste_encode_var( struct fb_var_screeninfo *var,
 
1836                                                   struct atafb_par *par )
 
1839         memset(var, 0, sizeof(struct fb_var_screeninfo));
 
1841         var->red.length = ATARIHW_PRESENT(EXTD_SHIFTER) ? 4 : 3;
 
1842         var->red.msb_right=0;
 
1845         var->pixclock=31041;
 
1846         var->left_margin=120;           /* these are incorrect */
 
1847         var->right_margin=100;
 
1848         var->upper_margin=8;
 
1849         var->lower_margin=16;
 
1856         if (!(par->hw.st.sync & 1))
 
1859                 var->sync=FB_SYNC_EXT;
 
1861         switch (par->hw.st.mode & 3) {
 
1863                 var->xres=sttt_xres/2;
 
1864                 var->yres=st_yres/2;
 
1865                 var->bits_per_pixel=4;
 
1868                 var->xres=sttt_xres;
 
1869                 var->yres=st_yres/2;
 
1870                 var->bits_per_pixel=2;
 
1873                 var->xres=sttt_xres;
 
1875                 var->bits_per_pixel=1;
 
1878         var->blue=var->green=var->red;
 
1879         var->transp.offset=0;
 
1880         var->transp.length=0;
 
1881         var->transp.msb_right=0;
 
1882         var->xres_virtual=sttt_xres_virtual;
 
1883         linelen=var->xres_virtual * var->bits_per_pixel / 8;
 
1884         ovsc_addlen=linelen*(sttt_yres_virtual - st_yres);
 
1887                 var->yres_virtual=var->yres;
 
1888         else if (screen_len) {
 
1889                 if (par->yres_virtual)
 
1890                         var->yres_virtual = par->yres_virtual;
 
1892                         /* yres_virtual==0 means use maximum */
 
1893                         var->yres_virtual = screen_len / linelen;
 
1897                         var->yres_virtual = 2 * var->yres;
 
1899                         var->yres_virtual=var->yres+hwscroll * 16;
 
1903                 var->yoffset=(par->screen_base - screen_base)/linelen;
 
1908         var->vmode=FB_VMODE_NONINTERLACED;
 
1913 static void stste_get_par( struct atafb_par *par )
 
1916         par->hw.st.mode=shifter_tt.st_shiftmode;
 
1917         par->hw.st.sync=shifter.syncmode;
 
1918         addr = ((shifter.bas_hi & 0xff) << 16) |
 
1919                ((shifter.bas_md & 0xff) << 8);
 
1920         if (ATARIHW_PRESENT(EXTD_SHIFTER))
 
1921                 addr |= (shifter.bas_lo & 0xff);
 
1922         par->screen_base = phys_to_virt(addr);
 
1925 static void stste_set_par( struct atafb_par *par )
 
1927         shifter_tt.st_shiftmode=par->hw.st.mode;
 
1928         shifter.syncmode=par->hw.st.sync;
 
1929         /* only set screen_base if really necessary */
 
1930         if (current_par.screen_base != par->screen_base)
 
1931                 fbhw->set_screen_base(par->screen_base);
 
1935 static int stste_getcolreg(unsigned regno, unsigned *red,
 
1936                            unsigned *green, unsigned *blue,
 
1937                            unsigned *transp, struct fb_info *info)
 
1943         col = shifter_tt.color_reg[regno];
 
1944         if (ATARIHW_PRESENT(EXTD_SHIFTER)) {
 
1945                 t = ((col >> 7) & 0xe) | ((col >> 11) & 1);
 
1947                 *red = t | (t << 8);
 
1948                 t = ((col >> 3) & 0xe) | ((col >> 7) & 1);
 
1950                 *green = t | (t << 8);
 
1951                 t = ((col << 1) & 0xe) | ((col >> 3) & 1);
 
1953                 *blue = t | (t << 8);
 
1956                 t = (col >> 7) & 0xe;
 
1958                 *red = t | (t << 8);
 
1959                 t = (col >> 3) & 0xe;
 
1961                 *green = t | (t << 8);
 
1962                 t = (col << 1) & 0xe;
 
1964                 *blue = t | (t << 8);
 
1971 static int stste_setcolreg(unsigned regno, unsigned red,
 
1972                            unsigned green, unsigned blue,
 
1973                            unsigned transp, struct fb_info *info)
 
1980         if (ATARIHW_PRESENT(EXTD_SHIFTER))
 
1981                 shifter_tt.color_reg[regno] =
 
1982                         (((red & 0xe) >> 1) | ((red & 1) << 3) << 8) |
 
1983                         (((green & 0xe) >> 1) | ((green & 1) << 3) << 4) |
 
1984                         ((blue & 0xe) >> 1) | ((blue & 1) << 3);
 
1986                 shifter_tt.color_reg[regno] =
 
1987                         ((red & 0xe) << 7) |
 
1988                         ((green & 0xe) << 3) |
 
1989                         ((blue & 0xe) >> 1);
 
1994 static int stste_detect( void )
 
1996 {       struct atafb_par par;
 
1998         /* Determine the connected monitor: The DMA sound must be
 
1999          * disabled before reading the MFP GPIP, because the Sound
 
2000          * Done Signal and the Monochrome Detect are XORed together!
 
2002         if (ATARIHW_PRESENT(PCM_8BIT)) {
 
2003                 tt_dmasnd.ctrl = DMASND_CTRL_OFF;
 
2004                 udelay(20);     /* wait a while for things to settle down */
 
2006         mono_moni = (mfp.par_dt_reg & 0x80) == 0;
 
2008         stste_get_par(&par);
 
2009         stste_encode_var(&atafb_predefined[0], &par);
 
2011         if (!ATARIHW_PRESENT(EXTD_SHIFTER))
 
2016 static void stste_set_screen_base(void *s_base)
 
2019         addr= virt_to_phys(s_base);
 
2020         /* Setup Screen Memory */
 
2021         shifter.bas_hi=(unsigned char) ((addr & 0xff0000) >> 16);
 
2022         shifter.bas_md=(unsigned char) ((addr & 0x00ff00) >> 8);
 
2023         if (ATARIHW_PRESENT(EXTD_SHIFTER))
 
2024                 shifter.bas_lo=(unsigned char)  (addr & 0x0000ff);
 
2027 #endif /* ATAFB_STE */
 
2029 /* Switching the screen size should be done during vsync, otherwise
 
2030  * the margins may get messed up. This is a well known problem of
 
2031  * the ST's video system.
 
2033  * Unfortunately there is hardly any way to find the vsync, as the
 
2034  * vertical blank interrupt is no longer in time on machines with
 
2035  * overscan type modifications.
 
2037  * We can, however, use Timer B to safely detect the black shoulder,
 
2038  * but then we've got to guess an appropriate delay to find the vsync.
 
2039  * This might not work on every machine.
 
2041  * martin_rogge @ ki.maus.de, 8th Aug 1995
 
2044 #define LINE_DELAY  (mono_moni ? 30 : 70)
 
2045 #define SYNC_DELAY  (mono_moni ? 1500 : 2000)
 
2047 /* SWITCH_ACIA may be used for Falcon (ScreenBlaster III internal!) */
 
2048 static void st_ovsc_switch(void)
 
2050     unsigned long flags;
 
2051     register unsigned char old, new;
 
2053     if (!(atari_switches & ATARI_SWITCH_OVSC_MASK))
 
2055     local_irq_save(flags);
 
2057     mfp.tim_ct_b = 0x10;
 
2058     mfp.active_edge |= 8;
 
2060     mfp.tim_dt_b = 0xf0;
 
2062     while (mfp.tim_dt_b > 1)    /* TOS does it this way, don't ask why */
 
2069     } while (old != new);
 
2070     mfp.tim_ct_b = 0x10;
 
2073     if (atari_switches & ATARI_SWITCH_OVSC_IKBD)
 
2074         acia.key_ctrl = ACIA_DIV64 | ACIA_D8N1S | ACIA_RHTID | ACIA_RIE;
 
2075     if (atari_switches & ATARI_SWITCH_OVSC_MIDI)
 
2076         acia.mid_ctrl = ACIA_DIV16 | ACIA_D8N1S | ACIA_RHTID;
 
2077     if (atari_switches & (ATARI_SWITCH_OVSC_SND6|ATARI_SWITCH_OVSC_SND7)) {
 
2078         sound_ym.rd_data_reg_sel = 14;
 
2079         sound_ym.wd_data = sound_ym.rd_data_reg_sel |
 
2080                            ((atari_switches&ATARI_SWITCH_OVSC_SND6) ? 0x40:0) |
 
2081                            ((atari_switches&ATARI_SWITCH_OVSC_SND7) ? 0x80:0);
 
2083     local_irq_restore(flags);
 
2086 /* ------------------- External Video ---------------------- */
 
2090 static int ext_encode_fix( struct fb_fix_screeninfo *fix,
 
2091                                                    struct atafb_par *par )
 
2094         strcpy(fix->id,"Unknown Extern");
 
2095         fix->smem_start = (unsigned long)external_addr;
 
2096         fix->smem_len = PAGE_ALIGN(external_len);
 
2097         if (external_depth == 1) {
 
2098                 fix->type = FB_TYPE_PACKED_PIXELS;
 
2099                 /* The letters 'n' and 'i' in the "atavideo=external:" stand
 
2100                  * for "normal" and "inverted", rsp., in the monochrome case */
 
2102                         (external_pmode == FB_TYPE_INTERLEAVED_PLANES ||
 
2103                          external_pmode == FB_TYPE_PACKED_PIXELS) ?
 
2108                 /* Use STATIC if we don't know how to access color registers */
 
2109                 int visual = external_vgaiobase ?
 
2110                                          FB_VISUAL_PSEUDOCOLOR :
 
2111                                          FB_VISUAL_STATIC_PSEUDOCOLOR;
 
2112                 switch (external_pmode) {
 
2113                     case -1:              /* truecolor */
 
2114                         fix->type=FB_TYPE_PACKED_PIXELS;
 
2115                         fix->visual=FB_VISUAL_TRUECOLOR;
 
2117                     case FB_TYPE_PACKED_PIXELS:
 
2118                         fix->type=FB_TYPE_PACKED_PIXELS;
 
2121                     case FB_TYPE_PLANES:
 
2122                         fix->type=FB_TYPE_PLANES;
 
2125                     case FB_TYPE_INTERLEAVED_PLANES:
 
2126                         fix->type=FB_TYPE_INTERLEAVED_PLANES;
 
2135         fix->line_length = 0;
 
2140 static int ext_decode_var( struct fb_var_screeninfo *var,
 
2141                                                    struct atafb_par *par )
 
2143         struct fb_var_screeninfo *myvar = &atafb_predefined[0];
 
2145         if (var->bits_per_pixel > myvar->bits_per_pixel ||
 
2146                 var->xres > myvar->xres ||
 
2147                 var->xres_virtual > myvar->xres_virtual ||
 
2148                 var->yres > myvar->yres ||
 
2156 static int ext_encode_var( struct fb_var_screeninfo *var,
 
2157                                                    struct atafb_par *par )
 
2159         memset(var, 0, sizeof(struct fb_var_screeninfo));
 
2161         var->red.length=(external_pmode == -1) ? external_depth/3 : 
 
2162                         (external_vgaiobase ? external_bitspercol : 0);
 
2163         var->red.msb_right=0;
 
2166         var->pixclock=31041;
 
2167         var->left_margin=120;           /* these are surely incorrect   */
 
2168         var->right_margin=100;
 
2169         var->upper_margin=8;
 
2170         var->lower_margin=16;
 
2179         var->xres = external_xres;
 
2180         var->yres = external_yres;
 
2181         var->xres_virtual = external_xres_virtual;
 
2182         var->bits_per_pixel = external_depth;
 
2184         var->blue=var->green=var->red;
 
2185         var->transp.offset=0;
 
2186         var->transp.length=0;
 
2187         var->transp.msb_right=0;
 
2188         var->yres_virtual=var->yres;
 
2193         var->vmode=FB_VMODE_NONINTERLACED;
 
2198 static void ext_get_par( struct atafb_par *par )
 
2200         par->screen_base = external_addr;
 
2203 static void ext_set_par( struct atafb_par *par )
 
2207 #define OUTB(port,val) \
 
2208         *((unsigned volatile char *) ((port)+external_vgaiobase))=(val)
 
2210         (*((unsigned volatile char *) ((port)+external_vgaiobase)))
 
2213                 unsigned char tmp=INB(0x3da);   \
 
2217 static int ext_getcolreg( unsigned regno, unsigned *red,
 
2218                                                   unsigned *green, unsigned *blue,
 
2219                                                   unsigned *transp, struct fb_info *info )
 
2221         if (! external_vgaiobase)
 
2224             *red   = ext_color[regno].red;
 
2225             *green = ext_color[regno].green;
 
2226             *blue  = ext_color[regno].blue;
 
2231 static int ext_setcolreg( unsigned regno, unsigned red,
 
2232                                                   unsigned green, unsigned blue,
 
2233                                                   unsigned transp, struct fb_info *info )
 
2235 {       unsigned char colmask = (1 << external_bitspercol) - 1;
 
2237         if (! external_vgaiobase)
 
2240         ext_color[regno].red = red;
 
2241         ext_color[regno].green = green;
 
2242         ext_color[regno].blue = blue;
 
2244         switch (external_card_type) {
 
2248             OUTB(0x3c9, red & colmask);
 
2250             OUTB(0x3c9, green & colmask);
 
2252             OUTB(0x3c9, blue & colmask);
 
2257             OUTB((MV300_reg[regno] << 2)+1, red);
 
2258             OUTB((MV300_reg[regno] << 2)+1, green);
 
2259             OUTB((MV300_reg[regno] << 2)+1, blue);
 
2268 static int ext_detect( void )
 
2271         struct fb_var_screeninfo *myvar = &atafb_predefined[0];
 
2272         struct atafb_par dummy_par;
 
2274         myvar->xres = external_xres;
 
2275         myvar->xres_virtual = external_xres_virtual;
 
2276         myvar->yres = external_yres;
 
2277         myvar->bits_per_pixel = external_depth;
 
2278         ext_encode_var(myvar, &dummy_par);
 
2282 #endif /* ATAFB_EXT */
 
2284 /* ------ This is the same for most hardware types -------- */
 
2286 static void set_screen_base(void *s_base)
 
2289         addr= virt_to_phys(s_base);
 
2290         /* Setup Screen Memory */
 
2291         shifter.bas_hi=(unsigned char) ((addr & 0xff0000) >> 16);
 
2292         shifter.bas_md=(unsigned char) ((addr & 0x00ff00) >> 8);
 
2293         shifter.bas_lo=(unsigned char)  (addr & 0x0000ff);
 
2297 static int pan_display( struct fb_var_screeninfo *var,
 
2298                         struct atafb_par *par )
 
2300         if (!fbhw->set_screen_base ||
 
2301                 (!ATARIHW_PRESENT(EXTD_SHIFTER) && var->xoffset))
 
2303         var->xoffset = up(var->xoffset, 16);
 
2304         par->screen_base = screen_base +
 
2305                 (var->yoffset * fb_display[fb_info.currcon].var.xres_virtual + var->xoffset)
 
2306                 * fb_display[fb_info.currcon].var.bits_per_pixel / 8;
 
2307         fbhw->set_screen_base (par->screen_base);
 
2312 /* ------------ Interfaces to hardware functions ------------ */
 
2316 static struct fb_hwswitch tt_switch = {
 
2317         tt_detect, tt_encode_fix, tt_decode_var, tt_encode_var,
 
2318         tt_get_par, tt_set_par, tt_getcolreg, 
 
2319         set_screen_base, NULL, pan_display
 
2324 static struct fb_hwswitch falcon_switch = {
 
2325         falcon_detect, falcon_encode_fix, falcon_decode_var, falcon_encode_var,
 
2326         falcon_get_par, falcon_set_par, falcon_getcolreg,
 
2327         set_screen_base, falcon_blank, falcon_pan_display
 
2332 static struct fb_hwswitch st_switch = {
 
2333         stste_detect, stste_encode_fix, stste_decode_var, stste_encode_var,
 
2334         stste_get_par, stste_set_par, stste_getcolreg,
 
2335         stste_set_screen_base, NULL, pan_display
 
2340 static struct fb_hwswitch ext_switch = {
 
2341         ext_detect, ext_encode_fix, ext_decode_var, ext_encode_var,
 
2342         ext_get_par, ext_set_par, ext_getcolreg, NULL, NULL, NULL
 
2348 static void atafb_get_par( struct atafb_par *par )
 
2350         if (current_par_valid) {
 
2358 static void atafb_set_par( struct atafb_par *par )
 
2362         current_par_valid=1;
 
2367 /* =========================================================== */
 
2368 /* ============== Hardware Independent Functions ============= */
 
2369 /* =========================================================== */
 
2372 /* used for hardware scrolling */
 
2375 fb_update_var(int con, struct fb_info *info)
 
2377         int off=fb_display[con].var.yoffset*fb_display[con].var.xres_virtual*
 
2378                         fb_display[con].var.bits_per_pixel>>3;
 
2380         current_par.screen_base=screen_base + off;
 
2382         if (fbhw->set_screen_base)
 
2383                 fbhw->set_screen_base(current_par.screen_base);
 
2388 do_fb_set_var(struct fb_var_screeninfo *var, int isactive)
 
2391         struct atafb_par par;
 
2392         if ((err=fbhw->decode_var(var, &par)))
 
2394         activate=var->activate;
 
2395         if (((var->activate & FB_ACTIVATE_MASK) == FB_ACTIVATE_NOW) && isactive)
 
2396                 atafb_set_par(&par);
 
2397         fbhw->encode_var(var, &par);
 
2398         var->activate=activate;
 
2403 atafb_get_fix(struct fb_fix_screeninfo *fix, int con, struct fb_info *info)
 
2405         struct atafb_par par;
 
2407                 atafb_get_par(&par);
 
2410                 if ((err=fbhw->decode_var(&fb_display[con].var,&par)))
 
2413         memset(fix, 0, sizeof(struct fb_fix_screeninfo));
 
2414         return fbhw->encode_fix(fix, &par);
 
2418 atafb_get_var(struct fb_var_screeninfo *var, int con, struct fb_info *info)
 
2420         struct atafb_par par;
 
2422                 atafb_get_par(&par);
 
2423                 fbhw->encode_var(var, &par);
 
2426                 *var=fb_display[con].var;
 
2431 atafb_set_disp(int con, struct fb_info *info)
 
2433         struct fb_fix_screeninfo fix;
 
2434         struct fb_var_screeninfo var;
 
2435         struct display *display;
 
2438                 display = &fb_display[con];
 
2440                 display = &disp;        /* used during initialization */
 
2442         atafb_get_fix(&fix, con, info);
 
2443         atafb_get_var(&var, con, info);
 
2446         info->screen_base = (void *)fix.smem_start;
 
2447         display->visual = fix.visual;
 
2448         display->type = fix.type;
 
2449         display->type_aux = fix.type_aux;
 
2450         display->ypanstep = fix.ypanstep;
 
2451         display->ywrapstep = fix.ywrapstep;
 
2452         display->line_length = fix.line_length;
 
2453         if (fix.visual != FB_VISUAL_PSEUDOCOLOR &&
 
2454                 fix.visual != FB_VISUAL_DIRECTCOLOR)
 
2455                 display->can_soft_blank = 0;
 
2457                 display->can_soft_blank = 1;
 
2459             (fix.visual == FB_VISUAL_MONO01 ? !inverse : inverse);
 
2461             case FB_TYPE_INTERLEAVED_PLANES:
 
2462                 switch (var.bits_per_pixel) {
 
2463 #ifdef FBCON_HAS_IPLAN2P2
 
2465                         display->dispsw = &fbcon_iplan2p2;
 
2468 #ifdef FBCON_HAS_IPLAN2P4
 
2470                         display->dispsw = &fbcon_iplan2p4;
 
2473 #ifdef FBCON_HAS_IPLAN2P8
 
2475                         display->dispsw = &fbcon_iplan2p8;
 
2480             case FB_TYPE_PACKED_PIXELS:
 
2481                 switch (var.bits_per_pixel) {
 
2482 #ifdef FBCON_HAS_MFB
 
2484                         display->dispsw = &fbcon_mfb;
 
2487 #ifdef FBCON_HAS_CFB8
 
2489                         display->dispsw = &fbcon_cfb8;
 
2492 #ifdef FBCON_HAS_CFB16
 
2494                         display->dispsw = &fbcon_cfb16;
 
2495                         display->dispsw_data = fbcon_cfb16_cmap;
 
2504 atafb_set_var(struct fb_var_screeninfo *var, int con, struct fb_info *info)
 
2506         int err,oldxres,oldyres,oldbpp,oldxres_virtual,
 
2507             oldyres_virtual,oldyoffset;
 
2508         if ((err=do_fb_set_var(var, con==info->currcon)))
 
2510         if ((var->activate & FB_ACTIVATE_MASK) == FB_ACTIVATE_NOW) {
 
2511                 oldxres=fb_display[con].var.xres;
 
2512                 oldyres=fb_display[con].var.yres;
 
2513                 oldxres_virtual=fb_display[con].var.xres_virtual;
 
2514                 oldyres_virtual=fb_display[con].var.yres_virtual;
 
2515                 oldbpp=fb_display[con].var.bits_per_pixel;
 
2516                 oldyoffset=fb_display[con].var.yoffset;
 
2517                 fb_display[con].var=*var;
 
2518                 if (oldxres != var->xres || oldyres != var->yres 
 
2519                     || oldxres_virtual != var->xres_virtual
 
2520                     || oldyres_virtual != var->yres_virtual
 
2521                     || oldbpp != var->bits_per_pixel
 
2522                     || oldyoffset != var->yoffset) {
 
2523                         atafb_set_disp(con, info);
 
2524                         (*fb_info.changevar)(con);
 
2525                         fb_alloc_cmap(&fb_display[con].cmap, 0, 0);
 
2526                         do_install_cmap(con, info);
 
2536 atafb_get_cmap(struct fb_cmap *cmap, int kspc, int con, struct fb_info *info)
 
2538         if (con == info->currcon) /* current console ? */
 
2539                 return fb_get_cmap(cmap, kspc, fbhw->getcolreg, info);
 
2541                 if (fb_display[con].cmap.len) /* non default colormap ? */
 
2542                         fb_copy_cmap(&fb_display[con].cmap, cmap, kspc ? 0 : 2);
 
2544                         fb_copy_cmap(fb_default_cmap(1<<fb_display[con].var.bits_per_pixel),
 
2545                                      cmap, kspc ? 0 : 2);
 
2550 atafb_pan_display(struct fb_var_screeninfo *var, int con, struct fb_info *info)
 
2552         int xoffset = var->xoffset;
 
2553         int yoffset = var->yoffset;
 
2556         if (   xoffset < 0 || xoffset + fb_display[con].var.xres > fb_display[con].var.xres_virtual
 
2557             || yoffset < 0 || yoffset + fb_display[con].var.yres > fb_display[con].var.yres_virtual)
 
2560         if (con == info->currcon) {
 
2561                 if (fbhw->pan_display) {
 
2562                         if ((err = fbhw->pan_display(var, ¤t_par)))
 
2568         fb_display[con].var.xoffset = var->xoffset;
 
2569         fb_display[con].var.yoffset = var->yoffset;
 
2574 atafb_ioctl(struct inode *inode, struct file *file, unsigned int cmd,
 
2575                unsigned long arg, int con, struct fb_info *info)
 
2578 #ifdef FBCMD_GET_CURRENTPAR
 
2579         case FBCMD_GET_CURRENTPAR:
 
2580                 if (copy_to_user((void *)arg, (void *)¤t_par,
 
2581                                  sizeof(struct atafb_par)))
 
2585 #ifdef FBCMD_SET_CURRENTPAR
 
2586         case FBCMD_SET_CURRENTPAR:
 
2587                 if (copy_from_user((void *)¤t_par, (void *)arg,
 
2588                                    sizeof(struct atafb_par)))
 
2590                 atafb_set_par(¤t_par);
 
2597 /* (un)blank/poweroff
 
2605 atafb_blank(int blank, struct fb_info *info)
 
2607         unsigned short black[16];
 
2608         struct fb_cmap cmap;
 
2609         if (fbhw->blank && !fbhw->blank(blank))
 
2612                 memset(black, 0, 16*sizeof(unsigned short));
 
2619                 fb_set_cmap(&cmap, 1, info);
 
2622                 do_install_cmap(info->currcon, info);
 
2626 static struct fb_ops atafb_ops = {
 
2627         .owner =        THIS_MODULE,
 
2628         .fb_get_fix =   atafb_get_fix,
 
2629         .fb_get_var =   atafb_get_var,
 
2630         .fb_set_var =   atafb_set_var,
 
2631         .fb_get_cmap =  atafb_get_cmap,
 
2632         .fb_set_cmap =  gen_set_cmap,
 
2633         .fb_pan_display =atafb_pan_display,
 
2634         .fb_blank =     atafb_blank,
 
2635         .fb_ioctl =     atafb_ioctl,
 
2639 check_default_par( int detected_mode )
 
2641         char default_name[10];
 
2643         struct fb_var_screeninfo var;
 
2644         unsigned long min_mem;
 
2646         /* First try the user supplied mode */
 
2648                 var=atafb_predefined[default_par-1];
 
2649                 var.activate = FB_ACTIVATE_TEST;
 
2650                 if (do_fb_set_var(&var,1))
 
2651                         default_par=0;          /* failed */
 
2653         /* Next is the autodetected one */
 
2654         if (! default_par) {
 
2655                 var=atafb_predefined[detected_mode-1]; /* autodetect */
 
2656                 var.activate = FB_ACTIVATE_TEST;
 
2657                 if (!do_fb_set_var(&var,1))
 
2658                         default_par=detected_mode;
 
2660         /* If that also failed, try some default modes... */
 
2661         if (! default_par) {
 
2662                 /* try default1, default2... */
 
2663                 for (i=1 ; i < 10 ; i++) {
 
2664                         sprintf(default_name,"default%d",i);
 
2665                         default_par=get_video_mode(default_name);
 
2667                                 panic("can't set default video mode");
 
2668                         var=atafb_predefined[default_par-1];
 
2669                         var.activate = FB_ACTIVATE_TEST;
 
2670                         if (! do_fb_set_var(&var,1))
 
2674         min_mem=var.xres_virtual * var.yres_virtual * var.bits_per_pixel/8;
 
2675         if (default_mem_req < min_mem)
 
2676                 default_mem_req=min_mem;
 
2680 atafb_switch(int con, struct fb_info *info)
 
2682         /* Do we have to save the colormap ? */
 
2683         if (fb_display[info->currcon].cmap.len)
 
2684                 fb_get_cmap(&fb_display[info->currcon].cmap, 1, fbhw->getcolreg,
 
2686         do_fb_set_var(&fb_display[con].var,1);
 
2688         /* Install new colormap */
 
2689         do_install_cmap(con, info);
 
2693 int __init atafb_init(void)
 
2697         unsigned long mem_req;
 
2704                 if (external_addr) {
 
2706                         atafb_ops.fb_setcolreg = &ext_setcolreg;
 
2711                 if (ATARIHW_PRESENT(TT_SHIFTER)) {
 
2713                         atafb_ops.fb_setcolreg = &tt_setcolreg;
 
2718                 if (ATARIHW_PRESENT(VIDEL_SHIFTER)) {
 
2719                         fbhw = &falcon_switch;
 
2720                         atafb_ops.fb_setcolreg = &falcon_setcolreg;
 
2721                         request_irq(IRQ_AUTO_4, falcon_vbl_switcher, IRQ_TYPE_PRIO,
 
2722                                     "framebuffer/modeswitch", falcon_vbl_switcher);
 
2727                 if (ATARIHW_PRESENT(STND_SHIFTER) ||
 
2728                     ATARIHW_PRESENT(EXTD_SHIFTER)) {
 
2730                         atafb_ops.fb_setcolreg = &stste_setcolreg;
 
2734                 atafb_ops.fb_setcolreg = &stste_setcolreg;
 
2735                 printk("Cannot determine video hardware; defaulting to ST(e)\n");
 
2736 #else /* ATAFB_STE */
 
2737                 /* no default driver included */
 
2738                 /* Nobody will ever see this message :-) */
 
2739                 panic("Cannot initialize video hardware");
 
2743         /* Multisync monitor capabilities */
 
2744         /* Atari-TOS defaults if no boot option present */
 
2745         if (fb_info.monspecs.hfmin == 0) {
 
2746             fb_info.monspecs.hfmin = 31000;
 
2747             fb_info.monspecs.hfmax = 32000;
 
2748             fb_info.monspecs.vfmin = 58;
 
2749             fb_info.monspecs.vfmax = 62;
 
2752         detected_mode = fbhw->detect();
 
2753         check_default_par(detected_mode);
 
2755         if (!external_addr) {
 
2756 #endif /* ATAFB_EXT */
 
2757                 mem_req = default_mem_req + ovsc_offset + ovsc_addlen;
 
2758                 mem_req = PAGE_ALIGN(mem_req) + PAGE_SIZE;
 
2759                 screen_base = atari_stram_alloc(mem_req, "atafb");
 
2761                         panic("Cannot allocate screen memory");
 
2762                 memset(screen_base, 0, mem_req);
 
2763                 pad = -(unsigned long)screen_base & (PAGE_SIZE-1);
 
2765                 real_screen_base=screen_base+ovsc_offset;
 
2766                 screen_len = (mem_req - pad - ovsc_offset) & PAGE_MASK;
 
2768                 if (CPU_IS_040_OR_060) {
 
2769                         /* On a '040+, the cache mode of video RAM must be set to
 
2770                          * write-through also for internal video hardware! */
 
2771                         cache_push(virt_to_phys(screen_base), screen_len);
 
2772                         kernel_set_cachemode(screen_base, screen_len,
 
2773                                              IOMAP_WRITETHROUGH);
 
2778                 /* Map the video memory (physical address given) to somewhere
 
2779                  * in the kernel address space.
 
2782                   ioremap_writethrough((unsigned long)external_addr,
 
2784                 if (external_vgaiobase)
 
2785                         external_vgaiobase =
 
2786                           (unsigned long)ioremap(external_vgaiobase, 0x10000);
 
2788                 real_screen_base = external_addr;
 
2789                 screen_len       = external_len & PAGE_MASK;
 
2790                 memset (screen_base, 0, external_len);
 
2792 #endif /* ATAFB_EXT */
 
2794         strcpy(fb_info.modename, "Atari Builtin ");
 
2795         fb_info.changevar = NULL;
 
2796         fb_info.fbops = &atafb_ops;
 
2797         fb_info.disp = &disp;
 
2798         fb_info.currcon = -1;
 
2799         fb_info.switch_con = &atafb_switch;
 
2800         fb_info.updatevar = &fb_update_var;
 
2801         fb_info.flags = FBINFO_FLAG_DEFAULT;
 
2802         do_fb_set_var(&atafb_predefined[default_par-1], 1);
 
2803         strcat(fb_info.modename, fb_var_names[default_par-1][0]);
 
2805         atafb_get_var(&disp.var, -1, &fb_info);
 
2806         atafb_set_disp(-1, &fb_info);
 
2807         do_install_cmap(0, &fb_info);
 
2809         if (register_framebuffer(&fb_info) < 0)
 
2812         printk("Determined %dx%d, depth %d\n",
 
2813                disp.var.xres, disp.var.yres, disp.var.bits_per_pixel);
 
2814         if ((disp.var.xres != disp.var.xres_virtual) ||
 
2815             (disp.var.yres != disp.var.yres_virtual))
 
2816            printk("   virtual %dx%d\n",
 
2817                           disp.var.xres_virtual, disp.var.yres_virtual);
 
2818         printk("fb%d: %s frame buffer device, using %dK of video memory\n",
 
2819                fb_info.node, fb_info.modename, screen_len>>10);
 
2821         /* TODO: This driver cannot be unloaded yet */
 
2827 static void __init atafb_setup_ext(char *spec)
 
2829         int             xres, xres_virtual, yres, depth, planes;
 
2830         unsigned long addr, len;
 
2833         /* Format is: <xres>;<yres>;<depth>;<plane organ.>;
 
2835          *            [;<screen mem length>[;<vgaiobase>[;<bits-per-col>[;<colorreg-type>
 
2836          *            [;<xres-virtual>]]]]]
 
2839          * <xres_virtual>:      hardware's x-resolution (f.e. ProMST)
 
2841          * Even xres_virtual is available, we neither support panning nor hw-scrolling!
 
2843         if (!(p = strsep(&spec, ";")) || !*p)
 
2845         xres_virtual = xres = simple_strtoul(p, NULL, 10);
 
2849         if (!(p = strsep(&spec, ";")) || !*p)
 
2851         yres = simple_strtoul(p, NULL, 10);
 
2855         if (!(p = strsep(&spec, ";")) || !*p)
 
2857         depth = simple_strtoul(p, NULL, 10);
 
2858         if (depth != 1 && depth != 2 && depth != 4 && depth != 8 &&
 
2859                 depth != 16 && depth != 24)
 
2862         if (!(p = strsep(&spec, ";")) || !*p)
 
2865                 planes = FB_TYPE_INTERLEAVED_PLANES;
 
2867                 planes = FB_TYPE_PACKED_PIXELS;
 
2869                 planes = FB_TYPE_PLANES;
 
2871                 planes = -1; /* true color */
 
2876         if (!(p = strsep(&spec, ";")) || !*p)
 
2878         addr = simple_strtoul(p, NULL, 0);
 
2880         if (!(p = strsep(&spec, ";")) || !*p)
 
2881                 len = xres*yres*depth/8;
 
2883                 len = simple_strtoul(p, NULL, 0);
 
2885         if ((p = strsep(&spec, ";")) && *p) {
 
2886                 external_vgaiobase=simple_strtoul(p, NULL, 0);
 
2889         if ((p = strsep(&spec, ";")) && *p) {
 
2890                 external_bitspercol = simple_strtoul(p, NULL, 0);
 
2891                 if (external_bitspercol > 8)
 
2892                         external_bitspercol = 8;
 
2893                 else if (external_bitspercol < 1)
 
2894                         external_bitspercol = 1;
 
2897         if ((p = strsep(&spec, ";")) && *p) {
 
2898                 if (!strcmp(p, "vga"))
 
2899                         external_card_type = IS_VGA;
 
2900                 if (!strcmp(p, "mv300"))
 
2901                         external_card_type = IS_MV300;
 
2904         if ((p = strsep(&spec, ";")) && *p) {
 
2905                 xres_virtual = simple_strtoul(p, NULL, 10);
 
2906                 if (xres_virtual < xres)
 
2907                         xres_virtual = xres;
 
2908                 if (xres_virtual*yres*depth/8 > len)
 
2909                         len=xres_virtual*yres*depth/8;
 
2912         external_xres  = xres;
 
2913         external_xres_virtual  = xres_virtual;
 
2914         external_yres  = yres;
 
2915         external_depth = depth;
 
2916         external_pmode = planes;
 
2917         external_addr  = (void *)addr;
 
2920         if (external_card_type == IS_MV300)
 
2921           switch (external_depth) {
 
2923               MV300_reg = MV300_reg_1bit;
 
2926               MV300_reg = MV300_reg_4bit;
 
2929               MV300_reg = MV300_reg_8bit;
 
2933 #endif /* ATAFB_EXT */
 
2936 static void __init atafb_setup_int(char *spec)
 
2938         /* Format to config extended internal video hardware like OverScan:
 
2939         "internal:<xres>;<yres>;<xres_max>;<yres_max>;<offset>"
 
2941         <xres>: x-resolution 
 
2942         <yres>: y-resolution
 
2943         The following are only needed if you have an overscan which
 
2944         needs a black border:
 
2945         <xres_max>: max. length of a line in pixels your OverScan hardware would allow
 
2946         <yres_max>: max. number of lines your OverScan hardware would allow
 
2947         <offset>: Offset from physical beginning to visible beginning
 
2953         if (!(p = strsep(&spec, ";")) || !*p)
 
2955         xres = simple_strtoul(p, NULL, 10);
 
2956         if (!(p = strsep(&spec, ";")) || !*p)
 
2959         tt_yres=st_yres=simple_strtoul(p, NULL, 10);
 
2960         if ((p=strsep(&spec, ";")) && *p) {
 
2961                 sttt_xres_virtual=simple_strtoul(p, NULL, 10);
 
2963         if ((p=strsep(&spec, ";")) && *p) {
 
2964                 sttt_yres_virtual=simple_strtoul(p, NULL, 0);
 
2966         if ((p=strsep(&spec, ";")) && *p) {
 
2967                 ovsc_offset=simple_strtoul(p, NULL, 0);
 
2970         if (ovsc_offset || (sttt_yres_virtual != st_yres))
 
2976 static void __init atafb_setup_mcap(char *spec)
 
2979         int vmin, vmax, hmin, hmax;
 
2981         /* Format for monitor capabilities is: <Vmin>;<Vmax>;<Hmin>;<Hmax>
 
2982          * <V*> vertical freq. in Hz
 
2983          * <H*> horizontal freq. in kHz
 
2985         if (!(p = strsep(&spec, ";")) || !*p)
 
2987         vmin = simple_strtoul(p, NULL, 10);
 
2990         if (!(p = strsep(&spec, ";")) || !*p)
 
2992         vmax = simple_strtoul(p, NULL, 10);
 
2993         if (vmax <= 0 || vmax <= vmin)
 
2995         if (!(p = strsep(&spec, ";")) || !*p)
 
2997         hmin = 1000 * simple_strtoul(p, NULL, 10);
 
3000         if (!(p = strsep(&spec, "")) || !*p)
 
3002         hmax = 1000 * simple_strtoul(p, NULL, 10);
 
3003         if (hmax <= 0 || hmax <= hmin)
 
3006         fb_info.monspecs.vfmin = vmin;
 
3007         fb_info.monspecs.vfmax = vmax;
 
3008         fb_info.monspecs.hfmin = hmin;
 
3009         fb_info.monspecs.hfmax = hmax;
 
3011 #endif /* ATAFB_FALCON */
 
3014 static void __init atafb_setup_user(char *spec)
 
3016         /* Format of user defined video mode is: <xres>;<yres>;<depth>
 
3019         int xres, yres, depth, temp;
 
3021         if (!(p = strsep(&spec, ";")) || !*p)
 
3023         xres = simple_strtoul(p, NULL, 10);
 
3024         if (!(p = strsep(&spec, ";")) || !*p)
 
3026         yres = simple_strtoul(p, NULL, 10);
 
3027         if (!(p = strsep(&spec, "")) || !*p)
 
3029         depth = simple_strtoul(p, NULL, 10);
 
3030         if ((temp=get_video_mode("user0"))) {
 
3032                 atafb_predefined[default_par-1].xres = xres;
 
3033                 atafb_predefined[default_par-1].yres = yres;
 
3034                 atafb_predefined[default_par-1].bits_per_pixel = depth;
 
3038 int __init atafb_setup( char *options )
 
3043     fb_info.fontname[0] = '\0';
 
3045     if (!options || !*options)
 
3048     while ((this_opt = strsep(&options, ",")) != NULL) {         
 
3049         if (!*this_opt) continue;
 
3050         if ((temp=get_video_mode(this_opt)))
 
3052         else if (! strcmp(this_opt, "inverse"))
 
3054         else if (!strncmp(this_opt, "font:", 5))
 
3055            strcpy(fb_info.fontname, this_opt+5);
 
3056         else if (! strncmp(this_opt, "hwscroll_",9)) {
 
3057                 hwscroll=simple_strtoul(this_opt+9, NULL, 10);
 
3064         else if (!strcmp(this_opt,"mv300")) {
 
3065                 external_bitspercol = 8;
 
3066                 external_card_type = IS_MV300;
 
3068         else if (!strncmp(this_opt,"external:",9))
 
3069                 atafb_setup_ext(this_opt+9);
 
3071         else if (!strncmp(this_opt,"internal:",9))
 
3072                 atafb_setup_int(this_opt+9);
 
3074         else if (!strncmp(this_opt, "eclock:", 7)) {
 
3075                 fext.f = simple_strtoul(this_opt+7, NULL, 10);
 
3076                 /* external pixelclock in kHz --> ps */
 
3077                 fext.t = 1000000000/fext.f;
 
3080         else if (!strncmp(this_opt, "monitorcap:", 11))
 
3081                 atafb_setup_mcap(this_opt+11);
 
3083         else if (!strcmp(this_opt, "keep"))
 
3085         else if (!strncmp(this_opt, "R", 1))
 
3086                 atafb_setup_user(this_opt+1);
 
3092 MODULE_LICENSE("GPL");
 
3094 int init_module(void)
 
3096         return atafb_init();