xref: /haiku/src/add-ons/media/media-add-ons/mixer/Resampler.cpp (revision 95bac3fda53a4cb21880712d7b43f8c21db32a2e)
1 /* Copyright (C) 2003 Marcus Overhagen
2  * Released under terms of the MIT license.
3  *
4  * A simple resampling class for the audio mixer.
5  * You pick the conversion function on object creation,
6  * and then call the Resample() function, specifying data pointer,
7  * offset (in bytes) to the next sample, and count of samples for
8  * both source and destination.
9  *
10  */
11 
12 #include <MediaDefs.h>
13 #include "Resampler.h"
14 #include "debug.h"
15 
16 Resampler::Resampler(uint32 src_format, uint32 dst_format)
17  :	fFunc(0)
18 {
19 	if (dst_format == media_raw_audio_format::B_AUDIO_FLOAT) {
20 		switch (src_format) {
21 			case media_raw_audio_format::B_AUDIO_FLOAT:
22 				fFunc = &Resampler::float_to_float;
23 				return;
24 			case media_raw_audio_format::B_AUDIO_INT:
25 				fFunc = &Resampler::int32_to_float;
26 				return;
27 			case media_raw_audio_format::B_AUDIO_SHORT:
28 				fFunc = &Resampler::int16_to_float;
29 				return;
30 			case media_raw_audio_format::B_AUDIO_CHAR:
31 				fFunc = &Resampler::int8_to_float;
32 				return;
33 			case media_raw_audio_format::B_AUDIO_UCHAR:
34 				fFunc = &Resampler::uint8_to_float;
35 				return;
36 			default:
37 				ERROR("Resampler::Resampler: unknown source format 0x%x\n", src_format);
38 				return;
39 		}
40 	}
41 
42 	if (src_format == media_raw_audio_format::B_AUDIO_FLOAT) {
43 		switch (dst_format) {
44 			// float=>float already handled above
45 			case media_raw_audio_format::B_AUDIO_INT:
46 				fFunc = &Resampler::float_to_int32;
47 				return;
48 			case media_raw_audio_format::B_AUDIO_SHORT:
49 				fFunc = &Resampler::float_to_int16;
50 				return;
51 			case media_raw_audio_format::B_AUDIO_CHAR:
52 				fFunc = &Resampler::float_to_int8;
53 				return;
54 			case media_raw_audio_format::B_AUDIO_UCHAR:
55 				fFunc = &Resampler::float_to_uint8;
56 				return;
57 			default:
58 				ERROR("Resampler::Resampler: unknown destination format 0x%x\n", dst_format);
59 				return;
60 		}
61 	}
62 
63 	ERROR("Resampler::Resampler: source or destination format must be B_AUDIO_FLOAT\n");
64 }
65 
66 Resampler::~Resampler()
67 {
68 }
69 
70 status_t
71 Resampler::InitCheck()
72 {
73 	return (fFunc != 0) ? B_OK : B_ERROR;
74 }
75 
76 void
77 Resampler::float_to_float(const void *_src, int32 src_sample_offset, int32 src_sample_count,
78 						  void *_dst, int32 dst_sample_offset, int32 dst_sample_count, float _gain)
79 {
80 	register const char * src = (const char *) _src;
81 	register char * dst = (char *) _dst;
82 	register int32 count = dst_sample_count;
83 	register float gain = _gain;
84 
85 	if (src_sample_count == dst_sample_count) {
86 		// optimized case for no resampling
87 		while (count--) {
88 			*(float *)dst = *(const float *)src * gain;
89 			src += src_sample_offset;
90 			dst += dst_sample_offset;
91 		}
92 		return;
93 	}
94 
95 	register float delta = float(src_sample_count) / float(dst_sample_count);
96 	register float current = 0.0f;
97 
98 	if (delta < 1.0) {
99 		// upsample
100 		while (count--) {
101 			*(float *)dst = *(const float *)src * gain;
102 			dst += dst_sample_offset;
103 			current += delta;
104 			if (current >= 1.0f) {
105 				current -= 1.0f;
106 				src += src_sample_offset;
107 			}
108 		}
109 	} else {
110 		// downsample
111 		while (count--) {
112 			*(float *)dst = *(const float *)src * gain;
113 			dst += dst_sample_offset;
114 			current += delta;
115 			register int32 skipcount = (int32)current;
116 			current -= skipcount;
117 			src += skipcount * src_sample_offset;
118 		}
119 	}
120 }
121 
122 void
123 Resampler::int32_to_float(const void *_src, int32 src_sample_offset, int32 src_sample_count,
124 						  void *_dst, int32 dst_sample_offset, int32 dst_sample_count, float _gain)
125 {
126 	register const char * src = (const char *) _src;
127 	register char * dst = (char *) _dst;
128 	register int32 count = dst_sample_count;
129 	register float gain = _gain / 2147483647.0;
130 
131 	if (src_sample_count == dst_sample_count) {
132 		// optimized case for no resampling
133 		while (count--) {
134 			*(float *)dst = *(const int32 *)src * gain;
135 			src += src_sample_offset;
136 			dst += dst_sample_offset;
137 		}
138 		return;
139 	}
140 
141 	register float delta = float(src_sample_count) / float(dst_sample_count);
142 	register float current = 0.0f;
143 
144 	if (delta < 1.0) {
145 		// upsample
146 		while (count--) {
147 			*(float *)dst = *(const int32 *)src * gain;
148 			dst += dst_sample_offset;
149 			current += delta;
150 			if (current >= 1.0f) {
151 				current -= 1.0f;
152 				src += src_sample_offset;
153 			}
154 		}
155 	} else {
156 		// downsample
157 		while (count--) {
158 			*(float *)dst = *(const int32 *)src * gain;
159 			dst += dst_sample_offset;
160 			current += delta;
161 			register int32 skipcount = (int32)current;
162 			current -= skipcount;
163 			src += skipcount * src_sample_offset;
164 		}
165 	}
166 }
167 
168 void
169 Resampler::int16_to_float(const void *_src, int32 src_sample_offset, int32 src_sample_count,
170 						  void *_dst, int32 dst_sample_offset, int32 dst_sample_count, float _gain)
171 {
172 	register const char * src = (const char *) _src;
173 	register char * dst = (char *) _dst;
174 	register int32 count = dst_sample_count;
175 	register float gain = _gain / 32767.0;
176 
177 	if (src_sample_count == dst_sample_count) {
178 		// optimized case for no resampling
179 		while (count--) {
180 			*(float *)dst = *(const int16 *)src * gain;
181 			src += src_sample_offset;
182 			dst += dst_sample_offset;
183 		}
184 		return;
185 	}
186 
187 	register float delta = float(src_sample_count) / float(dst_sample_count);
188 	register float current = 0.0f;
189 
190 	if (delta < 1.0) {
191 		// upsample
192 		while (count--) {
193 			*(float *)dst = *(const int16 *)src * gain;
194 			dst += dst_sample_offset;
195 			current += delta;
196 			if (current >= 1.0f) {
197 				current -= 1.0f;
198 				src += src_sample_offset;
199 			}
200 		}
201 	} else {
202 		// downsample
203 		while (count--) {
204 			*(float *)dst = *(const int16 *)src * gain;
205 			dst += dst_sample_offset;
206 			current += delta;
207 			register int32 skipcount = (int32)current;
208 			current -= skipcount;
209 			src += skipcount * src_sample_offset;
210 		}
211 	}
212 }
213 
214 void
215 Resampler::int8_to_float(const void *_src, int32 src_sample_offset, int32 src_sample_count,
216 				  		 void *_dst, int32 dst_sample_offset, int32 dst_sample_count, float _gain)
217 {
218 	register const char * src = (const char *) _src;
219 	register char * dst = (char *) _dst;
220 	register int32 count = dst_sample_count;
221 	register float gain = _gain / 127.0;
222 
223 	if (src_sample_count == dst_sample_count) {
224 		// optimized case for no resampling
225 		while (count--) {
226 			*(float *)dst = *(const int8 *)src * gain;
227 			src += src_sample_offset;
228 			dst += dst_sample_offset;
229 		}
230 		return;
231 	}
232 
233 	register float delta = float(src_sample_count) / float(dst_sample_count);
234 	register float current = 0.0f;
235 
236 	if (delta < 1.0) {
237 		// upsample
238 		while (count--) {
239 			*(float *)dst = *(const int8 *)src * gain;
240 			dst += dst_sample_offset;
241 			current += delta;
242 			if (current >= 1.0f) {
243 				current -= 1.0f;
244 				src += src_sample_offset;
245 			}
246 		}
247 	} else {
248 		// downsample
249 		while (count--) {
250 			*(float *)dst = *(const int8 *)src * gain;
251 			dst += dst_sample_offset;
252 			current += delta;
253 			register int32 skipcount = (int32)current;
254 			current -= skipcount;
255 			src += skipcount * src_sample_offset;
256 		}
257 	}
258 }
259 
260 void
261 Resampler::uint8_to_float(const void *_src, int32 src_sample_offset, int32 src_sample_count,
262 						  void *_dst, int32 dst_sample_offset, int32 dst_sample_count, float _gain)
263 {
264 	register const char * src = (const char *) _src;
265 	register char * dst = (char *) _dst;
266 	register int32 count = dst_sample_count;
267 	register float gain = _gain / 127.0;
268 
269 	if (src_sample_count == dst_sample_count) {
270 		// optimized case for no resampling
271 		while (count--) {
272 			*(float *)dst = (((int32) *(const uint8 *)src) - 128) * gain;
273 			src += src_sample_offset;
274 			dst += dst_sample_offset;
275 		}
276 		return;
277 	}
278 
279 	register float delta = float(src_sample_count) / float(dst_sample_count);
280 	register float current = 0.0f;
281 
282 	if (delta < 1.0) {
283 		// upsample
284 		while (count--) {
285 			*(float *)dst = (((int32) *(const uint8 *)src) - 128) * gain;
286 			dst += dst_sample_offset;
287 			current += delta;
288 			if (current >= 1.0f) {
289 				current -= 1.0f;
290 				src += src_sample_offset;
291 			}
292 		}
293 	} else {
294 		// downsample
295 		while (count--) {
296 			*(float *)dst = (((int32) *(const uint8 *)src) - 128) * gain;
297 			dst += dst_sample_offset;
298 			current += delta;
299 			register int32 skipcount = (int32)current;
300 			current -= skipcount;
301 			src += skipcount * src_sample_offset;
302 		}
303 	}
304 }
305 
306 void
307 Resampler::float_to_int32(const void *_src, int32 src_sample_offset, int32 src_sample_count,
308 						  void *_dst, int32 dst_sample_offset, int32 dst_sample_count, float _gain)
309 {
310 	register const char * src = (const char *) _src;
311 	register char * dst = (char *) _dst;
312 	register int32 count = dst_sample_count;
313 	register float gain = _gain * 2147483647.0;
314 
315 	if (src_sample_count == dst_sample_count) {
316 		// optimized case for no resampling
317 		while (count--) {
318 			register float sample = *(const float *)src * gain;
319 			if (sample > 2147483647.0f)
320 				*(int32 *)dst = 2147483647L;
321 			else if (sample < -2147483647.0f)
322 				*(int32 *)dst = -2147483647L;
323 			else
324 				*(int32 *)dst = (int32)sample;
325 			src += src_sample_offset;
326 			dst += dst_sample_offset;
327 		}
328 		return;
329 	}
330 
331 	register float delta = float(src_sample_count) / float(dst_sample_count);
332 	register float current = 0.0f;
333 
334 	if (delta < 1.0) {
335 		// upsample
336 		while (count--) {
337 			register float sample = *(const float *)src * gain;
338 			if (sample > 2147483647.0f)
339 				*(int32 *)dst = 2147483647L;
340 			else if (sample < -2147483647.0f)
341 				*(int32 *)dst = -2147483647L;
342 			else
343 				*(int32 *)dst = (int32)sample;
344 			dst += dst_sample_offset;
345 			current += delta;
346 			if (current >= 1.0f) {
347 				current -= 1.0f;
348 				src += src_sample_offset;
349 			}
350 		}
351 	} else {
352 		// downsample
353 		while (count--) {
354 			register float sample = *(const float *)src * gain;
355 			if (sample > 2147483647.0f)
356 				*(int32 *)dst = 2147483647L;
357 			else if (sample < -2147483647.0f)
358 				*(int32 *)dst = -2147483647L;
359 			else
360 				*(int32 *)dst = (int32)sample;
361 			dst += dst_sample_offset;
362 			current += delta;
363 			register int32 skipcount = (int32)current;
364 			current -= skipcount;
365 			src += skipcount * src_sample_offset;
366 		}
367 	}
368 }
369 
370 void
371 Resampler::float_to_int16(const void *_src, int32 src_sample_offset, int32 src_sample_count,
372 				  		  void *_dst, int32 dst_sample_offset, int32 dst_sample_count, float _gain)
373 {
374 	register const char * src = (const char *) _src;
375 	register char * dst = (char *) _dst;
376 	register int32 count = dst_sample_count;
377 	register float gain = _gain * 32767.0;
378 
379 	if (src_sample_count == dst_sample_count) {
380 		// optimized case for no resampling
381 		while (count--) {
382 			register float sample = *(const float *)src * gain;
383 			if (sample > 32767.0f)
384 				*(int16 *)dst = 32767;
385 			else if (sample < -32767.0f)
386 				*(int16 *)dst = -32767;
387 			else
388 				*(int16 *)dst = (int16)sample;
389 			src += src_sample_offset;
390 			dst += dst_sample_offset;
391 		}
392 		return;
393 	}
394 
395 	register float delta = float(src_sample_count) / float(dst_sample_count);
396 	register float current = 0.0f;
397 
398 	if (delta < 1.0) {
399 		// upsample
400 		while (count--) {
401 			register float sample = *(const float *)src * gain;
402 			if (sample > 32767.0f)
403 				*(int16 *)dst = 32767;
404 			else if (sample < -32767.0f)
405 				*(int16 *)dst = -32767;
406 			else
407 				*(int16 *)dst = (int16)sample;
408 			dst += dst_sample_offset;
409 			current += delta;
410 			if (current >= 1.0f) {
411 				current -= 1.0f;
412 				src += src_sample_offset;
413 			}
414 		}
415 	} else {
416 		// downsample
417 		while (count--) {
418 			register float sample = *(const float *)src * gain;
419 			if (sample > 32767.0f)
420 				*(int16 *)dst = 32767;
421 			else if (sample < -32767.0f)
422 				*(int16 *)dst = -32767;
423 			else
424 				*(int16 *)dst = (int16)sample;
425 			dst += dst_sample_offset;
426 			current += delta;
427 			register int32 skipcount = (int32)current;
428 			current -= skipcount;
429 			src += skipcount * src_sample_offset;
430 		}
431 	}
432 }
433 
434 void
435 Resampler::float_to_int8(const void *_src, int32 src_sample_offset, int32 src_sample_count,
436 			 			 void *_dst, int32 dst_sample_offset, int32 dst_sample_count, float _gain)
437 {
438 	register const char * src = (const char *) _src;
439 	register char * dst = (char *) _dst;
440 	register int32 count = dst_sample_count;
441 	register float gain = _gain * 127.0;
442 
443 	if (src_sample_count == dst_sample_count) {
444 		// optimized case for no resampling
445 		while (count--) {
446 			register float sample = *(const float *)src * gain;
447 			if (sample > 127.0f)
448 				*(int8 *)dst = 127;
449 			else if (sample < -127.0f)
450 				*(int8 *)dst = -127;
451 			else
452 				*(int8 *)dst = (int8)sample;
453 			src += src_sample_offset;
454 			dst += dst_sample_offset;
455 		}
456 		return;
457 	}
458 
459 	register float delta = float(src_sample_count) / float(dst_sample_count);
460 	register float current = 0.0f;
461 
462 	if (delta < 1.0) {
463 		// upsample
464 		while (count--) {
465 			register float sample = *(const float *)src * gain;
466 			if (sample > 127.0f)
467 				*(int8 *)dst = 127;
468 			else if (sample < -127.0f)
469 				*(int8 *)dst = -127;
470 			else
471 				*(int8 *)dst = (int8)sample;
472 			dst += dst_sample_offset;
473 			current += delta;
474 			if (current >= 1.0f) {
475 				current -= 1.0f;
476 				src += src_sample_offset;
477 			}
478 		}
479 	} else {
480 		// downsample
481 		while (count--) {
482 			register float sample = *(const float *)src * gain;
483 			if (sample > 127.0f)
484 				*(int8 *)dst = 127;
485 			else if (sample < -127.0f)
486 				*(int8 *)dst = -127;
487 			else
488 				*(int8 *)dst = (int8)sample;
489 			dst += dst_sample_offset;
490 			current += delta;
491 			register int32 skipcount = (int32)current;
492 			current -= skipcount;
493 			src += skipcount * src_sample_offset;
494 		}
495 	}
496 }
497 
498 void
499 Resampler::float_to_uint8(const void *_src, int32 src_sample_offset, int32 src_sample_count,
500 						  void *_dst, int32 dst_sample_offset, int32 dst_sample_count, float _gain)
501 {
502 	register const char * src = (const char *) _src;
503 	register char * dst = (char *) _dst;
504 	register int32 count = dst_sample_count;
505 	register float gain = _gain * 127.0;
506 
507 	if (src_sample_count == dst_sample_count) {
508 		// optimized case for no resampling
509 		while (count--) {
510 			register float sample = 128.0f + *(const float *)src * gain;
511 			if (sample > 255.0f)
512 				*(uint8 *)dst = 255;
513 			else if (sample < 1.0f)
514 				*(uint8 *)dst = 1;
515 			else
516 				*(uint8 *)dst = (uint8)sample;
517 			src += src_sample_offset;
518 			dst += dst_sample_offset;
519 		}
520 		return;
521 	}
522 
523 	register float delta = float(src_sample_count) / float(dst_sample_count);
524 	register float current = 0.0f;
525 
526 	if (delta < 1.0) {
527 		// upsample
528 		while (count--) {
529 			register float sample = 128.0f + *(const float *)src * gain;
530 			if (sample > 255.0f)
531 				*(uint8 *)dst = 255;
532 			else if (sample < 1.0f)
533 				*(uint8 *)dst = 1;
534 			else
535 				*(uint8 *)dst = (uint8)sample;
536 			dst += dst_sample_offset;
537 			current += delta;
538 			if (current >= 1.0f) {
539 				current -= 1.0f;
540 				src += src_sample_offset;
541 			}
542 		}
543 	} else {
544 		// downsample
545 		while (count--) {
546 			register float sample = 128.0f + *(const float *)src * gain;
547 			if (sample > 255.0f)
548 				*(uint8 *)dst = 255;
549 			else if (sample < 1.0f)
550 				*(uint8 *)dst = 1;
551 			else
552 				*(uint8 *)dst = (uint8)sample;
553 			dst += dst_sample_offset;
554 			current += delta;
555 			register int32 skipcount = (int32)current;
556 			current -= skipcount;
557 			src += skipcount * src_sample_offset;
558 		}
559 	}
560 }
561