Blame src/gd_nnquant.c

Packit Service df60bb
/* NeuQuant Neural-Net Quantization Algorithm
Packit Service df60bb
 * ------------------------------------------
Packit Service df60bb
 *
Packit Service df60bb
 * Copyright (c) 1994 Anthony Dekker
Packit Service df60bb
 *
Packit Service df60bb
 * NEUQUANT Neural-Net quantization algorithm by Anthony Dekker, 1994.
Packit Service df60bb
 * See "Kohonen neural networks for optimal colour quantization"
Packit Service df60bb
 * in "Network: Computation in Neural Systems" Vol. 5 (1994) pp 351-367.
Packit Service df60bb
 * for a discussion of the algorithm.
Packit Service df60bb
 * See also  http://members.ozemail.com.au/~dekker/NEUQUANT.HTML
Packit Service df60bb
 *
Packit Service df60bb
 * Any party obtaining a copy of these files from the author, directly or
Packit Service df60bb
 * indirectly, is granted, free of charge, a full and unrestricted irrevocable,
Packit Service df60bb
 * world-wide, paid up, royalty-free, nonexclusive right and license to deal
Packit Service df60bb
 * in this software and documentation files (the "Software"), including without
Packit Service df60bb
 * limitation the rights to use, copy, modify, merge, publish, distribute, sublicense,
Packit Service df60bb
 * and/or sell copies of the Software, and to permit persons who receive
Packit Service df60bb
 * copies from any such party to do so, with the only requirement being
Packit Service df60bb
 * that this copyright notice remain intact.
Packit Service df60bb
 *
Packit Service df60bb
 *
Packit Service df60bb
 * Modified to process 32bit RGBA images.
Packit Service df60bb
 * Stuart Coyle 2004-2007
Packit Service df60bb
 * From: http://pngnq.sourceforge.net/
Packit Service df60bb
 *
Packit Service df60bb
 * Ported to libgd by Pierre A. Joye
Packit Service df60bb
 * (and make it thread safety by droping static and  global variables)
Packit Service df60bb
 */
Packit Service df60bb
Packit Service df60bb
#ifdef HAVE_CONFIG_H
Packit Service df60bb
#include "config.h"
Packit Service df60bb
#endif /* HAVE_CONFIG_H */
Packit Service df60bb
Packit Service df60bb
#include <stdlib.h>
Packit Service df60bb
#include <string.h>
Packit Service df60bb
#include "gd.h"
Packit Service df60bb
#include "gdhelpers.h"
Packit Service df60bb
#include "gd_errors.h"
Packit Service df60bb
Packit Service df60bb
#include "gd_nnquant.h"
Packit Service df60bb
Packit Service df60bb
/* Network Definitions
Packit Service df60bb
   ------------------- */
