xref: /aosp_15_r20/external/liblc3/python/tools/decoder.py (revision 49fe348c0058011ee60b6957cdd9d52742df84bc)
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