⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 arbresample.c

📁 著名的 helix realplayer 基于手机 symbian 系统的 播放器全套源代码
💻 C
字号:
/* ***** BEGIN LICENSE BLOCK ***** 
 * Version: RCSL 1.0/RPSL 1.0 
 *  
 * Portions Copyright (c) 1995-2002 RealNetworks, Inc. All Rights Reserved. 
 *      
 * The contents of this file, and the files included with this file, are 
 * subject to the current version of the RealNetworks Public Source License 
 * Version 1.0 (the "RPSL") available at 
 * http://www.helixcommunity.org/content/rpsl unless you have licensed 
 * the file under the RealNetworks Community Source License Version 1.0 
 * (the "RCSL") available at http://www.helixcommunity.org/content/rcsl, 
 * in which case the RCSL will apply. You may also obtain the license terms 
 * directly from RealNetworks.  You may not use this file except in 
 * compliance with the RPSL or, if you have a valid RCSL with RealNetworks 
 * applicable to this file, the RCSL.  Please see the applicable RPSL or 
 * RCSL for the rights, obligations and limitations governing use of the 
 * contents of the file.  
 *  
 * This file is part of the Helix DNA Technology. RealNetworks is the 
 * developer of the Original Code and owns the copyrights in the portions 
 * it created. 
 *  
 * This file, and the files included with this file, is distributed and made 
 * available on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER 
 * EXPRESS OR IMPLIED, AND REALNETWORKS HEREBY DISCLAIMS ALL SUCH WARRANTIES, 
 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, FITNESS 
 * FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT. 
 * 
 * Technology Compatibility Kit Test Suite(s) Location: 
 *    http://www.helixcommunity.org/content/tck 
 * 
 * Contributor(s): 
 *  
 * ***** END LICENSE BLOCK ***** */ 

/*
 * Polyphase sampling rate conversion.
 * Arbitrary mode.
 *
 * by Ken Cooke
 */

#include "hlxclib/stdlib.h"
#include "hlxclib/string.h"

#include "kaiser.h"
#include "mymath.h"
#include "allresamplers.h"

/* leave a trace of the C source version in the object code */
static const char VERSION[] = "$Revision: 1.9 $" ;

#undef ASSERT
#define ASSERT(x)

/* oversampling factor */
#define UPBITS		9
#define UP			(1 << UPBITS)
#define FRACBITS	(32 - UPBITS)
#define FRACMASK	((1 << FRACBITS) - 1)
#define FRAC2FLT	(1.0f / (1 << FRACBITS))

/* filter state */
typedef struct {
	int up;
	int dn;
	int inrate;
	int outrate;
	int nchans;
	int nwing;
	int nhist;
//	int phase;
	int offset;
	int isCloned; // if 1, some of this filter data is owned by another filter instance.
	float *histbuf;
	float *pcmbuf;
	float *filter;
//	uchar *pcmstep;
	unsigned int time_f;
	unsigned int step_i;
	unsigned int step_f;
} state_t;

void *
RAInitResamplerArb(int inrate, int outrate, int nchans,
			  float atten, float passband, float stopband, float dcgain)
{
	state_t *s;
//	int divisor;
	int up, dn;
	int ntaps, nwing, nfilter, nhist;
	float beta, fpass, fstop;
	double *lpfilt;
	int i, phase;

	/* use defaults for values <= 0 */
	if (atten <= 0.0f)
		atten = DEF_ATTEN;
	if (passband <= 0.0f)
		passband = DEF_PASSBAND;
	if (stopband <= 0.0f)
		stopband = DEF_STOPBAND;
	if (dcgain <= 0.0f)
		dcgain = DEF_DCGAIN;

	if (nchans < 1 || nchans > 2)
		return NULL;
	if (passband >= stopband)
		return NULL;

	/* reduce to smallest fraction */
//	divisor = gcd(inrate, outrate);
//	up = outrate / divisor;
//	dn = inrate / divisor;
	up = UP;
	dn = 1;

	/* compute the filter specs */
//	fstop = 1.0f / MAX(up, dn);
	fstop = MIN(1.0f, (float)outrate/inrate) / up;
	fpass = passband * fstop;
	fstop = stopband * fstop;
	KaiserEstim(fpass, fstop, atten, &nfilter, &beta);

	ntaps = (nfilter + up-1) / up;	/* length of each filter phase */
	nwing = (ntaps + 1) / 2;
	ntaps = nwing * 2;				/* update ntaps */
	nfilter = nwing * up;			/* update nfilter */
	nhist = nchans * ntaps;

//	printf("up=%d down=%d ntaps=%d beta=%.2f atten=%.1fdB pass=%.0fHz stop=%.0fHz\n",
//		up, dn, ntaps, beta, atten, 0.5*inrate*fpass*UP, 0.5*inrate*fstop*UP);

	/* malloc buffers */
	lpfilt = (double *) malloc(nfilter * sizeof(double));

	s = (state_t *) malloc(sizeof(state_t));
	s->filter = (float *) malloc(nfilter * sizeof(float));
	s->isCloned = 0;
//	s->pcmstep = (uchar *) malloc(up * sizeof(uchar));
	s->histbuf = (float *) calloc((NBLOCK + nhist), sizeof(float));
	s->pcmbuf = s->histbuf + nhist;

	/* create the lowpass filter */
	KaiserLowpass(nfilter, 0.5f * (fpass + fstop), beta, (dcgain * up), lpfilt);

	/* deinterleave into phases */
	for (phase = 0; phase < up; phase++) {
		for (i = 0; i < nwing; i++) {
			s->filter[phase*nwing + i] = (float) lpfilt[i*up + phase];
		}
	}
	
	/* filter init */
	s->inrate = inrate;
	s->outrate = outrate;
	s->up = up;
	s->dn = dn;
	s->nchans = nchans;
	s->nwing = nwing;
	s->nhist = nhist;
//	s->phase = 0;
	s->offset = 0;
	s->time_f = 0;
	s->step_i = (unsigned int)((double)inrate / outrate);
	s->step_f = (unsigned int)(4294967296.0 * ((double)inrate / outrate - s->step_i));
	
	free(lpfilt);

	return (void *) s;
}

