File:  [RetroPC.NET] / xmil / vram / palettes.c
Revision 1.17: download - view: text, annotated - select for diffs
Fri Aug 20 17:41:41 2004 JST (21 years, 2 months ago) by yui
Branches: MAIN
CVS tags: HEAD
fix...

#include	"compiler.h"
#include	"scrnmng.h"
#include	"pccore.h"
#include	"iocore.h"
#include	"scrndraw.h"
#include	"palettes.h"
#include	"makescrn.h"


	UINT		xmil_palettes;
	RGB32		xmil_pal32[XMILPAL_MAX];			// xm_palette
#if defined(SUPPORT_16BPP)
	RGB16		xmil_pal16[XMILPAL_MAX];
#endif
#if defined(SUPPORT_PALEVENT)
	PALEVENT	palevent;
#endif

#if defined(SUPPORT_TURBOZ)
static const UINT16 pal4096banktbl[2][64] = {
			{	0x000, 0x008, 0x080, 0x088, 0x800, 0x808, 0x880, 0x888,
				0x004, 0x00C, 0x084, 0x08C, 0x804, 0x80C, 0x884, 0x88C,
				0x040, 0x048, 0x0C0, 0x0C8, 0x840, 0x848, 0x8C0, 0x8C8,
				0x044, 0x04C, 0x0C4, 0x0CC, 0x844, 0x84C, 0x8C4, 0x8CC,
				0x400, 0x408, 0x480, 0x488, 0xC00, 0xC08, 0xC80, 0xC88,
				0x404, 0x40C, 0x484, 0x48C, 0xC04, 0xC0C, 0xC84, 0xC8C,
				0x440, 0x448, 0x4C0, 0x4C8, 0xC40, 0xC48, 0xCC0, 0xCC8,
				0x444, 0x44C, 0x4C4, 0x4CC, 0xC44, 0xC4C, 0xCC4, 0xCCC},

			{	0x000, 0x002, 0x020, 0x022, 0x200, 0x202, 0x220, 0x222,
				0x001, 0x003, 0x021, 0x023, 0x201, 0x203, 0x221, 0x223,
				0x010, 0x012, 0x030, 0x032, 0x210, 0x212, 0x230, 0x232,
				0x011, 0x013, 0x031, 0x033, 0x211, 0x213, 0x231, 0x233,
				0x100, 0x102, 0x120, 0x122, 0x300, 0x302, 0x320, 0x322,
				0x101, 0x103, 0x121, 0x123, 0x301, 0x303, 0x321, 0x323,
				0x110, 0x112, 0x130, 0x132, 0x310, 0x312, 0x330, 0x332,
				0x111, 0x113, 0x131, 0x133, 0x311, 0x313, 0x331, 0x333}};

static void pal4096to64(RGB32 *pal, const UINT16 *map) {

	UINT	r;

	r = 64;
	do {
		pal->d = xmil_pal32[XMILPAL_4096G + (*map++)].d;
		pal++;
	} while(--r);
}
#endif


// ----

void pal_makegrad(RGB32 *pal, int pals, UINT32 bg, UINT32 fg) {

	int		i;

	if (pals >= 2) {
		pals--;
		for (i=0; i<=pals; i++) {
			pal[i].p.b = (UINT8)
				((((fg >> 0) & 0x0000ff) * i + 
				((bg >> 0) & 0x0000ff) * (pals-i)) / pals);
			pal[i].p.g = (UINT8)
				((((fg >> 8) & 0x0000ff) * i + 
				((bg >> 8) & 0x0000ff) * (pals-i)) / pals);
			pal[i].p.r = (UINT8)
				((((fg >> 16) & 0x0000ff) * i + 
				((bg >> 16) & 0x0000ff) * (pals-i)) / pals);
			pal[i].p.e = 0;
		}
	}
}


// ----

typedef struct {
	UINT8	blankcol;
	UINT8	padding;
	UINT16	skipline;
	RGB32	text[16];
#if defined(SUPPORT_TURBOZ)
	RGB32	grph[2][16];
	RGB32	grph64[2][64];
#endif
} PALS;

static	PALS	pals;


