/* libppm3.c - ppm utility library part 3 ** ** Colormap routines. ** ** Copyright (C) 1989, 1991 by Jef Poskanzer. ** ** Permission to use, copy, modify, and distribute this software and its ** documentation for any purpose and without fee is hereby granted, provided ** that the above copyright notice appear in all copies and that both that ** copyright notice and this permission notice appear in supporting ** documentation. This software is provided "as is" without express or ** implied warranty. */ #include "ppm.h" #include "ppmcmap.h" #include "libppm.h" #define HASH_SIZE 20023 #define ppm_hashpixel(p) ( ( ( (long) PPM_GETR(p) * 33023 + \ (long) PPM_GETG(p) * 30013 + \ (long) PPM_GETB(p) * 27011 ) \ & 0x7fffffff ) % HASH_SIZE ) colorhist_vector ppm_computecolorhist( pixel ** const pixels, const int cols, const int rows, const int maxcolors, int * const colorsP ) { colorhash_table cht; colorhist_vector chv; cht = ppm_computecolorhash(pixels, cols, rows, maxcolors, colorsP); if (cht ==NULL) return NULL; chv = ppm_colorhashtocolorhist(cht, maxcolors); ppm_freecolorhash(cht); return chv; } colorhist_vector ppm_computecolorhist2(FILE * const ifp, const int cols, const int rows, const pixval maxval, const int format, const int maxcolors, int * const colorsP ) { colorhash_table cht; colorhist_vector chv; cht = ppm_computecolorhash2(ifp, cols, rows, maxval, format, maxcolors, colorsP); if (cht ==NULL) return NULL; chv = ppm_colorhashtocolorhist(cht, maxcolors); ppm_freecolorhash(cht); return chv; } void ppm_addtocolorhist( colorhist_vector chv, int * const colorsP, const int maxcolors, const pixel * const colorP, const int value, const int position ) { int i, j; /* Search colorhist for the color. */ for ( i = 0; i < *colorsP; ++i ) if ( PPM_EQUAL( chv[i].color, *colorP ) ) { /* Found it - move to new slot. */ if ( position > i ) { for ( j = i; j < position; ++j ) chv[j] = chv[j + 1]; } else if ( position < i ) { for ( j = i; j > position; --j ) chv[j] = chv[j - 1]; } chv[position].color = *colorP; chv[position].value = value; return; } if ( *colorsP < maxcolors ) { /* Didn't find it, but there's room to add it; so do so. */ for ( i = *colorsP; i > position; --i ) chv[i] = chv[i - 1]; chv[position].color = *colorP; chv[position].value = value; ++(*colorsP); } } colorhash_table computecolorhash(pixel ** const pixels, const int cols, const int rows, const int maxcolors, int * const colorsP, FILE * const ifp, pixval const maxval, int const format) { /*---------------------------------------------------------------------------- Compute a color histogram from an image. The input is one of two types: 1) a two-dimensional array of pixels 'pixels'; In this case, 'pixels' is non-NULL and 'ifp' is NULL. 2) an open file, positioned to the image data. In this case, 'pixels' is NULL and 'ifp' is non-NULL. ifp is the stream descriptor for the input file, and 'maxval' and 'format' are parameters of the image data in it. We return with the file still open and its position undefined. In either case, the image is 'cols' by 'rows'. Return the number of colors found as *colorsP. However, if the number of colors is greater than 'maxcolors', return a null return value and *colorsP undefined. -----------------------------------------------------------------------------*/ colorhash_table cht; int row; pixel * rowbuffer; /* malloc'ed */ /* Buffer for a row read from the input file; undefined (but still allocated) if input is not from a file. */ cht = ppm_alloccolorhash( ); *colorsP = 0; /* initial value */ rowbuffer = ppm_allocrow(cols); /* Go through the entire image, building a hash table of colors. */ for (row = 0; row < rows; ++row) { int col; pixel * pixelrow; /* The row of pixels we are processing */ if (ifp) { ppm_readppmrow(ifp, rowbuffer, cols, maxval, format); pixelrow = rowbuffer; } else pixelrow = pixels[row]; for (col = 0; col < cols; ++col) { const pixel pixel = pixelrow[col]; const int hash = ppm_hashpixel(pixel); colorhist_list chl; for (chl = cht[hash]; chl != (colorhist_list) 0 && !PPM_EQUAL(chl->ch.color, pixel); chl = chl->next); if (chl) chl->ch.value++; else { /* It's not in the hash yet, so add it (if allowed) */ ++(*colorsP); if (maxcolors > 0 && *colorsP > maxcolors) { ppm_freecolorhash(cht); return NULL; } else { chl = (colorhist_list) malloc(sizeof(struct colorhist_list_item)); if (chl == NULL) pm_error("out of memory computing hash table"); chl->ch.color = pixel; chl->ch.value = 1; chl->next = cht[hash]; cht[hash] = chl; } } } } ppm_freerow(rowbuffer); return cht; } colorhash_table ppm_computecolorhash( pixel ** const pixels, const int cols, const int rows, const int maxcolors, int * const colorsP ) { return computecolorhash(pixels, cols, rows, maxcolors, colorsP, NULL, 0, 0); } colorhash_table ppm_computecolorhash2(FILE * const ifp, const int cols, const int rows, const pixval maxval, const int format, const int maxcolors, int * const colorsP ) { return computecolorhash(NULL, cols, rows, maxcolors, colorsP, ifp, maxval, format); } colorhash_table ppm_alloccolorhash( ) { colorhash_table cht; int i; cht = (colorhash_table) malloc( HASH_SIZE * sizeof(colorhist_list) ); if ( cht == 0 ) pm_error( "out of memory allocating hash table" ); for ( i = 0; i < HASH_SIZE; ++i ) cht[i] = (colorhist_list) 0; return cht; } int ppm_addtocolorhash( colorhash_table cht, const pixel * const colorP, const int value ) { register int hash; register colorhist_list chl; chl = (colorhist_list) malloc( sizeof(struct colorhist_list_item) ); if (chl == NULL) return -1; hash = ppm_hashpixel(*colorP); chl->ch.color = *colorP; chl->ch.value = value; chl->next = cht[hash]; cht[hash] = chl; return 0; } colorhist_vector ppm_colorhashtocolorhist( const colorhash_table cht, const int maxcolors ) { colorhist_vector chv; colorhist_list chl; int i, j; /* Now collate the hash table into a simple colorhist array. */ chv = (colorhist_vector) malloc(maxcolors * sizeof(struct colorhist_item)); /* (Leave room for expansion by caller.) */ if ( chv == (colorhist_vector) 0 ) pm_error( "out of memory generating histogram" ); /* Loop through the hash table. */ j = 0; for ( i = 0; i < HASH_SIZE; ++i ) for ( chl = cht[i]; chl != (colorhist_list) 0; chl = chl->next ) { /* Add the new entry. */ chv[j] = chl->ch; ++j; } /* All done. */ return chv; } colorhash_table ppm_colorhisttocolorhash( const colorhist_vector chv, const int colors ) { colorhash_table cht; int i, hash; pixel color; colorhist_list chl; cht = ppm_alloccolorhash( ); /* Initializes to NULLs */ for ( i = 0; i < colors; ++i ) { color = chv[i].color; hash = ppm_hashpixel( color ); for ( chl = cht[hash]; chl != (colorhist_list) 0; chl = chl->next ) if ( PPM_EQUAL( chl->ch.color, color ) ) pm_error( "same color found twice - %d %d %d", PPM_GETR(color), PPM_GETG(color), PPM_GETB(color) ); chl = (colorhist_list) malloc( sizeof(struct colorhist_list_item) ); if ( chl == (colorhist_list) 0 ) pm_error( "out of memory" ); chl->ch.color = color; chl->ch.value = i; chl->next = cht[hash]; cht[hash] = chl; } return cht; } int ppm_lookupcolor( const colorhash_table cht, const pixel * const colorP ) { int hash; colorhist_list chl; hash = ppm_hashpixel( *colorP ); for ( chl = cht[hash]; chl != (colorhist_list) 0; chl = chl->next ) if ( PPM_EQUAL( chl->ch.color, *colorP ) ) return chl->ch.value; return -1; } void ppm_freecolorhist( colorhist_vector chv ) { free( (char*) chv ); } void ppm_freecolorhash( colorhash_table cht ) { int i; colorhist_list chl, chlnext; for (i = 0; i < HASH_SIZE; ++i) for (chl = cht[i]; chl != (colorhist_list) 0; chl = chlnext) { chlnext = chl->next; free(chl); } free(cht); } /***************************************************************************** The following "color row" routines are taken from Ingo Wilken's ilbm package, dated December 1994. Since they're only used by ppmtoilbm and ilbmtoppm today, they aren't documented or well maintained, but they seem pretty useful and ought to be used in other programs. -Bryan 2001.03.10 ****************************************************************************/ /* libcmap2.c - support routines for color rows ** ** Copyright (C) 1994 Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de) ** ** Permission to use, copy, modify, and distribute this software and its ** documentation for any purpose and without fee is hereby granted, provided ** that the above copyright notice appear in all copies and that both that ** copyright notice and this permission notice appear in supporting ** documentation. This software is provided "as is" without express or ** implied warranty. */ colorhash_table ppm_colorrowtocolorhash(colorrow, ncolors) pixel *colorrow; int ncolors; { colorhash_table cht; int i; cht = ppm_alloccolorhash(); for( i = 0; i < ncolors; i++ ) { if( ppm_lookupcolor(cht, &colorrow[i]) < 0 ) { if( ppm_addtocolorhash(cht, &colorrow[i], i) < 0 ) pm_error("out of memory adding to hash table"); } } return cht; } pixel * ppm_computecolorrow(pixels, cols, rows, maxcolors, ncolorsP) pixel **pixels; int cols, rows, maxcolors; int *ncolorsP; { int ncolors, row, col; colorhash_table cht; pixel *pixrow; pixrow = ppm_allocrow(maxcolors); cht = ppm_alloccolorhash(); ncolors = 0; for( row = 0; row < rows; row++ ) { for( col = 0; col < cols; col++ ) { if( ppm_lookupcolor(cht, &pixels[row][col]) < 0 ) { if( ncolors >= maxcolors ) { ppm_freerow(pixrow); pixrow = (pixel *)0; ncolors = -1; goto fail; } if( ppm_addtocolorhash(cht, &pixels[row][col], ncolors) < 0 ) pm_error("out of memory adding to hash table"); pixrow[ncolors] = pixels[row][col]; ++ncolors; } } } fail: ppm_freecolorhash(cht); *ncolorsP = ncolors; return pixrow; } pixel * ppm_mapfiletocolorrow(file, maxcolors, ncolorsP, maxvalP) FILE *file; int maxcolors; int *ncolorsP; pixval *maxvalP; { int cols, rows, format, row, col, ncolors; pixel *pixrow, *temprow; colorhash_table cht; pixrow = ppm_allocrow(maxcolors); ppm_readppminit(file, &cols, &rows, maxvalP, &format); temprow = ppm_allocrow(cols); cht = ppm_alloccolorhash(); ncolors = 0; for( row = 0; row < rows; row++ ) { ppm_readppmrow(file, temprow, cols, *maxvalP, format); for( col = 0; col < cols; col++ ) { if( ppm_lookupcolor(cht, &temprow[col]) < 0 ) { if( ncolors >= maxcolors ) { ppm_freerow(pixrow); pixrow = (pixel *)0; ncolors = -1; goto fail; } if( ppm_addtocolorhash(cht, &temprow[col], ncolors) < 0 ) pm_error("out of memory adding to hash table"); pixrow[ncolors] = temprow[col]; ++ncolors; } } } fail: ppm_freecolorhash(cht); ppm_freerow(temprow); *ncolorsP = ncolors; return pixrow; } static int pixel_cmp(const void * const a, const void * const b) { const pixel *p1 = (const pixel *)a, *p2 = (const pixel *)b; int diff; diff = PPM_GETR(*p1) - PPM_GETR(*p2); if( diff == 0 ) { diff = PPM_GETG(*p1) - PPM_GETG(*p2); if( diff == 0 ) diff = PPM_GETB(*p1) - PPM_GETB(*p2); } return diff; } static int (*custom_cmp)(pixel *, pixel *); static int custom_stub(const void * const a, const void * const b) { return (*custom_cmp)((pixel *)a, (pixel *)b); } void ppm_sortcolorrow(pixel * const colorrow, const int ncolors, int (*cmpfunc)(pixel *, pixel *)) { if( cmpfunc ) { custom_cmp = cmpfunc; qsort((void *)colorrow, ncolors, sizeof(pixel), custom_stub); } else qsort((void *)colorrow, ncolors, sizeof(pixel), pixel_cmp); } int ppm_addtocolorrow(colorrow, ncolorsP, maxcolors, pixelP) pixel *colorrow; int *ncolorsP; int maxcolors; pixel *pixelP; { int i; pixval r, g, b; r = PPM_GETR(*pixelP); g = PPM_GETG(*pixelP); b = PPM_GETB(*pixelP); for( i = 0; i < *ncolorsP; i++ ) { if( PPM_GETR(colorrow[i]) == r && PPM_GETG(colorrow[i]) == g && PPM_GETB(colorrow[i]) == b ) return i; } i = *ncolorsP; if( i >= maxcolors ) return -1; colorrow[i] = *pixelP; ++*ncolorsP; return i; } int ppm_findclosestcolor(colormap, ncolors, pP) pixel *colormap; int ncolors; pixel *pP; { /* Search colormap for closest match. */ /* algorithm taken from ppmquant.c -IUW */ register int i, r1, g1, b1; int ind; long dist; r1 = PPM_GETR(*pP); g1 = PPM_GETG(*pP); b1 = PPM_GETB(*pP); dist = 2000000000; for( i = 0; i < ncolors; i++ ) { register int r2, g2, b2; long newdist; r2 = PPM_GETR(colormap[i]); g2 = PPM_GETG(colormap[i]); b2 = PPM_GETB(colormap[i]); newdist = ( r1 - r2 ) * ( r1 - r2 ) + ( g1 - g2 ) * ( g1 - g2 ) + ( b1 - b2 ) * ( b1 - b2 ); if( newdist < dist ) { ind = i; dist = newdist; } } return ind; } void #if __STDC__ ppm_colorrowtomapfile(FILE *ofp, pixel *colormap, int ncolors, pixval maxval) #else ppm_colorrowtomapfile(ofp, colormap, ncolors, maxval) FILE *ofp; pixel *colormap; int ncolors; pixval maxval; #endif { int i; ppm_writeppminit(ofp, ncolors, 1, maxval, 1); for( i = 0; i < ncolors; i++ ) ppm_writeppmrow(ofp, &colormap[i], 1, maxval, 1); }