aboutsummaryrefslogtreecommitdiffstats
path: root/src/replayGain.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/replayGain.c')
-rw-r--r--src/replayGain.c110
1 files changed, 80 insertions, 30 deletions
diff --git a/src/replayGain.c b/src/replayGain.c
index ea521d728..f052c68bc 100644
--- a/src/replayGain.c
+++ b/src/replayGain.c
@@ -107,13 +107,20 @@ void freeReplayGainInfo(ReplayGainInfo * info) {
void doReplayGain(ReplayGainInfo * info, char * buffer, int bufferSize,
AudioFormat * format)
{
- mpd_sint16 * buffer16;
- mpd_sint8 * buffer8;
- mpd_sint32 temp32;
+ float * bufferFloat = (float *)buffer;
+ mpd_sint32 * buffer32 = (mpd_sint32 *)buffer;
+ mpd_sint16 * buffer16 = (mpd_sint16 *)buffer;
+ mpd_sint8 * buffer8 = (mpd_sint8 *)buffer;
float scale;
+ mpd_sint32 iScale;
if(replayGainState == REPLAYGAIN_OFF || !info) return;
+/* DEBUG */
+ if(bufferSize % (format->channels * 4))
+ ERROR("doReplayGain: bufferSize=%i not multipel of %i\n",
+ bufferSize, format->channels);
+/* /DEBUG */
if(info->scale < 0) {
switch(replayGainState) {
case REPLAYGAIN_TRACK:
@@ -127,35 +134,78 @@ void doReplayGain(ReplayGainInfo * info, char * buffer, int bufferSize,
}
}
- if(info->scale <= 1.01 && info->scale >= 0.99) return;
-
- buffer16 = (mpd_sint16 *)buffer;
- buffer8 = (mpd_sint8 *)buffer;
+#ifdef MPD_FIXED_POINT
+ if(format->bits!=16 || format.channels!=2 || format->floatSamples) {
+ ERROR("Only 16 bit stereo is supported in fixed point mode!\n");
+ exit(EXIT_FAILURE);
+ }
+
+ /* If the volume change is very small then we hardly here the
+ * difference anyway, and if the change is positiv then clipping
+ * may occur. We don't want that. */
+ if(info->scale > 0.99) return;
+
+ iScale = scale * 32768.0; /* << 15 */
+
+ while(bufferSize) {
+ sample32 = (mpd_sint32)(*buffer16) * iScale;
+ /* no need to check boundaries - we only lower the volume*/
+ /* It would be good to add some kind of dither here... TODO?! */
+ *buffer16 = (sample32 >> 15);
+ bufferSize -= 2;
+ }
+ return;
+#else
scale = info->scale;
-
- switch(format->bits) {
- case 16:
- while(bufferSize > 0){
- temp32 = *buffer16;
- temp32 *= scale;
- *buffer16 = temp32>32767 ? 32767 :
- (temp32<-32768 ? -32768 : temp32);
- buffer16++;
- bufferSize-=2;
- }
- break;
- case 8:
- while(bufferSize>0){
- temp32 = *buffer8;
- temp32 *= scale;
- *buffer8 = temp32>127 ? 127 :
- (temp32<-128 ? -128 : temp32);
- buffer8++;
- bufferSize--;
+
+ if(format->floatSamples) {
+ if(format->bits==32) {
+ while(bufferSize) {
+ *bufferFloat *= scale;
+ bufferFloat++;
+ bufferSize-=4;
}
- break;
- default:
- ERROR("%i bits not supported by doReplaygain!\n", format->bits);
+ return;
+ }
+ else {
+ ERROR("%i bit float not supported by doReplaygain!\n",
+ format->bits);
+ exit(EXIT_FAILURE);
+ }
+ }
+
+ switch(format->bits) {
+ case 32:
+ while(bufferSize) {
+ double sample = (double)(*buffer32) * scale;
+ if(sample>2147483647.0) *buffer32 = 2147483647;
+ else if(sample<-2147483647.0) *buffer32 = -2147483647;
+ else *buffer32 = rintf(sample);
+ *buffer32++;
+ bufferSize-=4;
+ }
+ break;
+ case 16:
+ while(bufferSize){
+ float sample = *buffer16 * scale;
+ *buffer16 = sample>32767.0 ? 32767 :
+ (sample<-32768.0 ? -32768 : rintf(sample));
+ buffer16++;
+ bufferSize-=2;
+ }
+ break;
+ case 8:
+ while(bufferSize){
+ float sample = *buffer8 * scale;
+ *buffer8 = sample>127.0 ? 127 :
+ (sample<-128.0 ? -128 : rintf(sample));
+ buffer8++;
+ bufferSize--;
+ }
+ break;
+ default:
+ ERROR("%i bits not supported by doReplaygain!\n", format->bits);
}
+#endif
}