Packit Service df60bb
Packit Service df60bb
#define maxnetpos	(MAXNETSIZE-1)
Packit Service df60bb
#define netbiasshift	4			/* bias for colour values */
Packit Service df60bb
#define ncycles		100			/* no. of learning cycles */
Packit Service df60bb
Packit Service df60bb
/* defs for freq and bias */
Packit Service df60bb
#define intbiasshift    16			/* bias for fractions */
Packit Service df60bb
#define intbias		(((int) 1)<
Packit Service df60bb
#define gammashift  	10			/* gamma = 1024 */
Packit Service df60bb
#define gamma   	(((int) 1)<
Packit Service df60bb
#define betashift  	10
Packit Service df60bb
#define beta		(intbias>>betashift)	/* beta = 1/1024 */
Packit Service df60bb
#define betagamma	(intbias<<(gammashift-betashift))
Packit Service df60bb
Packit Service df60bb
/* defs for decreasing radius factor */
Packit Service df60bb
#define initrad		(MAXNETSIZE>>3)		/* for 256 cols, radius starts */
Packit Service df60bb
#define radiusbiasshift	6			/* at 32.0 biased by 6 bits */
Packit Service df60bb
#define radiusbias	(((int) 1)<
Packit Service df60bb
#define initradius	(initrad*radiusbias)	/* and decreases by a */
Packit Service df60bb
#define radiusdec	30			/* factor of 1/30 each cycle */
Packit Service df60bb
Packit Service df60bb
/* defs for decreasing alpha factor */
Packit Service df60bb
#define alphabiasshift	10			/* alpha starts at 1.0 */
Packit Service df60bb
#define initalpha	(((int) 1)<
Packit Service df60bb
int alphadec;
Packit Service df60bb
Packit Service df60bb
/* radbias and alpharadbias used for radpower calculation */
Packit Service df60bb
#define radbiasshift	8
Packit Service df60bb
#define radbias		(((int) 1)<
Packit Service df60bb
#define alpharadbshift  (alphabiasshift+radbiasshift)
Packit Service df60bb
#define alpharadbias    (((int) 1)<
Packit Service df60bb
Packit Service df60bb
#define ALPHA 0
Packit Service df60bb
#define RED 1
Packit Service df60bb
#define BLUE 2
Packit Service df60bb
#define GREEN 3
Packit Service df60bb
Packit Service df60bb
typedef int nq_pixel[5];
Packit Service df60bb
Packit Service df60bb
typedef struct {
Packit Service df60bb
	/* biased by 10 bits */
Packit Service df60bb
	int alphadec;
Packit Service df60bb
Packit Service df60bb
	/* lengthcount = H*W*3 */
Packit Service df60bb
	int lengthcount;
Packit Service df60bb
Packit Service df60bb
	/* sampling factor 1..30 */
Packit Service df60bb
	int samplefac;
Packit Service df60bb
Packit Service df60bb
	/* Number of colours to use. Made a global instead of #define */
Packit Service df60bb
	int netsize;
Packit Service df60bb
Packit Service df60bb
	/* for network lookup - really 256 */
Packit Service df60bb
	int netindex[256];
Packit Service df60bb
Packit Service df60bb
	/* ABGRc */
Packit Service df60bb
	/* the network itself */
Packit Service df60bb
	nq_pixel network[MAXNETSIZE];
Packit Service df60bb
Packit Service df60bb
	/* bias and freq arrays for learning */
Packit Service df60bb
	int bias[MAXNETSIZE];
Packit Service df60bb
	int freq[MAXNETSIZE];
Packit Service df60bb
Packit Service df60bb
	/* radpower for precomputation */
Packit Service df60bb
	int radpower[initrad];
Packit Service df60bb
Packit Service df60bb
	/* the input image itself */
Packit Service df60bb
	unsigned char *thepicture;
Packit Service df60bb
} nn_quant;
Packit Service df60bb
Packit Service df60bb
/* Initialise network in range (0,0,0,0) to (255,255,255,255) and set parameters
Packit Service df60bb
   ----------------------------------------------------------------------- */
Packit Service df60bb
void initnet(nnq, thepic, len, sample, colours)
Packit Service df60bb
nn_quant *nnq;
Packit Service df60bb
unsigned char *thepic;
Packit Service df60bb
int len;
Packit Service df60bb
int sample;
Packit Service df60bb
int colours;
Packit Service df60bb
{
Packit Service df60bb
	register int i;
Packit Service df60bb
	register int *p;
Packit Service df60bb
Packit Service df60bb
	/* Clear out network from previous runs */
Packit Service df60bb
	/* thanks to Chen Bin for this fix */
Packit Service df60bb
	memset((void*)nnq->network, 0, sizeof(nq_pixel)*MAXNETSIZE);
Packit Service df60bb
Packit Service df60bb
	nnq->thepicture = thepic;
Packit Service df60bb
	nnq->lengthcount = len;
Packit Service df60bb
	nnq->samplefac = sample;
Packit Service df60bb
	nnq->netsize = colours;
Packit Service df60bb
Packit Service df60bb
	for (i=0; i < nnq->netsize; i++) {
Packit Service df60bb
		p = nnq->network[i];
Packit Service df60bb
		p[0] = p[1] = p[2] = p[3] = (i << (netbiasshift+8)) / nnq->netsize;
Packit Service df60bb
		nnq->freq[i] = intbias / nnq->netsize;	/* 1/netsize */
Packit Service df60bb
		nnq->bias[i] = 0;
Packit Service df60bb
	}
Packit Service df60bb
}
Packit Service df60bb
Packit Service df60bb
/* -------------------------- */
Packit Service df60bb
Packit Service df60bb
/* Unbias network to give byte values 0..255 and record
Packit Service df60bb
 * position i to prepare for sort
Packit Service df60bb
 */
