xref: /OK3568_Linux_fs/kernel/sound/core/oss/rate.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun  *  Rate conversion Plug-In
3*4882a593Smuzhiyun  *  Copyright (c) 1999 by Jaroslav Kysela <perex@perex.cz>
4*4882a593Smuzhiyun  *
5*4882a593Smuzhiyun  *
6*4882a593Smuzhiyun  *   This library is free software; you can redistribute it and/or modify
7*4882a593Smuzhiyun  *   it under the terms of the GNU Library General Public License as
8*4882a593Smuzhiyun  *   published by the Free Software Foundation; either version 2 of
9*4882a593Smuzhiyun  *   the License, or (at your option) any later version.
10*4882a593Smuzhiyun  *
11*4882a593Smuzhiyun  *   This program is distributed in the hope that it will be useful,
12*4882a593Smuzhiyun  *   but WITHOUT ANY WARRANTY; without even the implied warranty of
13*4882a593Smuzhiyun  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14*4882a593Smuzhiyun  *   GNU Library General Public License for more details.
15*4882a593Smuzhiyun  *
16*4882a593Smuzhiyun  *   You should have received a copy of the GNU Library General Public
17*4882a593Smuzhiyun  *   License along with this library; if not, write to the Free Software
18*4882a593Smuzhiyun  *   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA
19*4882a593Smuzhiyun  *
20*4882a593Smuzhiyun  */
21*4882a593Smuzhiyun 
22*4882a593Smuzhiyun #include <linux/time.h>
23*4882a593Smuzhiyun #include <sound/core.h>
24*4882a593Smuzhiyun #include <sound/pcm.h>
25*4882a593Smuzhiyun #include "pcm_plugin.h"
26*4882a593Smuzhiyun 
27*4882a593Smuzhiyun #define SHIFT	11
28*4882a593Smuzhiyun #define BITS	(1<<SHIFT)
29*4882a593Smuzhiyun #define R_MASK	(BITS-1)
30*4882a593Smuzhiyun 
31*4882a593Smuzhiyun /*
32*4882a593Smuzhiyun  *  Basic rate conversion plugin
33*4882a593Smuzhiyun  */
34*4882a593Smuzhiyun 
35*4882a593Smuzhiyun struct rate_channel {
36*4882a593Smuzhiyun 	signed short last_S1;
37*4882a593Smuzhiyun 	signed short last_S2;
38*4882a593Smuzhiyun };
39*4882a593Smuzhiyun 
40*4882a593Smuzhiyun typedef void (*rate_f)(struct snd_pcm_plugin *plugin,
41*4882a593Smuzhiyun 		       const struct snd_pcm_plugin_channel *src_channels,
42*4882a593Smuzhiyun 		       struct snd_pcm_plugin_channel *dst_channels,
43*4882a593Smuzhiyun 		       int src_frames, int dst_frames);
44*4882a593Smuzhiyun 
45*4882a593Smuzhiyun struct rate_priv {
46*4882a593Smuzhiyun 	unsigned int pitch;
47*4882a593Smuzhiyun 	unsigned int pos;
48*4882a593Smuzhiyun 	rate_f func;
49*4882a593Smuzhiyun 	snd_pcm_sframes_t old_src_frames, old_dst_frames;
50*4882a593Smuzhiyun 	struct rate_channel channels[];
51*4882a593Smuzhiyun };
52*4882a593Smuzhiyun 
rate_init(struct snd_pcm_plugin * plugin)53*4882a593Smuzhiyun static void rate_init(struct snd_pcm_plugin *plugin)
54*4882a593Smuzhiyun {
55*4882a593Smuzhiyun 	unsigned int channel;
56*4882a593Smuzhiyun 	struct rate_priv *data = (struct rate_priv *)plugin->extra_data;
57*4882a593Smuzhiyun 	data->pos = 0;
58*4882a593Smuzhiyun 	for (channel = 0; channel < plugin->src_format.channels; channel++) {
59*4882a593Smuzhiyun 		data->channels[channel].last_S1 = 0;
60*4882a593Smuzhiyun 		data->channels[channel].last_S2 = 0;
61*4882a593Smuzhiyun 	}
62*4882a593Smuzhiyun }
63*4882a593Smuzhiyun 
resample_expand(struct snd_pcm_plugin * plugin,const struct snd_pcm_plugin_channel * src_channels,struct snd_pcm_plugin_channel * dst_channels,int src_frames,int dst_frames)64*4882a593Smuzhiyun static void resample_expand(struct snd_pcm_plugin *plugin,
65*4882a593Smuzhiyun 			    const struct snd_pcm_plugin_channel *src_channels,
66*4882a593Smuzhiyun 			    struct snd_pcm_plugin_channel *dst_channels,
67*4882a593Smuzhiyun 			    int src_frames, int dst_frames)
68*4882a593Smuzhiyun {
69*4882a593Smuzhiyun 	unsigned int pos = 0;
70*4882a593Smuzhiyun 	signed int val;
71*4882a593Smuzhiyun 	signed short S1, S2;
72*4882a593Smuzhiyun 	signed short *src, *dst;
73*4882a593Smuzhiyun 	unsigned int channel;
74*4882a593Smuzhiyun 	int src_step, dst_step;
75*4882a593Smuzhiyun 	int src_frames1, dst_frames1;
76*4882a593Smuzhiyun 	struct rate_priv *data = (struct rate_priv *)plugin->extra_data;
77*4882a593Smuzhiyun 	struct rate_channel *rchannels = data->channels;
78*4882a593Smuzhiyun 
79*4882a593Smuzhiyun 	for (channel = 0; channel < plugin->src_format.channels; channel++) {
80*4882a593Smuzhiyun 		pos = data->pos;
81*4882a593Smuzhiyun 		S1 = rchannels->last_S1;
82*4882a593Smuzhiyun 		S2 = rchannels->last_S2;
83*4882a593Smuzhiyun 		if (!src_channels[channel].enabled) {
84*4882a593Smuzhiyun 			if (dst_channels[channel].wanted)
85*4882a593Smuzhiyun 				snd_pcm_area_silence(&dst_channels[channel].area, 0, dst_frames, plugin->dst_format.format);
86*4882a593Smuzhiyun 			dst_channels[channel].enabled = 0;
87*4882a593Smuzhiyun 			continue;
88*4882a593Smuzhiyun 		}
89*4882a593Smuzhiyun 		dst_channels[channel].enabled = 1;
90*4882a593Smuzhiyun 		src = (signed short *)src_channels[channel].area.addr +
91*4882a593Smuzhiyun 			src_channels[channel].area.first / 8 / 2;
92*4882a593Smuzhiyun 		dst = (signed short *)dst_channels[channel].area.addr +
93*4882a593Smuzhiyun 			dst_channels[channel].area.first / 8 / 2;
94*4882a593Smuzhiyun 		src_step = src_channels[channel].area.step / 8 / 2;
95*4882a593Smuzhiyun 		dst_step = dst_channels[channel].area.step / 8 / 2;
96*4882a593Smuzhiyun 		src_frames1 = src_frames;
97*4882a593Smuzhiyun 		dst_frames1 = dst_frames;
98*4882a593Smuzhiyun 		while (dst_frames1-- > 0) {
99*4882a593Smuzhiyun 			if (pos & ~R_MASK) {
100*4882a593Smuzhiyun 				pos &= R_MASK;
101*4882a593Smuzhiyun 				S1 = S2;
102*4882a593Smuzhiyun 				if (src_frames1-- > 0) {
103*4882a593Smuzhiyun 					S2 = *src;
104*4882a593Smuzhiyun 					src += src_step;
105*4882a593Smuzhiyun 				}
106*4882a593Smuzhiyun 			}
107*4882a593Smuzhiyun 			val = S1 + ((S2 - S1) * (signed int)pos) / BITS;
108*4882a593Smuzhiyun 			if (val < -32768)
109*4882a593Smuzhiyun 				val = -32768;
110*4882a593Smuzhiyun 			else if (val > 32767)
111*4882a593Smuzhiyun 				val = 32767;
112*4882a593Smuzhiyun 			*dst = val;
113*4882a593Smuzhiyun 			dst += dst_step;
114*4882a593Smuzhiyun 			pos += data->pitch;
115*4882a593Smuzhiyun 		}
116*4882a593Smuzhiyun 		rchannels->last_S1 = S1;
117*4882a593Smuzhiyun 		rchannels->last_S2 = S2;
118*4882a593Smuzhiyun 		rchannels++;
119*4882a593Smuzhiyun 	}
120*4882a593Smuzhiyun 	data->pos = pos;
121*4882a593Smuzhiyun }
122*4882a593Smuzhiyun 
resample_shrink(struct snd_pcm_plugin * plugin,const struct snd_pcm_plugin_channel * src_channels,struct snd_pcm_plugin_channel * dst_channels,int src_frames,int dst_frames)123*4882a593Smuzhiyun static void resample_shrink(struct snd_pcm_plugin *plugin,
124*4882a593Smuzhiyun 			    const struct snd_pcm_plugin_channel *src_channels,
125*4882a593Smuzhiyun 			    struct snd_pcm_plugin_channel *dst_channels,
126*4882a593Smuzhiyun 			    int src_frames, int dst_frames)
127*4882a593Smuzhiyun {
128*4882a593Smuzhiyun 	unsigned int pos = 0;
129*4882a593Smuzhiyun 	signed int val;
130*4882a593Smuzhiyun 	signed short S1, S2;
131*4882a593Smuzhiyun 	signed short *src, *dst;
132*4882a593Smuzhiyun 	unsigned int channel;
133*4882a593Smuzhiyun 	int src_step, dst_step;
134*4882a593Smuzhiyun 	int src_frames1, dst_frames1;
135*4882a593Smuzhiyun 	struct rate_priv *data = (struct rate_priv *)plugin->extra_data;
136*4882a593Smuzhiyun 	struct rate_channel *rchannels = data->channels;
137*4882a593Smuzhiyun 
138*4882a593Smuzhiyun 	for (channel = 0; channel < plugin->src_format.channels; ++channel) {
139*4882a593Smuzhiyun 		pos = data->pos;
140*4882a593Smuzhiyun 		S1 = rchannels->last_S1;
141*4882a593Smuzhiyun 		S2 = rchannels->last_S2;
142*4882a593Smuzhiyun 		if (!src_channels[channel].enabled) {
143*4882a593Smuzhiyun 			if (dst_channels[channel].wanted)
144*4882a593Smuzhiyun 				snd_pcm_area_silence(&dst_channels[channel].area, 0, dst_frames, plugin->dst_format.format);
145*4882a593Smuzhiyun 			dst_channels[channel].enabled = 0;
146*4882a593Smuzhiyun 			continue;
147*4882a593Smuzhiyun 		}
148*4882a593Smuzhiyun 		dst_channels[channel].enabled = 1;
149*4882a593Smuzhiyun 		src = (signed short *)src_channels[channel].area.addr +
150*4882a593Smuzhiyun 			src_channels[channel].area.first / 8 / 2;
151*4882a593Smuzhiyun 		dst = (signed short *)dst_channels[channel].area.addr +
152*4882a593Smuzhiyun 			dst_channels[channel].area.first / 8 / 2;
153*4882a593Smuzhiyun 		src_step = src_channels[channel].area.step / 8 / 2;
154*4882a593Smuzhiyun 		dst_step = dst_channels[channel].area.step / 8 / 2;
155*4882a593Smuzhiyun 		src_frames1 = src_frames;
156*4882a593Smuzhiyun 		dst_frames1 = dst_frames;
157*4882a593Smuzhiyun 		while (dst_frames1 > 0) {
158*4882a593Smuzhiyun 			S1 = S2;
159*4882a593Smuzhiyun 			if (src_frames1-- > 0) {
160*4882a593Smuzhiyun 				S2 = *src;
161*4882a593Smuzhiyun 				src += src_step;
162*4882a593Smuzhiyun 			}
163*4882a593Smuzhiyun 			if (pos & ~R_MASK) {
164*4882a593Smuzhiyun 				pos &= R_MASK;
165*4882a593Smuzhiyun 				val = S1 + ((S2 - S1) * (signed int)pos) / BITS;
166*4882a593Smuzhiyun 				if (val < -32768)
167*4882a593Smuzhiyun 					val = -32768;
168*4882a593Smuzhiyun 				else if (val > 32767)
169*4882a593Smuzhiyun 					val = 32767;
170*4882a593Smuzhiyun 				*dst = val;
171*4882a593Smuzhiyun 				dst += dst_step;
172*4882a593Smuzhiyun 				dst_frames1--;
173*4882a593Smuzhiyun 			}
174*4882a593Smuzhiyun 			pos += data->pitch;
175*4882a593Smuzhiyun 		}
176*4882a593Smuzhiyun 		rchannels->last_S1 = S1;
177*4882a593Smuzhiyun 		rchannels->last_S2 = S2;
178*4882a593Smuzhiyun 		rchannels++;
179*4882a593Smuzhiyun 	}
180*4882a593Smuzhiyun 	data->pos = pos;
181*4882a593Smuzhiyun }
182*4882a593Smuzhiyun 
rate_src_frames(struct snd_pcm_plugin * plugin,snd_pcm_uframes_t frames)183*4882a593Smuzhiyun static snd_pcm_sframes_t rate_src_frames(struct snd_pcm_plugin *plugin, snd_pcm_uframes_t frames)
184*4882a593Smuzhiyun {
185*4882a593Smuzhiyun 	struct rate_priv *data;
186*4882a593Smuzhiyun 	snd_pcm_sframes_t res;
187*4882a593Smuzhiyun 
188*4882a593Smuzhiyun 	if (snd_BUG_ON(!plugin))
189*4882a593Smuzhiyun 		return -ENXIO;
190*4882a593Smuzhiyun 	if (frames == 0)
191*4882a593Smuzhiyun 		return 0;
192*4882a593Smuzhiyun 	data = (struct rate_priv *)plugin->extra_data;
193*4882a593Smuzhiyun 	if (plugin->src_format.rate < plugin->dst_format.rate) {
194*4882a593Smuzhiyun 		res = (((frames * data->pitch) + (BITS/2)) >> SHIFT);
195*4882a593Smuzhiyun 	} else {
196*4882a593Smuzhiyun 		res = (((frames << SHIFT) + (data->pitch / 2)) / data->pitch);
197*4882a593Smuzhiyun 	}
198*4882a593Smuzhiyun 	if (data->old_src_frames > 0) {
199*4882a593Smuzhiyun 		snd_pcm_sframes_t frames1 = frames, res1 = data->old_dst_frames;
200*4882a593Smuzhiyun 		while (data->old_src_frames < frames1) {
201*4882a593Smuzhiyun 			frames1 >>= 1;
202*4882a593Smuzhiyun 			res1 <<= 1;
203*4882a593Smuzhiyun 		}
204*4882a593Smuzhiyun 		while (data->old_src_frames > frames1) {
205*4882a593Smuzhiyun 			frames1 <<= 1;
206*4882a593Smuzhiyun 			res1 >>= 1;
207*4882a593Smuzhiyun 		}
208*4882a593Smuzhiyun 		if (data->old_src_frames == frames1)
209*4882a593Smuzhiyun 			return res1;
210*4882a593Smuzhiyun 	}
211*4882a593Smuzhiyun 	data->old_src_frames = frames;
212*4882a593Smuzhiyun 	data->old_dst_frames = res;
213*4882a593Smuzhiyun 	return res;
214*4882a593Smuzhiyun }
215*4882a593Smuzhiyun 
rate_dst_frames(struct snd_pcm_plugin * plugin,snd_pcm_uframes_t frames)216*4882a593Smuzhiyun static snd_pcm_sframes_t rate_dst_frames(struct snd_pcm_plugin *plugin, snd_pcm_uframes_t frames)
217*4882a593Smuzhiyun {
218*4882a593Smuzhiyun 	struct rate_priv *data;
219*4882a593Smuzhiyun 	snd_pcm_sframes_t res;
220*4882a593Smuzhiyun 
221*4882a593Smuzhiyun 	if (snd_BUG_ON(!plugin))
222*4882a593Smuzhiyun 		return -ENXIO;
223*4882a593Smuzhiyun 	if (frames == 0)
224*4882a593Smuzhiyun 		return 0;
225*4882a593Smuzhiyun 	data = (struct rate_priv *)plugin->extra_data;
226*4882a593Smuzhiyun 	if (plugin->src_format.rate < plugin->dst_format.rate) {
227*4882a593Smuzhiyun 		res = (((frames << SHIFT) + (data->pitch / 2)) / data->pitch);
228*4882a593Smuzhiyun 	} else {
229*4882a593Smuzhiyun 		res = (((frames * data->pitch) + (BITS/2)) >> SHIFT);
230*4882a593Smuzhiyun 	}
231*4882a593Smuzhiyun 	if (data->old_dst_frames > 0) {
232*4882a593Smuzhiyun 		snd_pcm_sframes_t frames1 = frames, res1 = data->old_src_frames;
233*4882a593Smuzhiyun 		while (data->old_dst_frames < frames1) {
234*4882a593Smuzhiyun 			frames1 >>= 1;
235*4882a593Smuzhiyun 			res1 <<= 1;
236*4882a593Smuzhiyun 		}
237*4882a593Smuzhiyun 		while (data->old_dst_frames > frames1) {
238*4882a593Smuzhiyun 			frames1 <<= 1;
239*4882a593Smuzhiyun 			res1 >>= 1;
240*4882a593Smuzhiyun 		}
241*4882a593Smuzhiyun 		if (data->old_dst_frames == frames1)
242*4882a593Smuzhiyun 			return res1;
243*4882a593Smuzhiyun 	}
244*4882a593Smuzhiyun 	data->old_dst_frames = frames;
245*4882a593Smuzhiyun 	data->old_src_frames = res;
246*4882a593Smuzhiyun 	return res;
247*4882a593Smuzhiyun }
248*4882a593Smuzhiyun 
rate_transfer(struct snd_pcm_plugin * plugin,const struct snd_pcm_plugin_channel * src_channels,struct snd_pcm_plugin_channel * dst_channels,snd_pcm_uframes_t frames)249*4882a593Smuzhiyun static snd_pcm_sframes_t rate_transfer(struct snd_pcm_plugin *plugin,
250*4882a593Smuzhiyun 			     const struct snd_pcm_plugin_channel *src_channels,
251*4882a593Smuzhiyun 			     struct snd_pcm_plugin_channel *dst_channels,
252*4882a593Smuzhiyun 			     snd_pcm_uframes_t frames)
253*4882a593Smuzhiyun {
254*4882a593Smuzhiyun 	snd_pcm_uframes_t dst_frames;
255*4882a593Smuzhiyun 	struct rate_priv *data;
256*4882a593Smuzhiyun 
257*4882a593Smuzhiyun 	if (snd_BUG_ON(!plugin || !src_channels || !dst_channels))
258*4882a593Smuzhiyun 		return -ENXIO;
259*4882a593Smuzhiyun 	if (frames == 0)
260*4882a593Smuzhiyun 		return 0;
261*4882a593Smuzhiyun #ifdef CONFIG_SND_DEBUG
262*4882a593Smuzhiyun 	{
263*4882a593Smuzhiyun 		unsigned int channel;
264*4882a593Smuzhiyun 		for (channel = 0; channel < plugin->src_format.channels; channel++) {
265*4882a593Smuzhiyun 			if (snd_BUG_ON(src_channels[channel].area.first % 8 ||
266*4882a593Smuzhiyun 				       src_channels[channel].area.step % 8))
267*4882a593Smuzhiyun 				return -ENXIO;
268*4882a593Smuzhiyun 			if (snd_BUG_ON(dst_channels[channel].area.first % 8 ||
269*4882a593Smuzhiyun 				       dst_channels[channel].area.step % 8))
270*4882a593Smuzhiyun 				return -ENXIO;
271*4882a593Smuzhiyun 		}
272*4882a593Smuzhiyun 	}
273*4882a593Smuzhiyun #endif
274*4882a593Smuzhiyun 
275*4882a593Smuzhiyun 	dst_frames = rate_dst_frames(plugin, frames);
276*4882a593Smuzhiyun 	if (dst_frames > dst_channels[0].frames)
277*4882a593Smuzhiyun 		dst_frames = dst_channels[0].frames;
278*4882a593Smuzhiyun 	data = (struct rate_priv *)plugin->extra_data;
279*4882a593Smuzhiyun 	data->func(plugin, src_channels, dst_channels, frames, dst_frames);
280*4882a593Smuzhiyun 	return dst_frames;
281*4882a593Smuzhiyun }
282*4882a593Smuzhiyun 
rate_action(struct snd_pcm_plugin * plugin,enum snd_pcm_plugin_action action,unsigned long udata)283*4882a593Smuzhiyun static int rate_action(struct snd_pcm_plugin *plugin,
284*4882a593Smuzhiyun 		       enum snd_pcm_plugin_action action,
285*4882a593Smuzhiyun 		       unsigned long udata)
286*4882a593Smuzhiyun {
287*4882a593Smuzhiyun 	if (snd_BUG_ON(!plugin))
288*4882a593Smuzhiyun 		return -ENXIO;
289*4882a593Smuzhiyun 	switch (action) {
290*4882a593Smuzhiyun 	case INIT:
291*4882a593Smuzhiyun 	case PREPARE:
292*4882a593Smuzhiyun 		rate_init(plugin);
293*4882a593Smuzhiyun 		break;
294*4882a593Smuzhiyun 	default:
295*4882a593Smuzhiyun 		break;
296*4882a593Smuzhiyun 	}
297*4882a593Smuzhiyun 	return 0;	/* silenty ignore other actions */
298*4882a593Smuzhiyun }
299*4882a593Smuzhiyun 
snd_pcm_plugin_build_rate(struct snd_pcm_substream * plug,struct snd_pcm_plugin_format * src_format,struct snd_pcm_plugin_format * dst_format,struct snd_pcm_plugin ** r_plugin)300*4882a593Smuzhiyun int snd_pcm_plugin_build_rate(struct snd_pcm_substream *plug,
301*4882a593Smuzhiyun 			      struct snd_pcm_plugin_format *src_format,
302*4882a593Smuzhiyun 			      struct snd_pcm_plugin_format *dst_format,
303*4882a593Smuzhiyun 			      struct snd_pcm_plugin **r_plugin)
304*4882a593Smuzhiyun {
305*4882a593Smuzhiyun 	int err;
306*4882a593Smuzhiyun 	struct rate_priv *data;
307*4882a593Smuzhiyun 	struct snd_pcm_plugin *plugin;
308*4882a593Smuzhiyun 
309*4882a593Smuzhiyun 	if (snd_BUG_ON(!r_plugin))
310*4882a593Smuzhiyun 		return -ENXIO;
311*4882a593Smuzhiyun 	*r_plugin = NULL;
312*4882a593Smuzhiyun 
313*4882a593Smuzhiyun 	if (snd_BUG_ON(src_format->channels != dst_format->channels))
314*4882a593Smuzhiyun 		return -ENXIO;
315*4882a593Smuzhiyun 	if (snd_BUG_ON(src_format->channels <= 0))
316*4882a593Smuzhiyun 		return -ENXIO;
317*4882a593Smuzhiyun 	if (snd_BUG_ON(src_format->format != SNDRV_PCM_FORMAT_S16))
318*4882a593Smuzhiyun 		return -ENXIO;
319*4882a593Smuzhiyun 	if (snd_BUG_ON(dst_format->format != SNDRV_PCM_FORMAT_S16))
320*4882a593Smuzhiyun 		return -ENXIO;
321*4882a593Smuzhiyun 	if (snd_BUG_ON(src_format->rate == dst_format->rate))
322*4882a593Smuzhiyun 		return -ENXIO;
323*4882a593Smuzhiyun 
324*4882a593Smuzhiyun 	err = snd_pcm_plugin_build(plug, "rate conversion",
325*4882a593Smuzhiyun 				   src_format, dst_format,
326*4882a593Smuzhiyun 				   struct_size(data, channels,
327*4882a593Smuzhiyun 					       src_format->channels),
328*4882a593Smuzhiyun 				   &plugin);
329*4882a593Smuzhiyun 	if (err < 0)
330*4882a593Smuzhiyun 		return err;
331*4882a593Smuzhiyun 	data = (struct rate_priv *)plugin->extra_data;
332*4882a593Smuzhiyun 	if (src_format->rate < dst_format->rate) {
333*4882a593Smuzhiyun 		data->pitch = ((src_format->rate << SHIFT) + (dst_format->rate >> 1)) / dst_format->rate;
334*4882a593Smuzhiyun 		data->func = resample_expand;
335*4882a593Smuzhiyun 	} else {
336*4882a593Smuzhiyun 		data->pitch = ((dst_format->rate << SHIFT) + (src_format->rate >> 1)) / src_format->rate;
337*4882a593Smuzhiyun 		data->func = resample_shrink;
338*4882a593Smuzhiyun 	}
339*4882a593Smuzhiyun 	data->pos = 0;
340*4882a593Smuzhiyun 	rate_init(plugin);
341*4882a593Smuzhiyun 	data->old_src_frames = data->old_dst_frames = 0;
342*4882a593Smuzhiyun 	plugin->transfer = rate_transfer;
343*4882a593Smuzhiyun 	plugin->src_frames = rate_src_frames;
344*4882a593Smuzhiyun 	plugin->dst_frames = rate_dst_frames;
345*4882a593Smuzhiyun 	plugin->action = rate_action;
346*4882a593Smuzhiyun 	*r_plugin = plugin;
347*4882a593Smuzhiyun 	return 0;
348*4882a593Smuzhiyun }
349