1"""Stuff to parse Sun and NeXT audio files.
2
3An audio file consists of a header followed by the data.  The structure
4of the header is as follows.
5
6        +---------------+
7        | magic word    |
8        +---------------+
9        | header size   |
10        +---------------+
11        | data size     |
12        +---------------+
13        | encoding      |
14        +---------------+
15        | sample rate   |
16        +---------------+
17        | # of channels |
18        +---------------+
19        | info          |
20        |               |
21        +---------------+
22
23The magic word consists of the 4 characters '.snd'.  Apart from the
24info field, all header fields are 4 bytes in size.  They are all
2532-bit unsigned integers encoded in big-endian byte order.
26
27The header size really gives the start of the data.
28The data size is the physical size of the data.  From the other
29parameters the number of frames can be calculated.
30The encoding gives the way in which audio samples are encoded.
31Possible values are listed below.
32The info field currently consists of an ASCII string giving a
33human-readable description of the audio file.  The info field is
34padded with NUL bytes to the header size.
35
36Usage.
37
38Reading audio files:
39        f = sunau.open(file, 'r')
40where file is either the name of a file or an open file pointer.
41The open file pointer must have methods read(), seek(), and close().
42When the setpos() and rewind() methods are not used, the seek()
43method is not  necessary.
44
45This returns an instance of a class with the following public methods:
46        getnchannels()  -- returns number of audio channels (1 for
47                           mono, 2 for stereo)
48        getsampwidth()  -- returns sample width in bytes
49        getframerate()  -- returns sampling frequency
50        getnframes()    -- returns number of audio frames
51        getcomptype()   -- returns compression type ('NONE' or 'ULAW')
52        getcompname()   -- returns human-readable version of
53                           compression type ('not compressed' matches 'NONE')
54        getparams()     -- returns a namedtuple consisting of all of the
55                           above in the above order
56        getmarkers()    -- returns None (for compatibility with the
57                           aifc module)
58        getmark(id)     -- raises an error since the mark does not
59                           exist (for compatibility with the aifc module)
60        readframes(n)   -- returns at most n frames of audio
61        rewind()        -- rewind to the beginning of the audio stream
62        setpos(pos)     -- seek to the specified position
63        tell()          -- return the current position
64        close()         -- close the instance (make it unusable)
65The position returned by tell() and the position given to setpos()
66are compatible and have nothing to do with the actual position in the
67file.
68The close() method is called automatically when the class instance
69is destroyed.
70
71Writing audio files:
72        f = sunau.open(file, 'w')
73where file is either the name of a file or an open file pointer.
74The open file pointer must have methods write(), tell(), seek(), and
75close().
76
77This returns an instance of a class with the following public methods:
78        setnchannels(n) -- set the number of channels
79        setsampwidth(n) -- set the sample width
80        setframerate(n) -- set the frame rate
81        setnframes(n)   -- set the number of frames
82        setcomptype(type, name)
83                        -- set the compression type and the
84                           human-readable compression type
85        setparams(tuple)-- set all parameters at once
86        tell()          -- return current position in output file
87        writeframesraw(data)
88                        -- write audio frames without pathing up the
89                           file header
90        writeframes(data)
91                        -- write audio frames and patch up the file header
92        close()         -- patch up the file header and close the
93                           output file
94You should set the parameters before the first writeframesraw or
95writeframes.  The total number of frames does not need to be set,
96but when it is set to the correct value, the header does not have to
97be patched up.
98It is best to first set all parameters, perhaps possibly the
99compression type, and then write audio frames using writeframesraw.
100When all frames have been written, either call writeframes(b'') or
101close() to patch up the sizes in the header.
102The close() method is called automatically when the class instance
103is destroyed.
104"""
105
106from collections import namedtuple
107import warnings
108
109_sunau_params = namedtuple('_sunau_params',
110                           'nchannels sampwidth framerate nframes comptype compname')
111
112# from <multimedia/audio_filehdr.h>
113AUDIO_FILE_MAGIC = 0x2e736e64
114AUDIO_FILE_ENCODING_MULAW_8 = 1
115AUDIO_FILE_ENCODING_LINEAR_8 = 2
116AUDIO_FILE_ENCODING_LINEAR_16 = 3
117AUDIO_FILE_ENCODING_LINEAR_24 = 4
118AUDIO_FILE_ENCODING_LINEAR_32 = 5
119AUDIO_FILE_ENCODING_FLOAT = 6
120AUDIO_FILE_ENCODING_DOUBLE = 7
121AUDIO_FILE_ENCODING_ADPCM_G721 = 23
122AUDIO_FILE_ENCODING_ADPCM_G722 = 24
123AUDIO_FILE_ENCODING_ADPCM_G723_3 = 25
124AUDIO_FILE_ENCODING_ADPCM_G723_5 = 26
125AUDIO_FILE_ENCODING_ALAW_8 = 27
126
127# from <multimedia/audio_hdr.h>
128AUDIO_UNKNOWN_SIZE = 0xFFFFFFFF        # ((unsigned)(~0))
129
130_simple_encodings = [AUDIO_FILE_ENCODING_MULAW_8,
131                     AUDIO_FILE_ENCODING_LINEAR_8,
132                     AUDIO_FILE_ENCODING_LINEAR_16,
133                     AUDIO_FILE_ENCODING_LINEAR_24,
134                     AUDIO_FILE_ENCODING_LINEAR_32,
135                     AUDIO_FILE_ENCODING_ALAW_8]
136
137class Error(Exception):
138    pass
139
140def _read_u32(file):
141    x = 0
142    for i in range(4):
143        byte = file.read(1)
144        if not byte:
145            raise EOFError
146        x = x*256 + ord(byte)
147    return x
148
149def _write_u32(file, x):
150    data = []
151    for i in range(4):
152        d, m = divmod(x, 256)
153        data.insert(0, int(m))
154        x = d
155    file.write(bytes(data))
156
157class Au_read:
158
159    def __init__(self, f):
160        if type(f) == type(''):
161            import builtins
162            f = builtins.open(f, 'rb')
163            self._opened = True
164        else:
165            self._opened = False
166        self.initfp(f)
167
168    def __del__(self):
169        if self._file:
170            self.close()
171
172    def __enter__(self):
173        return self
174
175    def __exit__(self, *args):
176        self.close()
177
178    def initfp(self, file):
179        self._file = file
180        self._soundpos = 0
181        magic = int(_read_u32(file))
182        if magic != AUDIO_FILE_MAGIC:
183            raise Error('bad magic number')
184        self._hdr_size = int(_read_u32(file))
185        if self._hdr_size < 24:
186            raise Error('header size too small')
187        if self._hdr_size > 100:
188            raise Error('header size ridiculously large')
189        self._data_size = _read_u32(file)
190        if self._data_size != AUDIO_UNKNOWN_SIZE:
191            self._data_size = int(self._data_size)
192        self._encoding = int(_read_u32(file))
193        if self._encoding not in _simple_encodings:
194            raise Error('encoding not (yet) supported')
195        if self._encoding in (AUDIO_FILE_ENCODING_MULAW_8,
196                  AUDIO_FILE_ENCODING_ALAW_8):
197            self._sampwidth = 2
198            self._framesize = 1
199        elif self._encoding == AUDIO_FILE_ENCODING_LINEAR_8:
200            self._framesize = self._sampwidth = 1
201        elif self._encoding == AUDIO_FILE_ENCODING_LINEAR_16:
202            self._framesize = self._sampwidth = 2
203        elif self._encoding == AUDIO_FILE_ENCODING_LINEAR_24:
204            self._framesize = self._sampwidth = 3
205        elif self._encoding == AUDIO_FILE_ENCODING_LINEAR_32:
206            self._framesize = self._sampwidth = 4
207        else:
208            raise Error('unknown encoding')
209        self._framerate = int(_read_u32(file))
210        self._nchannels = int(_read_u32(file))
211        if not self._nchannels:
212            raise Error('bad # of channels')
213        self._framesize = self._framesize * self._nchannels
214        if self._hdr_size > 24:
215            self._info = file.read(self._hdr_size - 24)
216            self._info, _, _ = self._info.partition(b'\0')
217        else:
218            self._info = b''
219        try:
220            self._data_pos = file.tell()
221        except (AttributeError, OSError):
222            self._data_pos = None
223
224    def getfp(self):
225        return self._file
226
227    def getnchannels(self):
228        return self._nchannels
229
230    def getsampwidth(self):
231        return self._sampwidth
232
233    def getframerate(self):
234        return self._framerate
235
236    def getnframes(self):
237        if self._data_size == AUDIO_UNKNOWN_SIZE:
238            return AUDIO_UNKNOWN_SIZE
239        if self._encoding in _simple_encodings:
240            return self._data_size // self._framesize
241        return 0                # XXX--must do some arithmetic here
242
243    def getcomptype(self):
244        if self._encoding == AUDIO_FILE_ENCODING_MULAW_8:
245            return 'ULAW'
246        elif self._encoding == AUDIO_FILE_ENCODING_ALAW_8:
247            return 'ALAW'
248        else:
249            return 'NONE'
250
251    def getcompname(self):
252        if self._encoding == AUDIO_FILE_ENCODING_MULAW_8:
253            return 'CCITT G.711 u-law'
254        elif self._encoding == AUDIO_FILE_ENCODING_ALAW_8:
255            return 'CCITT G.711 A-law'
256        else:
257            return 'not compressed'
258
259    def getparams(self):
260        return _sunau_params(self.getnchannels(), self.getsampwidth(),
261                  self.getframerate(), self.getnframes(),
262                  self.getcomptype(), self.getcompname())
263
264    def getmarkers(self):
265        return None
266
267    def getmark(self, id):
268        raise Error('no marks')
269
270    def readframes(self, nframes):
271        if self._encoding in _simple_encodings:
272            if nframes == AUDIO_UNKNOWN_SIZE:
273                data = self._file.read()
274            else:
275                data = self._file.read(nframes * self._framesize)
276            self._soundpos += len(data) // self._framesize
277            if self._encoding == AUDIO_FILE_ENCODING_MULAW_8:
278                import audioop
279                data = audioop.ulaw2lin(data, self._sampwidth)
280            return data
281        return None             # XXX--not implemented yet
282
283    def rewind(self):
284        if self._data_pos is None:
285            raise OSError('cannot seek')
286        self._file.seek(self._data_pos)
287        self._soundpos = 0
288
289    def tell(self):
290        return self._soundpos
291
292    def setpos(self, pos):
293        if pos < 0 or pos > self.getnframes():
294            raise Error('position not in range')
295        if self._data_pos is None:
296            raise OSError('cannot seek')
297        self._file.seek(self._data_pos + pos * self._framesize)
298        self._soundpos = pos
299
300    def close(self):
301        file = self._file
302        if file:
303            self._file = None
304            if self._opened:
305                file.close()
306
307class Au_write:
308
309    def __init__(self, f):
310        if type(f) == type(''):
311            import builtins
312            f = builtins.open(f, 'wb')
313            self._opened = True
314        else:
315            self._opened = False
316        self.initfp(f)
317
318    def __del__(self):
319        if self._file:
320            self.close()
321        self._file = None
322
323    def __enter__(self):
324        return self
325
326    def __exit__(self, *args):
327        self.close()
328
329    def initfp(self, file):
330        self._file = file
331        self._framerate = 0
332        self._nchannels = 0
333        self._sampwidth = 0
334        self._framesize = 0
335        self._nframes = AUDIO_UNKNOWN_SIZE
336        self._nframeswritten = 0
337        self._datawritten = 0
338        self._datalength = 0
339        self._info = b''
340        self._comptype = 'ULAW' # default is U-law
341
342    def setnchannels(self, nchannels):
343        if self._nframeswritten:
344            raise Error('cannot change parameters after starting to write')
345        if nchannels not in (1, 2, 4):
346            raise Error('only 1, 2, or 4 channels supported')
347        self._nchannels = nchannels
348
349    def getnchannels(self):
350        if not self._nchannels:
351            raise Error('number of channels not set')
352        return self._nchannels
353
354    def setsampwidth(self, sampwidth):
355        if self._nframeswritten:
356            raise Error('cannot change parameters after starting to write')
357        if sampwidth not in (1, 2, 3, 4):
358            raise Error('bad sample width')
359        self._sampwidth = sampwidth
360
361    def getsampwidth(self):
362        if not self._framerate:
363            raise Error('sample width not specified')
364        return self._sampwidth
365
366    def setframerate(self, framerate):
367        if self._nframeswritten:
368            raise Error('cannot change parameters after starting to write')
369        self._framerate = framerate
370
371    def getframerate(self):
372        if not self._framerate:
373            raise Error('frame rate not set')
374        return self._framerate
375
376    def setnframes(self, nframes):
377        if self._nframeswritten:
378            raise Error('cannot change parameters after starting to write')
379        if nframes < 0:
380            raise Error('# of frames cannot be negative')
381        self._nframes = nframes
382
383    def getnframes(self):
384        return self._nframeswritten
385
386    def setcomptype(self, type, name):
387        if type in ('NONE', 'ULAW'):
388            self._comptype = type
389        else:
390            raise Error('unknown compression type')
391
392    def getcomptype(self):
393        return self._comptype
394
395    def getcompname(self):
396        if self._comptype == 'ULAW':
397            return 'CCITT G.711 u-law'
398        elif self._comptype == 'ALAW':
399            return 'CCITT G.711 A-law'
400        else:
401            return 'not compressed'
402
403    def setparams(self, params):
404        nchannels, sampwidth, framerate, nframes, comptype, compname = params
405        self.setnchannels(nchannels)
406        self.setsampwidth(sampwidth)
407        self.setframerate(framerate)
408        self.setnframes(nframes)
409        self.setcomptype(comptype, compname)
410
411    def getparams(self):
412        return _sunau_params(self.getnchannels(), self.getsampwidth(),
413                  self.getframerate(), self.getnframes(),
414                  self.getcomptype(), self.getcompname())
415
416    def tell(self):
417        return self._nframeswritten
418
419    def writeframesraw(self, data):
420        if not isinstance(data, (bytes, bytearray)):
421            data = memoryview(data).cast('B')
422        self._ensure_header_written()
423        if self._comptype == 'ULAW':
424            import audioop
425            data = audioop.lin2ulaw(data, self._sampwidth)
426        nframes = len(data) // self._framesize
427        self._file.write(data)
428        self._nframeswritten = self._nframeswritten + nframes
429        self._datawritten = self._datawritten + len(data)
430
431    def writeframes(self, data):
432        self.writeframesraw(data)
433        if self._nframeswritten != self._nframes or \
434                  self._datalength != self._datawritten:
435            self._patchheader()
436
437    def close(self):
438        if self._file:
439            try:
440                self._ensure_header_written()
441                if self._nframeswritten != self._nframes or \
442                        self._datalength != self._datawritten:
443                    self._patchheader()
444                self._file.flush()
445            finally:
446                file = self._file
447                self._file = None
448                if self._opened:
449                    file.close()
450
451    #
452    # private methods
453    #
454
455    def _ensure_header_written(self):
456        if not self._nframeswritten:
457            if not self._nchannels:
458                raise Error('# of channels not specified')
459            if not self._sampwidth:
460                raise Error('sample width not specified')
461            if not self._framerate:
462                raise Error('frame rate not specified')
463            self._write_header()
464
465    def _write_header(self):
466        if self._comptype == 'NONE':
467            if self._sampwidth == 1:
468                encoding = AUDIO_FILE_ENCODING_LINEAR_8
469                self._framesize = 1
470            elif self._sampwidth == 2:
471                encoding = AUDIO_FILE_ENCODING_LINEAR_16
472                self._framesize = 2
473            elif self._sampwidth == 3:
474                encoding = AUDIO_FILE_ENCODING_LINEAR_24
475                self._framesize = 3
476            elif self._sampwidth == 4:
477                encoding = AUDIO_FILE_ENCODING_LINEAR_32
478                self._framesize = 4
479            else:
480                raise Error('internal error')
481        elif self._comptype == 'ULAW':
482            encoding = AUDIO_FILE_ENCODING_MULAW_8
483            self._framesize = 1
484        else:
485            raise Error('internal error')
486        self._framesize = self._framesize * self._nchannels
487        _write_u32(self._file, AUDIO_FILE_MAGIC)
488        header_size = 25 + len(self._info)
489        header_size = (header_size + 7) & ~7
490        _write_u32(self._file, header_size)
491        if self._nframes == AUDIO_UNKNOWN_SIZE:
492            length = AUDIO_UNKNOWN_SIZE
493        else:
494            length = self._nframes * self._framesize
495        try:
496            self._form_length_pos = self._file.tell()
497        except (AttributeError, OSError):
498            self._form_length_pos = None
499        _write_u32(self._file, length)
500        self._datalength = length
501        _write_u32(self._file, encoding)
502        _write_u32(self._file, self._framerate)
503        _write_u32(self._file, self._nchannels)
504        self._file.write(self._info)
505        self._file.write(b'\0'*(header_size - len(self._info) - 24))
506
507    def _patchheader(self):
508        if self._form_length_pos is None:
509            raise OSError('cannot seek')
510        self._file.seek(self._form_length_pos)
511        _write_u32(self._file, self._datawritten)
512        self._datalength = self._datawritten
513        self._file.seek(0, 2)
514
515def open(f, mode=None):
516    if mode is None:
517        if hasattr(f, 'mode'):
518            mode = f.mode
519        else:
520            mode = 'rb'
521    if mode in ('r', 'rb'):
522        return Au_read(f)
523    elif mode in ('w', 'wb'):
524        return Au_write(f)
525    else:
526        raise Error("mode must be 'r', 'rb', 'w', or 'wb'")
527
528def openfp(f, mode=None):
529    warnings.warn("sunau.openfp is deprecated since Python 3.7. "
530                  "Use sunau.open instead.", DeprecationWarning, stacklevel=2)
531    return open(f, mode=mode)
532