import pyaudio import socket import optparse import tarfile import wave import cStringIO as StringIO import array import time from packet import Packet, CMD, stoi, OBLIGATE_POLYPHONE parser = optparse.OptionParser() parser.add_option('-t', '--test', dest='test', action='store_true', help='As a test, play all samples then exit') parser.add_option('-v', '--verbose', dest='verbose', action='store_true', help='Be verbose') parser.add_option('-V', '--volume', dest='volume', type='float', default=1.0, help='Set the volume factor (nominally [0.0, 1.0], but >1.0 can be used to amplify with possible distortion)') parser.add_option('-r', '--rate', dest='rate', type='int', default=44100, help='Audio sample rate for output and of input files') parser.add_option('-u', '--uid', dest='uid', default='', help='User identifier of this client') parser.add_option('-p', '--port', dest='port', default=13676, type='int', help='UDP port to listen on') parser.add_option('--repeat', dest='repeat', action='store_true', help='If a note plays longer than a sample length, keep playing the sample') parser.add_option('--cut', dest='cut', action='store_true', help='If a note ends within a sample, stop playing that sample immediately') parser.add_option('-n', '--max-voices', dest='max_voices', default=-1, type='int', help='Only support this many notes playing simultaneously (earlier ones get dropped)') options, args = parser.parse_args() MAX = 0x7fffffff MIN = -0x80000000 IDENT = 'DRUM' if not args: print 'Need at least one drumpack (.tar.bz2) as an argument!' parser.print_usage() exit(1) DRUMS = {} for fname in args: print 'Reading', fname, '...' tf = tarfile.open(fname, 'r') names = tf.getnames() for nm in names: if not (nm.endswith('.wav') or nm.endswith('.raw')) or len(nm) < 5: continue frq = int(nm[:-4]) if options.verbose: print '\tLoading frq', frq, '...' fo = tf.extractfile(nm) if nm.endswith('.wav'): wf = wave.open(fo) if wf.getnchannels() != 1: print '\t\tWARNING: Channel count wrong: got', wf.getnchannels(), 'expecting 1' if wf.getsampwidth() != 4: print '\t\tWARNING: Sample width wrong: got', wf.getsampwidth(), 'expecting 4' if wf.getframerate() != options.rate: print '\t\tWARNING: Rate wrong: got', wf.getframerate(), 'expecting', options.rate, '(maybe try setting -r?)' frames = wf.getnframes() data = '' while len(data) < wf.getsampwidth() * frames: data += wf.readframes(frames - len(data) / wf.getsampwidth()) elif nm.endswith('.raw'): data = fo.read() frames = len(data) / 4 if options.verbose: print '\t\tData:', frames, 'samples,', len(data), 'bytes' if frq in DRUMS: print '\t\tWARNING: frequency', frq, 'already in map, overwriting...' DRUMS[frq] = data if options.verbose: print len(DRUMS), 'sounds loaded' PLAYING = [] class SampleReader(object): def __init__(self, buf, total, amp): self.buf = buf self.total = total self.cur = 0 self.amp = amp def read(self, bytes): if self.cur >= self.total: return '' res = '' while self.cur < self.total and len(res) < bytes: data = self.buf[self.cur % len(self.buf):self.cur % len(self.buf) + bytes - len(res)] self.cur += len(data) res += data arr = array.array('i') arr.fromstring(res) for i in range(len(arr)): arr[i] = int(arr[i] * self.amp) return arr.tostring() def __repr__(self): return ''%(len(self.buf), self.cur, self.total, self.amp) def gen_data(data, frames, tm, status): fdata = array.array('l', [0] * frames) torem = set() for src in set(PLAYING): buf = src.read(frames * 4) if not buf: torem.add(src) continue samps = array.array('i') samps.fromstring(buf) if len(samps) < frames: samps.extend([0] * (frames - len(samps))) for i in range(frames): fdata[i] += samps[i] for src in torem: PLAYING.remove(src) for i in range(frames): fdata[i] = max(MIN, min(MAX, fdata[i])) fdata = array.array('i', fdata) return (fdata.tostring(), pyaudio.paContinue) pa = pyaudio.PyAudio() stream = pa.open(rate=options.rate, channels=1, format=pyaudio.paInt32, output=True, frames_per_buffer=64, stream_callback=gen_data) if options.test: for frq in sorted(DRUMS.keys()): print 'Current playing:', PLAYING print 'Playing:', frq data = DRUMS[frq] PLAYING.append(SampleReader(data, len(data), options.volume)) time.sleep(len(data) / (4.0 * options.rate)) print 'Done' exit() sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) sock.bind(('', options.port)) #signal.signal(signal.SIGALRM, sigalrm) while True: data = '' while not data: try: data, cli = sock.recvfrom(4096) except socket.error: pass pkt = Packet.FromStr(data) print 'From', cli, 'command', pkt.cmd if pkt.cmd == CMD.KA: pass elif pkt.cmd == CMD.PING: sock.sendto(data, cli) elif pkt.cmd == CMD.QUIT: break elif pkt.cmd == CMD.PLAY: frq = pkt.data[2] if frq not in DRUMS: print 'WARNING: No such instrument', frq, ', ignoring...' continue rdata = DRUMS[frq] rframes = len(rdata) / 4 dur = pkt.data[0]+pkt.data[1]/1000000.0 dframes = int(dur * options.rate) if not options.repeat: dframes = max(dframes, rframes) if not options.cut: dframes = rframes * ((dframes + rframes - 1) / rframes) amp = max(min(options.volume * pkt.as_float(3), 1.0), 0.0) PLAYING.append(SampleReader(rdata, dframes * 4, amp)) if options.max_voices >= 0: while len(PLAYING) > options.max_voices: PLAYING.pop(0) #signal.setitimer(signal.ITIMER_REAL, dur) elif pkt.cmd == CMD.CAPS: data = [0] * 8 data[0] = OBLIGATE_POLYPHONE data[1] = stoi(IDENT) for i in xrange(len(options.uid)/4 + 1): data[i+2] = stoi(options.uid[4*i:4*(i+1)]) sock.sendto(str(Packet(CMD.CAPS, *data)), cli) # elif pkt.cmd == CMD.PCM: # fdata = data[4:] # fdata = struct.pack('16i', *[i<<16 for i in struct.unpack('16h', fdata)]) # QUEUED_PCM += fdata # print 'Now', len(QUEUED_PCM) / 4.0, 'frames queued' else: print 'Unknown cmd', pkt.cmd