Packit Service df60bb
/* -------------------------- */
Packit Service df60bb
Packit Service df60bb
void unbiasnet(nn_quant *nnq)
Packit Service df60bb
{
Packit Service df60bb
	int i,j,temp;
Packit Service df60bb
Packit Service df60bb
	for (i=0; i < nnq->netsize; i++) {
Packit Service df60bb
		for (j=0; j<4; j++) {
Packit Service df60bb
			/* OLD CODE: network[i][j] >>= netbiasshift; */
Packit Service df60bb
			/* Fix based on bug report by Juergen Weigert jw@suse.de */
Packit Service df60bb
			temp = (nnq->network[i][j] + (1 << (netbiasshift - 1))) >> netbiasshift;
Packit Service df60bb
			if (temp > 255) temp = 255;
Packit Service df60bb
			nnq->network[i][j] = temp;
Packit Service df60bb
		}
Packit Service df60bb
		nnq->network[i][4] = i;			/* record colour no */
Packit Service df60bb
	}
Packit Service df60bb
}
Packit Service df60bb
Packit Service df60bb
/* Output colour map
Packit Service df60bb
	 ----------------- */
Packit Service df60bb
void writecolourmap(nnq, f)
Packit Service df60bb
nn_quant *nnq;
Packit Service df60bb
FILE *f;
Packit Service df60bb
{
Packit Service df60bb
	int i,j;
Packit Service df60bb
Packit Service df60bb
	for (i=3; i>=0; i--)
Packit Service df60bb
		for (j=0; j < nnq->netsize; j++)
Packit Service df60bb
			putc(nnq->network[j][i], f);
Packit Service df60bb
}
Packit Service df60bb
Packit Service df60bb
/* Output colormap to unsigned char ptr in RGBA format */
Packit Service df60bb
void getcolormap(nnq, map)
Packit Service df60bb
nn_quant *nnq;
Packit Service df60bb
unsigned char *map;
Packit Service df60bb
{
Packit Service df60bb
	int i,j;
Packit Service df60bb
	for(j=0; j < nnq->netsize; j++) {
Packit Service df60bb
		for (i=3; i>=0; i--) {
Packit Service df60bb
			*map = nnq->network[j][i];
Packit Service df60bb
			map++;
Packit Service df60bb
		}
Packit Service df60bb
	}
Packit Service df60bb
}
Packit Service df60bb
Packit Service df60bb
/* Insertion sort of network and building of netindex[0..255] (to do after unbias)
Packit Service df60bb
   ------------------------------------------------------------------------------- */
Packit Service df60bb
void inxbuild(nn_quant *nnq)
Packit Service df60bb
{
Packit Service df60bb
	register int i,j,smallpos,smallval;
Packit Service df60bb
	register int *p,*q;
Packit Service df60bb
	int previouscol,startpos;
Packit Service df60bb
Packit Service df60bb
	previouscol = 0;
Packit Service df60bb
	startpos = 0;
Packit Service df60bb
	for (i=0; i < nnq->netsize; i++) {
Packit Service df60bb
		p = nnq->network[i];
Packit Service df60bb
		smallpos = i;
Packit Service df60bb
		smallval = p[2];			/* index on g */
Packit Service df60bb
		/* find smallest in i..netsize-1 */
Packit Service df60bb
		for (j=i+1; j < nnq->netsize; j++) {
Packit Service df60bb
			q = nnq->network[j];
Packit Service df60bb
			if (q[2] < smallval) {		/* index on g */
Packit Service df60bb
				smallpos = j;
Packit Service df60bb
				smallval = q[2];	/* index on g */
Packit Service df60bb
			}
Packit Service df60bb
		}
Packit Service df60bb
		q = nnq->network[smallpos];
Packit Service df60bb
		/* swap p (i) and q (smallpos) entries */
Packit Service df60bb
		if (i != smallpos) {
Packit Service df60bb
			j = q[0];
Packit Service df60bb
			q[0] = p[0];
Packit Service df60bb
			p[0] = j;
Packit Service df60bb
			j = q[1];
Packit Service df60bb
			q[1] = p[1];
Packit Service df60bb
			p[1] = j;
Packit Service df60bb
			j = q[2];
Packit Service df60bb
			q[2] = p[2];
Packit Service df60bb
			p[2] = j;
Packit Service df60bb
			j = q[3];
Packit Service df60bb
			q[3] = p[3];
Packit Service df60bb
			p[3] = j;
Packit Service df60bb
			j = q[4];
Packit Service df60bb
			q[4] = p[4];
Packit Service df60bb
			p[4] = j;
Packit Service df60bb
		}
Packit Service df60bb
		/* smallval entry is now in position i */
Packit Service df60bb
		if (smallval != previouscol) {
Packit Service df60bb
			nnq->netindex[previouscol] = (startpos+i)>>1;
Packit Service df60bb
			for (j=previouscol+1; j<smallval; j++) nnq->netindex[j] = i;
Packit Service df60bb
			previouscol = smallval;
Packit Service df60bb
			startpos = i;
Packit Service df60bb
		}
Packit Service df60bb
	}
Packit Service df60bb
	nnq->netindex[previouscol] = (startpos+maxnetpos)>>1;
Packit Service df60bb
	for (j=previouscol+1; j<256; j++) nnq->netindex[j] = maxnetpos; /* really 256 */
Packit Service df60bb
}
Packit Service df60bb
Packit Service df60bb
Packit Service df60bb
/* Search for ABGR values 0..255 (after net is unbiased) and return colour index
Packit Service df60bb
	 ---------------------------------------------------------------------------- */