void *
RAInitResamplerCopyArb(int nchans, const void *inst)
{
	state_t *s_in = (state_t *)inst;
	state_t *s_out = (state_t *)malloc(sizeof(state_t));
	if (s_in == NULL || s_out == NULL)
		return NULL;

	*s_out = *s_in ;
	s_out->isCloned = 1;
	s_out->nchans = nchans;
	s_out->nhist = s_in->nhist / s_in->nchans * nchans;

	s_out->histbuf = (float *) calloc((NBLOCK + s_out->nhist), sizeof(float));
	if (s_out->histbuf == NULL)
		return NULL;
	s_out->pcmbuf = s_out->histbuf + s_out->nhist;

	return s_out;
}

void
RAFreeResamplerArb(void *inst)
{
	state_t *s = (state_t *)inst;

	if (s != NULL) {
		if (!s->isCloned && s->filter)
			free(s->filter);
//		free(s->pcmstep);
		if (s->histbuf)
			free(s->histbuf);
		free(s);
	}
}

int
RAResampleStereoArb(void *inbuf, int insamps, tConverter *pCvt, short *outbuf, int outstride, void *inst)
{
	state_t *s = (state_t *)inst;
	float *pcmptr, *pcmend, *pcmptr2;
	float *rwingptr, *lwingptr;
	short *outptr;
	float acc0, acc1, acc2, acc3, frac;
	int i;
	int nwing = s->nwing;
	unsigned int step_i = s->step_i;
	unsigned int step_f = s->step_f;
	unsigned int time_f = s->time_f;
	unsigned int phase, phase2;

//	TICK();
//	ASSERT(!(insamps & 0x1));	/* stereo must be even */

	/* convert input to float */
	insamps = pCvt->pfCvt(s->pcmbuf,inbuf,insamps,pCvt->pStateMachine) ;

	/* restore filter state */
	phase = time_f >> FRACBITS;
	phase2 = (phase + 1) & (UP-1);
	pcmptr = s->pcmbuf - STEREO * (nwing-1);
	pcmend = pcmptr + insamps;
	pcmptr += s->offset;
	pcmptr2 = pcmptr + STEREO * (phase2 < phase);
	outptr = outbuf;

	/* filter */
	while (pcmptr2 < pcmend) {

		rwingptr = s->filter + nwing * phase;
		lwingptr = s->filter + nwing * (UP-1-phase);
		acc0 = acc1 = 0.0f;

		for (i = 0; i < nwing; i++) {
			int j = STEREO * i;
			acc0 += pcmptr[-j-2] * rwingptr[i];
			acc1 += pcmptr[-j-1] * rwingptr[i];
			acc0 += pcmptr[+j+0] * lwingptr[i];
			acc1 += pcmptr[+j+1] * lwingptr[i];
		}

		rwingptr = s->filter + nwing * phase2;
		lwingptr = s->filter + nwing * (UP-1-phase2);
		acc2 = acc3 = 0.0f;

		for (i = 0; i < nwing; i++) {
			int j = STEREO * i;
			acc2 += pcmptr2[-j-2] * rwingptr[i];
			acc3 += pcmptr2[-j-1] * rwingptr[i];
			acc2 += pcmptr2[+j+0] * lwingptr[i];
			acc3 += pcmptr2[+j+1] * lwingptr[i];
		}

		frac = (time_f & FRACMASK) * FRAC2FLT;
		acc0 += frac * (acc2 - acc0);
		acc1 += frac * (acc3 - acc1);
	
		time_f += step_f;
		phase = time_f >> FRACBITS;
		phase2 = (phase + 1) & (UP-1);
		pcmptr += STEREO * (step_i + (time_f < step_f));
		pcmptr2 = pcmptr + STEREO * (phase2 < phase);

		outptr[0] = RoundFtoS(acc0);
		outptr[1] = RoundFtoS(acc1);
		outptr += outstride;
	}

	/* save filter state */
	s->time_f = time_f;
	s->offset = pcmptr - pcmend;
	memmove(s->histbuf, s->histbuf + insamps, s->nhist * sizeof(float));

//	TOCK(outptr - outbuf);
	return (outptr - outbuf);
}