#if !defined(SUPPORT_TURBOZ)
static void pal_settext(REG8 num) {

	RGB32	rgb;

	rgb.p.e = 0;
	if (!num) {
		rgb.p.b = pals.blankcol;
		rgb.p.g = pals.blankcol;
		rgb.p.r = pals.blankcol;
	}
	else {
		rgb.p.b = (num & 1)?0xff:0x00;
		rgb.p.g = (num & 4)?0xff:0x00;
		rgb.p.r = (num & 2)?0xff:0x00;
	}
	pals.text[num].d = rgb.d;
	pals.text[num+8].p.b = (UINT8)((pals.skipline * rgb.p.b) >> 8);
	pals.text[num+8].p.g = (UINT8)((pals.skipline * rgb.p.g) >> 8);
	pals.text[num+8].p.r = (UINT8)((pals.skipline * rgb.p.r) >> 8);
	makescrn.palandply = 1;
}
#else
void pal_settext(REG8 num) {

	REG8	textpal;
	REG8	black;
	RGB32	rgb;

	textpal = crtc.p.text[num];
	black = 0;
	rgb.p.e = 0;
	rgb.p.b = ((textpal >> 0) & 3) * 0x55;
	if (rgb.p.b < pals.blankcol) {
		black |= 1;
	}
	rgb.p.g = ((textpal >> 4) & 3) * 0x55;
	if (rgb.p.g < pals.blankcol) {
		black |= 2;
	}
	rgb.p.r = ((textpal >> 2) & 3) * 0x55;
	if (rgb.p.r < pals.blankcol) {
		black |= 4;
	}
	if (black == 7) {
		rgb.p.b = pals.blankcol;
		rgb.p.g = pals.blankcol;
		rgb.p.r = pals.blankcol;
	}
	xmil_pal32[XMILPAL_4096T + num].d = rgb.d;
#if defined(SUPPORT_16BPP)
	if (scrnmng_getbpp() == 16) {
		xmil_pal16[XMILPAL_4096T + num] = scrnmng_makepal16(rgb);
	}
#endif
	pals.text[num].d = rgb.d;
	pals.text[num+8].p.b = (UINT8)((pals.skipline * rgb.p.b) >> 8);
	pals.text[num+8].p.g = (UINT8)((pals.skipline * rgb.p.g) >> 8);
	pals.text[num+8].p.r = (UINT8)((pals.skipline * rgb.p.r) >> 8);
	makescrn.palandply = 1;
}

void pal_setgrph(REG8 bank, REG8 num) {

	UINT	grphpal;
	REG8	black;
	RGB32	rgb;

	grphpal = crtc.p.grph[bank][num];
	black = 0;
	rgb.p.e = 0;
	rgb.p.b = ((grphpal >> 0) & 0x0f) * 0x11;
	if (rgb.p.b < pals.blankcol) {
		black |= 1;
	}
	rgb.p.g = ((grphpal >> 8) & 0x0f) * 0x11;
	if (rgb.p.g < pals.blankcol) {
		black |= 2;
	}
	rgb.p.r = ((grphpal >> 4) & 0x0f) * 0x11;
	if (rgb.p.r < pals.blankcol) {
		black |= 4;
	}
	if (black == 7) {
		rgb.p.b = pals.blankcol;
		rgb.p.g = pals.blankcol;
		rgb.p.r = pals.blankcol;
	}
	pals.grph64[bank][num].d = rgb.d;
	if (!(((num >> 3) ^ num) & 7)) {
		num &= 7;
		pals.grph[bank][num+0].d = rgb.d;
		pals.grph[bank][num+8].p.b = (UINT8)((pals.skipline * rgb.p.b) >> 8);
		pals.grph[bank][num+8].p.g = (UINT8)((pals.skipline * rgb.p.g) >> 8);
		pals.grph[bank][num+8].p.r = (UINT8)((pals.skipline * rgb.p.r) >> 8);
	}
	makescrn.palandply = 1;
}

void pal_setgrph4096(UINT num) {

	UINT	grphpal;
	REG8	black;
	RGB32	rgb;

	grphpal = crtc.p.grph4096[num];
	black = 0;
	rgb.p.e = 0;
	rgb.p.b = ((grphpal >> 0) & 0x0f) * 0x11;
	if (rgb.p.b < pals.blankcol) {
		black |= 1;
	}
	rgb.p.g = ((grphpal >> 8) & 0x0f) * 0x11;
	if (rgb.p.g < pals.blankcol) {
		black |= 2;
	}
	rgb.p.r = ((grphpal >> 4) & 0x0f) * 0x11;
	if (rgb.p.r < pals.blankcol) {
		black |= 4;
	}
	if (black == 7) {
		rgb.p.b = pals.blankcol;
		rgb.p.g = pals.blankcol;
		rgb.p.r = pals.blankcol;
	}
	xmil_pal32[XMILPAL_4096G + num].d = rgb.d;
#if defined(SUPPORT_16BPP)
	if (scrnmng_getbpp() == 16) {
		xmil_pal16[XMILPAL_4096G + num] = scrnmng_makepal16(rgb);
	}
#endif
	makescrn.palandply = 1;
}
#endif


// ----

