mac-rom/QuickDraw/PictUtilities/puMedian.c

608 lines
19 KiB
C
Raw Normal View History

/*
File: puMedian.c
Contains: This is the median color finding algorithm for the picture utilities package.
Written by: Dave Good. Some ideas stolen from Konstantin Othmer and Bruce Leak. Algorithm by Keith McGreggor.
Copyright: <EFBFBD> 1990 by Apple Computer, Inc., all rights reserved.
This file is used in these builds: BigBang
Change History (most recent first):
<11> 2/6/91 SMC DDG,#81681: Removed un-uncommented code in box splitting
routine, since it is never executed (or necessary).
<10> 2/4/91 SMC KON,#81681: Un-uncommented code in box splitting routine to
deal with a box filled with more than copy of the same color.
<9> 2/4/91 SMC KON,#81589: Fixed the median box splitting code. It didn't
seem to want to break a box on the blue axis.
<8> 12/17/90 KON <with DDG> Made the median color algorithm return better results
by splitting the box with the largest spread, instead of
splitting every box that we find.
<7> 9/21/90 DDG Made changes from code review. Fixed a possible stack overflow
problem, by allocating the boxInfo array as a buffer, instead of
a stack array. Minor changes to reflect the new interface to the
buffering routines.
<6> 8/16/90 DDG Cleaned up the comments.
<5> 8/5/90 DDG Added a parameter to the RecordMedianColors routine. This
routine should still never be called.
<4> 7/31/90 DDG Fixed a bug in the RecordMedianColors routine. Note that this
routine should never be called, but better safe than sorry.
<3> 7/30/90 DDG Added init, record, and kill routines to support the generic
colorPickMethod model. Also fixed a few bugs.
<2> 7/29/90 DDG Fixed header.
<1> 7/29/90 DDG First checked in using new structure.
To Do:
*/
/*----------------------------------------------------------------------------------------------------------*/
/*
| Includes
*/
#include "puPrivate.h"
/*----------------------------------------------------------------------------------------------------------*/
/*
| Private Defines
*/
#define noBoxToSplit -1
/*----------------------------------------------------------------------------------------------------------*/
/*
| Private Data Structures
*/
typedef struct BoxInfo
{
unsigned short count;
unsigned short minRed;
unsigned short maxRed;
unsigned short minGreen;
unsigned short maxGreen;
unsigned short minBlue;
unsigned short maxBlue;
unsigned long totalRed;
unsigned long totalGreen;
unsigned long totalBlue;
unsigned long padding1;
unsigned short padding2;
} BoxInfo;
typedef struct RangeInfo
{
unsigned short spread;
unsigned short start;
unsigned short end;
unsigned short step;
} RangeInfo;
/*----------------------------------------------------------------------------------------------------------*/
/*
| Private Function Declarations
*/
void SplitBox( BoxInfo *, short *, short, short );
void FillMedianTable( BoxInfo *, ColorSpec *, short );
/*----------------------------------------------------------------------------------------------------------*/
/*----------------------------------------------------------------------------------------------------------*/
/************************************************************************************************************/
/**** Functions Follow ****/
/************************************************************************************************************/
/*----------------------------------------------------------------------------------------------------------*/
/*----------------------------------------------------------------------------------------------------------*/
/*
| This routine is called to initialize the median method. It doesn<EFBFBD>t really do anything, except tell
| the main picture utilities package that it should store the colors in the ExactAnd555 format.
*/
pascal OSErr InitMedianMethod(
short colorsRequested,
long *dataHandlePtr,
short *colorBankTypePtr )
{
#pragma unused( colorsRequested )
*dataHandlePtr = nil;
*colorBankTypePtr = ColorBankIsExactAnd555;
return noErr;
}
/*----------------------------------------------------------------------------------------------------------*/
/*
| This routine is called to kill the median method. It doesn<EFBFBD>t do anything.
*/
pascal OSErr KillMedianMethod(
long dataHandle )
{
#pragma unused( dataHandle )
return noErr;
}
/*----------------------------------------------------------------------------------------------------------*/
/*
| This routine should never be called, so it returns an error.
*/
pascal OSErr RecordMedianColors(
long dataHandle,
RGBColor *colorPtr,
long colorCount,
long *uniqueColorsPtr )
{
#pragma unused( dataHandle, colorPtr, colorCount, uniqueColorsPtr )
return cantLoadPickMethodErr;
}
/*----------------------------------------------------------------------------------------------------------*/
/*
| This returns the colors that cover the widest range in the histogram.
*/
pascal OSErr CalcMedianTable(
long dataHandle,
short colorsRequested,
short *colorBankPtr,
ColorSpec *resultPtr )
{
#pragma unused( dataHandle )
register BoxInfo *boxInfoPtr;
register short *boxIDPtr;
register short index;
Buffer boxIDBuffer;
Buffer boxInfoBuffer;
short colorsToFill;
OSErr error;
/*
| Allocate two buffers, one to hold a box ID number for each color in the histogram and the second one to hold
| the actual box information (component range, etc). Then make <EFBFBD>boxInfoPtr<EFBFBD> a pointer to the first boxInfo in
| the boxInfo buffer. This variable won<EFBFBD>t get reset until after the main loop.
*/
if( error = NewBufferClear( &boxIDBuffer, HistogramTableSize, bufferFixedType ) )
return error;
if( error = NewBufferClear( &boxInfoBuffer, 256 * sizeof(BoxInfo), bufferFixedType ) )
return error;
boxInfoPtr = (BoxInfo *)(boxInfoBuffer.ptr);
/*
| Mark all the entries in the histogram as either being in box zero (if they have at least one color) or -1 if
| they had no colors. This will also give us the total number of colors in the histogram, which we must make
| sure to clip to the number of colors requested to get the number of colors to fill.
*/
{
register long histogramEnd;
short *histogramPtr;
boxIDPtr = (short *)(boxIDBuffer.ptr);
histogramPtr = colorBankPtr;
histogramEnd = (long)colorBankPtr + HistogramTableSize;
while( histogramPtr < (char *)histogramEnd )
{
if( *(histogramPtr++) > 0 )
{
*(boxIDPtr++) = 0;
boxInfoPtr->count++;
}
else
*(boxIDPtr++) = -1;
}
}
colorsToFill = (boxInfoPtr->count > colorsRequested) ? colorsRequested : boxInfoPtr->count;
/*
| Find the minimum and maximum range for each component of all the colors in box zero. Since all the colors are
| in box zero to start with, this finds the range for all the colors in the histogram. Note that we must
| initialize the minimum values for each box, but not the maximum ones. This is because we have cleared the
| boxInfo buffer, so that our maximum values are already initialized to zero for us. boxInfoPtr continues to
| point to the BoxInfo structure for block zero.
|
| Note that we declare a separate block to allow us to registerize the our component indexes.
*/
boxIDPtr = (short *)(boxIDBuffer.ptr);
{
register unsigned short red, green, blue;
boxInfoPtr->minRed = boxInfoPtr->minGreen = boxInfoPtr->minBlue = 0xFFFF;
for(red = 0; red < 32; red++)
{
for(green = 0; green < 32; green++)
{
for(blue = 0; blue < 32; blue++)
{
if( *(boxIDPtr++) == 0 )
{
if( red < boxInfoPtr->minRed ) boxInfoPtr->minRed = red;
if( green < boxInfoPtr->minGreen ) boxInfoPtr->minGreen = green;
if( blue < boxInfoPtr->minBlue ) boxInfoPtr->minBlue = blue;
if( red > boxInfoPtr->maxRed ) boxInfoPtr->maxRed = red;
if( green > boxInfoPtr->maxGreen ) boxInfoPtr->maxGreen = green;
if( blue > boxInfoPtr->maxBlue ) boxInfoPtr->maxBlue = blue;
}
}
}
}
}
/*
| We scan thru all the filled boxes multiple times, each time looking for the one that has the largest spread
| in either red, green or blue. Note that we only look at boxes that have more than one color, because we are
| going to split the box that we find into two sub-boxes. After we split the box, we loop back and scan thru
| all the filled boxes again (this time there is one more filled box than last time. We terminate both loops,
| when we have reached the number of colors to fill. Note that since we clipped colorsToFill to the total number
| of colors in the histogram, we will never be in a situation where we can<EFBFBD>t split a box, but we haven<EFBFBD>t used
| all our colors yet.
|
| Note that we declare a separate block to allow us to registerize some other variables.
*/
{
register short availableBox;
register short filledBoxes;
filledBoxes = availableBox = 1;
while( true /* filledBoxes < 256 */ )
{
register short theSpread;
register short spreadMax;
short maxIndex;
if( availableBox >= colorsToFill )
goto exitMain;
spreadMax = maxIndex = 0;
for(index = 0; (index < filledBoxes) && (spreadMax != 0x1F); index++)
{
boxInfoPtr = (BoxInfo *)( boxInfoBuffer.ptr + (index * sizeof(BoxInfo)) );
if( boxInfoPtr->count > 1 )
{
theSpread = boxInfoPtr->maxRed - boxInfoPtr->minRed;
if( theSpread > spreadMax )
{
spreadMax = theSpread;
maxIndex = index;
}
theSpread = boxInfoPtr->maxGreen - boxInfoPtr->minGreen;
if( theSpread > spreadMax )
{
spreadMax = theSpread;
maxIndex = index;
}
theSpread = boxInfoPtr->maxBlue - boxInfoPtr->minBlue;
if( theSpread > spreadMax )
{
spreadMax = theSpread;
maxIndex = index;
}
}
}
SplitBox((BoxInfo *)(boxInfoBuffer.ptr), (short *)(boxIDBuffer.ptr), maxIndex, availableBox++);
filledBoxes = availableBox;
}
}
exitMain:
/*
| Calculate the totals for each color component for all colors contained in each of the boxes. We first loop
| thru each color checking to see which box it belongs to. We then get a pointer to the BoxInfo for that
| particular box and for each component, we add the color to the appropriate total.
|
| Note that we declare a separate block to allow us to registerize the our component indexes.
*/
boxIDPtr = (short *)(boxIDBuffer.ptr);
{
register unsigned short red, green, blue;
for(red = 0; red < 32; red++)
{
for(green = 0; green < 32; green++)
{
for(blue = 0; blue < 32; blue++)
{
if( (index = *(boxIDPtr++)) >= 0 )
{
boxInfoPtr = (BoxInfo *)( boxInfoBuffer.ptr + (index * sizeof(BoxInfo)) );
boxInfoPtr->totalRed += red;
boxInfoPtr->totalGreen += green;
boxInfoPtr->totalBlue += blue;
}
}
}
}
}
/*
| Here is where we actually fill the color table that we are returning. We scan thru all the boxes in the
| boxInfo buffer, checking to see if their count is greater than zero. If it is, then we have a color to return,
| so for each component, we divide the total by the number of colors in this box to get the average component,
| then we smear the component out to fill a full 16-bit range for the color table. If the count for a particular
| color is zero, then we return black for that entry in the color table. Note that this will never happen until
| we have reached the end of the filled boxes, so at that point all we really want to do is fill the rest of the
| returned color table with black.
*/
FillMedianTable( (BoxInfo *)(boxInfoBuffer.ptr), resultPtr, colorsRequested );
/*
| Kill the buffers for the boxes and the boxIDs
*/
KillBuffer( &boxInfoBuffer );
KillBuffer( &boxIDBuffer );
return noErr;
}
/*----------------------------------------------------------------------------------------------------------*/
/*----------------------------------------------------------------------------------------------------------*/
/************************************************************************************************************/
/**** Internal Functions Follow ****/
/************************************************************************************************************/
/*----------------------------------------------------------------------------------------------------------*/
/*----------------------------------------------------------------------------------------------------------*/
static void SplitBox(
BoxInfo *boxInfoPtr,
short *mainBoxIDPtr,
short srcBox,
short dstBox )
{
RangeInfo first, second, third;
BoxInfo *slowSrcPtr, *slowDstPtr;
register unsigned short oneHalfCount;
/*
| Make pointers to the two boxes; the source box that we are taking colors from and the dest box
| that we are adding the taken colors to. These pointers are stored in non register locals, so
| that we can reload fast register pointers quickly.
*/
slowSrcPtr = boxInfoPtr + srcBox;
slowDstPtr = boxInfoPtr + dstBox;
/*
| The second box has a count that is equal to one half of the source box. Note that we subtract
| the number of colors we removed from the source box<EFBFBD>s count to ensure that even if the count is
| odd, we will not lose a color.
|
| We use a separate block in order to get register variables for our two box pointers (srcBoxPtr
| and dstBoxPtr).
*/
{
register BoxInfo *srcBoxPtr = slowSrcPtr;
register BoxInfo *dstBoxPtr = slowDstPtr;
dstBoxPtr->count = oneHalfCount = srcBoxPtr->count >> 1;
srcBoxPtr->count -= oneHalfCount;
/*
| Here we setup the three range blocks. This is some very tricky code that allows us to sort the
| component range blocks in order of the spread between the minimum and maximum values of the
| components. The variables that we need for each of the range blocks are the spread between the
| values of the components, the starting index (the minimum component value shifted over by the
| proper amount), the ending index (the maximum component value shifted over by the proper amount),
| and the step (the amount to increment the component index by each time thru the loop.
*/
first.spread = srcBoxPtr->maxRed - srcBoxPtr->minRed;
first.start = (srcBoxPtr->minRed) << 10;
first.end = (srcBoxPtr->maxRed) << 10;
first.step = 1024;
second.spread = srcBoxPtr->maxGreen - srcBoxPtr->minGreen;
second.start = (srcBoxPtr->minGreen) << 5;
second.end = (srcBoxPtr->maxGreen) << 5;
second.step = 32;
third.spread = srcBoxPtr->maxBlue - srcBoxPtr->minBlue;
third.start = srcBoxPtr->minBlue;
third.end = srcBoxPtr->maxBlue;
third.step = 1;
}
/*
| Here is where we sort the range blocks in order of the spread between the components. Notice that
| since we have a RangeInfo structure for each, we only need to swap pointers to swap two elements
| in this structure. Also notice that we don<EFBFBD>t have to actually swap the pointers, since we know
| the value of at least one of them at all times.
|
| We need a separate code block again so that we can registerize the pointers to each of our range
| info structures. We also registerize some other frequently used variables.
*/
{
register RangeInfo *firstPtr, *secondPtr, *thirdPtr;
RangeInfo *swapPtr; /*<9>*/
register unsigned short firstIndex, secondIndex, index;
firstPtr = &first;
secondPtr = &second;
thirdPtr = &third;
if( secondPtr->spread > firstPtr->spread )
{
secondPtr = firstPtr;
firstPtr = &second;
}
if( thirdPtr->spread > secondPtr->spread )
{
thirdPtr = secondPtr;
secondPtr = &third;
if( secondPtr->spread > firstPtr->spread )
{
swapPtr = firstPtr; /*<9>*/
firstPtr = secondPtr; /*<9>*/
secondPtr = swapPtr; /*<9>*/
}
}
/*
| This is where we actually use these range blocks that we set up and sorted . We look through
| all the colors that are within this defined range and we set half of them to be in the dest
| box instead of the src box. Note that mainBoxIDPtr is not a register; it would be nice if
| it was, but it is actually less important than the other variables.
*/
firstIndex = firstPtr->start;
while( firstIndex <= firstPtr->end )
{
secondIndex = secondPtr->start;
while( secondIndex <= secondPtr->end )
{
index = firstIndex + secondIndex + thirdPtr->start;
while( index <= (firstIndex + secondIndex + thirdPtr->end) )
{
if( mainBoxIDPtr[index] == srcBox )
{
mainBoxIDPtr[index] = dstBox;
if( --oneHalfCount == 0 )
goto exitLoop;
}
index += thirdPtr->step;
}
secondIndex += secondPtr->step;
}
firstIndex += firstPtr->step;
}
}
exitLoop:
/*
| We have now divided the original box into two parts. Now all we need to do is find the range of
| each of the components for both the original source box and the new destination box.
|
| Again, we have a new code block so that we can declare a completely different set of register
| variables. This time the src and dst box pointers are very important. The indexes for our loops
| are also important.
*/
{
BoxInfo initialBox;
register BoxInfo *srcBoxPtr = slowSrcPtr;
register BoxInfo *dstBoxPtr = slowDstPtr;
register short *boxIDPtr;
register unsigned short red, green, blue; /*<9>*/
register short boxIndex;
/*
| Copy the initial box structure (particularly the range information) into our local structure.
| This is because we need the initial values in the source box structure for our loop ending
| conditions, but we want to be able to fill the new source box structure directly in the loop.
*/
initialBox = *srcBoxPtr;
/*
| Initialize the minimum and maximum fields of the src and dst box structures.
*/
srcBoxPtr->minRed = srcBoxPtr->minGreen = srcBoxPtr->minBlue = 0xFFFF;
dstBoxPtr->minRed = dstBoxPtr->minGreen = dstBoxPtr->minBlue = 0xFFFF;
srcBoxPtr->maxRed = srcBoxPtr->maxGreen = srcBoxPtr->maxBlue = 0x0000;
dstBoxPtr->maxRed = dstBoxPtr->maxGreen = dstBoxPtr->maxBlue = 0x0000;
/*
| This is where we loop thru all the colors in the old range, checking them against either the
| source or destination box structure<EFBFBD>s minimums and maximums. Note that boxIDPtr is a pointer
| into the boxID buffer that tells which box each color belongs to. We can increment it by one
| as we move thru the blue component, but before each blue loop, we need to set it up properly.
*/
for( red = initialBox.minRed; red <= initialBox.maxRed; red++ )
{
for( green = initialBox.minGreen; green <= initialBox.maxGreen; green++ )
{
boxIDPtr = mainBoxIDPtr + (red << 10) + (green << 5) + initialBox.minBlue;
for( blue = initialBox.minBlue; blue <= initialBox.maxBlue; blue++ )
{
boxIndex = *(boxIDPtr++);
if( boxIndex == srcBox )
{
if( red < srcBoxPtr->minRed ) srcBoxPtr->minRed = red;
if( green < srcBoxPtr->minGreen ) srcBoxPtr->minGreen = green;
if( blue < srcBoxPtr->minBlue ) srcBoxPtr->minBlue = blue;
if( red > srcBoxPtr->maxRed ) srcBoxPtr->maxRed = red;
if( green > srcBoxPtr->maxGreen ) srcBoxPtr->maxGreen = green;
if( blue > srcBoxPtr->maxBlue ) srcBoxPtr->maxBlue = blue;
}
else if( boxIndex == dstBox )
{
if( red < dstBoxPtr->minRed ) dstBoxPtr->minRed = red;
if( green < dstBoxPtr->minGreen ) dstBoxPtr->minGreen = green;
if( blue < dstBoxPtr->minBlue ) dstBoxPtr->minBlue = blue;
if( red > dstBoxPtr->maxRed ) dstBoxPtr->maxRed = red;
if( green > dstBoxPtr->maxGreen ) dstBoxPtr->maxGreen = green;
if( blue > dstBoxPtr->maxBlue ) dstBoxPtr->maxBlue = blue;
}
}
}
}
}
}
/*----------------------------------------------------------------------------------------------------------*/