int
RAResampleMonoArb(void *inbuf, int insamps, tConverter *pCvt, short *outbuf, int outstride, void *inst)
{
    state_t *s = (state_t *)inst;
	float *pcmptr, *pcmend, *pcmptr2;
	float *rwingptr, *lwingptr;
	short *outptr;
	float acc0, acc1, acc2, acc3, frac;
	int i;
	int nwing = s->nwing;
	unsigned int step_i = s->step_i;
	unsigned int step_f = s->step_f;
	unsigned int time_f = s->time_f;
	unsigned int phase, phase2;

//	TICK();

	/* convert input to float */
	insamps = pCvt->pfCvt(s->pcmbuf,inbuf,insamps,pCvt->pStateMachine) ;

	/* restore filter state */
	phase = time_f >> FRACBITS;
	phase2 = (phase + 1) & (UP-1);
	pcmptr = s->pcmbuf - (nwing-1);
	pcmend = pcmptr + insamps;
	pcmptr += s->offset;
	pcmptr2 = pcmptr + (phase2 < phase);
	outptr = outbuf;

	/* filter */
	while (pcmptr2 < pcmend) {

		rwingptr = s->filter + nwing * phase;
		lwingptr = s->filter + nwing * (UP-1-phase);
		acc0 = acc1 = 0.0f;

		for (i = 0; i < nwing; i++) {
			acc0 += pcmptr[-i-1] * rwingptr[i];
			acc1 += pcmptr[+i+0] * lwingptr[i];
		}

		rwingptr = s->filter + nwing * phase2;
		lwingptr = s->filter + nwing * (UP-1-phase2);
		acc2 = acc3 = 0.0f;

		for (i = 0; i < nwing; i++) {
			acc2 += pcmptr2[-i-1] * rwingptr[i];
			acc3 += pcmptr2[+i+0] * lwingptr[i];
		}

		acc0 += acc1;
		acc2 += acc3;
		frac = (time_f & FRACMASK) * FRAC2FLT;
		acc0 += frac * (acc2 - acc0);

		time_f += step_f;
		phase = time_f >> FRACBITS;
		phase2 = (phase + 1) & (UP-1);
		pcmptr += step_i + (time_f < step_f);
		pcmptr2 = pcmptr + (phase2 < phase);

		outptr[0] = RoundFtoS(acc0);
		outptr += outstride;
	}

	/* save filter state */
	s->time_f = time_f;
	s->offset = pcmptr - pcmend;
	memmove(s->histbuf, s->histbuf + insamps, s->nhist * sizeof(float));

//	TOCK(outptr - outbuf);
	return (outptr - outbuf);
}

/*
 * The maximum output, in samples, that could be produced by
 * insamps input samples.
 */
int
RAGetMaxOutputArb(int insamps, void *inst)
{
    state_t *s = (state_t *)inst;
	int inframes, outframes;
	double step;

	inframes = (insamps + (s->nchans-1)) / s->nchans;
	step = s->step_i + (s->step_f / 4294967296.0);
	outframes = (int) myCeil((double)inframes / step);
	return outframes * s->nchans;
}

/*
 * The minimum input, in samples, that will produce at least
 * outsamps output samples.
 */
int
RAGetMinInputArb(int outsamps, void *inst)
{
    state_t *s = (state_t *)inst;
	int inframes, outframes;
	double step;

	outframes = (outsamps + (s->nchans-1)) / s->nchans;
	step = s->step_i + (s->step_f / 4294967296.0);
	inframes = (int) myCeil((double)outframes * step);

	return inframes * s->nchans;
}

int
RAGetDelayArb(void *inst)
{
	state_t *s = (state_t *)inst;
	return (int)(s->nwing * (float)s->outrate / s->inrate) ;
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -