mirror of
				https://github.com/python/cpython.git
				synced 2025-11-04 11:49:12 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			257 lines
		
	
	
	
		
			7.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			257 lines
		
	
	
	
		
			7.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
"""Classes for manipulating audio devices (currently only for Sun and SGI)"""
 | 
						|
 | 
						|
__all__ = ["error","AudioDev"]
 | 
						|
 | 
						|
class error(Exception):
 | 
						|
    pass
 | 
						|
 | 
						|
class Play_Audio_sgi:
 | 
						|
    # Private instance variables
 | 
						|
##      if 0: access frameratelist, nchannelslist, sampwidthlist, oldparams, \
 | 
						|
##                params, config, inited_outrate, inited_width, \
 | 
						|
##                inited_nchannels, port, converter, classinited: private
 | 
						|
 | 
						|
    classinited = 0
 | 
						|
    frameratelist = nchannelslist = sampwidthlist = None
 | 
						|
 | 
						|
    def initclass(self):
 | 
						|
        import AL
 | 
						|
        self.frameratelist = [
 | 
						|
                  (48000, AL.RATE_48000),
 | 
						|
                  (44100, AL.RATE_44100),
 | 
						|
                  (32000, AL.RATE_32000),
 | 
						|
                  (22050, AL.RATE_22050),
 | 
						|
                  (16000, AL.RATE_16000),
 | 
						|
                  (11025, AL.RATE_11025),
 | 
						|
                  ( 8000,  AL.RATE_8000),
 | 
						|
                  ]
 | 
						|
        self.nchannelslist = [
 | 
						|
                  (1, AL.MONO),
 | 
						|
                  (2, AL.STEREO),
 | 
						|
                  (4, AL.QUADRO),
 | 
						|
                  ]
 | 
						|
        self.sampwidthlist = [
 | 
						|
                  (1, AL.SAMPLE_8),
 | 
						|
                  (2, AL.SAMPLE_16),
 | 
						|
                  (3, AL.SAMPLE_24),
 | 
						|
                  ]
 | 
						|
        self.classinited = 1
 | 
						|
 | 
						|
    def __init__(self):
 | 
						|
        import al, AL
 | 
						|
        if not self.classinited:
 | 
						|
            self.initclass()
 | 
						|
        self.oldparams = []
 | 
						|
        self.params = [AL.OUTPUT_RATE, 0]
 | 
						|
        self.config = al.newconfig()
 | 
						|
        self.inited_outrate = 0
 | 
						|
        self.inited_width = 0
 | 
						|
        self.inited_nchannels = 0
 | 
						|
        self.converter = None
 | 
						|
        self.port = None
 | 
						|
        return
 | 
						|
 | 
						|
    def __del__(self):
 | 
						|
        if self.port:
 | 
						|
            self.stop()
 | 
						|
        if self.oldparams:
 | 
						|
            import al, AL
 | 
						|
            al.setparams(AL.DEFAULT_DEVICE, self.oldparams)
 | 
						|
            self.oldparams = []
 | 
						|
 | 
						|
    def wait(self):
 | 
						|
        if not self.port:
 | 
						|
            return
 | 
						|
        import time
 | 
						|
        while self.port.getfilled() > 0:
 | 
						|
            time.sleep(0.1)
 | 
						|
        self.stop()
 | 
						|
 | 
						|
    def stop(self):
 | 
						|
        if self.port:
 | 
						|
            self.port.closeport()
 | 
						|
            self.port = None
 | 
						|
        if self.oldparams:
 | 
						|
            import al, AL
 | 
						|
            al.setparams(AL.DEFAULT_DEVICE, self.oldparams)
 | 
						|
            self.oldparams = []
 | 
						|
 | 
						|
    def setoutrate(self, rate):
 | 
						|
        for (raw, cooked) in self.frameratelist:
 | 
						|
            if rate == raw:
 | 
						|
                self.params[1] = cooked
 | 
						|
                self.inited_outrate = 1
 | 
						|
                break
 | 
						|
        else:
 | 
						|
            raise error, 'bad output rate'
 | 
						|
 | 
						|
    def setsampwidth(self, width):
 | 
						|
        for (raw, cooked) in self.sampwidthlist:
 | 
						|
            if width == raw:
 | 
						|
                self.config.setwidth(cooked)
 | 
						|
                self.inited_width = 1
 | 
						|
                break
 | 
						|
        else:
 | 
						|
            if width == 0:
 | 
						|
                import AL
 | 
						|
                self.inited_width = 0
 | 
						|
                self.config.setwidth(AL.SAMPLE_16)
 | 
						|
                self.converter = self.ulaw2lin
 | 
						|
            else:
 | 
						|
                raise error, 'bad sample width'
 | 
						|
 | 
						|
    def setnchannels(self, nchannels):
 | 
						|
        for (raw, cooked) in self.nchannelslist:
 | 
						|
            if nchannels == raw:
 | 
						|
                self.config.setchannels(cooked)
 | 
						|
                self.inited_nchannels = 1
 | 
						|
                break
 | 
						|
        else:
 | 
						|
            raise error, 'bad # of channels'
 | 
						|
 | 
						|
    def writeframes(self, data):
 | 
						|
        if not (self.inited_outrate and self.inited_nchannels):
 | 
						|
            raise error, 'params not specified'
 | 
						|
        if not self.port:
 | 
						|
            import al, AL
 | 
						|
            self.port = al.openport('Python', 'w', self.config)
 | 
						|
            self.oldparams = self.params[:]
 | 
						|
            al.getparams(AL.DEFAULT_DEVICE, self.oldparams)
 | 
						|
            al.setparams(AL.DEFAULT_DEVICE, self.params)
 | 
						|
        if self.converter:
 | 
						|
            data = self.converter(data)
 | 
						|
        self.port.writesamps(data)
 | 
						|
 | 
						|
    def getfilled(self):
 | 
						|
        if self.port:
 | 
						|
            return self.port.getfilled()
 | 
						|
        else:
 | 
						|
            return 0
 | 
						|
 | 
						|
    def getfillable(self):
 | 
						|
        if self.port:
 | 
						|
            return self.port.getfillable()
 | 
						|
        else:
 | 
						|
            return self.config.getqueuesize()
 | 
						|
 | 
						|
    # private methods
 | 
						|
