## diffname pc/vgat2r4.c 1999/1007
## diff -e /dev/null /n/emeliedump/1999/1007/sys/src/brazil/pc/vgat2r4.c
0a
#include "u.h"
#include "../port/lib.h"
#include "mem.h"
#include "dat.h"
#include "fns.h"
#include "io.h"
#include "../port/error.h"
#define Image IMAGE
#include <draw.h>
#include <memdraw.h>
#include <cursor.h>
#include "screen.h"
/*
* #9 Ticket to Ride IV.
*/
enum {
IndexLo = 4,
IndexHi = 5,
Data = 6,
IndexCtl = 7,
};
enum { /* index registers */
CursorCtl = 0x30,
CursorXLo = 0x31,
CursorXHi = 0x32,
CursorYLo = 0x33,
CursorYHi = 0x34,
CursorHotX = 0x35,
CursorHotY = 0x36,
CursorR1 = 0x40,
CursorG1 = 0x41,
CursorB1 = 0x42,
CursorR2 = 0x43,
CursorG2 = 0x44,
CursorB2 = 0x45,
CursorR3 = 0x46,
CursorG3 = 0x47,
CursorB3 = 0x48,
CursorArray = 0x100,
};
static ulong
t2r4linear(VGAscr* scr, int* size, int* align)
{
ulong aperture, oaperture;
int oapsize, wasupamem;
Pcidev *p;
oaperture = scr->aperture;
oapsize = scr->apsize;
wasupamem = scr->isupamem;
aperture = 0;
if(p = pcimatch(nil, 0x105D, 0)){
switch(p->did){
case 0x5348:
aperture = p->mem[0].bar & ~0x0F;
*size = p->mem[0].size;
break;
default:
break;
}
}
if(wasupamem){
if(oaperture == aperture)
return oaperture;
upafree(oaperture, oapsize);
}
scr->isupamem = 0;
aperture = upamalloc(aperture, *size, *align);
if(aperture == 0){
if(wasupamem && upamalloc(oaperture, oapsize, 0)){
aperture = oaperture;
scr->isupamem = 1;
}
else
scr->isupamem = 0;
}
else
scr->isupamem = 1;
return aperture;
}
static void
t2r4enable(VGAscr* scr)
{
Pcidev *p;
Physseg seg;
int size, align;
ulong aperture;
/*
* Only once, can't be disabled for now.
* scr->io holds the virtual address of
* the MMIO registers.
*/
if(scr->io)
return;
if(p = pcimatch(nil, 0x105D, 0)){
switch(p->did){
case 0x5348:
break;
default:
return;
}
}
else
return;
scr->io = upamalloc(p->mem[4].bar & ~0x0F, p->mem[4].size, 0);
if(scr->io == 0)
return;
memset(&seg, 0, sizeof(seg));
seg.attr = SG_PHYSICAL;
seg.name = smalloc(NAMELEN);
snprint(seg.name, NAMELEN, "t2r4mmio");
seg.pa = scr->io;
seg.size = p->mem[4].size;
addphysseg(&seg);
scr->io = (ulong)KADDR(scr->io);
size = p->mem[0].size;
align = 0;
aperture = t2r4linear(scr, &size, &align);
if(aperture){
scr->aperture = aperture;
scr->apsize = size;
memset(&seg, 0, sizeof(seg));
seg.attr = SG_PHYSICAL;
seg.name = smalloc(NAMELEN);
snprint(seg.name, NAMELEN, "t2r4screen");
seg.pa = aperture;
seg.size = size;
addphysseg(&seg);
}
}
static void
t2r4xo(VGAscr* scr, int index, uchar data)
{
ulong *mmio;
mmio = (ulong*)scr->io;
mmio[IndexLo] = index & 0xFF;
mmio[IndexHi] = (index>>8) & 0xFF;
mmio[Data] = data;
}
static void
t2r4curdisable(VGAscr* scr)
{
if(scr->io == 0)
return;
t2r4xo(scr, CursorCtl, 0x00);
}
static void
t2r4curload(VGAscr* scr, Cursor* curs)
{
int x, y;
ulong *mmio;
uchar p, p0, p1;
if(scr->io == 0)
return;
mmio = (ulong*)scr->io;
/*
* Make sure cursor is off by initialising the cursor
* control to defaults.
*/
t2r4xo(scr, CursorCtl, 0x00);
/*
* Set auto-increment mode for index-register addressing
* and initialise the cursor array index.
*/
mmio[IndexCtl] = 0x01;
mmio[IndexLo] = CursorArray & 0xFF;
mmio[IndexHi] = (CursorArray>>8) & 0xFF;
/*
* Initialise the 32x32 cursor RAM array. There are 2 planes,
* p0 and p1. Data is written 4 pixels per byte, with p1 the
* MS bit of each pixel.
* The cursor is set in X-Windows mode which gives the following
* truth table:
* p1 p0 colour
* 0 0 underlying pixel colour
* 0 1 underlying pixel colour
* 1 0 cursor colour 1
* 1 1 cursor colour 2
* Put the cursor into the top-left of the 32x32 array.
*/
for(y = 0; y < 32; y++){
for(x = 0; x < 32/8; x++){
if(x < 16/8 && y < 16){
p0 = curs->clr[x+y*2];
p1 = curs->set[x+y*2];
p = 0x00;
if(p1 & 0x80)
p |= 0xC0;
else if(p0 & 0x80)
p |= 0x80;
if(p1 & 0x40)
p |= 0x30;
else if(p0 & 0x40)
p |= 0x20;
if(p1 & 0x20)
p |= 0x0C;
else if(p0 & 0x20)
p |= 0x08;
if(p1 & 0x10)
p |= 0x03;
else if(p0 & 0x10)
p |= 0x02;
mmio[Data] = p;
p = 0x00;
if(p1 & 0x08)
p |= 0xC0;
else if(p0 & 0x08)
p |= 0x80;
if(p1 & 0x04)
p |= 0x30;
else if(p0 & 0x04)
p |= 0x20;
if(p1 & 0x02)
p |= 0x0C;
else if(p0 & 0x02)
p |= 0x08;
if(p1 & 0x01)
p |= 0x03;
else if(p0 & 0x01)
p |= 0x02;
mmio[Data] = p;
}
else{
mmio[Data] = 0x00;
mmio[Data] = 0x00;
}
}
}
/*
* Initialise the cursor hotpoint,
* enable the cursor and restore state.
*/
t2r4xo(scr, CursorHotX, -curs->offset.x);
t2r4xo(scr, CursorHotY, -curs->offset.y);
t2r4xo(scr, CursorCtl, 0x23);
}
static int
t2r4curmove(VGAscr* scr, Point p)
{
if(scr->io == 0)
return 1;
t2r4xo(scr, CursorXLo, p.x & 0xFF);
t2r4xo(scr, CursorXHi, (p.x>>8) & 0x0F);
t2r4xo(scr, CursorYLo, p.y & 0xFF);
t2r4xo(scr, CursorYHi, (p.y>>8) & 0x0F);
return 0;
}
static void
t2r4curenable(VGAscr* scr)
{
t2r4enable(scr);
if(scr->io == 0)
return;
/*
* Make sure cursor is off by initialising the cursor
* control to defaults.
*/
t2r4xo(scr, CursorCtl, 0x00);
/*
* Cursor colour 1 (white),
* cursor colour 2 (black).
*/
t2r4xo(scr, CursorR1, Pwhite);
t2r4xo(scr, CursorG1, Pwhite);
t2r4xo(scr, CursorB1, Pwhite);
t2r4xo(scr, CursorR2, Pblack);
t2r4xo(scr, CursorG2, Pblack);
t2r4xo(scr, CursorB2, Pblack);
/*
* Load, locate and enable the cursor, 32x32, mode 2.
*/
t2r4curload(scr, &arrow);
t2r4curmove(scr, ZP);
t2r4xo(scr, CursorCtl, 0x23);
}
VGAdev vgat2r4dev = {
"t2r4",
t2r4enable,
nil,
nil,
t2r4linear,
};
VGAcur vgat2r4cur = {
"t2r4hwgc",
t2r4curenable,
t2r4curdisable,
t2r4curload,
t2r4curmove,
};
.
## diffname pc/vgat2r4.c 1999/1008
## diff -e /n/emeliedump/1999/1007/sys/src/brazil/pc/vgat2r4.c /n/emeliedump/1999/1008/sys/src/brazil/pc/vgat2r4.c
310c
t2r4xo(scr, CursorCtl, CursorMode);
.
306c
* Load, locate and enable the cursor, 64x64, mode 2.
.
284c
if(scr->mmio == 0)
.
276a
zoom = (scr->mmio[Zoom] & 0x0F)+1;
y = p.y*zoom;
t2r4xo(scr, CursorYLo, y & 0xFF);
t2r4xo(scr, CursorYHi, (y>>8) & 0x0F);
.
274,275d
269c
int y, zoom;
if(scr->mmio == 0)
.
263c
t2r4xo(scr, CursorCtl, CursorMode);
.
257,258c
* Initialise the hotpoint and enable the cursor.
.
254a
mmio[IndexCtl] = 0x00;
.
253a
y++;
.
206,252c
*data = 0xFF;
*data = 0xAA;
*data = 0xFF;
*data = 0xAA;
*data = 0x00;
*data = 0x00;
*data = 0x00;
*data = 0x00;
*data = 0x00;
*data = 0x00;
*data = 0x00;
*data = 0x00;
*data = 0x00;
*data = 0x00;
*data = 0x00;
*data = 0x00;
}
while(y < 64){
for(x = 0; x < 64/8; x++){
*data = 0x00;
*data = 0x00;
.
204a
data = (uchar*)&mmio[Data];
.
203c
* Put the cursor into the top-left of the array.
.
193c
* Initialise the cursor RAM array. There are 2 planes,
.
176d
174c
mmio = scr->mmio;
if(mmio == 0)
.
172c
uchar *data, p, p0, p1;
.
162c
if(scr->mmio == 0)
.
152c
mmio = scr->mmio;
.
146a
static uchar
t2r4xi(VGAscr* scr, int index)
{
ulong *mmio;
mmio = scr->mmio;
mmio[IndexLo] = index & 0xFF;
mmio[IndexHi] = (index>>8) & 0xFF;
return mmio[Data];
}
.
129c
scr->mmio = KADDR(mmio);
.
125c
seg.pa = mmio;
.
117,118c
mmio = upamalloc(p->mem[4].bar & ~0x0F, p->mem[4].size, 0);
if(mmio == 0)
.
105c
if(scr->mmio)
.
102c
* scr->mmio holds the virtual address of
.
98c
ulong aperture, mmio;
.
44a
CursorMode32x32 = 0x03,
CursorMode64x64 = 0x07,
CursorMode = CursorMode64x64,
.
19,22c
IndexLo = 0x10/4,
IndexHi = 0x14/4,
Data = 0x18/4,
IndexCtl = 0x1C/4,
Zoom = 0x54/4,
.
## diffname pc/vgat2r4.c 1999/1009
## diff -e /n/emeliedump/1999/1008/sys/src/brazil/pc/vgat2r4.c /n/emeliedump/1999/1009/sys/src/brazil/pc/vgat2r4.c
224,228c
for(y = 0; y < zoom; y++){
clr = (curs->clr[2*y]<<8)|curs->clr[y*2 + 1];
set = (curs->set[2*y]<<8)|curs->set[y*2 + 1];
pixels = 0;
for(x = 0; x < 16; x++){
if(set & (1<<x))
pixels |= 0x03<<(x*2);
else if(clr & (1<<x))
pixels |= 0x02<<(x*2);
}
*data = pixels>>24;
*data = pixels>>16;
*data = pixels>>8;
*data = pixels;
.
222a
if(mmio[Zoom] & 0x0F)
zoom = 32;
else
zoom = 16;
.
221a
*
* Although this looks a lot like the IBM RGB524 cursor, the
* scanlines appear to be twice as long as they should be and
* some of the other features are missing.
.
188,190c
uchar *data;
int x, y, zoom;
ulong clr, *mmio, pixels, set;
.
48,49c
CursorMode32x32 = 0x23,
CursorMode64x64 = 0x27,
.
## diffname pc/vgat2r4.c 1999/1105
## diff -e /n/emeliedump/1999/1009/sys/src/brazil/pc/vgat2r4.c /n/emeliedump/1999/1105/sys/src/9/pc/vgat2r4.c
337a
t2r4drawinit,
.
330a
static void
t2r4drawinit(VGAscr *scr)
{
scr->fill = nil;
scr->scroll = nil;
}
.
## diffname pc/vgat2r4.c 1999/1127
## diff -e /n/emeliedump/1999/1105/sys/src/9/pc/vgat2r4.c /n/emeliedump/1999/1127/sys/src/9/pc/vgat2r4.c
355a
.
334,335c
ulong pitch;
int depth;
int fmt;
ulong *d;
pitch = Dx(scr->gscreen->r);
depth = scr->gscreen->depth;
switch(scr->gscreen->chan){
case RGB16:
fmt = 3;
break;
case XRGB32:
fmt = 2;
break;
case RGB15:
fmt = 1;
break;
default:
scr->fill = nil;
scr->scroll = nil;
return;
}
d = scr->mmio + RBaseD;
d[BufCtl] = fmt<<24;
d[DeSorg] = 0;
d[DeDorg] = 0;
d[DeSptch] = (pitch*depth)/8;
d[DeDptch] = (pitch*depth)/8;
d[CmdClp] = 0; /* 2 = inside rectangle */
d[Mask] = ~0;
d[DeKey] = 0;
d[Clptl] = 0;
d[Clpbr] = 0xFFF0FFF0;
d[Alpha] = 0;
d[ACtl] = 0;
scr->fill = t2r4hwfill;
scr->scroll = t2r4hwscroll;
.
331a
waitforfifo(VGAscr *scr)
{
int x;
ulong *d;
x = 0;
d = scr->mmio + RBaseD;
while((d[Busy]&1) && x++ < 1000000)
;
if(x >= 1000000) /* shouldn't happen */
iprint("busy %8lux\n", d[Busy]);
}
/* wait until command has finished executing */
static void
waitforcmd(VGAscr *scr)
{
int x;
ulong *d;
x = 0;
d = scr->mmio + RBaseD;
while((d[Flow]&1) && x++ < 1000000)
;
if(x >= 1000000) /* shouldn't happen */
iprint("flow %8lux\n", d[Busy]);
}
/* wait until memory controller not busy (i.e. wait for writes to flush) */
static void
waitformem(VGAscr *scr)
{
int x;
ulong *d;
x = 0;
d = scr->mmio + RBaseD;
while((d[Flow]&2)&& x++ < 1000000)
;
if(x >= 1000000) /* shouldn't happen */
iprint("mem %8lux\n", d[Busy]);
}
static int
t2r4hwscroll(VGAscr *scr, Rectangle r, Rectangle sr)
{
int ctl;
Point dp, sp;
ulong *d;
waitformem(scr);
waitforfifo(scr);
d = scr->mmio + RBaseD;
ctl = 0;
if(r.min.x <= sr.min.x){
dp.x = r.min.x;
sp.x = sr.min.x;
}else{
ctl |= 2;
dp.x = r.max.x-1;
sp.x = sr.max.x-1;
}
if(r.min.y <= sr.min.y){
dp.y = r.min.y;
sp.y = sr.min.y;
}else{
ctl |= 1;
dp.y = r.max.y-1;
sp.y = sr.max.y-1;
}
d[CmdOpc] = 0x1; /* bitblt */
d[CmdRop] = 0xC; /* copy source */
d[CmdStyle] = 0;
d[CmdPatrn] = 0;
/* writing XY1 executes cmd */
d[XY3] = ctl;
d[XY0] = (sp.x<<16)|sp.y;
d[XY2] = (Dx(r)<<16)|Dy(r);
d[XY4] = 0;
d[XY1] = (dp.x<<16)|dp.y;
waitforcmd(scr);
return 1;
}
static int
t2r4hwfill(VGAscr *scr, Rectangle r, ulong sval)
{
ulong *d;
d = scr->mmio + RBaseD;
waitformem(scr);
waitforfifo(scr);
d[CmdOpc] = 0x1; /* bitblt */
d[CmdRop] = 0xC; /* copy source */
d[CmdStyle] = 1; /* use source from Fore register */
d[CmdPatrn] = 0; /* no stipple */
d[Fore] = sval;
/* writing XY1 executes cmd */
d[XY3] = 0;
d[XY0] = (r.min.x<<16)|r.min.y;
d[XY2] = (Dx(r)<<16)|Dy(r);
d[XY4] = 0;
d[XY1] = (r.min.x<<16)|r.min.y;
waitforcmd(scr);
return 1;
}
static void
.
330a
enum {
Flow = 0x08/4,
Busy = 0x0C/4,
BufCtl = 0x20/4,
DeSorg = 0x28/4,
DeDorg = 0x2C/4,
DeSptch = 0x40/4,
DeDptch = 0x44/4,
CmdOpc = 0x50/4,
CmdRop = 0x54/4,
CmdStyle = 0x58/4,
CmdPatrn = 0x5C/4,
CmdClp = 0x60/4,
CmdPf = 0x64/4,
Fore = 0x68/4,
Back = 0x6C/4,
Mask = 0x70/4,
DeKey = 0x74/4,
Lpat = 0x78/4,
Pctrl = 0x7C/4,
Clptl = 0x80/4,
Clpbr = 0x84/4,
XY0 = 0x88/4,
XY1 = 0x8C/4,
XY2 = 0x90/4,
XY3 = 0x94/4,
XY4 = 0x98/4,
Alpha = 0x128/4,
ACtl = 0x16C/4,
RBaseD = 0x4000/4,
};
/* wait until pipeline ready for new command */
.
## diffname pc/vgat2r4.c 1999/1212
## diff -e /n/emeliedump/1999/1127/sys/src/9/pc/vgat2r4.c /n/emeliedump/1999/1212/sys/src/9/pc/vgat2r4.c
467a
d[Back] = sval;
.
441a
d[Fore] = 0;
d[Back] = 0;
.
429c
if(r.min.y < sr.min.y){
.
415a
if(r.min.y == sr.min.y){ /* a purely horizontal scroll */
depth = scr->gscreen->depth;
switch(depth){
case 32:
/*
* Using the SGI flat panels with the Ticket-to-Ride IV, horizontal
* 32-bit scrolls don't work perfectly on rectangles of width <= 24.
* we bail on a bigger bound for padding.
*/
if(Dx(r) < 32)
return 0;
break;
case 16:
/*
* Using the SGI flat panels with the Ticket-to-Ride IV, horizontal
* 16-bit scrolls don't work perfectly on rectangles of width <= 96.
* we bail on a bigger bound for padding.
*/
if(Dx(r) < 104)
return 0;
break;
}
}
.
414a
int depth;
.
391c
iprint("flow %8lux\n", d[Flow]);
.
388c
while((d[Flow]&0x1B) && x++ < 1000000)
.
## diffname pc/vgat2r4.c 2000/0112
## diff -e /n/emeliedump/1999/1212/sys/src/9/pc/vgat2r4.c /n/emeliedump/2000/0112/sys/src/9/pc/vgat2r4.c
274c
zoom = (scr->mmio[Zoom] & 0x0F)+1;
t2r4xo(scr, CursorHotY, -curs->offset.y*zoom);
.
261,262c
if(CursorMode == CursorMode32x32)
size = 32;
else
size = 64;
while(y < size){
for(x = 0; x < size/8; x++){
.
256,259d
251a
if(CursorMode == CursorMode32x32 && zoom == 16)
continue;
*data = pixels>>24;
*data = pixels>>16;
*data = pixels>>8;
*data = pixels;
.
189c
int size, x, y, zoom;
.
50c
CursorMode = CursorMode32x32,
.
## diffname pc/vgat2r4.c 2000/0326
## diff -e /n/emeliedump/2000/0112/sys/src/9/pc/vgat2r4.c /n/emeliedump/2000/0326/sys/src/9/pc/vgat2r4.c
559a
scr->blank = t2r4blank;
.
516a
t2r4blank(VGAscr *scr, int blank)
{
uchar x;
x = t2r4xi(scr, CursorSyncCtl);
x &= ~0x0F;
if(blank)
x |= HsyncLo | VsyncLo;
t2r4xo(scr, CursorSyncCtl, x);
}
static void
.
27a
CursorSyncCtl = 0x03,
HsyncHi = 0x01,
HsyncLo = 0x02,
VsyncHi = 0x04,
VsyncLo = 0x08,
.
## diffname pc/vgat2r4.c 2001/0527
## diff -e /n/emeliedump/2000/0326/sys/src/9/pc/vgat2r4.c /n/emeliedump/2001/0527/sys/src/9/pc/vgat2r4.c
151,152c
kstrdup(&seg.name, "t2r4screen");
.
135,136c
kstrdup(&seg.name, "t2r4mmio");
.
## diffname pc/vgat2r4.c 2001/0908
## diff -e /n/emeliedump/2001/0527/sys/src/9/pc/vgat2r4.c /n/emeliedump/2001/0908/sys/src/9/pc/vgat2r4.c
148,153c
addvgaseg("t2r4screen", aperture, size);
.
133,139d
131a
addvgaseg("t2r4mmio", mmio, p->mem[4].size);
.
108d
## diffname pc/vgat2r4.c 2001/1015
## diff -e /n/emeliedump/2001/0908/sys/src/9/pc/vgat2r4.c /n/emeliedump/2001/1015/sys/src/9/pc/vgat2r4.c
564a
hwblank = 1;
.
|