439 lines
10 KiB
C
439 lines
10 KiB
C
/* local histogram equalisation
|
|
*
|
|
* Copyright: 1991, N. Dessipris
|
|
*
|
|
* Author: N. Dessipris
|
|
* Written on: 24/10/1991
|
|
* Modified on :
|
|
* 25/1/96 JC
|
|
* - rewritten, adapting im_spcor()
|
|
* - correct result, 2x faster, partial, simpler, better arg checking
|
|
* 8/7/04
|
|
* - expand input rather than output with new im_embed() mode
|
|
* - _raw() output is one pixel larger
|
|
* - sets Xoffset/Yoffset
|
|
* 23/6/08
|
|
* - check for window too small as well
|
|
* 25/3/10
|
|
* - gtkdoc
|
|
* - small cleanups
|
|
* 5/9/13
|
|
* - redo as a class
|
|
* 9/9/13
|
|
* - any number of bands
|
|
* 20/1/17
|
|
* - add contrast limit
|
|
* - sum to <= target, not < target, since cumulative hists include the
|
|
* current value
|
|
* - scale result by 255, not 256, to avoid overflow
|
|
* - off by 1 fix for odd window widths
|
|
*/
|
|
|
|
/*
|
|
|
|
This file is part of VIPS.
|
|
|
|
VIPS is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU Lesser General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
|
02110-1301 USA
|
|
|
|
*/
|
|
|
|
/*
|
|
|
|
These files are distributed with VIPS - http://www.vips.ecs.soton.ac.uk
|
|
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif /*HAVE_CONFIG_H*/
|
|
#include <glib/gi18n-lib.h>
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <vips/vips.h>
|
|
#include <vips/internal.h>
|
|
|
|
typedef struct _VipsHistLocal {
|
|
VipsOperation parent_instance;
|
|
|
|
VipsImage *in;
|
|
VipsImage *out;
|
|
|
|
int width;
|
|
int height;
|
|
|
|
int max_slope;
|
|
|
|
} VipsHistLocal;
|
|
|
|
typedef VipsOperationClass VipsHistLocalClass;
|
|
|
|
G_DEFINE_TYPE( VipsHistLocal, vips_hist_local, VIPS_TYPE_OPERATION );
|
|
|
|
/* Our sequence value: the region this sequence is using, and local stats.
|
|
*/
|
|
typedef struct {
|
|
VipsRegion *ir; /* Input region */
|
|
|
|
/* A 256-element hist for every band.
|
|
*/
|
|
unsigned int **hist;
|
|
} VipsHistLocalSequence;
|
|
|
|
static int
|
|
vips_hist_local_stop( void *vseq, void *a, void *b )
|
|
{
|
|
VipsHistLocalSequence *seq = (VipsHistLocalSequence *) vseq;
|
|
VipsImage *in = (VipsImage *) a;
|
|
|
|
VIPS_UNREF( seq->ir );
|
|
if( seq->hist &&
|
|
in ) {
|
|
int i;
|
|
|
|
for( i = 0; i < in->Bands; i++ )
|
|
VIPS_FREE( seq->hist[i] );
|
|
}
|
|
VIPS_FREE( seq->hist );
|
|
VIPS_FREE( seq );
|
|
|
|
return( 0 );
|
|
}
|
|
|
|
static void *
|
|
vips_hist_local_start( VipsImage *out, void *a, void *b )
|
|
{
|
|
VipsImage *in = (VipsImage *) a;
|
|
VipsHistLocalSequence *seq;
|
|
|
|
int i;
|
|
|
|
if( !(seq = VIPS_NEW( NULL, VipsHistLocalSequence )) )
|
|
return( NULL );
|
|
seq->ir = NULL;
|
|
seq->hist = NULL;
|
|
|
|
if( !(seq->ir = vips_region_new( in )) ||
|
|
!(seq->hist = VIPS_ARRAY( NULL, in->Bands, unsigned int * )) ) {
|
|
vips_hist_local_stop( seq, NULL, NULL );
|
|
return( NULL );
|
|
}
|
|
|
|
for( i = 0; i < in->Bands; i++ )
|
|
if( !(seq->hist[i] = VIPS_ARRAY( NULL, 256, unsigned int )) ) {
|
|
vips_hist_local_stop( seq, NULL, NULL );
|
|
return( NULL );
|
|
}
|
|
|
|
return( seq );
|
|
}
|
|
|
|
static int
|
|
vips_hist_local_generate( VipsRegion *or,
|
|
void *vseq, void *a, void *b, gboolean *stop )
|
|
{
|
|
VipsHistLocalSequence *seq = (VipsHistLocalSequence *) vseq;
|
|
VipsImage *in = (VipsImage *) a;
|
|
const VipsHistLocal *local = (VipsHistLocal *) b;
|
|
VipsRect *r = &or->valid;
|
|
const int bands = in->Bands;
|
|
const int max_slope = local->max_slope;
|
|
|
|
VipsRect irect;
|
|
int y;
|
|
int lsk;
|
|
int centre; /* Offset to move to centre of window */
|
|
|
|
/* What part of ir do we need?
|
|
*/
|
|
irect.left = r->left;
|
|
irect.top = r->top;
|
|
irect.width = r->width + local->width;
|
|
irect.height = r->height + local->height;
|
|
if( vips_region_prepare( seq->ir, &irect ) )
|
|
return( -1 );
|
|
|
|
lsk = VIPS_REGION_LSKIP( seq->ir );
|
|
centre = lsk * (local->height / 2) + bands * (local->width / 2);
|
|
|
|
for( y = 0; y < r->height; y++ ) {
|
|
/* Get input and output pointers for this line.
|
|
*/
|
|
VipsPel * restrict p =
|
|
VIPS_REGION_ADDR( seq->ir, r->left, r->top + y );
|
|
VipsPel * restrict q =
|
|
VIPS_REGION_ADDR( or, r->left, r->top + y );
|
|
|
|
VipsPel * restrict p1;
|
|
int x, i, j, b;
|
|
|
|
/* Find histogram for the start of this line.
|
|
*/
|
|
for( b = 0; b < bands; b++ )
|
|
memset( seq->hist[b], 0, 256 * sizeof( unsigned int ) );
|
|
p1 = p;
|
|
for( j = 0; j < local->height; j++ ) {
|
|
for( i = 0, x = 0; x < local->width; x++ )
|
|
for( b = 0; b < bands; b++, i++ )
|
|
seq->hist[b][p1[i]] += 1;
|
|
|
|
p1 += lsk;
|
|
}
|
|
|
|
/* Loop for output pels.
|
|
*/
|
|
for( x = 0; x < r->width; x++ ) {
|
|
for( b = 0; b < bands; b++ ) {
|
|
/* Sum histogram up to current pel.
|
|
*/
|
|
unsigned int * restrict hist = seq->hist[b];
|
|
const int target = p[centre + b];
|
|
|
|
int sum;
|
|
|
|
sum = 0;
|
|
|
|
/* For CLAHE we need to limit the height of the
|
|
* hist to limit the amount we boost the
|
|
* contrast by.
|
|
*/
|
|
if( max_slope > 0 ) {
|
|
int sum_over;
|
|
|
|
sum_over = 0;
|
|
|
|
/* Must be <= target, since a cum hist
|
|
* always includes the current element.
|
|
*/
|
|
for( i = 0; i <= target; i++ ) {
|
|
if( hist[i] > max_slope ) {
|
|
sum_over += hist[i] -
|
|
max_slope;
|
|
sum += max_slope;
|
|
}
|
|
else
|
|
sum += hist[i];
|
|
}
|
|
|
|
for( ; i < 256; i++ ) {
|
|
if( hist[i] > max_slope )
|
|
sum_over += hist[i] -
|
|
max_slope;
|
|
}
|
|
|
|
/* The extra clipped off bit from the
|
|
* top of the hist is spread over all
|
|
* bins equally, then summed to target.
|
|
*/
|
|
sum += (target + 1) * sum_over / 256;
|
|
}
|
|
else {
|
|
sum = 0;
|
|
for( i = 0; i <= target; i++ )
|
|
sum += hist[i];
|
|
}
|
|
|
|
/* This can't overflow, even in
|
|
* contrast-limited mode.
|
|
*
|
|
* Scale by 255, not 256, or we'll get
|
|
* overflow.
|
|
*/
|
|
q[b] = 255 * sum /
|
|
(local->width * local->height);
|
|
|
|
/* Adapt histogram --- remove the pels from
|
|
* the left hand column, add in pels for a
|
|
* new right-hand column.
|
|
*/
|
|
p1 = p + b;
|
|
for( j = 0; j < local->height; j++ ) {
|
|
hist[p1[0]] -= 1;
|
|
hist[p1[bands * local->width]] += 1;
|
|
|
|
p1 += lsk;
|
|
}
|
|
}
|
|
|
|
p += bands;
|
|
q += bands;
|
|
}
|
|
}
|
|
|
|
return( 0 );
|
|
}
|
|
|
|
static int
|
|
vips_hist_local_build( VipsObject *object )
|
|
{
|
|
VipsObjectClass *class = VIPS_OBJECT_GET_CLASS( object );
|
|
VipsHistLocal *local = (VipsHistLocal *) object;
|
|
VipsImage **t = (VipsImage **) vips_object_local_array( object, 3 );
|
|
|
|
VipsImage *in;
|
|
|
|
if( VIPS_OBJECT_CLASS( vips_hist_local_parent_class )->build( object ) )
|
|
return( -1 );
|
|
|
|
in = local->in;
|
|
|
|
if( vips_image_decode( in, &t[0] ) )
|
|
return( -1 );
|
|
in = t[0];
|
|
|
|
if( vips_check_format( class->nickname, in, VIPS_FORMAT_UCHAR ) )
|
|
return( -1 );
|
|
|
|
if( local->width > in->Xsize ||
|
|
local->height > in->Ysize ) {
|
|
vips_error( class->nickname, "%s", _( "window too large" ) );
|
|
return( -1 );
|
|
}
|
|
|
|
/* Expand the input.
|
|
*/
|
|
if( vips_embed( in, &t[1],
|
|
local->width / 2, local->height / 2,
|
|
in->Xsize + local->width - 1, in->Ysize + local->height - 1,
|
|
"extend", VIPS_EXTEND_MIRROR,
|
|
NULL ) )
|
|
return( -1 );
|
|
in = t[1];
|
|
|
|
g_object_set( object, "out", vips_image_new(), NULL );
|
|
|
|
/* Set demand hints. FATSTRIP is good for us, as THINSTRIP will cause
|
|
* too many recalculations on overlaps.
|
|
*/
|
|
if( vips_image_pipelinev( local->out,
|
|
VIPS_DEMAND_STYLE_FATSTRIP, in, NULL ) )
|
|
return( -1 );
|
|
local->out->Xsize -= local->width - 1;
|
|
local->out->Ysize -= local->height - 1;
|
|
|
|
if( vips_image_generate( local->out,
|
|
vips_hist_local_start,
|
|
vips_hist_local_generate,
|
|
vips_hist_local_stop,
|
|
in, local ) )
|
|
return( -1 );
|
|
|
|
local->out->Xoffset = 0;
|
|
local->out->Yoffset = 0;
|
|
|
|
vips_reorder_margin_hint( local->out, local->width * local->height );
|
|
|
|
return( 0 );
|
|
}
|
|
|
|
static void
|
|
vips_hist_local_class_init( VipsHistLocalClass *class )
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS( class );
|
|
VipsObjectClass *object_class = (VipsObjectClass *) class;
|
|
VipsOperationClass *operation_class = VIPS_OPERATION_CLASS( class );
|
|
|
|
gobject_class->set_property = vips_object_set_property;
|
|
gobject_class->get_property = vips_object_get_property;
|
|
|
|
object_class->nickname = "hist_local";
|
|
object_class->description = _( "local histogram equalisation" );
|
|
object_class->build = vips_hist_local_build;
|
|
|
|
operation_class->flags = VIPS_OPERATION_SEQUENTIAL;
|
|
|
|
VIPS_ARG_IMAGE( class, "in", 1,
|
|
_( "Input" ),
|
|
_( "Input image" ),
|
|
VIPS_ARGUMENT_REQUIRED_INPUT,
|
|
G_STRUCT_OFFSET( VipsHistLocal, in ) );
|
|
|
|
VIPS_ARG_IMAGE( class, "out", 2,
|
|
_( "Output" ),
|
|
_( "Output image" ),
|
|
VIPS_ARGUMENT_REQUIRED_OUTPUT,
|
|
G_STRUCT_OFFSET( VipsHistLocal, out ) );
|
|
|
|
VIPS_ARG_INT( class, "width", 4,
|
|
_( "Width" ),
|
|
_( "Window width in pixels" ),
|
|
VIPS_ARGUMENT_REQUIRED_INPUT,
|
|
G_STRUCT_OFFSET( VipsHistLocal, width ),
|
|
1, VIPS_MAX_COORD, 1 );
|
|
|
|
VIPS_ARG_INT( class, "height", 5,
|
|
_( "Height" ),
|
|
_( "Window height in pixels" ),
|
|
VIPS_ARGUMENT_REQUIRED_INPUT,
|
|
G_STRUCT_OFFSET( VipsHistLocal, height ),
|
|
1, VIPS_MAX_COORD, 1 );
|
|
|
|
VIPS_ARG_INT( class, "max_slope", 6,
|
|
_( "Max slope" ),
|
|
_( "Maximum slope (CLAHE)" ),
|
|
VIPS_ARGUMENT_OPTIONAL_INPUT,
|
|
G_STRUCT_OFFSET( VipsHistLocal, max_slope ),
|
|
0, 100, 0 );
|
|
|
|
}
|
|
|
|
static void
|
|
vips_hist_local_init( VipsHistLocal *local )
|
|
{
|
|
}
|
|
|
|
/**
|
|
* vips_hist_local: (method)
|
|
* @in: input image
|
|
* @out: (out): output image
|
|
* @width: width of region
|
|
* @height: height of region
|
|
* @...: %NULL-terminated list of optional named arguments
|
|
*
|
|
* Optional arguments:
|
|
*
|
|
* * @max_slope: maximum brightening
|
|
*
|
|
* Performs local histogram equalisation on @in using a
|
|
* window of size @width by @height centered on the input pixel.
|
|
*
|
|
* The output image is the same size as the input image. The edge pixels are
|
|
* created by mirroring the input image outwards.
|
|
*
|
|
* If @max_slope is greater than 0, it sets the maximum value for the slope of
|
|
* the cumulative histogram, that is, the maximum brightening that is
|
|
* performed. A value of 3 is often used. Local histogram equalization with
|
|
* contrast limiting is usually called CLAHE.
|
|
*
|
|
* See also: vips_hist_equal().
|
|
*
|
|
* Returns: 0 on success, -1 on error
|
|
*/
|
|
int
|
|
vips_hist_local( VipsImage *in, VipsImage **out, int width, int height, ... )
|
|
{
|
|
va_list ap;
|
|
int result;
|
|
|
|
va_start( ap, height );
|
|
result = vips_call_split( "hist_local", ap, in, out, width, height );
|
|
va_end( ap );
|
|
|
|
return( result );
|
|
}
|