view 3rdparty/utils/viewgif/setmap.c @ 3129:ce9f4ff0a380

l1 rel.asm: Do the module padding in a more portable way Add comments to describe what's happening and why.
author Neal Crook <foofoobedoo@gmail.com>
date Wed, 30 Nov 2016 22:10:56 +0000
parents aaae5eac20e1
children
line wrap: on
line source

/*
 * SetMap - Palette creator for ViewGIF 2.0
 * These routines are the heart of the color analysis.  They take the
 * global bitmap and, based on the dithering factor, produce a CoCo color
 * palette.
 */

#include "viewgif.h"

int		toler2(), exact();
bool	nearcolor();

/*
 * setmap() -- determine whether, given the specified dithering factor,
 * there are CLUTs for the screens we've been told to use that will let
 * us come reasonably close to representing the colors in the GIF image.
 * We change the semantics from the original version, returning FALSE
 * if not all the CLUT etnries can be fit in, or TRUE if they all work.
 *
 * Method: look for approximations to each color in use in terms of colors
 * already in the CLUTs.  If none is close enough, then add the color
 * (or some decomposition thereof, in the multiple screen case) to the
 * CLUTs.  If there's no room left to add the color, we return FALSE.
 *
 * Note: we're trying to trade space for speed in the innermost loop of
 * doline() by always leaving an end marker, avoiding double testing in
 * said loop.  We'll see how it works out.
 */
bool
setmap()
{
	rgbcolor			next;
	register int		add;
	register rgbcolor	*gcscan;
	register xlate		*trscan;
	int         		numtrans, tolrnce, x, uplim, lowlim;

	/* should be a loop if we really do change NSCREENS */
	screen[0].clutsize = screen[1].clutsize = 0;

	lowlim = low0;
	uplim = up0;

	if (dfactor > 0) {
		tolrnce = 0;
		lowlim -= dfactor;
	} else
		tolrnce = -dfactor;

	if (dfactor > (int) flicker)
		uplim += dfactor;

	for (gcscan = globclut, x = 0; x < globcolors; gcscan++, x++) {
		if (coloruse[x]) {
			numtrans = 0;
			trscan = &transtab[x][0];
			for (add = lowlim; add < uplim; add = minadd(gcscan, add)) {
				trscan->addval = add;
				addoff(&next, gcscan, add);
				if (!(*approx)(&next, trscan, tolrnce))
					return FALSE;
				if (++numtrans > 4)
					fatal("BUG: numtrans>4");
				trscan++;
			}
			/* mark end of transtab entries for this color */
			trscan->addval = BOGUSDITH;
		}
	}
	return TRUE;
}

addoff(color, color0, offset)
register BYTE	*color, *color0;
int				offset;
{
	register int	accum, x;

	for (x = 3; --x >= 0; ) {
		if ((accum = arith(*color0++) + offset) > 0xff)
			accum = 0xff;
		*color++ = accum;
	}
}

/*
 * approx1() -- the approximation seeker if there's only one screen
 */
bool
approx1(goal, trans, toler)
rgbcolor	*goal;
xlate		*trans;
int			toler;
{
	register rgbcolor	*cscan, *nearest;
	int					x, mintoler, tv;

	mintoler = toler + 1;
	nearest = NULL;
	for (cscan = screen[0].clut, x = screen[0].clutsize; --x >= 0; cscan++) {
		if ((tv = tolerval(cscan, goal)) < mintoler) {
			nearest = cscan;
			if ((mintoler = tv) == 0)
				break;
		}
	}

	if (nearest == NULL) {
		if (++screen[0].clutsize > MCCLUT)
			return FALSE;
		cscan->red   = arith(goal->red) / SCALE1;
		cscan->green = arith(goal->green) / SCALE1;
		cscan->blue  = arith(goal->blue) / SCALE1;
		nearest = cscan;
	}
	trans->clutval[0] = nearest - screen[0].clut;
	return TRUE;
}