Packit Service df60bb
unsigned int inxsearch(nnq, al,b,g,r)
Packit Service df60bb
nn_quant *nnq;
Packit Service df60bb
register int al, b, g, r;
Packit Service df60bb
{
Packit Service df60bb
	register int i, j, dist, a, bestd;
Packit Service df60bb
	register int *p;
Packit Service df60bb
	unsigned int best;
Packit Service df60bb
Packit Service df60bb
	bestd = 1000;		/* biggest possible dist is 256*3 */
Packit Service df60bb
	best = 0;
Packit Service df60bb
	i = nnq->netindex[g];	/* index on g */
Packit Service df60bb
	j = i-1;		/* start at netindex[g] and work outwards */
Packit Service df60bb
Packit Service df60bb
	while ((i<nnq->netsize) || (j>=0)) {
Packit Service df60bb
		if (i< nnq->netsize) {
Packit Service df60bb
			p = nnq->network[i];
Packit Service df60bb
			dist = p[2] - g;		/* inx key */
Packit Service df60bb
			if (dist >= bestd) i = nnq->netsize;	/* stop iter */
Packit Service df60bb
			else {
Packit Service df60bb
				i++;
Packit Service df60bb
				if (dist<0) dist = -dist;
Packit Service df60bb
				a = p[1] - b;
Packit Service df60bb
				if (a<0) a = -a;
Packit Service df60bb
				dist += a;
Packit Service df60bb
				if (dist
Packit Service df60bb
					a = p[3] - r;
Packit Service df60bb
					if (a<0) a = -a;
Packit Service df60bb
					dist += a;
Packit Service df60bb
				}
Packit Service df60bb
				if(dist
Packit Service df60bb
					a = p[0] - al;
Packit Service df60bb
					if (a<0) a = -a;
Packit Service df60bb
					dist += a;
Packit Service df60bb
				}
Packit Service df60bb
				if (dist
Packit Service df60bb
					bestd=dist;
Packit Service df60bb
					best=p[4];
Packit Service df60bb
				}
Packit Service df60bb
			}
Packit Service df60bb
		}
Packit Service df60bb
Packit Service df60bb
		if (j>=0) {
Packit Service df60bb
			p = nnq->network[j];
Packit Service df60bb
			dist = g - p[2]; /* inx key - reverse dif */
Packit Service df60bb
			if (dist >= bestd) j = -1; /* stop iter */
Packit Service df60bb
			else {
Packit Service df60bb
				j--;
Packit Service df60bb
				if (dist<0) dist = -dist;
Packit Service df60bb
				a = p[1] - b;
Packit Service df60bb
				if (a<0) a = -a;
Packit Service df60bb
				dist += a;
Packit Service df60bb
				if (dist
Packit Service df60bb
					a = p[3] - r;
Packit Service df60bb
					if (a<0) a = -a;
Packit Service df60bb
					dist += a;
Packit Service df60bb
				}
Packit Service df60bb
				if(dist
Packit Service df60bb
					a = p[0] - al;
Packit Service df60bb
					if (a<0) a = -a;
Packit Service df60bb
					dist += a;
Packit Service df60bb
				}
Packit Service df60bb
				if (dist
Packit Service df60bb
					bestd=dist;
Packit Service df60bb
					best=p[4];
Packit Service df60bb
				}
Packit Service df60bb
			}
Packit Service df60bb
		}
Packit Service df60bb
	}
Packit Service df60bb
Packit Service df60bb
	return(best);
Packit Service df60bb
}
Packit Service df60bb
Packit Service df60bb
/* Search for biased ABGR values
Packit Service df60bb
   ---------------------------- */
