Rev 3 |
Go to most recent revision |
Blame |
Compare with Previous |
Last modification |
View Log
| RSS feed
#include <ll/i386/hw-data.h>
#include <ll/i386/mem.h>
#include <ll/i386/cons.h>
#include <ll/sys/ll/ll-func.h>
#include <drivers/llpci.h>
#include <drivers/pci.h>
#include <drivers/gd.h>
#include "drivers/s3.h"
#include "drivers/trident.h"
#include "drivers/vesa.h"
#include "drivers/chips.h"
BYTE bus
, dev
;
ModeInfoBlock vbemi
;
BYTE actbank
;
BYTE type
;
DWORD flb
=0;
int linear
;
BYTE trylinear
;
/*BYTE gmode = FALSE;*/
#define NMODES 13
static struct gmode allmodes
[NMODES
] = {
{640, 400, 640, 8, 0x100},
{640, 480, 640, 8, 0x101},
{800, 600, 800, 8, 0x103},
{1024, 768, 1024, 8, 0x105},
{640, 480, 1280, 15, 0x110},
{800, 600, 1600, 15, 0x113},
{1024, 768, 2048, 15, 0x116},
{640, 480, 1280, 16, 0x111},
{800, 600, 1600, 16, 0x114},
{1024, 768, 2048, 16, 0x117},
{640, 480, 1920, 24, 0x112},
{800, 600, 2400, 24, 0x115},
{1024, 768, 3072, 24, 0x118}
};
static struct gmode actualmode
;
static int myceil
(float f
)
{
int i
= (int) f
;
float f1
= f
- i
;
if (f1
!= 0) {
return i
+ 1;
} else {
return i
;
}
}
/************************************************************************/
/* GD Common Interface */
/************************************************************************/
int gd_init
(WORD flags
)
{
struct pci_regs
*r
;
int dummy
;
memset(&actualmode
, 0, sizeof(struct gmode
));
type
= UNKNOWN
;
if (pci_init
() == 1) {
if ((r
= pci_class
((0x0300) << 8, 0, &bus
, &dev
)) != NULL
) {
cprintf
("PCI video card found!!!\n");
cprintf
("Vendor: %x\n", r
->VendorId
);
cprintf
("Card: %x\n", r
->DeviceId
);
/* flb = r->IoBaseAddress & 0xFF800000; */
dummy
= pcibios_read_config_dword
(bus
, dev
, 16, &flb
);
flb
= flb
& 0xFF800000;
/* Err... Have to set type!!! */
cprintf
("FLB @ %lx\n", flb
);
if (flb
< 32 * 1024 * 1024) {
/*flb = 32 * 0x1000000;*/
/*flb = 0xF3000000;*/
flb
= 0x02000000;
cprintf
("Wrong FLB!!! Forced to %lx\n", flb
);
pcibios_write_config_dword
(bus
, dev
, 16, flb
);
}
}
}
/* VESA driver needs the V86 Mode... */
vm86_init
();
/* First, check VESA... */
if (vbe_getinfo
() == 1) {
type
= VESA
;
if (flags
& NOLINEAR
) {
trylinear
= 0;
} else {
trylinear
= 1;
}
}
/* Then, the other graph cards... Override VESA!!! */
if (trident_test
() == 1) {
type
= TRIDENT
;
trylinear
= 0;
}
if (s3_test
() == 1) {
type
= S3
;
if (flags
& NOLINEAR
) {
trylinear
= 0;
} else {
trylinear
= 1;
}
return 1;
}
if (type
== UNKNOWN
) {
return -1;
}
if (type
== VESA
) {
return vbe_check_id
();
}
return -2;
}
void gd_showmodeinfo
(void)
{
cprintf
("Mode number %x\n", actualmode.
modenum);
cprintf
("MaxX: %u\n", actualmode.
xdim);
cprintf
("MaxY: %u\n", actualmode.
ydim);
cprintf
("BPR: %u\n", actualmode.
bpr);
}
void gd_showinfo
(void)
{
DWORD addr
;
addr
= gd_getflb
();
if (addr
!= 0) {
cprintf
("\t gd: Frame Linear Buffer @ %lx (%luM)\n",
addr
, (addr
/ 0x1000000));
}
if (type
== VESA
) {
cprintf
("Vesa SVGA card\n");
vbe_showinfo
();
}
if (type
== TRIDENT
) {
cprintf
("Trident graphic card\n");
trident_showinfo
();
}
if (type
== S3
) {
cprintf
("S3 graphic card\n");
s3_showinfo
();
}
}
int gd_setmode
(WORD mode
)
{
WORD app
;
int i
;
#ifndef VM86
BYTE p1
, p2
;
#endif
if (mode
== 0) {
X_REGS16 inregs
, outregs
;
X_SREGS16 sregs
;
if (actualmode.
modenum != 0) {
inregs.
x.
ax = 0x03;
#ifndef VM86
p1
= inp
(0x21);
p2
= inp
(0xA1);
outp
(0x21,0xFF);
outp
(0xA1,0xFF);
X_callBIOS
(0x10, &inregs
, &outregs
, &sregs
);
outp
(0x21,p1
);
outp
(0xA1,p2
);
#else
vm86_callBIOS
(0x10, &inregs
, &outregs
, &sregs
);
#endif
}
return 1;
}
for (i
= 0; i
< NMODES
; i
++) {
if ((mode
== allmodes
[i
].
modenum)) {
memcpy(&actualmode
, &(allmodes
[i
]), sizeof(struct gmode
));
}
}
if (type
== VESA
) {
if (vbe_getmodeinfo
(&vbemi
, mode
) < 0) {
return -1;
}
app
= mode
;
if (trylinear
) {
/* try linear... */
cprintf
("Mode: %x\n", mode
);
mode
= mode
| 0x4000;
cprintf
("Lin_Mode: %x\n", mode
);
linear
= TRUE
;
if (vbe_setmode
(mode
) == 1) {
return linear
;
}
}
actbank
= 0;
Load_Write_Bank_256
(0);
linear
= FALSE
;
mode
= app
;
if (vbe_setmode
(mode
) < 0) {
return -1;
}
return linear
;
}
if (type
== S3
) {
/* Still use VESA to open graph... */
if (vbe_getmodeinfo
(&vbemi
, mode
) < 0) {
return -1;
}
if (vbe_setmode
(mode
) < 0) {
return -1;
}
if ((flb
!= 0) && trylinear
) {
linear
= TRUE
;
s3_linear
(flb
);
pcibios_write_config_dword
(bus
, dev
, 16, flb
);
return TRUE
;
}
linear
= FALSE
;
actbank
= 0;
Load_Write_Bank_256
(0);
return FALSE
;
}
if (type
== TRIDENT
) {
/* Still use VESA to open graph... */
if (vbe_getmodeinfo
(&vbemi
, mode
) < 0) {
return -1;
}
if (vbe_setmode
(mode
) < 0) {
return -1;
}
actbank
= 0;
Load_Write_Bank_256
(0);
return FALSE
;
}
return -2;
}
DWORD gd_getflb
(void)
{
if (type
== VESA
) {
return vbe_getflb
();
}
if (type
== S3
) {
return flb
;
}
return 0;
}
DWORD gd_getmem
(void)
{
if (type
== VESA
) {
return vbe_getmem
();
}
if (type
== S3
) {
return s3_getmem
();
}
if (type
== TRIDENT
) {
return trident_getmem
();
}
return 0;
}
WORD gd_getbpr
(void)
{
if (actualmode.
bpr) return actualmode.
bpr;
return -2;
}
int gd_getmodeinfo
(vga_modeinfo
*m
)
{
if (actualmode.
modenum == 0) {
return -1;
}
m
->width
= actualmode.
xdim;
m
->height
= actualmode.
ydim;
m
->bytesperpixel
= (myceil
((float)actualmode.
bpp / 8));
/* m->colors = myexp(2, actualmode.bpp); */
m
->colors
= 1 << actualmode.
bpp;
m
->linewidth
= gd_getbpr
();
m
->maxlogicalwidth
= gd_getbpr
(); /* 4 the moment... */
m
->startaddressrange
= 0; /* don't know :( */
m
->maxpixels
= gd_getmem
() / m
->bytesperpixel
;
m
->haveblit
= 0;
m
->flags
= 0;
/* Extended fields: */
m
->chiptype
= type
; /* Chiptype detected */
m
->memory
= gd_getmem
();
m
->linewidth_unit
= 0; /* don't know :( */
if (linear
) {
m
->linear_aperture
= (LIN_ADDR
)gd_getflb
();
m
->aperture_size
= gd_getmem
();
} else {
m
->linear_aperture
= 0;
m
->aperture_size
= 0xFFFF;
}
m
->set_aperture_page
= NULL
;
m
->extensions
= NULL
;
return 1;
}
void Load_Write_Bank_256
(BYTE bank
)
{
if (bank
!= actbank
) {
actbank
= bank
;
if (type
== VESA
) {
if (vbe_setbank
(&vbemi
, bank
) < 0)
ll_abort
(259);
}
if (type
== S3
) {
s3_setpage
(bank
);
}
if (type
== TRIDENT
) {
trident_setpage
(bank
);
}
}
}
/************************************************************************/
/* BANK HANDLING FUNCS */
/************************************************************************/
LIN_ADDR start_address
(WORD bpr
, WORD x
, WORD y
)
{
LIN_ADDR a
;
a
= (LIN_ADDR
)(y
* bpr
+x
);
return a
;
}
void Seg_Off_256
(WORD x
, WORD y
, WORD pitch
, WORD
*offs
, WORD
*seg
)
{
DWORD a
;
a
= y
* pitch
+ x
;
*offs
= a
& 0x0000FFFF;
*seg
= a
>> 16;
}
void Seg_Off_Hi
(WORD x
, WORD y
, WORD pitch
, WORD
*offs
, WORD
*seg
)
{
DWORD a
;
a
= y
* pitch
+ x
* 2;
*offs
= a
& 0x0000FFFF;
*seg
= a
>> 16;
}
int gd_modenum
(WORD x
, WORD y
, BYTE depth
)
{
int mode
, i
;
mode
= -1;
for (i
= 0; i
< NMODES
; i
++) {
if ((depth
== allmodes
[i
].
bpp) && (x
== allmodes
[i
].
xdim) && (y
== allmodes
[i
].
ydim)) {
mode
= allmodes
[i
].
modenum;
} else {
}
}
if ((mode
!= -1) && (vbe_checkmode
(mode
) != -1))
return mode
;
return -1;
}