/*
 * approx2() -- a two-screen approximator
 *
 * The general idea here takes multiple passes:
 * 1. Iterate over sums of pairs of colors taken one from each screen.
 * 2. If that fails, then for each screen, try adding something that
 *    should come close with each of the colors in the CLUT for the
 *    other screen.  (Try it with the smallest CLUT first, to save space.)
 * 3. If *that* fails, extend both CLUTs with the new color (at half
 *    intensity).
 *
 * Since in step 1, which one would think eats the most time, viewgif was
 * making repeated calls to nearcolor(), we've switched over to keeping
 * a table remembering which colors in the palettes are near as determined
 * by nearcolor().  (Donald Michie would be proud of us.)
 */

static bool	neartab[MCCLUT][MCCLUT];

#define ROW		1
#define COLUMN	2

bool
approx2(goal, trans, toler)
rgbcolor	*goal;
xlate		*trans;
int			toler;
{
	register int		c1, c0;
	rgbcolor			*scan0, *scan1, *near0, *near1;
	cocoscreen			*extend, *exam, *temp;
	int					(*tolfun)();
	char				*near_row;
	rgbcolor			scalergb;
	int					mintoler, tv;
	int					x;
	bool				found;

	scalergb.red   = arith(goal->red) / SCALE2;
	scalergb.green = arith(goal->green) / SCALE2;
	scalergb.blue  = arith(goal->blue) / SCALE2;

	if (toler > 0)
		tolfun = toler2;
	else {
		goal = &scalergb;
		tolfun = exact;
	}

	found = FALSE;
	mintoler = toler + 1;

	for (scan0 = &screen[0].clut[c0 = screen[0].clutsize];
		 scan0--, --c0 >= 0; )
	{
		near_row = neartab[c0];
		for (c1 = screen[1].clutsize; --c1 >= 0; ) {
			if (near_row[c1]) {
				scan1 = &screen[1].clut[c1];
				if ((tv = (*tolfun)(scan0, scan1, goal)) < mintoler) {
					near0 = scan0;
					near1 = scan1;
					found = TRUE;
					if ((mintoler = tv) == 0)
						goto out;
				}
			}
		}
	}

out:
	if (!found) {
		if (screen[1].clutsize < screen[0].clutsize) {
			extend = &screen[1];
			exam = &screen[0];
		} else {
			extend = &screen[0];
			exam = &screen[1];
		}
		for (x = 2; --x >= 0; ) {
			if (extend->clutsize < MCCLUT) {
				near0 = &extend->clut[extend->clutsize];
				for (scan1 = &exam->clut[c1 = exam->clutsize];
					 scan1--, --c1 >= 0; )
				{
					forcenear(near0, scan1, &scalergb);
					if ((tv = (*tolfun)(near0, scan1, goal)) < mintoler) {
						near1 = scan1;
						found = TRUE;
						if ((mintoler = tv) == 0)
							break;
					}
				}
				if (found)
					break;
			}
			temp = extend;
			extend = exam;
			exam = temp;
		}
	
		if (found) {
			if (mintoler > 0)
				forcenear(near0, near1, &scalergb);
			extend->clutsize++;
			if (extend == &screen[0])
				nearfill(ROW);
			else {
				temp = (cocoscreen *) near0;
				near0 = near1;
				near1 = (rgbcolor *) temp;
				nearfill(COLUMN);
			}
		} else {
			if (screen[0].clutsize >= MCCLUT ||
				screen[1].clutsize >= MCCLUT)
				return FALSE;
			near0 = &screen[0].clut[screen[0].clutsize++];
			near0->red   = arith(scalergb.red) / 2;
			near0->green = arith(scalergb.green) / 2;
			near0->blue  = arith(scalergb.blue) / 2;
			near1 = &screen[1].clut[screen[1].clutsize++];
			near1->red   = (arith(scalergb.red) + 1) / 2;
			near1->green = (arith(scalergb.green) + 1) / 2;
			near1->blue  = (arith(scalergb.blue) + 1) / 2;
			nearfill(ROW | COLUMN);
		}
	}
	trans->clutval[0] = near0 - screen[0].clut;
	trans->clutval[1] = near1 - screen[1].clut;
	return TRUE;
}

