ref: d916a4c3823f55227ffae35738c2497256e307b5
dir: /os/boot/mpc/screen.c/
#include "all.h" #include <libg.h> #include <gnot.h> enum { Colldepth = 3, Colmaxx = 640, Colmaxxvis = 640, Colmaxy = 480, }; #define MINX 8 extern GSubfont defont0; struct{ Point pos; int bwid; }out; typedef struct Mode Mode; struct Mode { int x; int y; int d; char* aperture; int apsize; }; GBitmap gscreen; Point gchar(GBitmap*, Point, GFont*, int, Fcode); int setcolor(ulong, ulong, ulong, ulong); static void lcdinit(Mode*); void screeninit(void) { Mode m; m.x = Colmaxx; m.y = Colmaxy; m.d = Colldepth; m.aperture = 0; lcdinit(&m); if(m.aperture == 0) return; gscreen.ldepth = 3; gscreen.base = (ulong*)m.aperture; gscreen.width = Colmaxx/BY2WD; gscreen.r = Rect(0, 0, Colmaxxvis, Colmaxy); gscreen.clipr = gscreen.r; /* * For now, just use a fixed colormap: * 0 == white and 255 == black */ setcolor(0, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF); setcolor(255, 0x00000000, 0x00000000, 0x00000000); gbitblt(&gscreen, Pt(0, 0), &gscreen, gscreen.r, Zero); out.pos.x = MINX; out.pos.y = 0; out.bwid = defont0.info[' '].width; } void screenputc(int c) { Fontchar *i; Point p; if(gscreen.base == nil) return; switch(c){ case '\n': out.pos.x = MINX; out.pos.y += defont0.height; if(out.pos.y > gscreen.r.max.y-defont0.height) out.pos.y = gscreen.r.min.y; gbitblt(&gscreen, Pt(0, out.pos.y), &gscreen, Rect(0, out.pos.y, gscreen.r.max.x, out.pos.y+2*defont0.height), Zero); break; case '\t': out.pos.x += (8-((out.pos.x-MINX)/out.bwid&7))*out.bwid; if(out.pos.x >= gscreen.r.max.x) screenputc('\n'); break; case '\b': if(out.pos.x >= out.bwid+MINX){ out.pos.x -= out.bwid; screenputc(' '); out.pos.x -= out.bwid; } break; default: if(out.pos.x >= gscreen.r.max.x-out.bwid) screenputc('\n'); c &= 0x7f; if(c <= 0 || c >= defont0.n) break; i = defont0.info + c; p = out.pos; gbitblt(&gscreen, Pt(p.x+i->left, p.y), defont0.bits, Rect(i[0].x, 0, i[1].x, defont0.height), S); out.pos.x = p.x + i->width; break; } } void screenputs(char *s, int n) { while(n-- > 0) screenputc(*s++); } /* * See section 5.2.1 (page 5-6) of the MPC823 manual */ static uchar lcdclock[17] = { /* (a<<2)|b => divisor of (1<<a)*((b<<1)+1) */ 0, 0, (1<<2), 1, (2<<2), 2, (1<<2)|1, 3, (3<<2), (1<<2)|2, (1<<2)|2, (2<<2)|1, (2<<2)|1, (1<<2)|3, (1<<2)|3, (4<<2), (4<<2) }; /* * support for the Sharp LQ64D341 TFT colour display */ enum { COLS = 640, ROWS = 480, LDEPTH = 3, /* screen depth */ LCDFREQ = 25000000, /* lccr */ ClockLow = 1<<11, OELow = 1<<10, HsyncLow = 1<<9, VsyncLow = 1<<8, DataLow = 1<<7, Passive8 = 1<<4, DualScan = 1<<3, IsColour = 1<<2, IsTFT = 1<<1, Enable = 1<<0, /* lchcr */ BigEndian = 1<<24, AT7 = 7<<21, /* access type */ /* sdcr */ LAM = 1<<6, /* ``LCD aggressive mode'' */ }; /* * TO DO: most of the data could come from a table */ static void lcdinit(Mode *mode) { IMM *io; int i, d; long hz; io = m->iomem; mode->y = ROWS; mode->x = COLS; mode->d = LDEPTH; mode->aperture = ialloc(mode->x*mode->y, 16); mode->apsize = mode->x*mode->y; io->sdcr &= ~LAM; /* MPC823 errata: turn off LAM before disabling controller */ io->lcfaa = PADDR(mode->aperture); io->lccr = (((mode->x*mode->y*(1<<LDEPTH)+127)/128) << 17) | (LDEPTH << 5) | IsColour | IsTFT | OELow | VsyncLow | ClockLow; switch(LDEPTH){ default: case 0: /* monochrome/greyscale identity map */ for(i=0; i<16; i++) io->lcdmap[i] = i; break; case 2: /* 4-bit grey scale map */ for(i=0; i<16; i++) io->lcdmap[0] = (i<<8)|(i<<4)|i; break; case 3: /* 8-bit linear map */ for(i=0; i<256; i++) io->lcdmap[i] = (i<<8)|(i<<4)|i; break; } io->lcvcr = (mode->y << 11) | (1<<28) | 33; /* 2 line vsync pulse, 34 line wait between frames */ io->lchcr = (mode->x<<10) | BigEndian | 228; /* clock cycles between lines */ hz = m->cpuhz; d = hz/LCDFREQ; if(hz/d > LCDFREQ) d++; if(d >= 16) d = 16; /* * enable LCD outputs */ io->pddat = 0; io->pdpar = 0x1fff; io->pdpar &= ~SIBIT(6); /* 823 bug fix? */ io->pddir = 0x1fff; io->pbpar |= IBIT(31) | IBIT(19) | IBIT(17); io->pbdir |= IBIT(31) | IBIT(19) | IBIT(17); io->pbodr &= ~(IBIT(31) | IBIT(19) | IBIT(17)); eieio(); io->sccrk = KEEP_ALIVE_KEY; eieio(); io->sccr = (io->sccr & ~0x1F) | lcdclock[d]; eieio(); io->sccrk = ~KEEP_ALIVE_KEY; eieio(); gscreen.width = gscreen.width; /* access external memory before enabling (mpc823 errata) */ io->lcsr = 7; /* clear status */ eieio(); io->lccr |= Enable; archbacklight(1); } int setcolor(ulong p, ulong r, ulong g, ulong b) { r >>= 28; g >>= 28; b >>= 28; m->iomem->lcdmap[~p&0xFF] = (r<<8) | (g<<4) | b; /* TO DO: it's a function of the ldepth */ return 1; }