void pal_update1(const UINT8 *rgbp) {

	UINT	i;
	UINT	j;
	REG8	bit;
	UINT	c;
	UINT	skip8;
	UINT	bcnt;

	skip8 = 0;
	if (!(crtc.s.SCRN_BITS & SCRN_24KHZ)) {
		skip8 = 8;
	}
	for (i=0, bit=1; i<8; i++, bit<<=1) {
#if defined(SUPPORT_TURBOZ)
		if (!(crtc.s.EXTPALMODE & 0x80)) {
#endif
			c = 0;
			if (rgbp[CRTC_PALB] & bit) {
				c += 1;
			}
			if (rgbp[CRTC_PALR] & bit) {
				c += 2;
			}
			if (rgbp[CRTC_PALG] & bit) {
				c += 4;
			}
#if defined(SUPPORT_TURBOZ)
		}
		else {
			c = i;
		}
#endif
#if !defined(SUPPORT_TURBOZ)
		xmil_pal32[i].d = pals.text[c].d;
		xmil_pal32[i+64].d = pals.text[c + skip8].d;
#else
		xmil_pal32[i].d = pals.grph[crtc.e.pal_disp][c].d;
		xmil_pal32[i+64].d = pals.grph[crtc.e.pal_disp][c + skip8].d;
#endif
		if (rgbp[CRTC_PLY] & bit) {
			for (j=i+8; j<64; j+=8) {
				xmil_pal32[j].d = xmil_pal32[i].d;
				xmil_pal32[j+64].d = xmil_pal32[i+64].d;
			}
		}
		else {
			bcnt = (rgbp[CRTC_BLACK] & 15) - 8;
			for (j=i+8; j<64; j+=8) {
				bcnt--;
				if (bcnt) {
					c = j >> 3;
				}
				else {
					c = 0;
				}
				xmil_pal32[j].d = pals.text[c].d;
				xmil_pal32[j+64].d = pals.text[c + skip8].d;
			}
		}
	}
#if defined(SUPPORT_16BPP)
	if (scrnmng_getbpp() == 16) {
		for (i=0; i<xmil_palettes; i++) {
			xmil_pal16[i] = scrnmng_makepal16(xmil_pal32[i]);
		}
	}
#endif
}

void pal_update(void) {

#if defined(SUPPORT_TURBOZ) || defined(SUPPORT_16BPP)
	UINT	i;
#endif

#if !defined(SUPPORT_TURBOZ)
	pal_update1(crtc.s.rgbp);
	xmil_palettes = 64 + 64;
#else
	if (!(crtc.e.dispmode & SCRN64_ENABLE)) {
		pal_update1(crtc.s.rgbp);
		xmil_palettes = 64 + 64;
		scrndraw_changepalette();
		return;
	}
	else if (crtc.e.pal_disp & PAL_4096) {
		switch(crtc.e.pal_disp & 0xf) {
			case PAL_4096H:
				pal4096to64(xmil_pal32, pal4096banktbl[0]);
				xmil_palettes = 64;
				break;

			case PAL_4096L:
				pal4096to64(xmil_pal32, pal4096banktbl[1]);
				xmil_palettes = 64;
				break;

			case (PAL_4096H | PAL_64x2):
				pal4096to64(&xmil_pal32[ 0], pal4096banktbl[0]);
				pal4096to64(&xmil_pal32[64], pal4096banktbl[1]);
				xmil_palettes = 128;
				break;

			case (PAL_4096L | PAL_64x2):
				pal4096to64(&xmil_pal32[ 0], pal4096banktbl[0]);
				pal4096to64(&xmil_pal32[64], pal4096banktbl[1]);
				xmil_palettes = 128;
				break;

			default:						// fullcolor!
				return;
		}
		for (i=0; i<8; i++) {
			xmil_pal32[xmil_palettes++].d = pals.text[i].d;
		}
	}
	else {
		for (i=0; i<64; i++) {
			xmil_pal32[i].d = pals.grph64[crtc.e.pal_disp][i].d;
		}
		for (i=0; i<8; i++) {
			xmil_pal32[i+64].d = pals.text[i].d;
		}
		xmil_palettes = 64 + 8;
	}
#endif

#if defined(SUPPORT_16BPP)
	if (scrnmng_getbpp() == 16) {
		for (i=0; i<xmil_palettes; i++) {
			xmil_pal16[i] = scrnmng_makepal16(xmil_pal32[i]);
		}
	}
#endif
	scrndraw_changepalette();
}

#if defined(SUPPORT_PALEVENT)
void pal_eventclear(void) {

	if ((!corestat.drawframe) ||
		(!xmilcfg.RASTER) || (scrnmng_getbpp() == 8)) {
		palevent.events = SUPPORT_PALEVENT;
	}
	else {
		CopyMemory(palevent.rgbp, crtc.s.rgbp, CRTC_RGBPMAX);
		palevent.events = 0;
	}
}
#endif


// ----

void pal_reset(void) {

	REG8	i;
#if defined(SUPPORT_TURBOZ)
	REG8	j;
	UINT	k;
#endif

	pals.blankcol = 0;
	pals.skipline = (xmilcfg.skipline)?xmilcfg.skiplight:0;
	for (i=0; i<8; i++) {
		pal_settext(i);
	}
#if defined(SUPPORT_TURBOZ)
	for (i=0; i<2; i++) {
		for (j=0; j<64; j++) {
			pal_setgrph(i, j);
		}
	}
	for (k=0; k<4096; k++) {
		pal_setgrph4096(k);
	}
#endif
}


RetroPC.NET-CVS <cvs@retropc.net>