##      if 0: access *: private
 | 
						|
 | 
						|
    def ulaw2lin(self, data):
 | 
						|
        import audioop
 | 
						|
        return audioop.ulaw2lin(data, 2)
 | 
						|
 | 
						|
class Play_Audio_sun:
 | 
						|
##      if 0: access outrate, sampwidth, nchannels, inited_outrate, inited_width, \
 | 
						|
##                inited_nchannels, converter: private
 | 
						|
 | 
						|
    def __init__(self):
 | 
						|
        self.outrate = 0
 | 
						|
        self.sampwidth = 0
 | 
						|
        self.nchannels = 0
 | 
						|
        self.inited_outrate = 0
 | 
						|
        self.inited_width = 0
 | 
						|
        self.inited_nchannels = 0
 | 
						|
        self.converter = None
 | 
						|
        self.port = None
 | 
						|
        return
 | 
						|
 | 
						|
    def __del__(self):
 | 
						|
        self.stop()
 | 
						|
 | 
						|
    def setoutrate(self, rate):
 | 
						|
        self.outrate = rate
 | 
						|
        self.inited_outrate = 1
 | 
						|
 | 
						|
    def setsampwidth(self, width):
 | 
						|
        self.sampwidth = width
 | 
						|
        self.inited_width = 1
 | 
						|
 | 
						|
    def setnchannels(self, nchannels):
 | 
						|
        self.nchannels = nchannels
 | 
						|
        self.inited_nchannels = 1
 | 
						|
 | 
						|
    def writeframes(self, data):
 | 
						|
        if not (self.inited_outrate and self.inited_width and self.inited_nchannels):
 | 
						|
            raise error, 'params not specified'
 | 
						|
        if not self.port:
 | 
						|
            import sunaudiodev, SUNAUDIODEV
 | 
						|
            self.port = sunaudiodev.open('w')
 | 
						|
            info = self.port.getinfo()
 | 
						|
            info.o_sample_rate = self.outrate
 | 
						|
            info.o_channels = self.nchannels
 | 
						|
            if self.sampwidth == 0:
 | 
						|
                info.o_precision = 8
 | 
						|
                self.o_encoding = SUNAUDIODEV.ENCODING_ULAW
 | 
						|
                # XXX Hack, hack -- leave defaults
 | 
						|
            else:
 | 
						|
                info.o_precision = 8 * self.sampwidth
 | 
						|
                info.o_encoding = SUNAUDIODEV.ENCODING_LINEAR
 | 
						|
                self.port.setinfo(info)
 | 
						|
        if self.converter:
 | 
						|
            data = self.converter(data)
 | 
						|
        self.port.write(data)
 | 
						|
 | 
						|
    def wait(self):
 | 
						|
        if not self.port:
 | 
						|
            return
 | 
						|
        self.port.drain()
 | 
						|
        self.stop()
 | 
						|
 | 
						|
    def stop(self):
 | 
						|
        if self.port:
 | 
						|
            self.port.flush()
 | 
						|
            self.port.close()
 | 
						|
            self.port = None
 | 
						|
 | 
						|
    def getfilled(self):
 | 
						|
        if self.port:
 | 
						|
            return self.port.obufcount()
 | 
						|
        else:
 | 
						|
            return 0
 | 
						|
 | 
						|
