1*49fe348cSAndroid Build Coastguard Worker#!/usr/bin/env python3 2*49fe348cSAndroid Build Coastguard Worker# 3*49fe348cSAndroid Build Coastguard Worker# Copyright 2024 Google LLC 4*49fe348cSAndroid Build Coastguard Worker# 5*49fe348cSAndroid Build Coastguard Worker# Licensed under the Apache License, Version 2.0 (the "License"); 6*49fe348cSAndroid Build Coastguard Worker# you may not use this file except in compliance with the License. 7*49fe348cSAndroid Build Coastguard Worker# You may obtain a copy of the License at 8*49fe348cSAndroid Build Coastguard Worker# 9*49fe348cSAndroid Build Coastguard Worker# http://www.apache.org/licenses/LICENSE-2.0 10*49fe348cSAndroid Build Coastguard Worker# 11*49fe348cSAndroid Build Coastguard Worker# Unless required by applicable law or agreed to in writing, software 12*49fe348cSAndroid Build Coastguard Worker# distributed under the License is distributed on an "AS IS" BASIS, 13*49fe348cSAndroid Build Coastguard Worker# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14*49fe348cSAndroid Build Coastguard Worker# See the License for the specific language governing permissions and 15*49fe348cSAndroid Build Coastguard Worker# limitations under the License. 16*49fe348cSAndroid Build Coastguard Worker# 17*49fe348cSAndroid Build Coastguard Worker 18*49fe348cSAndroid Build Coastguard Workerimport argparse 19*49fe348cSAndroid Build Coastguard Workerimport lc3 20*49fe348cSAndroid Build Coastguard Workerimport struct 21*49fe348cSAndroid Build Coastguard Workerimport sys 22*49fe348cSAndroid Build Coastguard Workerimport wave 23*49fe348cSAndroid Build Coastguard Worker 24*49fe348cSAndroid Build Coastguard Workerparser = argparse.ArgumentParser(description='LC3 Decoder') 25*49fe348cSAndroid Build Coastguard Worker 26*49fe348cSAndroid Build Coastguard Workerparser.add_argument( 27*49fe348cSAndroid Build Coastguard Worker 'lc3_file', nargs='?', 28*49fe348cSAndroid Build Coastguard Worker help='Input bitstream file, default is stdin', 29*49fe348cSAndroid Build Coastguard Worker type=argparse.FileType('rb'), default=sys.stdin.buffer) 30*49fe348cSAndroid Build Coastguard Worker 31*49fe348cSAndroid Build Coastguard Workerparser.add_argument( 32*49fe348cSAndroid Build Coastguard Worker 'wav_file', nargs='?', 33*49fe348cSAndroid Build Coastguard Worker help='Output wave file, default is stdout', 34*49fe348cSAndroid Build Coastguard Worker type=argparse.FileType('wb'), default=sys.stdout.buffer) 35*49fe348cSAndroid Build Coastguard Worker 36*49fe348cSAndroid Build Coastguard Workerparser.add_argument( 37*49fe348cSAndroid Build Coastguard Worker '--bitdepth', 38*49fe348cSAndroid Build Coastguard Worker help='Output bitdepth, default is 16 bits', 39*49fe348cSAndroid Build Coastguard Worker type=int, choices=[16, 24], default=16) 40*49fe348cSAndroid Build Coastguard Worker 41*49fe348cSAndroid Build Coastguard Workerparser.add_argument( 42*49fe348cSAndroid Build Coastguard Worker '--libpath', help='LC3 Library path') 43*49fe348cSAndroid Build Coastguard Worker 44*49fe348cSAndroid Build Coastguard Workerargs = parser.parse_args() 45*49fe348cSAndroid Build Coastguard Worker 46*49fe348cSAndroid Build Coastguard Worker# --- LC3 File input --- 47*49fe348cSAndroid Build Coastguard Worker 48*49fe348cSAndroid Build Coastguard Workerf_lc3 = args.lc3_file 49*49fe348cSAndroid Build Coastguard Worker 50*49fe348cSAndroid Build Coastguard Workerheader = struct.unpack('=HHHHHHHI', f_lc3.read(18)) 51*49fe348cSAndroid Build Coastguard Workerif header[0] != 0xcc1c: 52*49fe348cSAndroid Build Coastguard Worker raise ValueError('Invalid bitstream file') 53*49fe348cSAndroid Build Coastguard Worker 54*49fe348cSAndroid Build Coastguard Workersamplerate = header[2] * 100 55*49fe348cSAndroid Build Coastguard Workernchannels = header[4] 56*49fe348cSAndroid Build Coastguard Workerframe_duration = header[5] / 100 57*49fe348cSAndroid Build Coastguard Workerstream_length = header[7] 58*49fe348cSAndroid Build Coastguard Worker 59*49fe348cSAndroid Build Coastguard Worker# --- Setup output --- 60*49fe348cSAndroid Build Coastguard Worker 61*49fe348cSAndroid Build Coastguard Workerbitdepth = args.bitdepth 62*49fe348cSAndroid Build Coastguard Workerpcm_size = nchannels * (bitdepth // 8) 63*49fe348cSAndroid Build Coastguard Worker 64*49fe348cSAndroid Build Coastguard Workerf_wav = args.wav_file 65*49fe348cSAndroid Build Coastguard Workerwavfile = wave.open(f_wav) 66*49fe348cSAndroid Build Coastguard Workerwavfile.setnchannels(nchannels) 67*49fe348cSAndroid Build Coastguard Workerwavfile.setsampwidth(bitdepth // 8) 68*49fe348cSAndroid Build Coastguard Workerwavfile.setframerate(samplerate) 69*49fe348cSAndroid Build Coastguard Workerwavfile.setnframes(stream_length) 70*49fe348cSAndroid Build Coastguard Worker 71*49fe348cSAndroid Build Coastguard Worker# --- Setup decoder --- 72*49fe348cSAndroid Build Coastguard Worker 73*49fe348cSAndroid Build Coastguard Workerdec = lc3.Decoder( 74*49fe348cSAndroid Build Coastguard Worker frame_duration, samplerate, nchannels, libpath=args.libpath) 75*49fe348cSAndroid Build Coastguard Workerframe_length = dec.get_frame_samples() 76*49fe348cSAndroid Build Coastguard Workerencoded_length = stream_length + dec.get_delay_samples() 77*49fe348cSAndroid Build Coastguard Worker 78*49fe348cSAndroid Build Coastguard Worker# --- Decoding loop --- 79*49fe348cSAndroid Build Coastguard Worker 80*49fe348cSAndroid Build Coastguard Workerfor i in range(0, encoded_length, frame_length): 81*49fe348cSAndroid Build Coastguard Worker 82*49fe348cSAndroid Build Coastguard Worker lc3_frame_size = struct.unpack('=H', f_lc3.read(2))[0] 83*49fe348cSAndroid Build Coastguard Worker pcm = dec.decode(f_lc3.read(lc3_frame_size), bitdepth=bitdepth) 84*49fe348cSAndroid Build Coastguard Worker 85*49fe348cSAndroid Build Coastguard Worker pcm = pcm[max(encoded_length - stream_length - i, 0) * pcm_size: 86*49fe348cSAndroid Build Coastguard Worker min(encoded_length - i, frame_length) * pcm_size] 87*49fe348cSAndroid Build Coastguard Worker 88*49fe348cSAndroid Build Coastguard Worker wavfile.writeframesraw(pcm) 89*49fe348cSAndroid Build Coastguard Worker 90*49fe348cSAndroid Build Coastguard Worker# --- Cleanup --- 91*49fe348cSAndroid Build Coastguard Worker 92*49fe348cSAndroid Build Coastguard Workerwavfile.close() 93*49fe348cSAndroid Build Coastguard Worker 94*49fe348cSAndroid Build Coastguard Workerfor f in (f_lc3, f_wav): 95*49fe348cSAndroid Build Coastguard Worker if f is not sys.stdout.buffer: 96*49fe348cSAndroid Build Coastguard Worker f.close() 97