nearfill(section)
int	section;
{
	register int	i, j;
	rgbcolor		*newrgb;

	if (section & ROW) {
		newrgb = &screen[0].clut[j = screen[0].clutsize - 1];
		for (i = screen[1].clutsize; --i >= 0; )
			neartab[j][i] = nearcolor(newrgb, &screen[1].clut[i]);
	}
	if (section & COLUMN) {
		newrgb = &screen[1].clut[j = screen[1].clutsize - 1];
		for (i = screen[0].clutsize; --i >= 0; )
			neartab[i][j] = nearcolor(&screen[0].clut[i], newrgb);
	}
}

/*
 * nearcolor() -- tell approx2() whether two colors are close enough
 * to be considered for approximate summation to a color in the GIF CLUT.
 */
bool
nearcolor(c1, c2)
register rgbcolor	*c1, *c2;
{
	return abs(arith(c2->red)   - arith(c1->red))   < 2 &&
		   abs(arith(c2->green) - arith(c1->green)) < 2 &&
		   abs(arith(c2->blue)  - arith(c1->blue))  < 2;
}

/*
 * forcenear() -- stuff a color into a CLUT that is guaranteed to
 * pass the nearcolor test with the other fixed color (so we don't
 * have to call nearcolor() explicitly), and should come close to
 * adding to the fixed color (already in the other screen's CLUT)
 * to give a specified goal color from the GIF CLUT.
 */
forcenear(vary, fixed, goal)
register BYTE   vary[], fixed[], goal[];
{
	register int    d, x;

	for (x = 3; --x >= 0; ) {
		if ((d = arith(*vary = *goal++ - *fixed) - arith(*fixed)) < 0)
			d = -1;
		else if (d > 0)
			d = 1;
		*vary++ = *fixed++ + d;
	}
}

int
minadd(color, add)
register BYTE	*color;
int             add;
{
	int             maxval, tryadd, x;

	maxval = 0;
	for (x = 3; --x >= 0; ) {
		if ((tryadd = (arith(*color++) + add) % minmod) > maxval)
			maxval = tryadd;
	}
	return minmod - maxval + add;
}

int
tolerval(ccolor, gcolor)
register BYTE	*ccolor, *gcolor;
{
	int     x, v, mv;

	mv = 0;
	for (x = 3; --x >= 0; ) {
		v = abs(SCALE1 * *ccolor++ + SCALE1 / 2 - arith(*gcolor++)) -
			SCALE1 / 2;
		if (v > mv)
			mv = v;
	}
	return mv;
}

int
toler2(ccolor1, ccolor2, gcolor)
register BYTE	*ccolor1, *ccolor2, *gcolor;
{
	int     x, v, mv;

	mv = 0;
	for (x = 3; --x >= 0; ) {
		v = abs(SCALE2 * (*ccolor1++ + *ccolor2++) + SCALE2 / 2 -
				arith(*gcolor++)) - SCALE2 / 2;
		if (v > mv)
			mv = v;
	}
	return mv;
}

int
exact(ccolor1, ccolor2, gcolor)
register BYTE	*ccolor1, *ccolor2, *gcolor;
{
	if (arith(ccolor1[0]) + arith(ccolor2[0]) != arith(gcolor[0]))
		return 2;
	if (arith(ccolor1[1]) + arith(ccolor2[1]) != arith(gcolor[1]))
		return 2;
	if (arith(ccolor1[2]) + arith(ccolor2[2]) != arith(gcolor[2]))
		return 2;
	return 0;
}