2009-11-04 00:47:28 +11:00
/*
* macfb . c : Generic framebuffer for Macs whose colourmaps / modes we
* don ' t know how to set .
*
* ( c ) 1999 David Huggins - Daines < dhd @ debian . org >
*
* Primarily based on vesafb . c , by Gerd Knorr
* ( c ) 1998 Gerd Knorr < kraxel @ cs . tu - berlin . de >
*
* Also uses information and code from :
*
* The original macfb . c from Linux / mac68k 2.0 , by Alan Cox , Juergen
* Mellinger , Mikael Forselius , Michael Schmitz , and others .
*
* valkyriefb . c , by Martin Costabel , Kevin Schoedel , Barry Nathan , Dan
* Jacobowitz , Paul Mackerras , Fabio Riccardi , and Geert Uytterhoeven .
*
* The VideoToolbox " Bugs " web page at
* http : //rajsky.psych.nyu.edu/Tips/VideoBugs.html
*
* This code is free software . You may copy , modify , and distribute
* it subject to the terms and conditions of the GNU General Public
* License , version 2 , or any later version , at your convenience .
*/
2005-04-16 15:20:36 -07:00
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/errno.h>
# include <linux/string.h>
# include <linux/mm.h>
# include <linux/delay.h>
# include <linux/nubus.h>
# include <linux/init.h>
# include <linux/fb.h>
# include <asm/setup.h>
# include <asm/bootinfo.h>
# include <asm/macintosh.h>
# include <asm/io.h>
/* Common DAC base address for the LC, RBV, Valkyrie, and IIvx */
# define DAC_BASE 0x50f24000
/* Some addresses for the DAFB */
# define DAFB_BASE 0xf9800200
/* Address for the built-in Civic framebuffer in Quadra AVs */
2009-11-04 00:47:28 +11:00
# define CIVIC_BASE 0x50f30800
2005-04-16 15:20:36 -07:00
/* GSC (Gray Scale Controller) base address */
# define GSC_BASE 0x50F20000
/* CSC (Color Screen Controller) base address */
# define CSC_BASE 0x50F20000
2009-11-04 00:47:28 +11:00
static int ( * macfb_setpalette ) ( unsigned int regno , unsigned int red ,
unsigned int green , unsigned int blue ,
struct fb_info * info ) ;
2005-04-16 15:20:36 -07:00
2008-11-18 20:45:22 +01:00
static struct {
2005-04-16 15:20:36 -07:00
unsigned char addr ;
unsigned char lut ;
2008-11-18 20:45:22 +01:00
} __iomem * v8_brazil_cmap_regs ;
2005-04-16 15:20:36 -07:00
2008-11-18 20:45:22 +01:00
static struct {
2005-04-16 15:20:36 -07:00
unsigned char addr ;
char pad1 [ 3 ] ; /* word aligned */
unsigned char lut ;
char pad2 [ 3 ] ; /* word aligned */
unsigned char cntl ; /* a guess as to purpose */
2008-11-18 20:45:22 +01:00
} __iomem * rbv_cmap_regs ;
2005-04-16 15:20:36 -07:00
2008-11-18 20:45:22 +01:00
static struct {
2005-04-16 15:20:36 -07:00
unsigned long reset ;
unsigned long pad1 [ 3 ] ;
unsigned char pad2 [ 3 ] ;
unsigned char lut ;
2008-11-18 20:45:22 +01:00
} __iomem * dafb_cmap_regs ;
2005-04-16 15:20:36 -07:00
2008-11-18 20:45:22 +01:00
static struct {
2005-04-16 15:20:36 -07:00
unsigned char addr ; /* OFFSET: 0x00 */
unsigned char pad1 [ 15 ] ;
unsigned char lut ; /* OFFSET: 0x10 */
unsigned char pad2 [ 15 ] ;
unsigned char status ; /* OFFSET: 0x20 */
unsigned char pad3 [ 7 ] ;
unsigned long vbl_addr ; /* OFFSET: 0x28 */
unsigned int status2 ; /* OFFSET: 0x2C */
2008-11-18 20:45:22 +01:00
} __iomem * civic_cmap_regs ;
2005-04-16 15:20:36 -07:00
2008-11-18 20:45:22 +01:00
static struct {
2009-11-04 00:47:28 +11:00
char pad1 [ 0x40 ] ;
unsigned char clut_waddr ; /* 0x40 */
char pad2 ;
unsigned char clut_data ; /* 0x42 */
char pad3 [ 0x3 ] ;
unsigned char clut_raddr ; /* 0x46 */
2008-11-18 20:45:22 +01:00
} __iomem * csc_cmap_regs ;
2005-04-16 15:20:36 -07:00
2009-11-04 00:47:28 +11:00
/* The registers in these structs are in NuBus slot space */
2005-04-16 15:20:36 -07:00
struct mdc_cmap_regs {
char pad1 [ 0x200200 ] ;
unsigned char addr ;
char pad2 [ 6 ] ;
unsigned char lut ;
} ;
struct toby_cmap_regs {
char pad1 [ 0x90018 ] ;
unsigned char lut ; /* TFBClutWDataReg, offset 0x90018 */
char pad2 [ 3 ] ;
unsigned char addr ; /* TFBClutAddrReg, offset 0x9001C */
} ;
struct jet_cmap_regs {
char pad1 [ 0xe0e000 ] ;
unsigned char addr ;
unsigned char lut ;
} ;
2009-11-04 00:47:28 +11:00
# define PIXEL_TO_MM(a) (((a)*10) / 28) /* width in mm at 72 dpi */
2005-04-16 15:20:36 -07:00
static struct fb_var_screeninfo macfb_defined = {
2009-11-04 00:47:28 +11:00
. bits_per_pixel = 8 ,
2005-04-16 15:20:36 -07:00
. activate = FB_ACTIVATE_NOW ,
. width = - 1 ,
. height = - 1 ,
. right_margin = 32 ,
. upper_margin = 16 ,
. lower_margin = 4 ,
. vsync_len = 4 ,
. vmode = FB_VMODE_NONINTERLACED ,
} ;
static struct fb_fix_screeninfo macfb_fix = {
. type = FB_TYPE_PACKED_PIXELS ,
. accel = FB_ACCEL_NONE ,
} ;
2009-11-04 00:51:18 +11:00
static void * slot_addr ;
2005-04-16 15:20:36 -07:00
static struct fb_info fb_info ;
2007-07-17 04:05:37 -07:00
static u32 pseudo_palette [ 16 ] ;
2009-11-04 00:47:28 +11:00
static int inverse ;
static int vidtest ;
/*
* Unlike the Valkyrie , the DAFB cannot set individual colormap
* registers . Therefore , we do what the MacOS driver does ( no
* kidding ! ) and simply set them one by one until we hit the one we
* want .
*/
static int dafb_setpalette ( unsigned int regno , unsigned int red ,
unsigned int green , unsigned int blue ,
struct fb_info * info )
2005-04-16 15:20:36 -07:00
{
static int lastreg = - 1 ;
unsigned long flags ;
local_irq_save ( flags ) ;
2009-11-04 00:47:28 +11:00
/*
* fbdev will set an entire colourmap , but X won ' t . Hopefully
* this should accommodate both of them
*/
if ( regno ! = lastreg + 1 ) {
2005-04-16 15:20:36 -07:00
int i ;
2009-11-04 00:47:28 +11:00
2005-04-16 15:20:36 -07:00
/* Stab in the dark trying to reset the CLUT pointer */
nubus_writel ( 0 , & dafb_cmap_regs - > reset ) ;
nop ( ) ;
2009-11-04 00:47:28 +11:00
2005-04-16 15:20:36 -07:00
/* Loop until we get to the register we want */
for ( i = 0 ; i < regno ; i + + ) {
2009-11-04 00:47:28 +11:00
nubus_writeb ( info - > cmap . red [ i ] > > 8 ,
& dafb_cmap_regs - > lut ) ;
2005-04-16 15:20:36 -07:00
nop ( ) ;
2009-11-04 00:47:28 +11:00
nubus_writeb ( info - > cmap . green [ i ] > > 8 ,
& dafb_cmap_regs - > lut ) ;
2005-04-16 15:20:36 -07:00
nop ( ) ;
2009-11-04 00:47:28 +11:00
nubus_writeb ( info - > cmap . blue [ i ] > > 8 ,
& dafb_cmap_regs - > lut ) ;
2005-04-16 15:20:36 -07:00
nop ( ) ;
}
}
2009-11-04 00:47:28 +11:00
2005-04-16 15:20:36 -07:00
nubus_writeb ( red , & dafb_cmap_regs - > lut ) ;
nop ( ) ;
nubus_writeb ( green , & dafb_cmap_regs - > lut ) ;
nop ( ) ;
nubus_writeb ( blue , & dafb_cmap_regs - > lut ) ;
2009-11-04 00:47:28 +11:00
2005-04-16 15:20:36 -07:00
local_irq_restore ( flags ) ;
lastreg = regno ;
return 0 ;
}
/* V8 and Brazil seem to use the same DAC. Sonora does as well. */
2009-11-04 00:47:28 +11:00
static int v8_brazil_setpalette ( unsigned int regno , unsigned int red ,
unsigned int green , unsigned int blue ,
struct fb_info * info )
2005-04-16 15:20:36 -07:00
{
unsigned int bpp = info - > var . bits_per_pixel ;
unsigned long flags ;
2009-11-04 00:47:28 +11:00
if ( bpp > 8 )
return 1 ; /* failsafe */
2005-04-16 15:20:36 -07:00
local_irq_save ( flags ) ;
/* On these chips, the CLUT register numbers are spread out
2009-11-04 00:47:28 +11:00
* across the register space . Thus :
* In 8 bpp , all regnos are valid .
* In 4 bpp , the regnos are 0x0f , 0x1f , 0x2f , etc , etc
* In 2 bpp , the regnos are 0x3f , 0x7f , 0xbf , 0xff
*/
2009-11-04 00:51:18 +11:00
regno = ( regno < < ( 8 - bpp ) ) | ( 0xFF > > bpp ) ;
nubus_writeb ( regno , & v8_brazil_cmap_regs - > addr ) ;
2009-11-04 00:47:28 +11:00
nop ( ) ;
2005-04-16 15:20:36 -07:00
/* send one color channel at a time */
2009-11-04 00:51:18 +11:00
nubus_writeb ( red , & v8_brazil_cmap_regs - > lut ) ;
2009-11-04 00:47:28 +11:00
nop ( ) ;
2009-11-04 00:51:18 +11:00
nubus_writeb ( green , & v8_brazil_cmap_regs - > lut ) ;
2009-11-04 00:47:28 +11:00
nop ( ) ;
2009-11-04 00:51:18 +11:00
nubus_writeb ( blue , & v8_brazil_cmap_regs - > lut ) ;
2005-04-16 15:20:36 -07:00
2009-11-04 00:47:28 +11:00
local_irq_restore ( flags ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2009-11-04 00:47:28 +11:00
/* RAM-Based Video */
static int rbv_setpalette ( unsigned int regno , unsigned int red ,
unsigned int green , unsigned int blue ,
struct fb_info * info )
2005-04-16 15:20:36 -07:00
{
unsigned long flags ;
2009-11-04 00:47:28 +11:00
if ( info - > var . bits_per_pixel > 8 )
return 1 ; /* failsafe */
2005-04-16 15:20:36 -07:00
local_irq_save ( flags ) ;
2009-11-04 00:47:28 +11:00
2005-04-16 15:20:36 -07:00
/* From the VideoToolbox driver. Seems to be saying that
* regno # 254 and # 255 are the important ones for 1 - bit color ,
* regno # 252 - 255 are the important ones for 2 - bit color , etc .
*/
2009-11-04 00:51:18 +11:00
regno + = 256 - ( 1 < < info - > var . bits_per_pixel ) ;
2005-04-16 15:20:36 -07:00
/* reset clut? (VideoToolbox sez "not necessary") */
2009-11-04 00:47:28 +11:00
nubus_writeb ( 0xFF , & rbv_cmap_regs - > cntl ) ;
nop ( ) ;
2005-04-16 15:20:36 -07:00
/* tell clut which address to use. */
2009-11-04 00:51:18 +11:00
nubus_writeb ( regno , & rbv_cmap_regs - > addr ) ;
2009-11-04 00:47:28 +11:00
nop ( ) ;
2005-04-16 15:20:36 -07:00
/* send one color channel at a time. */
2009-11-04 00:51:18 +11:00
nubus_writeb ( red , & rbv_cmap_regs - > lut ) ;
2009-11-04 00:47:28 +11:00
nop ( ) ;
2009-11-04 00:51:18 +11:00
nubus_writeb ( green , & rbv_cmap_regs - > lut ) ;
2009-11-04 00:47:28 +11:00
nop ( ) ;
2009-11-04 00:51:18 +11:00
nubus_writeb ( blue , & rbv_cmap_regs - > lut ) ;
2009-11-04 00:47:28 +11:00
local_irq_restore ( flags ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2009-11-04 00:47:28 +11:00
/* Macintosh Display Card (8*24) */
2005-04-16 15:20:36 -07:00
static int mdc_setpalette ( unsigned int regno , unsigned int red ,
unsigned int green , unsigned int blue ,
struct fb_info * info )
{
2009-11-04 00:51:18 +11:00
struct mdc_cmap_regs * cmap_regs = slot_addr ;
2005-04-16 15:20:36 -07:00
unsigned long flags ;
local_irq_save ( flags ) ;
2009-11-04 00:47:28 +11:00
2005-04-16 15:20:36 -07:00
/* the nop's are there to order writes. */
2009-11-04 00:51:18 +11:00
nubus_writeb ( regno , & cmap_regs - > addr ) ;
2009-11-04 00:47:28 +11:00
nop ( ) ;
2009-11-04 00:51:18 +11:00
nubus_writeb ( red , & cmap_regs - > lut ) ;
2009-11-04 00:47:28 +11:00
nop ( ) ;
2009-11-04 00:51:18 +11:00
nubus_writeb ( green , & cmap_regs - > lut ) ;
2009-11-04 00:47:28 +11:00
nop ( ) ;
2009-11-04 00:51:18 +11:00
nubus_writeb ( blue , & cmap_regs - > lut ) ;
2005-04-16 15:20:36 -07:00
local_irq_restore ( flags ) ;
return 0 ;
}
/* Toby frame buffer */
static int toby_setpalette ( unsigned int regno , unsigned int red ,
unsigned int green , unsigned int blue ,
2009-11-04 00:47:28 +11:00
struct fb_info * info )
2005-04-16 15:20:36 -07:00
{
2009-11-04 00:51:18 +11:00
struct toby_cmap_regs * cmap_regs = slot_addr ;
2005-04-16 15:20:36 -07:00
unsigned int bpp = info - > var . bits_per_pixel ;
unsigned long flags ;
2009-11-04 00:51:18 +11:00
red = ~ red ;
green = ~ green ;
blue = ~ blue ;
regno = ( regno < < ( 8 - bpp ) ) | ( 0xFF > > bpp ) ;
2005-04-16 15:20:36 -07:00
local_irq_save ( flags ) ;
2009-11-04 00:47:28 +11:00
2009-11-04 00:51:18 +11:00
nubus_writeb ( regno , & cmap_regs - > addr ) ;
2009-11-04 00:47:28 +11:00
nop ( ) ;
2009-11-04 00:51:18 +11:00
nubus_writeb ( red , & cmap_regs - > lut ) ;
2009-11-04 00:47:28 +11:00
nop ( ) ;
2009-11-04 00:51:18 +11:00
nubus_writeb ( green , & cmap_regs - > lut ) ;
2009-11-04 00:47:28 +11:00
nop ( ) ;
2009-11-04 00:51:18 +11:00
nubus_writeb ( blue , & cmap_regs - > lut ) ;
2005-04-16 15:20:36 -07:00
local_irq_restore ( flags ) ;
return 0 ;
}
/* Jet frame buffer */
static int jet_setpalette ( unsigned int regno , unsigned int red ,
unsigned int green , unsigned int blue ,
struct fb_info * info )
{
2009-11-04 00:51:18 +11:00
struct jet_cmap_regs * cmap_regs = slot_addr ;
2005-04-16 15:20:36 -07:00
unsigned long flags ;
local_irq_save ( flags ) ;
2009-11-04 00:47:28 +11:00
nubus_writeb ( regno , & cmap_regs - > addr ) ;
nop ( ) ;
2009-11-04 00:51:18 +11:00
nubus_writeb ( red , & cmap_regs - > lut ) ;
2009-11-04 00:47:28 +11:00
nop ( ) ;
2009-11-04 00:51:18 +11:00
nubus_writeb ( green , & cmap_regs - > lut ) ;
2009-11-04 00:47:28 +11:00
nop ( ) ;
2009-11-04 00:51:18 +11:00
nubus_writeb ( blue , & cmap_regs - > lut ) ;
2005-04-16 15:20:36 -07:00
local_irq_restore ( flags ) ;
return 0 ;
}
/*
* Civic framebuffer - - Quadra AV built - in video . A chip
* called Sebastian holds the actual color palettes , and
2009-11-04 00:47:28 +11:00
* apparently , there are two different banks of 512 K RAM
2005-04-16 15:20:36 -07:00
* which can act as separate framebuffers for doing video
* input and viewing the screen at the same time ! The 840 AV
2009-11-04 00:47:28 +11:00
* Can add another 1 MB RAM to give the two framebuffers
2005-04-16 15:20:36 -07:00
* 1 MB RAM apiece .
*/
2009-11-04 00:47:28 +11:00
static int civic_setpalette ( unsigned int regno , unsigned int red ,
unsigned int green , unsigned int blue ,
struct fb_info * info )
2005-04-16 15:20:36 -07:00
{
unsigned long flags ;
int clut_status ;
2009-11-04 00:47:28 +11:00
if ( info - > var . bits_per_pixel > 8 )
return 1 ; /* failsafe */
2005-04-16 15:20:36 -07:00
local_irq_save ( flags ) ;
2009-11-04 00:47:28 +11:00
/* Set the register address */
nubus_writeb ( regno , & civic_cmap_regs - > addr ) ;
nop ( ) ;
2005-04-16 15:20:36 -07:00
/*
* Grab a status word and do some checking ;
* Then finally write the clut !
*/
clut_status = nubus_readb ( & civic_cmap_regs - > status2 ) ;
if ( ( clut_status & 0x0008 ) = = 0 )
{
#if 0
if ( ( clut_status & 0x000D ) ! = 0 )
{
2009-11-04 00:47:28 +11:00
nubus_writeb ( 0x00 , & civic_cmap_regs - > lut ) ;
nop ( ) ;
nubus_writeb ( 0x00 , & civic_cmap_regs - > lut ) ;
nop ( ) ;
2005-04-16 15:20:36 -07:00
}
# endif
2009-11-04 00:47:28 +11:00
nubus_writeb ( red , & civic_cmap_regs - > lut ) ;
nop ( ) ;
nubus_writeb ( green , & civic_cmap_regs - > lut ) ;
nop ( ) ;
nubus_writeb ( blue , & civic_cmap_regs - > lut ) ;
nop ( ) ;
nubus_writeb ( 0x00 , & civic_cmap_regs - > lut ) ;
2005-04-16 15:20:36 -07:00
}
else
{
unsigned char junk ;
2009-11-04 00:47:28 +11:00
junk = nubus_readb ( & civic_cmap_regs - > lut ) ;
nop ( ) ;
junk = nubus_readb ( & civic_cmap_regs - > lut ) ;
nop ( ) ;
junk = nubus_readb ( & civic_cmap_regs - > lut ) ;
nop ( ) ;
junk = nubus_readb ( & civic_cmap_regs - > lut ) ;
nop ( ) ;
2005-04-16 15:20:36 -07:00
if ( ( clut_status & 0x000D ) ! = 0 )
{
2009-11-04 00:47:28 +11:00
nubus_writeb ( 0x00 , & civic_cmap_regs - > lut ) ;
nop ( ) ;
nubus_writeb ( 0x00 , & civic_cmap_regs - > lut ) ;
nop ( ) ;
2005-04-16 15:20:36 -07:00
}
2009-11-04 00:47:28 +11:00
nubus_writeb ( red , & civic_cmap_regs - > lut ) ;
nop ( ) ;
nubus_writeb ( green , & civic_cmap_regs - > lut ) ;
nop ( ) ;
nubus_writeb ( blue , & civic_cmap_regs - > lut ) ;
nop ( ) ;
nubus_writeb ( junk , & civic_cmap_regs - > lut ) ;
2005-04-16 15:20:36 -07:00
}
local_irq_restore ( flags ) ;
return 0 ;
}
/*
* The CSC is the framebuffer on the PowerBook 190 series
* ( and the 5300 too , but that ' s a PowerMac ) . This function
* brought to you in part by the ECSC driver for MkLinux .
*/
2009-11-04 00:47:28 +11:00
static int csc_setpalette ( unsigned int regno , unsigned int red ,
unsigned int green , unsigned int blue ,
struct fb_info * info )
2005-04-16 15:20:36 -07:00
{
2009-11-04 00:51:18 +11:00
unsigned long flags ;
local_irq_save ( flags ) ;
udelay ( 1 ) ; /* mklinux on PB 5300 waits for 260 ns */
2008-11-18 20:45:22 +01:00
nubus_writeb ( regno , & csc_cmap_regs - > clut_waddr ) ;
2009-11-04 00:51:18 +11:00
nubus_writeb ( red , & csc_cmap_regs - > clut_data ) ;
2008-11-18 20:45:22 +01:00
nubus_writeb ( green , & csc_cmap_regs - > clut_data ) ;
2009-11-04 00:51:18 +11:00
nubus_writeb ( blue , & csc_cmap_regs - > clut_data ) ;
local_irq_restore ( flags ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
static int macfb_setcolreg ( unsigned regno , unsigned red , unsigned green ,
unsigned blue , unsigned transp ,
struct fb_info * fb_info )
{
/*
2009-11-04 00:47:28 +11:00
* Set a single color register . The values supplied are
* already rounded down to the hardware ' s capabilities
* ( according to the entries in the ` var ' structure ) .
* Return non - zero for invalid regno .
2005-04-16 15:20:36 -07:00
*/
if ( regno > = fb_info - > cmap . len )
return 1 ;
2007-07-17 04:05:37 -07:00
if ( fb_info - > var . bits_per_pixel < = 8 ) {
switch ( fb_info - > var . bits_per_pixel ) {
case 1 :
/* We shouldn't get here */
break ;
case 2 :
case 4 :
case 8 :
if ( macfb_setpalette )
2009-11-04 00:51:18 +11:00
macfb_setpalette ( regno , red > > 8 , green > > 8 ,
blue > > 8 , fb_info ) ;
2007-07-17 04:05:37 -07:00
else
return 1 ;
break ;
}
} else if ( regno < 16 ) {
switch ( fb_info - > var . bits_per_pixel ) {
case 16 :
if ( fb_info - > var . red . offset = = 10 ) {
/* 1:5:5:5 */
( ( u32 * ) ( fb_info - > pseudo_palette ) ) [ regno ] =
2005-04-16 15:20:36 -07:00
( ( red & 0xf800 ) > > 1 ) |
( ( green & 0xf800 ) > > 6 ) |
( ( blue & 0xf800 ) > > 11 ) |
( ( transp ! = 0 ) < < 15 ) ;
2007-07-17 04:05:37 -07:00
} else {
/* 0:5:6:5 */
( ( u32 * ) ( fb_info - > pseudo_palette ) ) [ regno ] =
2009-11-04 00:47:28 +11:00
( ( red & 0xf800 ) > > 0 ) |
2005-04-16 15:20:36 -07:00
( ( green & 0xfc00 ) > > 5 ) |
( ( blue & 0xf800 ) > > 11 ) ;
2007-07-17 04:05:37 -07:00
}
break ;
2009-11-04 00:47:28 +11:00
/*
* 24 - bit colour almost doesn ' t exist on 68 k Macs - -
* http : //support.apple.com/kb/TA28634 (Old Article: 10992)
*/
2007-07-17 04:05:37 -07:00
case 24 :
case 32 :
red > > = 8 ;
green > > = 8 ;
blue > > = 8 ;
( ( u32 * ) ( fb_info - > pseudo_palette ) ) [ regno ] =
2009-11-04 00:47:28 +11:00
( red < < fb_info - > var . red . offset ) |
2007-07-17 04:05:37 -07:00
( green < < fb_info - > var . green . offset ) |
( blue < < fb_info - > var . blue . offset ) ;
break ;
2005-04-16 15:20:36 -07:00
}
2007-07-17 04:05:37 -07:00
}
return 0 ;
2005-04-16 15:20:36 -07:00
}
static struct fb_ops macfb_ops = {
. owner = THIS_MODULE ,
. fb_setcolreg = macfb_setcolreg ,
. fb_fillrect = cfb_fillrect ,
. fb_copyarea = cfb_copyarea ,
. fb_imageblit = cfb_imageblit ,
} ;
2008-07-25 19:46:26 -07:00
static void __init macfb_setup ( char * options )
2005-04-16 15:20:36 -07:00
{
char * this_opt ;
2009-11-04 00:47:28 +11:00
2005-04-16 15:20:36 -07:00
if ( ! options | | ! * options )
return ;
2009-11-04 00:47:28 +11:00
2005-04-16 15:20:36 -07:00
while ( ( this_opt = strsep ( & options , " , " ) ) ! = NULL ) {
2009-11-04 00:47:28 +11:00
if ( ! * this_opt )
continue ;
if ( ! strcmp ( this_opt , " inverse " ) )
inverse = 1 ;
else
if ( ! strcmp ( this_opt , " vidtest " ) )
vidtest = 1 ; /* enable experimental CLUT code */
2005-04-16 15:20:36 -07:00
}
}
2006-12-08 02:40:09 -08:00
static void __init iounmap_macfb ( void )
{
if ( dafb_cmap_regs )
iounmap ( dafb_cmap_regs ) ;
if ( v8_brazil_cmap_regs )
iounmap ( v8_brazil_cmap_regs ) ;
if ( rbv_cmap_regs )
iounmap ( rbv_cmap_regs ) ;
if ( civic_cmap_regs )
iounmap ( civic_cmap_regs ) ;
if ( csc_cmap_regs )
iounmap ( csc_cmap_regs ) ;
}
2006-01-12 01:06:39 -08:00
static int __init macfb_init ( void )
2005-04-16 15:20:36 -07:00
{
int video_cmap_len , video_is_nubus = 0 ;
struct nubus_dev * ndev = NULL ;
char * option = NULL ;
2006-01-12 01:06:39 -08:00
int err ;
2005-04-16 15:20:36 -07:00
if ( fb_get_options ( " macfb " , & option ) )
return - ENODEV ;
macfb_setup ( option ) ;
if ( ! MACH_IS_MAC )
2006-01-12 01:06:39 -08:00
return - ENODEV ;
2005-04-16 15:20:36 -07:00
2009-11-04 00:43:52 +11:00
if ( mac_bi_data . id = = MAC_MODEL_Q630 | |
mac_bi_data . id = = MAC_MODEL_P588 )
return - ENODEV ; /* See valkyriefb.c */
2005-04-16 15:20:36 -07:00
macfb_defined . xres = mac_bi_data . dimensions & 0xFFFF ;
macfb_defined . yres = mac_bi_data . dimensions > > 16 ;
macfb_defined . bits_per_pixel = mac_bi_data . videodepth ;
2009-11-04 00:47:28 +11:00
2005-04-16 15:20:36 -07:00
macfb_fix . line_length = mac_bi_data . videorow ;
2009-11-04 00:47:28 +11:00
macfb_fix . smem_len = macfb_fix . line_length * macfb_defined . yres ;
2005-04-16 15:20:36 -07:00
/* Note: physical address (since 2.1.127) */
2009-11-04 00:47:28 +11:00
macfb_fix . smem_start = mac_bi_data . videoaddr ;
/*
* This is actually redundant with the initial mappings .
* However , there are some non - obvious aspects to the way
* those mappings are set up , so this is in fact the safest
* way to ensure that this driver will work on every possible Mac
*/
2009-11-04 00:51:18 +11:00
fb_info . screen_base = ioremap ( mac_bi_data . videoaddr ,
macfb_fix . smem_len ) ;
if ( ! fb_info . screen_base )
return - ENODEV ;
2005-04-16 15:20:36 -07:00
printk ( " macfb: framebuffer at 0x%08lx, mapped to 0x%p, size %dk \n " ,
2009-11-04 00:47:28 +11:00
macfb_fix . smem_start , fb_info . screen_base ,
macfb_fix . smem_len / 1024 ) ;
2005-04-16 15:20:36 -07:00
printk ( " macfb: mode is %dx%dx%d, linelength=%d \n " ,
2009-11-04 00:47:28 +11:00
macfb_defined . xres , macfb_defined . yres ,
macfb_defined . bits_per_pixel , macfb_fix . line_length ) ;
2009-11-04 00:51:18 +11:00
/* Fill in the available video resolution */
2009-11-04 00:47:28 +11:00
macfb_defined . xres_virtual = macfb_defined . xres ;
macfb_defined . yres_virtual = macfb_defined . yres ;
macfb_defined . height = PIXEL_TO_MM ( macfb_defined . yres ) ;
macfb_defined . width = PIXEL_TO_MM ( macfb_defined . xres ) ;
2005-04-16 15:20:36 -07:00
2009-11-04 00:51:18 +11:00
/* Some dummy values for timing to make fbset happy */
2009-11-04 00:47:28 +11:00
macfb_defined . pixclock = 10000000 / macfb_defined . xres *
1000 / macfb_defined . yres ;
2005-04-16 15:20:36 -07:00
macfb_defined . left_margin = ( macfb_defined . xres / 8 ) & 0xf8 ;
macfb_defined . hsync_len = ( macfb_defined . xres / 8 ) & 0xf8 ;
switch ( macfb_defined . bits_per_pixel ) {
case 1 :
2009-11-04 00:47:28 +11:00
/*
* XXX : I think this will catch any program that tries
* to do FBIO_PUTCMAP when the visual is monochrome .
*/
2005-04-16 15:20:36 -07:00
macfb_defined . red . length = macfb_defined . bits_per_pixel ;
macfb_defined . green . length = macfb_defined . bits_per_pixel ;
macfb_defined . blue . length = macfb_defined . bits_per_pixel ;
video_cmap_len = 0 ;
macfb_fix . visual = FB_VISUAL_MONO01 ;
break ;
case 2 :
case 4 :
case 8 :
macfb_defined . red . length = macfb_defined . bits_per_pixel ;
macfb_defined . green . length = macfb_defined . bits_per_pixel ;
macfb_defined . blue . length = macfb_defined . bits_per_pixel ;
video_cmap_len = 1 < < macfb_defined . bits_per_pixel ;
macfb_fix . visual = FB_VISUAL_PSEUDOCOLOR ;
break ;
case 16 :
macfb_defined . transp . offset = 15 ;
macfb_defined . transp . length = 1 ;
macfb_defined . red . offset = 10 ;
macfb_defined . red . length = 5 ;
macfb_defined . green . offset = 5 ;
macfb_defined . green . length = 5 ;
macfb_defined . blue . offset = 0 ;
macfb_defined . blue . length = 5 ;
video_cmap_len = 16 ;
2009-11-04 00:47:28 +11:00
/*
* Should actually be FB_VISUAL_DIRECTCOLOR , but this
* works too
*/
2005-04-16 15:20:36 -07:00
macfb_fix . visual = FB_VISUAL_TRUECOLOR ;
break ;
case 24 :
case 32 :
macfb_defined . red . offset = 16 ;
macfb_defined . red . length = 8 ;
macfb_defined . green . offset = 8 ;
macfb_defined . green . length = 8 ;
macfb_defined . blue . offset = 0 ;
macfb_defined . blue . length = 8 ;
video_cmap_len = 16 ;
macfb_fix . visual = FB_VISUAL_TRUECOLOR ;
2009-11-04 00:51:18 +11:00
break ;
2005-04-16 15:20:36 -07:00
default :
video_cmap_len = 0 ;
macfb_fix . visual = FB_VISUAL_MONO01 ;
2009-11-04 00:47:28 +11:00
printk ( " macfb: unknown or unsupported bit depth: %d \n " ,
macfb_defined . bits_per_pixel ) ;
2005-04-16 15:20:36 -07:00
break ;
}
2009-11-04 00:47:28 +11:00
/*
* We take a wild guess that if the video physical address is
* in nubus slot space , that the nubus card is driving video .
* Penguin really ought to tell us whether we are using internal
* video or not .
* Hopefully we only find one of them . Otherwise our NuBus
* code is really broken : - )
2005-04-16 15:20:36 -07:00
*/
2009-11-04 00:51:18 +11:00
while ( ( ndev = nubus_find_type ( NUBUS_CAT_DISPLAY ,
NUBUS_TYPE_VIDEO , ndev ) ) )
2005-04-16 15:20:36 -07:00
{
2009-11-04 00:51:18 +11:00
unsigned long base = ndev - > board - > slot_addr ;
if ( mac_bi_data . videoaddr < base | |
mac_bi_data . videoaddr - base > 0xFFFFFF )
2005-04-16 15:20:36 -07:00
continue ;
2009-11-04 00:51:18 +11:00
2005-04-16 15:20:36 -07:00
video_is_nubus = 1 ;
2009-11-04 00:51:18 +11:00
slot_addr = ( unsigned char * ) base ;
2005-04-16 15:20:36 -07:00
switch ( ndev - > dr_hw ) {
case NUBUS_DRHW_APPLE_MDC :
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " Mac Disp. Card " ) ;
2005-04-16 15:20:36 -07:00
macfb_setpalette = mdc_setpalette ;
macfb_defined . activate = FB_ACTIVATE_NOW ;
break ;
case NUBUS_DRHW_APPLE_TFB :
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " Toby " ) ;
2005-04-16 15:20:36 -07:00
macfb_setpalette = toby_setpalette ;
macfb_defined . activate = FB_ACTIVATE_NOW ;
break ;
case NUBUS_DRHW_APPLE_JET :
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " Jet " ) ;
2005-04-16 15:20:36 -07:00
macfb_setpalette = jet_setpalette ;
macfb_defined . activate = FB_ACTIVATE_NOW ;
2009-11-04 00:47:28 +11:00
break ;
2005-04-16 15:20:36 -07:00
default :
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " Generic NuBus " ) ;
2005-04-16 15:20:36 -07:00
break ;
}
}
/* If it's not a NuBus card, it must be internal video */
if ( ! video_is_nubus )
2009-11-04 00:47:28 +11:00
switch ( mac_bi_data . id ) {
/*
* DAFB Quadras
* Note : these first four have the v7 DAFB , which is
* known to be rather unlike the ones used in the
* other models
*/
2005-04-16 15:20:36 -07:00
case MAC_MODEL_P475 :
case MAC_MODEL_P475F :
case MAC_MODEL_P575 :
case MAC_MODEL_Q605 :
2009-11-04 00:47:28 +11:00
2005-04-16 15:20:36 -07:00
case MAC_MODEL_Q800 :
case MAC_MODEL_Q650 :
case MAC_MODEL_Q610 :
case MAC_MODEL_C650 :
case MAC_MODEL_C610 :
case MAC_MODEL_Q700 :
case MAC_MODEL_Q900 :
case MAC_MODEL_Q950 :
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " DAFB " ) ;
2005-04-16 15:20:36 -07:00
macfb_setpalette = dafb_setpalette ;
macfb_defined . activate = FB_ACTIVATE_NOW ;
dafb_cmap_regs = ioremap ( DAFB_BASE , 0x1000 ) ;
break ;
2009-11-04 00:47:28 +11:00
/*
* LC II uses the V8 framebuffer
*/
2005-04-16 15:20:36 -07:00
case MAC_MODEL_LCII :
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " V8 " ) ;
2005-04-16 15:20:36 -07:00
macfb_setpalette = v8_brazil_setpalette ;
macfb_defined . activate = FB_ACTIVATE_NOW ;
v8_brazil_cmap_regs = ioremap ( DAC_BASE , 0x1000 ) ;
break ;
2009-11-04 00:47:28 +11:00
/*
* IIvi , IIvx use the " Brazil " framebuffer ( which is
* very much like the V8 , it seems , and probably uses
* the same DAC )
*/
2005-04-16 15:20:36 -07:00
case MAC_MODEL_IIVI :
case MAC_MODEL_IIVX :
case MAC_MODEL_P600 :
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " Brazil " ) ;
2005-04-16 15:20:36 -07:00
macfb_setpalette = v8_brazil_setpalette ;
macfb_defined . activate = FB_ACTIVATE_NOW ;
v8_brazil_cmap_regs = ioremap ( DAC_BASE , 0x1000 ) ;
break ;
2009-11-04 00:47:28 +11:00
/*
* LC III ( and friends ) use the Sonora framebuffer
* Incidentally this is also used in the non - AV models
* of the x100 PowerMacs
* These do in fact seem to use the same DAC interface
* as the LC II .
*/
2005-04-16 15:20:36 -07:00
case MAC_MODEL_LCIII :
case MAC_MODEL_P520 :
case MAC_MODEL_P550 :
case MAC_MODEL_P460 :
macfb_setpalette = v8_brazil_setpalette ;
macfb_defined . activate = FB_ACTIVATE_NOW ;
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " Sonora " ) ;
2005-04-16 15:20:36 -07:00
v8_brazil_cmap_regs = ioremap ( DAC_BASE , 0x1000 ) ;
break ;
2009-11-04 00:47:28 +11:00
/*
* IIci and IIsi use the infamous RBV chip
* ( the IIsi is just a rebadged and crippled
* IIci in a different case , BTW )
*/
2005-04-16 15:20:36 -07:00
case MAC_MODEL_IICI :
case MAC_MODEL_IISI :
macfb_setpalette = rbv_setpalette ;
macfb_defined . activate = FB_ACTIVATE_NOW ;
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " RBV " ) ;
2005-04-16 15:20:36 -07:00
rbv_cmap_regs = ioremap ( DAC_BASE , 0x1000 ) ;
break ;
2009-11-04 00:47:28 +11:00
/*
* AVs use the Civic framebuffer
*/
2005-04-16 15:20:36 -07:00
case MAC_MODEL_Q840 :
case MAC_MODEL_C660 :
macfb_setpalette = civic_setpalette ;
macfb_defined . activate = FB_ACTIVATE_NOW ;
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " Civic " ) ;
2005-04-16 15:20:36 -07:00
civic_cmap_regs = ioremap ( CIVIC_BASE , 0x1000 ) ;
break ;
2009-11-04 00:47:28 +11:00
/*
* Assorted weirdos
* We think this may be like the LC II
*/
2005-04-16 15:20:36 -07:00
case MAC_MODEL_LC :
if ( vidtest ) {
macfb_setpalette = v8_brazil_setpalette ;
macfb_defined . activate = FB_ACTIVATE_NOW ;
v8_brazil_cmap_regs =
ioremap ( DAC_BASE , 0x1000 ) ;
}
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " LC " ) ;
2005-04-16 15:20:36 -07:00
break ;
2009-11-04 00:47:28 +11:00
/*
* We think this may be like the LC II
*/
2005-04-16 15:20:36 -07:00
case MAC_MODEL_CCL :
if ( vidtest ) {
macfb_setpalette = v8_brazil_setpalette ;
macfb_defined . activate = FB_ACTIVATE_NOW ;
v8_brazil_cmap_regs =
ioremap ( DAC_BASE , 0x1000 ) ;
}
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " Color Classic " ) ;
2005-04-16 15:20:36 -07:00
break ;
2009-11-04 00:47:28 +11:00
/*
* And we * do * mean " weirdos "
*/
2005-04-16 15:20:36 -07:00
case MAC_MODEL_TV :
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " Mac TV " ) ;
2005-04-16 15:20:36 -07:00
break ;
2009-11-04 00:47:28 +11:00
/*
* These don ' t have colour , so no need to worry
*/
2005-04-16 15:20:36 -07:00
case MAC_MODEL_SE30 :
case MAC_MODEL_CLII :
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " Monochrome " ) ;
2005-04-16 15:20:36 -07:00
break ;
2009-11-04 00:47:28 +11:00
/*
* Powerbooks are particularly difficult . Many of
* them have separate framebuffers for external and
* internal video , which is admittedly pretty cool ,
* but will be a bit of a headache to support here .
* Also , many of them are grayscale , and we don ' t
* really support that .
*/
2005-04-16 15:20:36 -07:00
2009-11-04 00:51:18 +11:00
/*
* Slot 0 ROM says TIM . No external video . B & W .
*/
2005-04-16 15:20:36 -07:00
case MAC_MODEL_PB140 :
case MAC_MODEL_PB145 :
case MAC_MODEL_PB170 :
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " DDC " ) ;
2005-04-16 15:20:36 -07:00
break ;
2009-11-04 00:47:28 +11:00
/*
* Internal is GSC , External ( if present ) is ViSC
*/
2005-04-16 15:20:36 -07:00
case MAC_MODEL_PB150 : /* no external video */
case MAC_MODEL_PB160 :
case MAC_MODEL_PB165 :
case MAC_MODEL_PB180 :
case MAC_MODEL_PB210 :
case MAC_MODEL_PB230 :
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " GSC " ) ;
2005-04-16 15:20:36 -07:00
break ;
2009-11-04 00:47:28 +11:00
/*
* Internal is TIM , External is ViSC
*/
2005-04-16 15:20:36 -07:00
case MAC_MODEL_PB165C :
case MAC_MODEL_PB180C :
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " TIM " ) ;
2005-04-16 15:20:36 -07:00
break ;
2009-11-04 00:47:28 +11:00
/*
* Internal is CSC , External is Keystone + Ariel .
*/
2005-04-16 15:20:36 -07:00
case MAC_MODEL_PB190 : /* external video is optional */
case MAC_MODEL_PB520 :
case MAC_MODEL_PB250 :
case MAC_MODEL_PB270C :
case MAC_MODEL_PB280 :
case MAC_MODEL_PB280C :
macfb_setpalette = csc_setpalette ;
macfb_defined . activate = FB_ACTIVATE_NOW ;
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " CSC " ) ;
2005-04-16 15:20:36 -07:00
csc_cmap_regs = ioremap ( CSC_BASE , 0x1000 ) ;
break ;
2009-11-04 00:47:28 +11:00
2005-04-16 15:20:36 -07:00
default :
2008-11-18 20:40:40 +01:00
strcpy ( macfb_fix . id , " Unknown " ) ;
2005-04-16 15:20:36 -07:00
break ;
}
fb_info . fbops = & macfb_ops ;
fb_info . var = macfb_defined ;
fb_info . fix = macfb_fix ;
fb_info . pseudo_palette = pseudo_palette ;
fb_info . flags = FBINFO_DEFAULT ;
2008-11-18 20:40:40 +01:00
err = fb_alloc_cmap ( & fb_info . cmap , video_cmap_len , 0 ) ;
if ( err )
goto fail_unmap ;
2009-11-04 00:47:28 +11:00
2006-01-12 01:06:39 -08:00
err = register_framebuffer ( & fb_info ) ;
2008-11-18 20:40:40 +01:00
if ( err )
goto fail_dealloc ;
printk ( " fb%d: %s frame buffer device \n " ,
fb_info . node , fb_info . fix . id ) ;
return 0 ;
fail_dealloc :
fb_dealloc_cmap ( & fb_info . cmap ) ;
fail_unmap :
iounmap ( fb_info . screen_base ) ;
iounmap_macfb ( ) ;
2006-01-12 01:06:39 -08:00
return err ;
2005-04-16 15:20:36 -07:00
}
module_init ( macfb_init ) ;
MODULE_LICENSE ( " GPL " ) ;