pulseaudio/src/modules/bluetooth
Siarhei Siamashka 00602537ba bluetooth: sbc: overflow bugfix and audio decoding quality improvement
The "(((audio_sample << 1) | 1) << frame->scale_factor[ch][sb])"
part of expression
    "frame->sb_sample[blk][ch][sb] =
        (((audio_sample << 1) | 1) << frame->scale_factor[ch][sb]) /
        levels[ch][sb] - (1 << frame->scale_factor[ch][sb])"
in "sbc_unpack_frame" function can sometimes overflow 32-bit signed int.
This problem can be reproduced by first using bitpool 128 and encoding
some random noise data, and then feeding it to sbc decoder. The obvious
thing to do would be to change "audio_sample" variable type to uint32_t.

However the problem is a little bit more complicated. According
to the section "12.6.2 Scale Factors" of A2DP spec:
    scalefactor[ch][sb] = pow(2.0, (scale_factor[ch][sb] + 1))

And according to "12.6.4 Reconstruction of the Subband Samples":
    sb_sample[blk][ch][sb] = scalefactor[ch][sb] *
        ((audio_sample[blk][ch][sb]*2.0+1.0) / levels[ch][sb]-1.0);

Hence the current code for calculating "sb_sample[blk][ch][sb]" is
not quite correct, because it loses one least significant bit of
sample data and passes twice smaller sample values to the synthesis
filter (the filter also deviates from the spec to compensate this).
This all has quite a noticeable impact on audio quality. Moreover,
it makes sense to keep a few extra bits of precision here in order
to minimize rounding errors. So the proposed patch introduces a new
SBCDEC_FIXED_EXTRA_BITS constant and uses uint64_t data type
for intermediate calculations in order to safeguard against
overflows. This patch intentionally addresses only the quality
issue, but performance can be also improved later (like replacing
division with multiplication by reciprocal).

Test for the difference of sbc encoding/decoding roundtrip vs.
the original audio file for joint stereo, bitpool 128, 8 subbands
and http://media.xiph.org/sintel/sintel-master-st.flac sample
demonstrates some quality improvement:

=== before ===
    --- comparing original / sbc_encoder.exe + sbcdec ---
    stddev:    4.64 PSNR: 82.97 bytes:170495708/170496000
=== after ===
    --- comparing original / sbc_encoder.exe + sbcdec ---
    stddev:    1.95 PSNR: 90.50 bytes:170495708/170496000
2011-10-28 15:43:48 +02:00
..
sbc bluetooth: sbc: overflow bugfix and audio decoding quality improvement 2011-10-28 15:43:48 +02:00
a2dp-codecs.h bluetooth: audio: Update license for shared header files 2011-10-28 15:43:42 +02:00
bluetooth-util.c bluetooth: Bump DBus version to 1.3.0 and drop conditional code. 2011-09-07 20:33:13 +01:00
bluetooth-util.h Remove unnecessary #includes 2011-06-22 23:12:20 +01:00
ipc.c bluetooth: audio: Update license for shared header files 2011-10-28 15:43:42 +02:00
ipc.h bluetooth: audio: Update license for shared header files 2011-10-28 15:43:42 +02:00
module-bluetooth-device.c Move i18n.[ch] to src/pulsecore 2011-08-11 13:23:42 +02:00
module-bluetooth-discover.c Remove unnecessary #includes 2011-06-22 23:12:20 +01:00
module-bluetooth-proximity.c Update PA_MODULE_USAGE to be in line with actual implementation 2011-03-18 12:07:02 +00:00
proximity-helper.c Move bluetooth proximity module to src/modules/bluetooth/ 2008-09-11 01:12:10 +03:00
rtp.h bluetooth: Run 'make update-sbc' 2011-03-20 12:49:49 +00:00