[vlc-devel] [PATCH v2 1/9] Add monotone cubic interpolation support
Jean-Baptiste Kempf
jb at videolan.org
Sun Nov 29 00:23:05 CET 2015
Is there any way to have test on those?
On 28 Nov, Ronald Wright wrote :
> ---
> include/vlc_interpolation.h | 127 ++++++++++++++++++++++++++
> src/misc/interpolation.c | 216 ++++++++++++++++++++++++++++++++++++++++++++
> 2 files changed, 343 insertions(+)
> create mode 100644 include/vlc_interpolation.h
> create mode 100644 src/misc/interpolation.c
>
> diff --git a/include/vlc_interpolation.h b/include/vlc_interpolation.h
> new file mode 100644
> index 0000000..ac54e6e
> --- /dev/null
> +++ b/include/vlc_interpolation.h
> @@ -0,0 +1,127 @@
> +/*****************************************************************************
> + * vlc_interpolation.h: Interpolation utility
> + *****************************************************************************
> + * Copyright (C) 2015 Ronald Wright
> + * $Id$
> + *
> + * Author: Ronald Wright <logiconcepts819 at gmail.com>
> + *
> + * This program 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.1 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.
> + *****************************************************************************/
> +
> +#ifndef VLC_INTERPOLATION_H
> +#define VLC_INTERPOLATION_H
> +
> +/**
> + * \file
> + * \brief Interface definition for monotone cubic spline interpolation
> + * (implementation in src/misc/interpolation.c)
> + *
> + * \section background Background
> + *
> + * This implements a modification of Fritsch-Carlson monotone cubic spline
> + * interpolation. Fritsch-Carlson interpolation creates an interpolant as
> + * follows:
> + *
> + * 1. The slopes between successive data points are computed.
> + * 2. The tangents at each data point are computed as the average of the
> + * secants.
> + * 3. To preserve monotonicity, slopes are set to zero wherever two successive
> + * points are equal.
> + * 4. To prevent overshoot, two successive slopes, normalized by the tangent
> + * at the leftmost point, are restricted so that the sum of their squares do
> + * not exceed a certain threshold (9 in this algorithm).
> + *
> + * The interpolation is modified such that all sample points left of the spline
> + * have values equal to the left-most data point and all sample points right of
> + * the spline have values equal to the right-most data point. Hence, the
> + * resulting line extending outward from the left-most data point has zero
> + * slope, and the resulting line extending outward from the right-most data
> + * point has zero slope.
> + *
> + * \section example Example usage
> + *
> + * \code{.cpp}
> + * #define NUM_POINTS 4
> + * ...
> + * my_function()
> + * {
> + * ...
> + * float points[NUM_POINTS] = { -0.5, 1.0, 2.5, 5.0 };
> + * float values[NUM_POINTS] = { 8.0, -5.0, 0.0, 3.0 };
> + * ...
> + * vlc_interpolant_t * itp;
> + * if( vlc_interpolant_create( &itp, points, values, NUM_POINTS ) == VLC_SUCCESS )
> + * {
> + * ...
> + * float y = vlc_interpolate( itp, 0.5 );
> + * ...
> + * vlc_interpolant_destroy( itp );
> + * ...
> + * }
> + * ...
> + * }
> + * \endcode
> + */
> +
> +/*****************************************************************************
> + * Interpolant type
> + *****************************************************************************/
> +
> +/**
> + * The interpolant data type. This is a pointer to a private structure defined
> + * in misc/interpolation.c.
> + */
> +
> +typedef struct _vlc_interpolant_t vlc_interpolant_t;
> +
> +/*****************************************************************************
> + * Interpolation routines
> + *****************************************************************************/
> +
> +/**
> + * Create a new interpolant from the given data points and data point values.
> + *
> + * \param pp_interpolant The object that shall hold interpolant data
> + * \param pf_points The set of data points (or data x-values) to use
> + * \param pf_values The set of data point values (or data y-values) to use
> + * \param i_point_count The number of points and values
> + * \retval VLC_SUCCESS Success
> + * \retval VLC_ENOMEM Memory failure
> + * \retval VLC_EBADVAR Not enough data points to create the spline
> + */
> +VLC_API int vlc_interpolant_create( vlc_interpolant_t ** pp_interpolant,
> + const float * pf_points,
> + const float * pf_values,
> + int i_point_count );
> +
> +/**
> + * Sample the spline at the given sample point.
> + *
> + * \param p_interpolant The object that holds interpolant data
> + * \param f_sample_point The point at which the spline is to be sampled
> + * \return The value of the spline at the given sample point
> + */
> +VLC_API float vlc_interpolate( const vlc_interpolant_t * p_interpolant,
> + float f_sample_point );
> +
> +/**
> + * Destroy an interpolant that is no longer in use.
> + *
> + * \param p_interpolant The object that holds interpolant data
> + */
> +VLC_API void vlc_interpolant_destroy( vlc_interpolant_t * p_interpolant );
> +
> +#endif
> diff --git a/src/misc/interpolation.c b/src/misc/interpolation.c
> new file mode 100644
> index 0000000..1bfbd63
> --- /dev/null
> +++ b/src/misc/interpolation.c
> @@ -0,0 +1,216 @@
> +/*****************************************************************************
> + * vlc_interpolation.h: Interpolation utility
> + *****************************************************************************
> + * Copyright (C) 2015 Ronald Wright
> + * $Id$
> + *
> + * Author: Ronald Wright <logiconcepts819 at gmail.com>
> + *
> + * This program 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.1 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.
> + *****************************************************************************/
> +
> +/*****************************************************************************
> + * Preamble
> + *****************************************************************************/
> +
> +#ifdef HAVE_CONFIG_H
> +# include "config.h"
> +#endif
> +
> +#include <vlc_common.h>
> +#include <vlc_interpolation.h>
> +
> +/*****************************************************************************
> + * Documentation : Read vlc_interpolation.h
> + *****************************************************************************/
> +
> +/*****************************************************************************
> + * Private types.
> + *****************************************************************************/
> +
> +struct _vlc_interpolant_t
> +{
> + float * pf_points;
> + float * pf_values;
> +
> + float * pf_c1s;
> + float * pf_c2s;
> + float * pf_c3s;
> +
> + int i_point_count;
> +
> + float f_cs[];
> +};
> +
> +/*****************************************************************************
> + * Implementation of interpolation routines.
> + *****************************************************************************/
> +
> +int vlc_interpolant_create( vlc_interpolant_t ** pp_interpolant,
> + const float * pf_points, const float * pf_values,
> + int i_point_count )
> +{
> + /* This subroutine assumes that each sample point in the sequence
> + * referenced by pf_points is unique and that the sequence is sorted. */
> +
> + /* We should have at least two data points for the computation */
> + if( i_point_count < 2 )
> + return VLC_EBADVAR;
> +
> + int i_segment_count = i_point_count - 1;
> +
> + /* Allocate memory for pf_dys, pf_dxs, and pf_ms, which will all be arrays
> + * of length i_segment_count. */
> +
> + float * pf_dys = malloc( 3 * i_segment_count * sizeof( *pf_dys ) );
> + if( unlikely( !pf_dys ) )
> + return VLC_ENOMEM;
> +
> + float * pf_dxs = pf_dys + i_segment_count;
> + float * pf_ms = pf_dxs + i_segment_count;
> +
> + /* Allocate memory for the interpolant struct and its data members. pf_c1s,
> + * pf_points_, and pf_values_ will be arrays of length i_point_count, and
> + * pf_c2s and pf_c3s will be arrays of length i_segment_count. */
> +
> + vlc_interpolant_t * p_interpolant;
> + float * pf_c1s, * pf_c2s, * pf_c3s, * pf_points_, * pf_values_;
> +
> + int i_ret_val = VLC_ENOMEM;
> + int i_data_len = 3 * i_point_count + 2 * i_segment_count;
> + int i_total_size = i_data_len * sizeof( *pf_c1s ) + sizeof( *p_interpolant );
> +
> + p_interpolant = malloc( i_total_size );
> + if( likely( p_interpolant != NULL ) )
> + {
> + i_ret_val = VLC_SUCCESS;
> +
> + pf_c1s = p_interpolant->f_cs;
> + pf_c2s = pf_c1s + i_point_count;
> + pf_c3s = pf_c2s + i_segment_count;
> + pf_points_ = pf_c3s + i_segment_count;
> + pf_values_ = pf_points_ + i_point_count;
> +
> + /* Get consecutive differences and slopes */
> + for( int i = 0; i < i_segment_count; i++ )
> + {
> + float f_dx = pf_points[i + 1] - pf_points[i];
> + float f_dy = pf_values[i + 1] - pf_values[i];
> + pf_dxs[i] = f_dx;
> + pf_dys[i] = f_dy;
> + pf_ms[i] = f_dy / f_dx;
> + }
> +
> + /* Get degree-1 coefficients */
> + pf_c1s[0] = pf_ms[0];
> + for( int i = 0; i < i_segment_count - 1; i++ )
> + {
> + float f_m = pf_ms[i];
> + float f_mNext = pf_ms[i + 1];
> + float f_c1;
> + if( f_m * f_mNext <= 0 )
> + {
> + f_c1 = 0.0f;
> + }
> + else
> + {
> + float f_dx = pf_dxs[i];
> + float f_dxNext = pf_dxs[i + 1];
> + float f_common = f_dx + f_dxNext;
> + f_c1 = 3.0f * f_common / ( ( f_common + f_dxNext ) / f_m
> + + ( f_common + f_dx ) / f_mNext );
> + }
> + pf_c1s[i + 1] = f_c1;
> + }
> + pf_c1s[i_segment_count] = pf_ms[i_segment_count - 1];
> +
> + /* Get degree-2 and degree-3 coefficients */
> + for( int i = 0; i < i_segment_count; i++ )
> + {
> + float f_c1 = pf_c1s[i];
> + float f_m = pf_ms[i];
> + float f_invDx = 1.0f / pf_dxs[i];
> + float f_common = f_c1 + pf_c1s[i + 1] - f_m - f_m;
> + pf_c2s[i] = ( f_m - f_c1 - f_common ) * f_invDx;
> + pf_c3s[i] = f_common * f_invDx * f_invDx;
> + }
> +
> + /* Set up structure */
> + memcpy( pf_points_, pf_points, i_point_count * sizeof( *pf_points ) );
> + memcpy( pf_values_, pf_values, i_point_count * sizeof( *pf_values ) );
> + p_interpolant->pf_c1s = pf_c1s;
> + p_interpolant->pf_c2s = pf_c2s;
> + p_interpolant->pf_c3s = pf_c3s;
> + p_interpolant->pf_points = pf_points_;
> + p_interpolant->pf_values = pf_values_;
> + p_interpolant->i_point_count = i_point_count;
> + *pp_interpolant = p_interpolant;
> + }
> + free( pf_dys );
> + return i_ret_val;
> +}
> +
> +float vlc_interpolate( const vlc_interpolant_t * p_interpolant,
> + float f_sample_point )
> +{
> + const float * pf_points = p_interpolant->pf_points;
> + const float * pf_values = p_interpolant->pf_values;
> +
> + /* Here, the implementation is modified to give the closest endpoints for
> + * sample points that lie outside the spline region */
> + int i = p_interpolant->i_point_count - 1;
> + if( f_sample_point >= pf_points[i] )
> + {
> + return pf_values[i];
> + }
> + if( f_sample_point <= pf_points[0] )
> + {
> + return pf_values[0];
> + }
> +
> + /* Use binary search to find the matching data value or the region where
> + * the matching value is to be interpolated */
> + int i_low = 0;
> + int i_high = i - 1;
> + while( i_low <= i_high )
> + {
> + int i_mid = ( i_low + i_high ) >> 1;
> + float f_point_here = pf_points[i_mid];
> + if( f_point_here < f_sample_point )
> + {
> + i_low = i_mid + 1;
> + }
> + else if( f_point_here > f_sample_point )
> + {
> + i_high = i_mid - 1;
> + }
> + else
> + {
> + return pf_values[i_mid];
> + }
> + }
> + i = i_high > 0 ? i_high : 0;
> +
> + /* Interpolate */
> + float f_diff = f_sample_point - pf_points[i];
> + return pf_values[i] + f_diff * ( p_interpolant->pf_c1s[i]
> + + f_diff * ( p_interpolant->pf_c2s[i]
> + + f_diff * p_interpolant->pf_c3s[i] ) );
> +}
> +
> +void vlc_interpolant_destroy( vlc_interpolant_t * p_interpolant )
> +{
> + free( p_interpolant );
> +}
> --
> 1.9.1
>
> _______________________________________________
> vlc-devel mailing list
> To unsubscribe or modify your subscription options:
> https://mailman.videolan.org/listinfo/vlc-devel
--
With my kindest regards,
--
Jean-Baptiste Kempf
http://www.jbkempf.com/ - +33 672 704 734
Sent from my Electronic Device
More information about the vlc-devel
mailing list