Packit Service df60bb
int contest(nnq, al,b,g,r)
Packit Service df60bb
nn_quant *nnq;
Packit Service df60bb
register int al,b,g,r;
Packit Service df60bb
{
Packit Service df60bb
	/* finds closest neuron (min dist) and updates freq */
Packit Service df60bb
	/* finds best neuron (min dist-bias) and returns position */
Packit Service df60bb
	/* for frequently chosen neurons, freq[i] is high and bias[i] is negative */
Packit Service df60bb
	/* bias[i] = gamma*((1/netsize)-freq[i]) */
Packit Service df60bb
Packit Service df60bb
	register int i,dist,a,biasdist,betafreq;
Packit Service df60bb
	unsigned int bestpos,bestbiaspos;
Packit Service df60bb
	double bestd,bestbiasd;
Packit Service df60bb
	register int *p,*f, *n;
Packit Service df60bb
Packit Service df60bb
	bestd = ~(((int) 1)<<31);
Packit Service df60bb
	bestbiasd = bestd;
Packit Service df60bb
	bestpos = 0;
Packit Service df60bb
	bestbiaspos = bestpos;
Packit Service df60bb
	p = nnq->bias;
Packit Service df60bb
	f = nnq->freq;
Packit Service df60bb
Packit Service df60bb
	for (i=0; i< nnq->netsize; i++) {
Packit Service df60bb
		n = nnq->network[i];
Packit Service df60bb
		dist = n[0] - al;
Packit Service df60bb
		if (dist<0) dist = -dist;
Packit Service df60bb
		a = n[1] - b;
Packit Service df60bb
		if (a<0) a = -a;
Packit Service df60bb
		dist += a;
Packit Service df60bb
		a = n[2] - g;
Packit Service df60bb
		if (a<0) a = -a;
Packit Service df60bb
		dist += a;
Packit Service df60bb
		a = n[3] - r;
Packit Service df60bb
		if (a<0) a = -a;
Packit Service df60bb
		dist += a;
Packit Service df60bb
		if (dist
Packit Service df60bb
			bestd=dist;
Packit Service df60bb
			bestpos=i;
Packit Service df60bb
		}
Packit Service df60bb
		biasdist = dist - ((*p)>>(intbiasshift-netbiasshift));
Packit Service df60bb
		if (biasdist
Packit Service df60bb
			bestbiasd=biasdist;
Packit Service df60bb
			bestbiaspos=i;
Packit Service df60bb
		}
Packit Service df60bb
		betafreq = (*f >> betashift);
Packit Service df60bb
		*f++ -= betafreq;
Packit Service df60bb
		*p++ += (betafreq<
Packit Service df60bb
	}
Packit Service df60bb
	nnq->freq[bestpos] += beta;
Packit Service df60bb
	nnq->bias[bestpos] -= betagamma;
Packit Service df60bb
	return(bestbiaspos);
Packit Service df60bb
}
Packit Service df60bb
Packit Service df60bb
Packit Service df60bb
/* Move neuron i towards biased (a,b,g,r) by factor alpha
Packit Service df60bb
	 ---------------------------------------------------- */
Packit Service df60bb
Packit Service df60bb
void altersingle(nnq, alpha,i,al,b,g,r)
Packit Service df60bb
nn_quant *nnq;
Packit Service df60bb
register int alpha,i,al,b,g,r;
Packit Service df60bb
{
Packit Service df60bb
	register int *n;
Packit Service df60bb
Packit Service df60bb
	n = nnq->network[i];	/* alter hit neuron */
Packit Service df60bb
	*n -= (alpha*(*n - al)) / initalpha;
Packit Service df60bb
	n++;
Packit Service df60bb
	*n -= (alpha*(*n - b)) / initalpha;
Packit Service df60bb
	n++;
Packit Service df60bb
	*n -= (alpha*(*n - g)) / initalpha;
Packit Service df60bb
	n++;
Packit Service df60bb
	*n -= (alpha*(*n - r)) / initalpha;
Packit Service df60bb
}
Packit Service df60bb
Packit Service df60bb
Packit Service df60bb
/* Move adjacent neurons by precomputed alpha*(1-((i-j)^2/[r]^2)) in radpower[|i-j|]
Packit Service df60bb
	 --------------------------------------------------------------------------------- */
Packit Service df60bb
Packit Service df60bb
void alterneigh(nnq, rad,i,al,b,g,r)
Packit Service df60bb
nn_quant *nnq;
Packit Service df60bb
int rad,i;
Packit Service df60bb
register int al,b,g,r;
Packit Service df60bb
{
Packit Service df60bb
	register int j,k,lo,hi,a;
Packit Service df60bb
	register int *p, *q;
Packit Service df60bb
Packit Service df60bb
	lo = i-rad;
Packit Service df60bb
	if (lo<-1) lo=-1;
Packit Service df60bb
	hi = i+rad;
Packit Service df60bb
	if (hi>nnq->netsize) hi=nnq->netsize;
Packit Service df60bb
Packit Service df60bb
	j = i+1;
Packit Service df60bb
	k = i-1;
Packit Service df60bb
	q = nnq->radpower;
Packit Service df60bb
	while ((j<hi) || (k>lo)) {
Packit Service df60bb
		a = (*(++q));
Packit Service df60bb
		if (j
Packit Service df60bb
			p = nnq->network[j];
Packit Service df60bb
			*p -= (a*(*p - al)) / alpharadbias;
Packit Service df60bb
			p++;
Packit Service df60bb
			*p -= (a*(*p - b)) / alpharadbias;
Packit Service df60bb
			p++;
Packit Service df60bb
			*p -= (a*(*p - g)) / alpharadbias;
Packit Service df60bb
			p++;
Packit Service df60bb
			*p -= (a*(*p - r)) / alpharadbias;
Packit Service df60bb
			j++;
Packit Service df60bb
		}
Packit Service df60bb
		if (k>lo) {
Packit Service df60bb
			p = nnq->network[k];
Packit Service df60bb
			*p -= (a*(*p - al)) / alpharadbias;
Packit Service df60bb
			p++;
Packit Service df60bb
			*p -= (a*(*p - b)) / alpharadbias;
Packit Service df60bb
			p++;
Packit Service df60bb
			*p -= (a*(*p - g)) / alpharadbias;
Packit Service df60bb
			p++;
Packit Service df60bb
			*p -= (a*(*p - r)) / alpharadbias;
Packit Service df60bb
			k--;
Packit Service df60bb
		}
Packit Service df60bb
	}
Packit Service df60bb
}
Packit Service df60bb
Packit Service df60bb
Packit Service df60bb
/* Main Learning Loop
Packit Service df60bb
   ------------------ */
Packit Service df60bb
Packit Service df60bb
void learn(nnq, verbose) /* Stu: N.B. added parameter so that main() could control verbosity. */
Packit Service df60bb
nn_quant *nnq;
Packit Service df60bb
int verbose;
Packit Service df60bb
{
Packit Service df60bb
	register int i,j,al,b,g,r;
Packit Service df60bb
	int radius,rad,alpha,step,delta,samplepixels;
Packit Service df60bb
	register unsigned char *p;
Packit Service df60bb
	unsigned char *lim;
Packit Service df60bb
Packit Service df60bb
	nnq->alphadec = 30 + ((nnq->samplefac-1)/3);
Packit Service df60bb
	p = nnq->thepicture;
Packit Service df60bb
	lim = nnq->thepicture + nnq->lengthcount;
Packit Service df60bb
	samplepixels = nnq->lengthcount/(4 * nnq->samplefac);
Packit Service df60bb
	/* here's a problem with small images: samplepixels < ncycles => delta = 0 */
Packit Service df60bb
	delta = samplepixels/ncycles;
Packit Service df60bb
	/* kludge to fix */
Packit Service df60bb
	if(delta==0) delta = 1;
Packit Service df60bb
	alpha = initalpha;
Packit Service df60bb
	radius = initradius;
Packit Service df60bb
Packit Service df60bb
	rad = radius >> radiusbiasshift;
Packit Service df60bb
	
Packit Service df60bb
	for (i=0; i
Packit Service df60bb
		nnq->radpower[i] = alpha*(((rad*rad - i*i)*radbias)/(rad*rad));
Packit Service df60bb
Packit Service df60bb
	if (verbose) gd_error_ex(GD_NOTICE, "beginning 1D learning: initial radius=%d\n", rad);
Packit Service df60bb
Packit Service df60bb
	if ((nnq->lengthcount%prime1) != 0) step = 4*prime1;
Packit Service df60bb
	else {
Packit Service df60bb
		if ((nnq->lengthcount%prime2) !=0) step = 4*prime2;
Packit Service df60bb
		else {
Packit Service df60bb
			if ((nnq->lengthcount%prime3) !=0) step = 4*prime3;
Packit Service df60bb
			else step = 4*prime4;
Packit Service df60bb
		}
Packit Service df60bb
	}
Packit Service df60bb
Packit Service df60bb
	i = 0;
Packit Service df60bb
	while (i < samplepixels) {
Packit Service df60bb
		al = p[ALPHA] << netbiasshift;
Packit Service df60bb
		b = p[BLUE] << netbiasshift;
Packit Service df60bb
		g = p[GREEN] << netbiasshift;
Packit Service df60bb
		r = p[RED] << netbiasshift;
Packit Service df60bb
		j = contest(nnq, al,b,g,r);
Packit Service df60bb
Packit Service df60bb
		altersingle(nnq, alpha,j,al,b,g,r);
Packit Service df60bb
		if (rad) alterneigh(nnq, rad,j,al,b,g,r);   /* alter neighbours */
Packit Service df60bb
Packit Service df60bb
		p += step;
Packit Service df60bb
		while (p >= lim) p -= nnq->lengthcount;
Packit Service df60bb
Packit Service df60bb
		i++;
Packit Service df60bb
		if (i%delta == 0) {                    /* FPE here if delta=0*/
Packit Service df60bb
			alpha -= alpha / nnq->alphadec;
Packit Service df60bb
			radius -= radius / radiusdec;
Packit Service df60bb
			rad = radius >> radiusbiasshift;
Packit Service df60bb
			if (rad <= 1) rad = 0;
Packit Service df60bb
			for (j=0; j
Packit Service df60bb
				nnq->radpower[j] = alpha*(((rad*rad - j*j)*radbias)/(rad*rad));
Packit Service df60bb
		}
Packit Service df60bb
	}
Packit Service df60bb
	if (verbose) gd_error_ex(GD_NOTICE, "finished 1D learning: final alpha=%f !\n",((float)alpha)/initalpha);
Packit Service df60bb
}
Packit Service df60bb
Packit Service df60bb
/**
Packit Service df60bb
 * Function: gdImageNeuQuant
Packit Service df60bb
 *
Packit Service df60bb
 * Creates a new palette image from a truecolor image
Packit Service df60bb
 *
Packit Service df60bb
 * This is the same as calling <gdImageCreatePaletteFromTrueColor> with the
Packit Service df60bb
 * quantization method <GD_QUANT_NEUQUANT>.
Packit Service df60bb
 *
Packit Service df60bb
 * Parameters:
Packit Service df60bb
 *   im            - The image.
Packit Service df60bb
 *   max_color     - The number of desired palette entries.
Packit Service df60bb
 *   sample_factor - The quantization precision between 1 (highest quality) and
Packit Service df60bb
 *                   10 (fastest).
Packit Service df60bb
 *
Packit Service df60bb
 * Returns:
Packit Service df60bb
 *   A newly create palette image; NULL on failure.
Packit Service df60bb
 */
Packit Service df60bb
BGD_DECLARE(gdImagePtr) gdImageNeuQuant(gdImagePtr im, const int max_color, int sample_factor)
Packit Service df60bb
{
Packit Service df60bb
	const int newcolors = max_color;
Packit Service df60bb
	const int verbose = 1;
Packit Service df60bb
Packit Service df60bb
	int bot_idx, top_idx; /* for remapping of indices */
Packit Service df60bb
	int remap[MAXNETSIZE];
Packit Service df60bb
	int i,x;
Packit Service df60bb
Packit Service df60bb
	unsigned char map[MAXNETSIZE][4];
Packit Service df60bb
	unsigned char *d;
Packit Service df60bb
Packit Service df60bb
	nn_quant *nnq = NULL;
Packit Service df60bb
Packit Service df60bb
	int row;
Packit Service df60bb
	unsigned char *rgba = NULL;
Packit Service df60bb
	gdImagePtr dst = NULL;
Packit Service df60bb
Packit Service df60bb
	/* Default it to 3 */
Packit Service df60bb
	if (sample_factor < 1) {
Packit Service df60bb
		sample_factor = 3;
Packit Service df60bb
	}
Packit Service df60bb
	/* Start neuquant */
Packit Service df60bb
	/* Pierre:
Packit Service df60bb
	 * This implementation works with aligned contiguous buffer only
Packit Service df60bb
	 * Upcoming new buffers are contiguous and will be much faster.
Packit Service df60bb
	 * let don't bloat this code to support our good "old" 31bit format.
Packit Service df60bb
	 * It alos lets us convert palette image, if one likes to reduce
Packit Service df60bb
	 * a palette
Packit Service df60bb
	 */
Packit Service df60bb
	if (overflow2(gdImageSX(im), gdImageSY(im))
Packit Service df60bb
	        || overflow2(gdImageSX(im) * gdImageSY(im), 4)) {
Packit Service df60bb
		goto done;
Packit Service df60bb
	}
Packit Service df60bb
	rgba = (unsigned char *) gdMalloc(gdImageSX(im) * gdImageSY(im) * 4);
Packit Service df60bb
	if (!rgba) {
Packit Service df60bb
		goto done;
Packit Service df60bb
	}
Packit Service df60bb
Packit Service df60bb
	d = rgba;
Packit Service df60bb
	for (row = 0; row < gdImageSY(im); row++) {
Packit Service df60bb
		int *p = im->tpixels[row];
Packit Service df60bb
		register int c;
Packit Service df60bb
Packit Service df60bb
		for (i = 0; i < gdImageSX(im); i++) {
Packit Service df60bb
			c = *p;
Packit Service df60bb
			*d++ = gdImageAlpha(im, c);
Packit Service df60bb
			*d++ = gdImageRed(im, c);
Packit Service df60bb
			*d++ = gdImageBlue(im, c);
Packit Service df60bb
			*d++ = gdImageGreen(im, c);
Packit Service df60bb
			p++;
Packit Service df60bb
		}
Packit Service df60bb
	}
Packit Service df60bb
Packit Service df60bb
	nnq = (nn_quant *) gdMalloc(sizeof(nn_quant));
Packit Service df60bb
	if (!nnq) {
Packit Service df60bb
		goto done;
Packit Service df60bb
	}
Packit Service df60bb
Packit Service df60bb
	initnet(nnq, rgba, gdImageSY(im) * gdImageSX(im) * 4, sample_factor, newcolors);
Packit Service df60bb
Packit Service df60bb
	learn(nnq, verbose);
Packit Service df60bb
	unbiasnet(nnq);
Packit Service df60bb
	getcolormap(nnq, (unsigned char*)map);
Packit Service df60bb
	inxbuild(nnq);
Packit Service df60bb
	/* remapping colormap to eliminate opaque tRNS-chunk entries... */
Packit Service df60bb
	for (top_idx = newcolors-1, bot_idx = x = 0;  x < newcolors;  ++x) {
Packit Service df60bb
		if (map[x][3] == 255) { /* maxval */
Packit Service df60bb
			remap[x] = top_idx--;
Packit Service df60bb
		} else {
Packit Service df60bb
			remap[x] = bot_idx++;
Packit Service df60bb
		}
Packit Service df60bb
	}
Packit Service df60bb
	if (bot_idx != top_idx + 1) {
Packit Service df60bb
		gd_error("  internal logic error: remapped bot_idx = %d, top_idx = %d\n",
Packit Service df60bb
			 bot_idx, top_idx);
Packit Service df60bb
		goto done;
Packit Service df60bb
	}
Packit Service df60bb
Packit Service df60bb
	dst = gdImageCreate(gdImageSX(im), gdImageSY(im));
Packit Service df60bb
	if (!dst) {
Packit Service df60bb
		goto done;
Packit Service df60bb
	}
Packit Service df60bb
Packit Service df60bb
	for (x = 0; x < newcolors; ++x) {
Packit Service df60bb
		dst->red[remap[x]] = map[x][0];
Packit Service df60bb
		dst->green[remap[x]] = map[x][1];
Packit Service df60bb
		dst->blue[remap[x]] = map[x][2];
Packit Service df60bb
		dst->alpha[remap[x]] = map[x][3];
Packit Service df60bb
		dst->open[remap[x]] = 0;
Packit Service df60bb
		dst->colorsTotal++;
Packit Service df60bb
	}
Packit Service df60bb
Packit Service df60bb
	/* Do each image row */
Packit Service df60bb
	for ( row = 0; row < gdImageSY(im); ++row ) {
Packit Service df60bb
		int offset;
Packit Service df60bb
		unsigned char *p = dst->pixels[row];
Packit Service df60bb
Packit Service df60bb
		/* Assign the new colors */
Packit Service df60bb
		offset = row * gdImageSX(im) * 4;
Packit Service df60bb
		for(i=0; i < gdImageSX(im); i++) {
Packit Service df60bb
			p[i] = remap[
Packit Service df60bb
			           inxsearch(nnq, rgba[i * 4 + offset + ALPHA],
Packit Service df60bb
			                     rgba[i * 4 + offset + BLUE],
Packit Service df60bb
			                     rgba[i * 4 + offset + GREEN],
Packit Service df60bb
			                     rgba[i * 4 + offset + RED])
Packit Service df60bb
			       ];
Packit Service df60bb
		}
Packit Service df60bb
	}
Packit Service df60bb
Packit Service df60bb
done:
Packit Service df60bb
	if (rgba) {
Packit Service df60bb
		gdFree(rgba);
Packit Service df60bb
	}
Packit Service df60bb
Packit Service df60bb
	if (nnq) {
Packit Service df60bb
		gdFree(nnq);
Packit Service df60bb
	}
Packit Service df60bb
	return dst;
Packit Service df60bb
}