##    # Nobody remembers what this method does, and it's broken. :-(
 | 
						|
##    def getfillable(self):
 | 
						|
##        return BUFFERSIZE - self.getfilled()
 | 
						|
 | 
						|
def AudioDev():
 | 
						|
    # Dynamically try to import and use a platform specific module.
 | 
						|
    try:
 | 
						|
        import al
 | 
						|
    except ImportError:
 | 
						|
        try:
 | 
						|
            import sunaudiodev
 | 
						|
            return Play_Audio_sun()
 | 
						|
        except ImportError:
 | 
						|
            try:
 | 
						|
                import Audio_mac
 | 
						|
            except ImportError:
 | 
						|
                raise error, 'no audio device'
 | 
						|
            else:
 | 
						|
                return Audio_mac.Play_Audio_mac()
 | 
						|
    else:
 | 
						|
        return Play_Audio_sgi()
 | 
						|
 | 
						|
def test(fn = None):
 | 
						|
    import sys
 | 
						|
    if sys.argv[1:]:
 | 
						|
        fn = sys.argv[1]
 | 
						|
    else:
 | 
						|
        fn = 'f:just samples:just.aif'
 | 
						|
    import aifc
 | 
						|
    af = aifc.open(fn, 'r')
 | 
						|
    print fn, af.getparams()
 | 
						|
    p = AudioDev()
 | 
						|
    p.setoutrate(af.getframerate())
 | 
						|
    p.setsampwidth(af.getsampwidth())
 | 
						|
    p.setnchannels(af.getnchannels())
 | 
						|
    BUFSIZ = af.getframerate()/af.getsampwidth()/af.getnchannels()
 | 
						|
    while 1:
 | 
						|
        data = af.readframes(BUFSIZ)
 | 
						|
        if not data: break
 | 
						|
        print len(data)
 | 
						|
        p.writeframes(data)
 | 
						|
    p.wait()
 | 
						|
 | 
						|
if __name__ == '__main__':
 | 
						|
    test()
 |