forked from W3AXL/python-radio-console
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmulaw.py
106 lines (83 loc) · 3.16 KB
/
mulaw.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
import numpy as np
class MuLaw:
bias = 0x84
clip = 32635
encodeTable = [
0,0,1,1,2,2,2,2,3,3,3,3,3,3,3,3,
4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,
5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,
5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,
6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,
6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,
6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,
6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,
7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,
7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,
7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,
7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,
7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,
7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,
7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,
7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7
]
decodeTable = [0,132,396,924,1980,4092,8316,16764]
def __decodeSample(muLawSample):
"""
Decode a mu-law encoded uint8 to an int16 audio sample
Args:
muLawSample (uint8): input mu-law encoded uint8 sample
Returns:
int16: output audio sample
"""
# make sure we have a python uint8
muLawSample = muLawSample.astype(np.uint8).item()
# Do the decoding
muLawSample = ~muLawSample
sign = (muLawSample & 0x80)
exponent = (muLawSample >> 4) & 0x07
mantissa = muLawSample & 0x0f
sample = MuLaw.decodeTable[exponent] + (mantissa << (exponent + 3))
if (sign != 0): sample = -sample
return sample
def decode(muLawSamples):
"""
Decode 8-bit mu-law samples to 32-bit floats
Args:
samples (Uint8[]): array of Uint8 samples
Returns:
np.float32[]: array of float samples
"""
# Decode the samples
output = np.array(list(map(MuLaw.__decodeSample, muLawSamples)), dtype=np.int16)
# convert to float32 from int16 and return
return output.astype(np.float32, order='C') / 32768.0
def __encodeSample(sample):
"""
Convert an int16 sample to a mu-law encoded uint8 sample
Args:
sample (np.int16): input audio sample
Returns:
uint8: output mu-law encoded sample
"""
# Convert numpy int16 to python int16
sample = sample.item()
sign = (sample >> 8) & 0x80
if (sign != 0): sample = -sample
sample = sample + MuLaw.bias
if (sample > MuLaw.clip): sample = MuLaw.clip
exponent = MuLaw.encodeTable[(sample >> 7) & 0xff]
mantissa = (sample >> (exponent + 3)) & 0x0f
return ~(sign | (exponent << 4) | mantissa)
def encode(samples):
"""
Encode Float32 samples to 8-bit mu-law samples
Args:
samples (np.float32[]): float32 array of samples
Returns:
np.uint8[]: array of 8-bit mu-law samples
"""
# convert float32 to int16 (list comprehension should be faster than a for loop)
int16samples = np.clip([n * 32768 for n in samples], -32768, 32768).astype(np.int16)
# Encode samples to mu-law
output = np.array(list(map(MuLaw.__encodeSample, int16samples)), dtype=np.uint8)
return output