aboutsummaryrefslogtreecommitdiff
path: root/broadcast.py
diff options
context:
space:
mode:
Diffstat (limited to 'broadcast.py')
-rw-r--r--broadcast.py62
1 files changed, 46 insertions, 16 deletions
diff --git a/broadcast.py b/broadcast.py
index 5606137..3b163b1 100644
--- a/broadcast.py
+++ b/broadcast.py
@@ -24,12 +24,13 @@ parser.add_option('-P', '--play-async', dest='play_async', action='store_true',
parser.add_option('-D', '--duration', dest='duration', type='float', help='How long to play this note for')
parser.add_option('-V', '--volume', dest='volume', type='int', help='Master volume (0-255)')
parser.add_option('-s', '--silence', dest='silence', action='store_true', help='Instruct all clients to stop playing any active tones')
+parser.add_option('-S', '--seek', dest='seek', type='float', help='Start time in seconds (scaled by --factor)')
parser.add_option('-f', '--factor', dest='factor', type='float', help='Rescale time by this factor (0<f<1 are faster; 0.5 is twice the speed, 2 is half)')
parser.add_option('-r', '--route', dest='routes', action='append', help='Add a routing directive (see --route-help)')
parser.add_option('-v', '--verbose', dest='verbose', action='store_true', help='Be verbose; dump events and actual time (can slow down performance!)')
parser.add_option('-W', '--wait-time', dest='wait_time', type='float', help='How long to wait for clients to initially respond (delays all broadcasts)')
parser.add_option('--help-routes', dest='help_routes', action='store_true', help='Show help about routing directives')
-parser.set_defaults(routes=[], random=0.0, rand_low=80, rand_high=2000, live=None, factor=1.0, duration=1.0, volume=255, wait_time=0.25, play=[], transpose=0)
+parser.set_defaults(routes=[], random=0.0, rand_low=80, rand_high=2000, live=None, factor=1.0, duration=1.0, volume=255, wait_time=0.25, play=[], transpose=0, seek=0.0)
options, args = parser.parse_args()
if options.help_routes:
@@ -70,6 +71,8 @@ try:
except socket.timeout:
pass
+print len(clients), 'detected clients'
+
print 'Clients:'
for cl in clients:
print cl,
@@ -86,10 +89,10 @@ for cl in clients:
uid_groups.setdefault(uid, []).append(cl)
type_groups.setdefault(tp, []).append(cl)
if options.test:
- s.sendto(str(Packet(CMD.PLAY, 0, 250000, 440, 255)), cl)
+ s.sendto(str(Packet(CMD.PLAY, 0, 250000, 440, options.volume)), cl)
if not options.sync_test:
time.sleep(0.25)
- s.sendto(str(Packet(CMD.PLAY, 0, 250000, 880, 255)), cl)
+ s.sendto(str(Packet(CMD.PLAY, 0, 250000, 880, options.volume)), cl)
if options.quit:
s.sendto(str(Packet(CMD.QUIT)), cl)
if options.silence:
@@ -132,7 +135,7 @@ if options.live or options.list_live:
exit()
seq = sequencer.SequencerRead(sequencer_resolution=120)
client_set = set(clients)
- active_set = {} # note (pitch) -> client
+ active_set = {} # note (pitch) -> [client]
deferred_set = set() # pitches held due to sustain
sustain_status = False
client, _, port = options.live.partition(',')
@@ -154,7 +157,7 @@ if options.live or options.list_live:
elif ev.type == S.SND_SEQ_EVENT_CONTROLLER:
event = midi.ControlChangeEvent(channel = ev.data.control.channel, control = ev.data.control.param, value = ev.data.control.value)
elif ev.type == S.SND_SEQ_EVENT_PGMCHANGE:
- event = midi.ProgramChangeEvent(channel = ev.data.control.channel, pitch = ev.data.control.value)
+ event = midi.ProgramChangeEvent(channel = ev.data.control.channel, value = ev.data.control.value)
elif ev.type == S.SND_SEQ_EVENT_PITCHBEND:
event = midi.PitchWheelEvent(channel = ev.data.control.channel, pitch = ev.data.control.value)
elif options.verbose:
@@ -169,34 +172,36 @@ if options.live or options.list_live:
if event.pitch in active_set:
if sustain_status:
deferred_set.discard(event.pitch)
- else:
- print 'WARNING: Note already activated: %r'%(event.pitch,),
- continue
- inactive_set = client_set - set(active_set.values())
+ inactive_set = client_set - set(sum(active_set.values(), []))
if not inactive_set:
print 'WARNING: Out of clients to do note %r; dropped'%(event.pitch,)
continue
- cli = random.choice(list(inactive_set))
+ cli = sorted(inactive_set)[0]
s.sendto(str(Packet(CMD.PLAY, 65535, 0, int(440.0 * 2**((event.pitch-69)/12.0)), 2*event.velocity)), cli)
- active_set[event.pitch] = cli
+ active_set.setdefault(event.pitch, []).append(cli)
+ if options.verbose:
+ print 'LIVE:', event.pitch, '+ =>', active_set[event.pitch]
elif isinstance(event, midi.NoteOffEvent):
- if event.pitch not in active_set:
+ if event.pitch not in active_set or not active_set[event.pitch]:
print 'WARNING: Deactivating inactive note %r'%(event.pitch,)
continue
if sustain_status:
deferred_set.add(event.pitch)
continue
- s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), active_set[event.pitch])
- del active_set[event.pitch]
+ cli = active_set[event.pitch].pop()
+ s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), cli)
+ if options.verbose:
+ print 'LIVE:', event.pitch, '- =>', active_set[event.pitch]
elif isinstance(event, midi.ControlChangeEvent):
if event.control == 64:
sustain_status = (event.value >= 64)
if not sustain_status:
for pitch in deferred_set:
- if pitch not in active_set:
+ if pitch not in active_set or not active_set[pitch]:
print 'WARNING: Attempted deferred removal of inactive note %r'%(pitch,)
continue
- s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), active_set[pitch])
+ for cli in active_set[pitch]:
+ s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), cli)
del active_set[pitch]
deferred_set.clear()
@@ -315,7 +320,20 @@ for fname in args:
for route in routeset.routes:
print route
+<<<<<<< HEAD
+class NSThread(threading.Thread):
+ def drop_missed(self):
+ nsq, cl = self._Thread__args
+ cnt = 0
+ while nsq and float(nsq[0].get('time'))*factor < time.time() - BASETIME:
+ nsq.pop(0)
+ cnt += 1
+ if options.verbose:
+ print self, 'dropped', cnt, 'notes due to miss'
+ self._Thread__args = (nsq, cl)
+=======
class NSThread(threading.Thread):
+>>>>>>> 7c9661d892f6145d123d91924b720d9d87b69502
def wait_for(self, t):
if t <= 0:
return
@@ -347,6 +365,17 @@ for fname in args:
print 'Playback threads:'
for thr in threads:
print thr._Thread__args[1]
+<<<<<<< HEAD
+
+ BASETIME = time.time() - (options.seek*factor)
+ if options.seek > 0:
+ for thr in threads:
+ thr.drop_missed()
+ for thr in threads:
+ thr.start()
+ for thr in threads:
+ thr.join()
+=======
BASETIME = time.time()
for thr in threads:
@@ -354,4 +383,5 @@ for fname in args:
for thr in threads:
thr.join()
+>>>>>>> 7c9661d892f6145d123d91924b720d9d87b69502
print fname, ': Done!'