#!/usr/bin/env python3 import numpy as np import wave import struct import sys from sbc import * from sbc_encoder import * from sbc_decoder import * error = 0.99 max_error = -1 def sbc_compare_pcm(frame_count, actual_frame, expected_frame): global error, max_error for ch in range(actual_frame.nr_channels): M = mse(actual_frame.pcm[ch], expected_frame.pcm[ch]) if M > max_error: max_error = M if max_error > error: print("pcm error (%d, %f ) " % (frame_count, max_error)) return -1 return 0 def sbc_compare_headers(frame_count, actual_frame, expected_frame): if actual_frame.sampling_frequency != expected_frame.sampling_frequency: print("sampling_frequency wrong ", actual_frame.sampling_frequency) return -1 if actual_frame.nr_blocks != expected_frame.nr_blocks: print("nr_blocks wrong ", actual_frame.nr_blocks) return -1 if actual_frame.channel_mode != expected_frame.channel_mode: print("channel_mode wrong ", actual_frame.channel_mode) return -1 if actual_frame.nr_channels != expected_frame.nr_channels: print("nr_channels wrong ", actual_frame.nr_channels) return -1 if actual_frame.allocation_method != expected_frame.allocation_method: print("allocation_method wrong ", actual_frame.allocation_method) return -1 if actual_frame.nr_subbands != expected_frame.nr_subbands: print("nr_subbands wrong ", actual_frame.nr_subbands) return -1 if actual_frame.bitpool != expected_frame.bitpool: print("bitpool wrong (E: %d, D: %d)" % (actual_frame.bitpool, expected_frame.bitpool)) return -1 return 0 file_size = 0 def get_actual_frame(fin, implementation, frame_count): global file_size actual_frame = SBCFrame() sbc_unpack_frame(fin, file_size - fin.tell(), actual_frame) sbc_reconstruct_subband_samples(actual_frame) if subband_frame_count == 0: sbc_init_sythesis(actual_frame.nr_subbands, implementation) print(actual_frame) sbc_synthesis(actual_frame, implementation) return actual_frame def get_expected_frame(fin_expected, nr_blocks, nr_subbands, nr_channels, sampling_frequency, bitpool, allocation_method): expected_frame = SBCFrame(nr_blocks, nr_subbands, nr_channels, sampling_frequency, bitpool, allocation_method) fetch_samples_for_next_sbc_frame(fin_expected, expected_frame) calculate_channel_mode_and_scale_factors(expected_frame, 0) return expected_frame usage = ''' Usage: ./sbc_decoder_test.py decoder_input.sbc force_channel_mode[No=0, Stereo=2, Joint Stereo=3] implementation[SIG, V1] decoder_expected_output.wav Example: ./sbc_decoder_test.py fanfare-4sb.sbc 0 fanfare-4sb-decoded.wav ''' if (len(sys.argv) < 5): print(usage) sys.exit(1) try: decoder_input_sbc = sys.argv[1] force_channel_mode = int(sys.argv[2]) implementation = sys.argv[3] decoder_expected_wav = sys.argv[4] if not decoder_input_sbc.endswith('.sbc'): print(usage) sys.exit(1) if not decoder_expected_wav.endswith('.wav'): print(usage) sys.exit(1) fin_expected = wave.open(decoder_expected_wav, 'rb') nr_channels, sampwidth, sampling_frequency, nr_audio_frames, comptype, compname = fin_expected.getparams() with open(decoder_input_sbc, 'rb') as fin: try: subband_frame_count = 0 fin.seek(0,2) file_size = fin.tell() fin.seek(0,0) while True: if subband_frame_count % 200 == 0: print("== Frame %d ==" % subband_frame_count) actual_frame = get_actual_frame(fin, implementation, subband_frame_count) expected_frame = get_expected_frame(fin_expected, actual_frame.nr_blocks, actual_frame.nr_subbands, nr_channels, actual_frame.bitpool, sampling_frequency, actual_frame.allocation_method) err = sbc_compare_headers(subband_frame_count, actual_frame, expected_frame) if err < 0: print("Frame %d: Headers differ \n%s\n%s" % (subband_frame_count, actual_frame, expected_frame)) sys.exit(1) err = sbc_compare_pcm(subband_frame_count, actual_frame, expected_frame) if err < 0: print("Frame %d: PCMs differ %f \n%s\n%s" % (subband_frame_count, max_error, actual_frame.pcm, expected_frame.pcm)) sys.exit(1) subband_frame_count += 1 except TypeError: fin_expected.close() fin.close() print("DONE, max MSE PCM error %f" % max_error) except IOError as e: print(usage) sys.exit(1)