Commit e9776274 authored by Jim Hague's avatar Jim Hague Committed by Linus Torvalds

[PATCH] pm2fb: fix big-endian (Sparc) support

A collection of changes that taken together makes the driver work correctly on
big-endian systems, specifically Sparc.

- Remove old PM2FB_BE_APERTURE define and use __BIG_ENDIAN
  throughout. PM2FB_BE_APERTURE wasn't defined on Sparc; this
  was incorrect.
- Replace colour mode magic numbers with suitable constants,
  and tidy colour mode handling. Use BGR not RGB at 24bit on
  big-endian.
- Replace aperture magic numbers with constants. Rearrange code to
  emphasise second aperture is not used. Add short explanations
  of aperture settings for big endian.
- Update comments to note that the driver now works on Sparc.
- Revisit 32bit depth colour offsets and ensure transp is set.
Signed-off-by: default avatarJim Hague <jim.hague@acm.org>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent ec84153d
...@@ -11,11 +11,11 @@ ...@@ -11,11 +11,11 @@
* and additional input from James Simmon's port of Hannu Mallat's tdfx * and additional input from James Simmon's port of Hannu Mallat's tdfx
* driver. * driver.
* *
* $Id$ * I have a Creative Graphics Blaster Exxtreme card - pm2fb on x86. I
* * have no access to other pm2fb implementations. Sparc (and thus
* I have a Creative Graphics Blaster Exxtreme card - pm2fb on x86. * hopefully other big-endian) devices now work, thanks to a lot of
* I have no access to other pm2fb implementations, and cannot test * testing work by Ron Murray. I have no access to CVision hardware,
* on them. Therefore for now I am omitting Sparc and CVision code. * and therefore for now I am omitting the CVision code.
* *
* Multiple boards support has been on the TODO list for ages. * Multiple boards support has been on the TODO list for ages.
* Don't expect this to change. * Don't expect this to change.
...@@ -48,10 +48,6 @@ ...@@ -48,10 +48,6 @@
#error "The endianness of the target host has not been defined." #error "The endianness of the target host has not been defined."
#endif #endif
#if defined(__BIG_ENDIAN) && !defined(__sparc__)
#define PM2FB_BE_APERTURE
#endif
#if !defined(CONFIG_PCI) #if !defined(CONFIG_PCI)
#error "Only generic PCI cards supported." #error "Only generic PCI cards supported."
#endif #endif
...@@ -424,27 +420,36 @@ static void reset_config(struct pm2fb_par* p) ...@@ -424,27 +420,36 @@ static void reset_config(struct pm2fb_par* p)
static void set_aperture(struct pm2fb_par* p, u32 depth) static void set_aperture(struct pm2fb_par* p, u32 depth)
{ {
/*
* The hardware is little-endian. When used in big-endian
* hosts, the on-chip aperture settings are used where
* possible to translate from host to card byte order.
*/
WAIT_FIFO(p, 4); WAIT_FIFO(p, 4);
#ifdef __LITTLE_ENDIAN #ifdef __LITTLE_ENDIAN
pm2_WR(p, PM2R_APERTURE_ONE, 0); pm2_WR(p, PM2R_APERTURE_ONE, PM2F_APERTURE_STANDARD);
pm2_WR(p, PM2R_APERTURE_TWO, 0);
#else #else
switch (depth) { switch (depth) {
case 8: case 24: /* RGB->BGR */
case 24: /*
pm2_WR(p, PM2R_APERTURE_ONE, 0); * We can't use the aperture to translate host to
pm2_WR(p, PM2R_APERTURE_TWO, 1); * card byte order here, so we switch to BGR mode
* in pm2fb_set_par().
*/
case 8: /* B->B */
pm2_WR(p, PM2R_APERTURE_ONE, PM2F_APERTURE_STANDARD);
break; break;
case 16: case 16: /* HL->LH */
pm2_WR(p, PM2R_APERTURE_ONE, 2); pm2_WR(p, PM2R_APERTURE_ONE, PM2F_APERTURE_HALFWORDSWAP);
pm2_WR(p, PM2R_APERTURE_TWO, 1);
break; break;
case 32: case 32: /* RGBA->ABGR */
pm2_WR(p, PM2R_APERTURE_ONE, 1); pm2_WR(p, PM2R_APERTURE_ONE, PM2F_APERTURE_BYTESWAP);
pm2_WR(p, PM2R_APERTURE_TWO, 1);
break; break;
} }
#endif #endif
// We don't use aperture two, so this may be superflous
pm2_WR(p, PM2R_APERTURE_TWO, PM2F_APERTURE_STANDARD);
} }
static void set_color(struct pm2fb_par* p, unsigned char regno, static void set_color(struct pm2fb_par* p, unsigned char regno,
...@@ -638,12 +643,10 @@ static int pm2fb_check_var(struct fb_var_screeninfo *var, struct fb_info *info) ...@@ -638,12 +643,10 @@ static int pm2fb_check_var(struct fb_var_screeninfo *var, struct fb_info *info)
var->blue.offset = 0; var->blue.offset = 0;
var->blue.length = 5; var->blue.length = 5;
break; break;
case 24:
var->red.offset = 16;
var->green.offset = 8;
var->blue.offset = 0;
var->red.length = var->green.length = var->blue.length = 8;
case 32: case 32:
var->transp.offset = 24;
var->transp.length = 8;
case 24:
var->red.offset = 16; var->red.offset = 16;
var->green.offset = 8; var->green.offset = 8;
var->blue.offset = 0; var->blue.offset = 0;
...@@ -676,7 +679,7 @@ static int pm2fb_set_par(struct fb_info *info) ...@@ -676,7 +679,7 @@ static int pm2fb_set_par(struct fb_info *info)
u32 stride; u32 stride;
u32 base; u32 base;
u32 video = 0; u32 video = 0;
u32 clrmode = PM2F_RD_COLOR_MODE_RGB; u32 clrmode = PM2F_RD_COLOR_MODE_RGB | PM2F_RD_GUI_ACTIVE;
u32 txtmap = 0; u32 txtmap = 0;
u32 pixsize = 0; u32 pixsize = 0;
u32 clrformat = 0; u32 clrformat = 0;
...@@ -771,22 +774,23 @@ static int pm2fb_set_par(struct fb_info *info) ...@@ -771,22 +774,23 @@ static int pm2fb_set_par(struct fb_info *info)
break; break;
case 16: case 16:
pm2_WR(par, PM2R_FB_READ_PIXEL, 1); pm2_WR(par, PM2R_FB_READ_PIXEL, 1);
clrmode |= PM2F_RD_TRUECOLOR | 0x06; clrmode |= PM2F_RD_TRUECOLOR | PM2F_RD_PIXELFORMAT_RGB565;
txtmap = PM2F_TEXTEL_SIZE_16; txtmap = PM2F_TEXTEL_SIZE_16;
pixsize = 1; pixsize = 1;
clrformat = 0x70; clrformat = 0x70;
break; break;
case 32: case 32:
pm2_WR(par, PM2R_FB_READ_PIXEL, 2); pm2_WR(par, PM2R_FB_READ_PIXEL, 2);
clrmode |= PM2F_RD_TRUECOLOR | 0x08; clrmode |= PM2F_RD_TRUECOLOR | PM2F_RD_PIXELFORMAT_RGBA8888;
txtmap = PM2F_TEXTEL_SIZE_32; txtmap = PM2F_TEXTEL_SIZE_32;
pixsize = 2; pixsize = 2;
clrformat = 0x20; clrformat = 0x20;
break; break;
case 24: case 24:
pm2_WR(par, PM2R_FB_READ_PIXEL, 4); pm2_WR(par, PM2R_FB_READ_PIXEL, 4);
clrmode |= PM2F_RD_TRUECOLOR | 0x09; clrmode |= PM2F_RD_TRUECOLOR | PM2F_RD_PIXELFORMAT_RGB888;
#ifndef PM2FB_BE_APERTURE #ifdef __BIG_ENDIAN
/* Use BGR not RGB */
clrmode &= ~PM2F_RD_COLOR_MODE_RGB; clrmode &= ~PM2F_RD_COLOR_MODE_RGB;
#endif #endif
txtmap = PM2F_TEXTEL_SIZE_24; txtmap = PM2F_TEXTEL_SIZE_24;
...@@ -819,8 +823,7 @@ static int pm2fb_set_par(struct fb_info *info) ...@@ -819,8 +823,7 @@ static int pm2fb_set_par(struct fb_info *info)
WAIT_FIFO(par, 4); WAIT_FIFO(par, 4);
switch (par->type) { switch (par->type) {
case PM2_TYPE_PERMEDIA2: case PM2_TYPE_PERMEDIA2:
pm2_RDAC_WR(par, PM2I_RD_COLOR_MODE, pm2_RDAC_WR(par, PM2I_RD_COLOR_MODE, clrmode);
PM2F_RD_COLOR_MODE_RGB | PM2F_RD_GUI_ACTIVE | clrmode);
break; break;
case PM2_TYPE_PERMEDIA2V: case PM2_TYPE_PERMEDIA2V:
pm2v_RDAC_WR(par, PM2VI_RD_PIXEL_SIZE, pixsize); pm2v_RDAC_WR(par, PM2VI_RD_PIXEL_SIZE, pixsize);
...@@ -1086,9 +1089,15 @@ static int __devinit pm2fb_probe(struct pci_dev *pdev, ...@@ -1086,9 +1089,15 @@ static int __devinit pm2fb_probe(struct pci_dev *pdev,
pm2fb_fix.mmio_start = pci_resource_start(pdev, 0); pm2fb_fix.mmio_start = pci_resource_start(pdev, 0);
pm2fb_fix.mmio_len = PM2_REGS_SIZE; pm2fb_fix.mmio_len = PM2_REGS_SIZE;
#ifdef PM2FB_BE_APERTURE #if defined(__BIG_ENDIAN)
/*
* PM2 has a 64k register file, mapped twice in 128k. Lower
* map is little-endian, upper map is big-endian.
*/
pm2fb_fix.mmio_start += PM2_REGS_SIZE; pm2fb_fix.mmio_start += PM2_REGS_SIZE;
DPRINTK("Adjusting register base for big-endian.\n");
#endif #endif
DPRINTK("Register base at 0x%lx\n", pm2fb_fix.mmio_start);
/* Registers - request region and map it. */ /* Registers - request region and map it. */
if ( !request_mem_region(pm2fb_fix.mmio_start, pm2fb_fix.mmio_len, if ( !request_mem_region(pm2fb_fix.mmio_start, pm2fb_fix.mmio_len,
......
...@@ -188,6 +188,14 @@ ...@@ -188,6 +188,14 @@
#define PM2F_VSYNC_ACT_LOW 0x60 #define PM2F_VSYNC_ACT_LOW 0x60
#define PM2F_LINE_DOUBLE 0x04 #define PM2F_LINE_DOUBLE 0x04
#define PM2F_VIDEO_ENABLE 0x01 #define PM2F_VIDEO_ENABLE 0x01
#define PM2F_RD_PIXELFORMAT_SVGA 0x01
#define PM2F_RD_PIXELFORMAT_RGB232OFFSET 0x02
#define PM2F_RD_PIXELFORMAT_RGBA2321 0x03
#define PM2F_RD_PIXELFORMAT_RGBA5551 0x04
#define PM2F_RD_PIXELFORMAT_RGBA4444 0x05
#define PM2F_RD_PIXELFORMAT_RGB565 0x06
#define PM2F_RD_PIXELFORMAT_RGBA8888 0x08
#define PM2F_RD_PIXELFORMAT_RGB888 0x09
#define PM2F_RD_GUI_ACTIVE 0x10 #define PM2F_RD_GUI_ACTIVE 0x10
#define PM2F_RD_COLOR_MODE_RGB 0x20 #define PM2F_RD_COLOR_MODE_RGB 0x20
#define PM2F_DELTA_ORDER_RGB (1L<<18) #define PM2F_DELTA_ORDER_RGB (1L<<18)
...@@ -209,6 +217,9 @@ ...@@ -209,6 +217,9 @@
#define PM2F_MEM_BANKS_2 (1L<<29) #define PM2F_MEM_BANKS_2 (1L<<29)
#define PM2F_MEM_BANKS_3 (2L<<29) #define PM2F_MEM_BANKS_3 (2L<<29)
#define PM2F_MEM_BANKS_4 (3L<<29) #define PM2F_MEM_BANKS_4 (3L<<29)
#define PM2F_APERTURE_STANDARD 0
#define PM2F_APERTURE_BYTESWAP 1
#define PM2F_APERTURE_HALFWORDSWAP 2
typedef enum { typedef enum {
PM2_TYPE_PERMEDIA2, PM2_TYPE_PERMEDIA2,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment