From 62e18f0871161cc0481e700aacc5fdc7d2e32a0d Mon Sep 17 00:00:00 2001 From: Grissess Date: Thu, 20 Aug 2015 23:06:11 -0400 Subject: Partial commit (sorry :( ) --- broadcast.py | 4 ++- mkarduino.py | 29 +++++++++++++++ shiv.py | 44 +++++++++++++++++++++++ voice.py | 116 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 4 files changed, 192 insertions(+), 1 deletion(-) create mode 100644 mkarduino.py create mode 100644 shiv.py create mode 100644 voice.py diff --git a/broadcast.py b/broadcast.py index f0b987b..70e6e98 100644 --- a/broadcast.py +++ b/broadcast.py @@ -142,6 +142,8 @@ if options.live or options.list_live: ev = S.event_input(seq.client) event = None if ev: + if options.verbose: + print 'SEQ:', ev if ev < 0: seq._error(ev) if ev.type == S.SND_SEQ_EVENT_NOTEON: @@ -159,7 +161,7 @@ if options.live or options.list_live: continue if event is not None: if isinstance(event, midi.NoteOnEvent) and event.velocity == 0: - ev.__class__ = midi.NoteOffEvent + event.__class__ = midi.NoteOffEvent if options.verbose: print 'EVENT:', event if isinstance(event, midi.NoteOnEvent): diff --git a/mkarduino.py b/mkarduino.py new file mode 100644 index 0000000..39926c5 --- /dev/null +++ b/mkarduino.py @@ -0,0 +1,29 @@ +#IV to arduino array computer + +import xml.etree.ElementTree as ET +import sys + +iv = ET.parse(sys.argv[1]).getroot() + +streams = iv.findall('./streams/stream[@type="ns"]') +if len(streams) > 3: + print 'WARNING: Too many streams' + +for i in xrange(min(3, len(streams))): + stream = streams[i] + notes = stream.findall('note') + +# First, the header + sys.stdout.write('const uint16_t track%d[] PROGMEM = {\n'%(i,)) + +# For the first note, write out the delay needed to get there + if notes[0].get('time') > 0: + sys.stdout.write('%d, 0,\n'%(int(float(notes[0].get('time'))*1000),)) + + for idx, note in enumerate(notes): + sys.stdout.write('%d, FREQ(%d),\n'%(int(float(note.get('dur'))*1000), int(440.0 * 2**((int(note.get('pitch'))-69)/12.0)))) + if idx < len(notes)-1 and float(note.get('time'))+float(note.get('dur')) < float(notes[idx+1].get('time')): + sys.stdout.write('%d, 0,\n'%(int(1000*(float(notes[idx+1].get('time')) - (float(note.get('time')) + float(note.get('dur'))))),)) + +# Finish up the stream + sys.stdout.write('};\n\n') diff --git a/shiv.py b/shiv.py new file mode 100644 index 0000000..96d8ea2 --- /dev/null +++ b/shiv.py @@ -0,0 +1,44 @@ +# IV file viewer + +import xml.etree.ElementTree as ET +import optparse +import sys + +parser = optparse.OptionParser() +parser.add_option('-n', '--number', dest='number', action='store_true', help='Show number of tracks') +parser.add_option('-g', '--groups', dest='groups', action='store_true', help='Show group names') +parser.add_option('-N', '--notes', dest='notes', action='store_true', help='Show number of notes') +parser.add_option('-m', '--meta', dest='meta', action='store_true', help='Show meta track information') +parser.add_option('-h', '--histogram', dest='histogram', action='store_true', help='Show a histogram distribution of pitches') +parser.add_option('-H', '--histogram-tracks', dest='histogram_tracks', action='store_true', help='Show a histogram distribution of pitches per track') +parser.add_option('-d', '--duration', dest='duration', action='store_true', help='Show the duration of the piece') +parser.add_option('-D', '--duty-cycle', dest='duty_cycle', action='store_true', help='Show the duration of the notes within tracks, and as a percentage of the piece duration') + +parser.add_option('-a', '--almost-all', dest='almost_all', action='store_true', help='Show useful information') +parser.add_option('-A', '--all', dest='all', action='store_true', help='Show everything') + +options, args = parser.parse_args() + +if options.almost_all or options.all: + options.number = True + options.groups = True + options.notes = True + options.histogram = True + options.duration = True + if options.all: + options.meta = True + options.histogram_tracks= True + options.duty_cycle = True + +for fname in args: + try: + iv = ET.parse(fname).getroot() + except IOError: + import traceback + traceback.print_exc() + print 'Bad file :', fname, ', skipping...' + continue + print + print 'File :', fname + print '\t' + diff --git a/voice.py b/voice.py new file mode 100644 index 0000000..eb0b43f --- /dev/null +++ b/voice.py @@ -0,0 +1,116 @@ +''' +voice -- Voices + +A voice is a simple, singular unit of sound generation that encompasses the following +properties: +-A *generator*: some function that generates a waveform. As an input, it receives theta, + the phase of the signal it is to generate (in [0, 2pi)) and, as an output, it produces + the sample at that point, a normalized amplitude value in [-1, 1]. +-An *envelope*: a function that receives a boolean (the status of whether or not a note + is playing now) and the change in time, and outputs a factor in [0, 1] that represents + a modification to the volume of the generator (pre-output mix). +All of these functions may internally store state or other data, usually by being +implemented as a class with a __call__ method. + +Voices are meant to generate audio data. This can be done in a number of ways, least to +most abstracted: +-A sample at a certain phase (theta) may be gotten from the generator; this can be done + by calling the voice outright; +-A set of samples can be generated via the .samples() method, which receives the number + of samples to generate and the phase velocity (a function of the sample rate and the + desired frequency of the waveform's period; this can be calculated using the static + method .phase_vel()); +-Audio data with enveloping can be generated using the .data() method, which calls the + envelope function as if the note is depressed at the given phase velocity; if the + freq is specified as None, then the note is treated as released. Note that + this will often be necessary for envelopes, as many of them are stateful (as they + depend on the first derivative of time). Also, at this level, the Voice will maintain + some state (namely, the phase at the end of generation) which will ensure (C0) smooth + transitions between already smooth generator functions, even if the frequency changes. +-Finally, a pyaudio-compatible stream callback can be provided with .pyaudio_scb(), a + method that returns a function that arranges to call .data() with the appropriate values. + The freq input to .data() will be taken from the .freq member of the voice in a possibly + non-atomic manner. +''' + +import math +import pyaudio +import struct + +def norm_theta(theta): + return theta % (2*math.pi) + +def norm_amp(amp): + return min(1.0, max(-1.0, amp)) + +def theta2lin(theta): + return theta / (2*math.pi) + +def lin2theta(lin): + return lin * 2*math.pi + +class ParamInfo(object): + PT_ANY = 0x0000 + PT_CONST = 0x0001 + PT_SPECIAL = 0x0002 + PT_INT = 0x0100 + PT_FLOAT = 0x0200 + PT_STR = 0x0400 + PT_THETA = 0x0102 + PT_TIME_SEC = 0x0202 + PT_SAMPLES = 0x0302 + PT_REALTIME = 0x0402 + def __init__(self, name, tp=PT_ANY): + self.name = name + self.tp = tp + +class GenInfo(object): + def __init__(self, name, *params): + self.name = name + self.params = list(params) + +class Generator(object): + class __metaclass__(type): + def __init__(self + +class Voice(object): + @classmethod + def register_gen(cls, name, params): + def __init__(self, generator=None, envelope=None): + self.generator = generator or self.DEFAULT_GENERATOR + self.envelope = envelope or self.DEFAULT_ENVELOPE + self.phase = 0 + self.freq = None + def __call__(self, theta): + return norm_amp(self.generator(norm_theta(theta))) + @staticmethod + def phase_vel(freq, samp_rate): + return 2 * math.pi * freq / samp_rate + def samples(self, frames, pvel): + for i in xrange(frames): + yield self(self.phase) + self.phase = norm_theta(self.phase + pvel) + def data(self, frames, freq, samp_rate): + period = 1.0/samp_rate + status = freq is not None + for samp in self.samples(frames, self.phase_vel(freq, samp_rate)): + yield samp * self.envelope(status, period) + def pyaudio_scb(self, rate, fmt=pyaudio.paInt16): + samp_size = pyaudio.get_sample_size(fmt) + maxint = (1 << (8*samp_size)) - 1 + dtype = ['!', 'h', 'i', '!', 'l', '!', '!', '!', 'q'][samp_size] + def __callback(data, frames, time, status, self=self, rate=rate, maxint=maxint, dtype=dtype): + return struct.pack(dtype*frames, *[maxint*int(i) for i in self.data(frames, self.freq, rate)]) + return __callback + +class VMeanMixer(Voice): + def __init__(self, *voices): + self.voices = list(voices) + def __call__(self, theta): + return sum([i(theta)/len(self.voices) for i in self.voices]) + +class VSumMixer(Voice): + def __init__(self, *voices): + self.voices = list(voices) + def __call__(self, theta): + return sum([i(theta) for i in self.voices]) -- cgit v1.2.3-70-g09d2 From aea344e6f7c013dbc0abfe89bad430cf29136f7e Mon Sep 17 00:00:00 2001 From: Grissess Date: Sun, 23 Aug 2015 23:49:11 -0400 Subject: Partial commit (last from cosi02!) --- broadcast.py | 265 +++++++++++++++--------------- mkiv.py | 8 +- piano.py | 529 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ voice.py | 10 +- 4 files changed, 678 insertions(+), 134 deletions(-) create mode 100644 piano.py diff --git a/broadcast.py b/broadcast.py index 70e6e98..2a3e352 100644 --- a/broadcast.py +++ b/broadcast.py @@ -199,155 +199,158 @@ if options.live or options.list_live: del active_set[pitch] deferred_set.clear() -try: - iv = ET.parse(args[0]).getroot() -except IOError: +for fname in args: + try: + iv = ET.parse(fname).getroot() + except IOError: import traceback traceback.print_exc() - print 'Bad file' - exit() + print fname, ': Bad file' + continue -notestreams = iv.findall("./streams/stream[@type='ns']") -groups = set([ns.get('group') for ns in notestreams if 'group' in ns.keys()]) -print len(notestreams), 'notestreams' -print len(clients), 'clients' -print len(groups), 'groups' + notestreams = iv.findall("./streams/stream[@type='ns']") + groups = set([ns.get('group') for ns in notestreams if 'group' in ns.keys()]) + print len(notestreams), 'notestreams' + print len(clients), 'clients' + print len(groups), 'groups' -class Route(object): - def __init__(self, fattr, fvalue, group, excl=False): - if fattr == 'U': - self.map = uid_groups - elif fattr == 'T': - self.map = type_groups - else: - raise ValueError('Not a valid attribute specifier: %r'%(fattr,)) - self.value = fvalue - if group is not None and group not in groups: - raise ValueError('Not a present group: %r'%(group,)) - self.group = group - self.excl = excl - @classmethod - def Parse(cls, s): - fspecs, _, grpspecs = map(lambda x: x.strip(), s.partition('=')) - fpairs = [] - ret = [] - for fspec in [i.strip() for i in fspecs.split(',')]: - fattr, _, fvalue = map(lambda x: x.strip(), fspec.partition(':')) - fpairs.append((fattr, fvalue)) - for part in [i.strip() for i in grpspecs.split(',')]: - for fattr, fvalue in fpairs: - if part[0] == '+': - ret.append(Route(fattr, fvalue, part[1:], False)) - elif part[0] == '-': - ret.append(Route(fattr, fvalue, part[1:], True)) - elif part[0] == '0': - ret.append(Route(fattr, fvalue, None, True)) - else: - raise ValueError('Not an exclusivity: %r'%(part[0],)) - return ret - def Apply(self, cli): - return cli in self.map.get(self.value, []) - def __repr__(self): - return ''%(self.group, ('U' if self.map is uid_groups else 'T'), self.value) + class Route(object): + def __init__(self, fattr, fvalue, group, excl=False): + if fattr == 'U': + self.map = uid_groups + elif fattr == 'T': + self.map = type_groups + else: + raise ValueError('Not a valid attribute specifier: %r'%(fattr,)) + self.value = fvalue + if group is not None and group not in groups: + raise ValueError('Not a present group: %r'%(group,)) + self.group = group + self.excl = excl + @classmethod + def Parse(cls, s): + fspecs, _, grpspecs = map(lambda x: x.strip(), s.partition('=')) + fpairs = [] + ret = [] + for fspec in [i.strip() for i in fspecs.split(',')]: + fattr, _, fvalue = map(lambda x: x.strip(), fspec.partition(':')) + fpairs.append((fattr, fvalue)) + for part in [i.strip() for i in grpspecs.split(',')]: + for fattr, fvalue in fpairs: + if part[0] == '+': + ret.append(Route(fattr, fvalue, part[1:], False)) + elif part[0] == '-': + ret.append(Route(fattr, fvalue, part[1:], True)) + elif part[0] == '0': + ret.append(Route(fattr, fvalue, None, True)) + else: + raise ValueError('Not an exclusivity: %r'%(part[0],)) + return ret + def Apply(self, cli): + return cli in self.map.get(self.value, []) + def __repr__(self): + return ''%(self.group, ('U' if self.map is uid_groups else 'T'), self.value) -class RouteSet(object): - def __init__(self, clis=None): - if clis is None: - clis = clients[:] - self.clients = clis - self.routes = [] - def Route(self, stream): - testset = self.clients[:] - grp = stream.get('group', 'ALL') - if options.verbose: - print 'Routing', grp, '...' - excl = False - for route in self.routes: - if route.group == grp: - if options.verbose: - print '\tMatches route', route - excl = excl or route.excl - matches = filter(lambda x, route=route: route.Apply(x), testset) - if matches: + class RouteSet(object): + def __init__(self, clis=None): + if clis is None: + clis = clients[:] + self.clients = clis + self.routes = [] + def Route(self, stream): + testset = self.clients[:] + grp = stream.get('group', 'ALL') + if options.verbose: + print 'Routing', grp, '...' + excl = False + for route in self.routes: + if route.group == grp: + if options.verbose: + print '\tMatches route', route + excl = excl or route.excl + matches = filter(lambda x, route=route: route.Apply(x), testset) + if matches: + if options.verbose: + print '\tUsing client', matches[0] + self.clients.remove(matches[0]) + return matches[0] if options.verbose: - print '\tUsing client', matches[0] - self.clients.remove(matches[0]) - return matches[0] + print '\tNo matches, moving on...' + if route.group is None: + if options.verbose: + print 'Encountered NULL route, removing from search space...' + toremove = [] + for cli in testset: + if route.Apply(cli): + toremove.append(cli) + for cli in toremove: + if options.verbose: + print '\tRemoving', cli, '...' + testset.remove(cli) + if excl: if options.verbose: - print '\tNo matches, moving on...' - if route.group is None: + print '\tExclusively routed, no route matched.' + return None + if not testset: if options.verbose: - print 'Encountered NULL route, removing from search space...' - toremove = [] - for cli in testset: - if route.Apply(cli): - toremove.append(cli) - for cli in toremove: - if options.verbose: - print '\tRemoving', cli, '...' - testset.remove(cli) - if excl: - if options.verbose: - print '\tExclusively routed, no route matched.' - return None - if not testset: + print '\tOut of clients, no route matched.' + return None + cli = testset[0] + self.clients.remove(cli) if options.verbose: - print '\tOut of clients, no route matched.' - return None - cli = testset[0] - self.clients.remove(cli) - if options.verbose: - print '\tDefault route to', cli - return cli + print '\tDefault route to', cli + return cli -routeset = RouteSet() -for rspec in options.routes: - try: - routeset.routes.extend(Route.Parse(rspec)) - except Exception: - import traceback - traceback.print_exc() + routeset = RouteSet() + for rspec in options.routes: + try: + routeset.routes.extend(Route.Parse(rspec)) + except Exception: + import traceback + traceback.print_exc() -if options.verbose: - print 'All routes:' - for route in routeset.routes: - print route + if options.verbose: + print 'All routes:' + for route in routeset.routes: + print route -class NSThread(threading.Thread): + class NSThread(threading.Thread): def wait_for(self, t): if t <= 0: return time.sleep(t) - def run(self): - nsq, cl = self._Thread__args - for note in nsq: - ttime = float(note.get('time')) - pitch = int(note.get('pitch')) - vel = int(note.get('vel')) - dur = factor*float(note.get('dur')) - while time.time() - BASETIME < factor*ttime: - self.wait_for(factor*ttime - (time.time() - BASETIME)) - s.sendto(str(Packet(CMD.PLAY, int(dur), int((dur*1000000)%1000000), int(440.0 * 2**((pitch-69)/12.0)), vel*2)), cl) - if options.verbose: - print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel - self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) + def run(self): + nsq, cl = self._Thread__args + for note in nsq: + ttime = float(note.get('time')) + pitch = int(note.get('pitch')) + vel = int(note.get('vel')) + dur = factor*float(note.get('dur')) + while time.time() - BASETIME < factor*ttime: + self.wait_for(factor*ttime - (time.time() - BASETIME)) + s.sendto(str(Packet(CMD.PLAY, int(dur), int((dur*1000000)%1000000), int(440.0 * 2**((pitch-69)/12.0)), vel*2)), cl) if options.verbose: - print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' + print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel + self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) + if options.verbose: + print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' -threads = [] -for ns in notestreams: - cli = routeset.Route(ns) - if cli: - nsq = ns.findall('note') - threads.append(NSThread(args=(nsq, cli))) + threads = [] + for ns in notestreams: + cli = routeset.Route(ns) + if cli: + nsq = ns.findall('note') + threads.append(NSThread(args=(nsq, cli))) -if options.verbose: - print 'Playback threads:' + if options.verbose: + print 'Playback threads:' + for thr in threads: + print thr._Thread__args[1] + + BASETIME = time.time() + for thr in threads: + thr.start() for thr in threads: - print thr._Thread__args[1] + thr.join() -BASETIME = time.time() -for thr in threads: - thr.start() -for thr in threads: - thr.join() + print fname, ': Done!' diff --git a/mkiv.py b/mkiv.py index 5f5ea38..1cee6ed 100644 --- a/mkiv.py +++ b/mkiv.py @@ -72,7 +72,13 @@ if options.fuckit: midi.read_midifile = fuckit(midi.read_midifile) for fname in args: - pat = midi.read_midifile(fname) + try: + pat = midi.read_midifile(fname) + except Exception: + import traceback + traceback.print_exc() + print fname, ': Exception occurred, skipping...' + continue if pat is None: print fname, ': Too fucked to continue' continue diff --git a/piano.py b/piano.py new file mode 100644 index 0000000..8c8b9a7 --- /dev/null +++ b/piano.py @@ -0,0 +1,529 @@ +import socket +import sys +import struct +import time +import xml.etree.ElementTree as ET +import threading +import optparse +import random + +from packet import Packet, CMD, itos + +parser = optparse.OptionParser() +parser.add_option('-t', '--test', dest='test', action='store_true', help='Play a test tone (440, 880) on all clients in sequence (the last overlaps with the first of the next)') +parser.add_option('-T', '--sync-test', dest='sync_test', action='store_true', help='Don\'t wait for clients to play tones properly--have them all test tone at the same time') +parser.add_option('-R', '--random', dest='random', type='float', help='Generate random notes at approximately this period') +parser.add_option('--rand-low', dest='rand_low', type='int', help='Low frequency to randomly sample') +parser.add_option('--rand-high', dest='rand_high', type='int', help='High frequency to randomly sample') +parser.add_option('-l', '--live', dest='live', help='Enter live mode (play from a controller in real time), specifying the port to connect to as "client,port"; use just "," to manually subscribe later') +parser.add_option('-L', '--list-live', dest='list_live', action='store_true', help='List all the clients and ports that can be connected to for live performance') +parser.add_option('-q', '--quit', dest='quit', action='store_true', help='Instruct all clients to quit') +parser.add_option('-p', '--play', dest='play', action='append', help='Play a single tone or chord (specified multiple times) on all listening clients (either "midi pitch" or "@frequency")') +parser.add_option('-P', '--play-async', dest='play_async', action='store_true', help='Don\'t wait for the tone to finish using the local clock') +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='How loud to play this note (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('-f', '--factor', dest='factor', type='float', help='Rescale time by this factor (0 0: + while True: + for cl in clients: + s.sendto(str(Packet(CMD.PLAY, int(options.random), int(1000000*(options.random-int(options.random))), random.randint(options.rand_low, options.rand_high), 255)), cl) + time.sleep(options.random) + +if options.live or options.list_live: + import midi + from midi import sequencer + S = sequencer.S + if options.list_live: + print sequencer.SequencerHardware() + exit() + seq = sequencer.SequencerRead(sequencer_resolution=120) + client_set = set(clients) + active_set = {} # note (pitch) -> client + deferred_set = set() # pitches held due to sustain + sustain_status = False + client, _, port = options.live.partition(',') + if client or port: + seq.subscribe_port(client, port) + seq.start_sequencer() + while True: + ev = S.event_input(seq.client) + event = None + if ev: + if options.verbose: + print 'SEQ:', ev + if ev < 0: + seq._error(ev) + if ev.type == S.SND_SEQ_EVENT_NOTEON: + event = midi.NoteOnEvent(channel = ev.data.note.channel, pitch = ev.data.note.note, velocity = ev.data.note.velocity) + elif ev.type == S.SND_SEQ_EVENT_NOTEOFF: + event = midi.NoteOffEvent(channel = ev.data.note.channel, pitch = ev.data.note.note, velocity = ev.data.note.velocity) + 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) + elif ev.type == S.SND_SEQ_EVENT_PITCHBEND: + event = midi.PitchWheelEvent(channel = ev.data.control.channel, pitch = ev.data.control.value) + elif options.verbose: + print 'WARNING: Unparsed event, type %r'%(ev.type,) + continue + if event is not None: + if isinstance(event, midi.NoteOnEvent) and event.velocity == 0: + event.__class__ = midi.NoteOffEvent + if options.verbose: + print 'EVENT:', event + if isinstance(event, midi.NoteOnEvent): + 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()) + if not inactive_set: + print 'WARNING: Out of clients to do note %r; dropped'%(event.pitch,) + continue + cli = random.choice(list(inactive_set)) + 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 + elif isinstance(event, midi.NoteOffEvent): + if event.pitch not in active_set: + 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] + 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: + print 'WARNING: Attempted deferred removal of inactive note %r'%(pitch,) + continue + s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), active_set[pitch]) + del active_set[pitch] + deferred_set.clear() + + + + + +if options.keyboard: + import pygame + import midi + from pygame import * + pygame.init() + size = width , height = 640,360 + screen = pygame.display.set_mode(size) + picture = pygame.image.load("aaaa.png") + surface = pygame.display.get_surface() + pitch = 60 + velocity = 127 + client_set = set(clients) + active_set = {} # note (pitch) -> client + sustain_status = False + sharp = 0 + while True: + surface.blit(picture,(0,0)) + pygame.display.update() + for event in pygame.event.get(): + if event.type == pygame.QUIT: + pygame.quit() + sys.exit() + elif event.type == pygame.KEYDOWN: + if event.key == K_a: + pitch = 60 + if event.key == K_s: + pitch = 62 + if event.key == K_d: + pitch = 64 + if event.key == K_f: + pitch = 65 + if event.key == K_g: + pitch = 67 + if event.key == K_h: + pitch = 69 + if event.key == K_j: + pitch = 71 + if event.key == K_k: + pitch = 72 + if event.key == K_l: + pitch = 74 + if event.key == K_z: + pitch = 48 + if event.key == K_x: + pitch = 50 + if event.key == K_c: + pitch = 52 + if event.key == K_v: + pitch = 53 + if event.key == K_b: + pitch = 55 + if event.key == K_n: + pitch = 57 + if event.key == K_m: + pitch = 59 + if event.key == K_q: + pitch = 76 + if event.key == K_w: + pitch = 77 + if event.key == K_e: + pitch = 79 + if event.key == K_r: + pitch = 81 + if event.key == K_t: + pitch = 83 + if event.key == K_y: + pitch = 84 + if event.key == K_u: + pitch = 86 + if event.key == K_i: + pitch = 88 + if event.key == K_o: + pitch = 89 + if event.key == K_p: + pitch = 91 + if event.key == K_LSHIFT: + sharp = 1 + continue + pitch = pitch + sharp + mevent = midi.NoteOnEvent(channel = 0, pitch = pitch, velocity = velocity) + if mevent.pitch in active_set: + if sustain_status: + deferred_set.discard(mevent.pitch) + else: + print 'WARNING: Note already activated: %r \n'%(mevent.pitch,), + continue + inactive_set = client_set - set(active_set.values()) + if not inactive_set: + print 'WARNING: Out of clients to do note %r; dropped'%(mevent.pitch,) + continue + cli = random.choice(list(inactive_set)) + s.sendto(str(Packet(CMD.PLAY, 65535, 0, int(440.0 * 2**((mevent.pitch-69)/12.0)), 2*mevent.velocity)), cli) + active_set[mevent.pitch] = cli + elif event.type == pygame.KEYUP: + if event.key == K_a: + pitch = 60 + if event.key == K_s: + pitch = 62 + if event.key == K_d: + pitch = 64 + if event.key == K_f: + pitch = 65 + if event.key == K_g: + pitch = 67 + if event.key == K_h: + pitch = 69 + if event.key == K_j: + pitch = 71 + if event.key == K_k: + pitch = 72 + if event.key == K_l: + pitch = 74 + if event.key == K_z: + pitch = 48 + if event.key == K_x: + pitch = 50 + if event.key == K_c: + pitch = 52 + if event.key == K_v: + pitch = 53 + if event.key == K_b: + pitch = 55 + if event.key == K_n: + pitch = 57 + if event.key == K_m: + pitch = 59 + if event.key == K_q: + pitch = 76 + if event.key == K_w: + pitch = 77 + if event.key == K_e: + pitch = 79 + if event.key == K_r: + pitch = 81 + if event.key == K_t: + pitch = 83 + if event.key == K_y: + pitch = 84 + if event.key == K_u: + pitch = 86 + if event.key == K_i: + pitch = 88 + if event.key == K_o: + pitch = 89 + if event.key == K_p: + pitch = 91 + if event.key == K_LSHIFT: + sharp = 0 + continue + mevent = midi.NoteOffEvent(channel = 0, pitch = pitch, velocity = velocity) + if mevent.pitch not in active_set: + print 'WARNING: Deactivating inactive note %r'%(mevent.pitch,) + continue + if sustain_status: + deferred_set.add(mevent.pitch) + continue + s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), active_set[mevent.pitch]) + del active_set[mevent.pitch] + mevent = midi.NoteOffEvent(channel = 0, pitch = pitch + 1, velocity = velocity) + if mevent.pitch not in active_set: + print 'WARNING: Deactivating inactive note %r'%(mevent.pitch,) + continue + if sustain_status: + deferred_set.add(mevent.pitch) + continue + s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), active_set[mevent.pitch]) + del active_set[mevent.pitch] + +for fname in args: + try: + iv = ET.parse(fname).getroot() + except IOError: + import traceback + traceback.print_exc() + print fname, ': Bad file' + continue + + notestreams = iv.findall("./streams/stream[@type='ns']") + groups = set([ns.get('group') for ns in notestreams if 'group' in ns.keys()]) + print len(notestreams), 'notestreams' + print len(clients), 'clients' + print len(groups), 'groups' + + class Route(object): + def __init__(self, fattr, fvalue, group, excl=False): + if fattr == 'U': + self.map = uid_groups + elif fattr == 'T': + self.map = type_groups + else: + raise ValueError('Not a valid attribute specifier: %r'%(fattr,)) + self.value = fvalue + if group is not None and group not in groups: + raise ValueError('Not a present group: %r'%(group,)) + self.group = group + self.excl = excl + @classmethod + def Parse(cls, s): + fspecs, _, grpspecs = map(lambda x: x.strip(), s.partition('=')) + fpairs = [] + ret = [] + for fspec in [i.strip() for i in fspecs.split(',')]: + fattr, _, fvalue = map(lambda x: x.strip(), fspec.partition(':')) + fpairs.append((fattr, fvalue)) + for part in [i.strip() for i in grpspecs.split(',')]: + for fattr, fvalue in fpairs: + if part[0] == '+': + ret.append(Route(fattr, fvalue, part[1:], False)) + elif part[0] == '-': + ret.append(Route(fattr, fvalue, part[1:], True)) + elif part[0] == '0': + ret.append(Route(fattr, fvalue, None, True)) + else: + raise ValueError('Not an exclusivity: %r'%(part[0],)) + return ret + def Apply(self, cli): + return cli in self.map.get(self.value, []) + def __repr__(self): + return ''%(self.group, ('U' if self.map is uid_groups else 'T'), self.value) + + class RouteSet(object): + def __init__(self, clis=None): + if clis is None: + clis = clients[:] + self.clients = clis + self.routes = [] + def Route(self, stream): + testset = self.clients[:] + grp = stream.get('group', 'ALL') + if options.verbose: + print 'Routing', grp, '...' + excl = False + for route in self.routes: + if route.group == grp: + if options.verbose: + print '\tMatches route', route + excl = excl or route.excl + matches = filter(lambda x, route=route: route.Apply(x), testset) + if matches: + if options.verbose: + print '\tUsing client', matches[0] + self.clients.remove(matches[0]) + return matches[0] + if options.verbose: + print '\tNo matches, moving on...' + if route.group is None: + if options.verbose: + print 'Encountered NULL route, removing from search space...' + toremove = [] + for cli in testset: + if route.Apply(cli): + toremove.append(cli) + for cli in toremove: + if options.verbose: + print '\tRemoving', cli, '...' + testset.remove(cli) + if excl: + if options.verbose: + print '\tExclusively routed, no route matched.' + return None + if not testset: + if options.verbose: + print '\tOut of clients, no route matched.' + return None + cli = testset[0] + self.clients.remove(cli) + if options.verbose: + print '\tDefault route to', cli + return cli + + routeset = RouteSet() + for rspec in options.routes: + try: + routeset.routes.extend(Route.Parse(rspec)) + except Exception: + import traceback + traceback.print_exc() + + if options.verbose: + print 'All routes:' + for route in routeset.routes: + print route + + class NSThread(threading.Thread): + def wait_for(self, t): + if t <= 0: + return + time.sleep(t) + def run(self): + nsq, cl = self._Thread__args + for note in nsq: + ttime = float(note.get('time')) + pitch = int(note.get('pitch')) + vel = int(note.get('vel')) + dur = factor*float(note.get('dur')) + while time.time() - BASETIME < factor*ttime: + self.wait_for(factor*ttime - (time.time() - BASETIME)) + s.sendto(str(Packet(CMD.PLAY, int(dur), int((dur*1000000)%1000000), int(440.0 * 2**((pitch-69)/12.0)), vel*2)), cl) + if options.verbose: + print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel + self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) + if options.verbose: + print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' + + threads = [] + for ns in notestreams: + cli = routeset.Route(ns) + if cli: + nsq = ns.findall('note') + threads.append(NSThread(args=(nsq, cli))) + + if options.verbose: + print 'Playback threads:' + for thr in threads: + print thr._Thread__args[1] + + BASETIME = time.time() + for thr in threads: + thr.start() + for thr in threads: + thr.join() + + print fname, ': Done!' diff --git a/voice.py b/voice.py index eb0b43f..14961c1 100644 --- a/voice.py +++ b/voice.py @@ -107,10 +107,16 @@ class VMeanMixer(Voice): def __init__(self, *voices): self.voices = list(voices) def __call__(self, theta): - return sum([i(theta)/len(self.voices) for i in self.voices]) + return norm_amp(sum([i(theta)/len(self.voices) for i in self.voices])) class VSumMixer(Voice): def __init__(self, *voices): self.voices = list(voices) def __call__(self, theta): - return sum([i(theta) for i in self.voices]) + return norm_amp(sum([i(theta) for i in self.voices])) + +class object(object): + def __init__(self): + this_obj = object() + +foo = object() -- cgit v1.2.3-70-g09d2 From 12c016ceb5ca9c120b3c3264f6dafaee03ceb451 Mon Sep 17 00:00:00 2001 From: Graham Northup Date: Sat, 3 Oct 2015 02:58:37 -0400 Subject: Some pygame stuff --- piano.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/piano.py b/piano.py index 8c8b9a7..2c8dd22 100644 --- a/piano.py +++ b/piano.py @@ -211,8 +211,8 @@ if options.keyboard: pygame.init() size = width , height = 640,360 screen = pygame.display.set_mode(size) - picture = pygame.image.load("aaaa.png") - surface = pygame.display.get_surface() + #picture = pygame.image.load("aaaa.png") + #surface = pygame.display.get_surface() pitch = 60 velocity = 127 client_set = set(clients) @@ -220,8 +220,8 @@ if options.keyboard: sustain_status = False sharp = 0 while True: - surface.blit(picture,(0,0)) - pygame.display.update() + #surface.blit(picture,(0,0)) + #pygame.display.update() for event in pygame.event.get(): if event.type == pygame.QUIT: pygame.quit() -- cgit v1.2.3-70-g09d2 From bc04100457505823140dbfe1df5d22c22e1435b9 Mon Sep 17 00:00:00 2001 From: Grissess Date: Thu, 20 Aug 2015 23:06:11 -0400 Subject: Partial commit (sorry :( ) --- broadcast.py | 4 ++- mkarduino.py | 29 +++++++++++++++ shiv.py | 44 +++++++++++++++++++++++ voice.py | 116 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 4 files changed, 192 insertions(+), 1 deletion(-) create mode 100644 mkarduino.py create mode 100644 shiv.py create mode 100644 voice.py diff --git a/broadcast.py b/broadcast.py index 2903e7a..b494645 100644 --- a/broadcast.py +++ b/broadcast.py @@ -143,6 +143,8 @@ if options.live or options.list_live: ev = S.event_input(seq.client) event = None if ev: + if options.verbose: + print 'SEQ:', ev if ev < 0: seq._error(ev) if ev.type == S.SND_SEQ_EVENT_NOTEON: @@ -160,7 +162,7 @@ if options.live or options.list_live: continue if event is not None: if isinstance(event, midi.NoteOnEvent) and event.velocity == 0: - ev.__class__ = midi.NoteOffEvent + event.__class__ = midi.NoteOffEvent if options.verbose: print 'EVENT:', event if isinstance(event, midi.NoteOnEvent): diff --git a/mkarduino.py b/mkarduino.py new file mode 100644 index 0000000..39926c5 --- /dev/null +++ b/mkarduino.py @@ -0,0 +1,29 @@ +#IV to arduino array computer + +import xml.etree.ElementTree as ET +import sys + +iv = ET.parse(sys.argv[1]).getroot() + +streams = iv.findall('./streams/stream[@type="ns"]') +if len(streams) > 3: + print 'WARNING: Too many streams' + +for i in xrange(min(3, len(streams))): + stream = streams[i] + notes = stream.findall('note') + +# First, the header + sys.stdout.write('const uint16_t track%d[] PROGMEM = {\n'%(i,)) + +# For the first note, write out the delay needed to get there + if notes[0].get('time') > 0: + sys.stdout.write('%d, 0,\n'%(int(float(notes[0].get('time'))*1000),)) + + for idx, note in enumerate(notes): + sys.stdout.write('%d, FREQ(%d),\n'%(int(float(note.get('dur'))*1000), int(440.0 * 2**((int(note.get('pitch'))-69)/12.0)))) + if idx < len(notes)-1 and float(note.get('time'))+float(note.get('dur')) < float(notes[idx+1].get('time')): + sys.stdout.write('%d, 0,\n'%(int(1000*(float(notes[idx+1].get('time')) - (float(note.get('time')) + float(note.get('dur'))))),)) + +# Finish up the stream + sys.stdout.write('};\n\n') diff --git a/shiv.py b/shiv.py new file mode 100644 index 0000000..96d8ea2 --- /dev/null +++ b/shiv.py @@ -0,0 +1,44 @@ +# IV file viewer + +import xml.etree.ElementTree as ET +import optparse +import sys + +parser = optparse.OptionParser() +parser.add_option('-n', '--number', dest='number', action='store_true', help='Show number of tracks') +parser.add_option('-g', '--groups', dest='groups', action='store_true', help='Show group names') +parser.add_option('-N', '--notes', dest='notes', action='store_true', help='Show number of notes') +parser.add_option('-m', '--meta', dest='meta', action='store_true', help='Show meta track information') +parser.add_option('-h', '--histogram', dest='histogram', action='store_true', help='Show a histogram distribution of pitches') +parser.add_option('-H', '--histogram-tracks', dest='histogram_tracks', action='store_true', help='Show a histogram distribution of pitches per track') +parser.add_option('-d', '--duration', dest='duration', action='store_true', help='Show the duration of the piece') +parser.add_option('-D', '--duty-cycle', dest='duty_cycle', action='store_true', help='Show the duration of the notes within tracks, and as a percentage of the piece duration') + +parser.add_option('-a', '--almost-all', dest='almost_all', action='store_true', help='Show useful information') +parser.add_option('-A', '--all', dest='all', action='store_true', help='Show everything') + +options, args = parser.parse_args() + +if options.almost_all or options.all: + options.number = True + options.groups = True + options.notes = True + options.histogram = True + options.duration = True + if options.all: + options.meta = True + options.histogram_tracks= True + options.duty_cycle = True + +for fname in args: + try: + iv = ET.parse(fname).getroot() + except IOError: + import traceback + traceback.print_exc() + print 'Bad file :', fname, ', skipping...' + continue + print + print 'File :', fname + print '\t' + diff --git a/voice.py b/voice.py new file mode 100644 index 0000000..eb0b43f --- /dev/null +++ b/voice.py @@ -0,0 +1,116 @@ +''' +voice -- Voices + +A voice is a simple, singular unit of sound generation that encompasses the following +properties: +-A *generator*: some function that generates a waveform. As an input, it receives theta, + the phase of the signal it is to generate (in [0, 2pi)) and, as an output, it produces + the sample at that point, a normalized amplitude value in [-1, 1]. +-An *envelope*: a function that receives a boolean (the status of whether or not a note + is playing now) and the change in time, and outputs a factor in [0, 1] that represents + a modification to the volume of the generator (pre-output mix). +All of these functions may internally store state or other data, usually by being +implemented as a class with a __call__ method. + +Voices are meant to generate audio data. This can be done in a number of ways, least to +most abstracted: +-A sample at a certain phase (theta) may be gotten from the generator; this can be done + by calling the voice outright; +-A set of samples can be generated via the .samples() method, which receives the number + of samples to generate and the phase velocity (a function of the sample rate and the + desired frequency of the waveform's period; this can be calculated using the static + method .phase_vel()); +-Audio data with enveloping can be generated using the .data() method, which calls the + envelope function as if the note is depressed at the given phase velocity; if the + freq is specified as None, then the note is treated as released. Note that + this will often be necessary for envelopes, as many of them are stateful (as they + depend on the first derivative of time). Also, at this level, the Voice will maintain + some state (namely, the phase at the end of generation) which will ensure (C0) smooth + transitions between already smooth generator functions, even if the frequency changes. +-Finally, a pyaudio-compatible stream callback can be provided with .pyaudio_scb(), a + method that returns a function that arranges to call .data() with the appropriate values. + The freq input to .data() will be taken from the .freq member of the voice in a possibly + non-atomic manner. +''' + +import math +import pyaudio +import struct + +def norm_theta(theta): + return theta % (2*math.pi) + +def norm_amp(amp): + return min(1.0, max(-1.0, amp)) + +def theta2lin(theta): + return theta / (2*math.pi) + +def lin2theta(lin): + return lin * 2*math.pi + +class ParamInfo(object): + PT_ANY = 0x0000 + PT_CONST = 0x0001 + PT_SPECIAL = 0x0002 + PT_INT = 0x0100 + PT_FLOAT = 0x0200 + PT_STR = 0x0400 + PT_THETA = 0x0102 + PT_TIME_SEC = 0x0202 + PT_SAMPLES = 0x0302 + PT_REALTIME = 0x0402 + def __init__(self, name, tp=PT_ANY): + self.name = name + self.tp = tp + +class GenInfo(object): + def __init__(self, name, *params): + self.name = name + self.params = list(params) + +class Generator(object): + class __metaclass__(type): + def __init__(self + +class Voice(object): + @classmethod + def register_gen(cls, name, params): + def __init__(self, generator=None, envelope=None): + self.generator = generator or self.DEFAULT_GENERATOR + self.envelope = envelope or self.DEFAULT_ENVELOPE + self.phase = 0 + self.freq = None + def __call__(self, theta): + return norm_amp(self.generator(norm_theta(theta))) + @staticmethod + def phase_vel(freq, samp_rate): + return 2 * math.pi * freq / samp_rate + def samples(self, frames, pvel): + for i in xrange(frames): + yield self(self.phase) + self.phase = norm_theta(self.phase + pvel) + def data(self, frames, freq, samp_rate): + period = 1.0/samp_rate + status = freq is not None + for samp in self.samples(frames, self.phase_vel(freq, samp_rate)): + yield samp * self.envelope(status, period) + def pyaudio_scb(self, rate, fmt=pyaudio.paInt16): + samp_size = pyaudio.get_sample_size(fmt) + maxint = (1 << (8*samp_size)) - 1 + dtype = ['!', 'h', 'i', '!', 'l', '!', '!', '!', 'q'][samp_size] + def __callback(data, frames, time, status, self=self, rate=rate, maxint=maxint, dtype=dtype): + return struct.pack(dtype*frames, *[maxint*int(i) for i in self.data(frames, self.freq, rate)]) + return __callback + +class VMeanMixer(Voice): + def __init__(self, *voices): + self.voices = list(voices) + def __call__(self, theta): + return sum([i(theta)/len(self.voices) for i in self.voices]) + +class VSumMixer(Voice): + def __init__(self, *voices): + self.voices = list(voices) + def __call__(self, theta): + return sum([i(theta) for i in self.voices]) -- cgit v1.2.3-70-g09d2 From 9f9ec226c5dbcbe11a9f4b7d530f2148968832e3 Mon Sep 17 00:00:00 2001 From: Grissess Date: Sun, 23 Aug 2015 23:49:11 -0400 Subject: Partial commit (last from cosi02!) --- broadcast.py | 242 ++++++++++++++------------- piano.py | 529 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ voice.py | 10 +- 3 files changed, 661 insertions(+), 120 deletions(-) create mode 100644 piano.py diff --git a/broadcast.py b/broadcast.py index b494645..1e52f14 100644 --- a/broadcast.py +++ b/broadcast.py @@ -200,121 +200,122 @@ if options.live or options.list_live: del active_set[pitch] deferred_set.clear() -try: - iv = ET.parse(args[0]).getroot() -except IOError: +for fname in args: + try: + iv = ET.parse(fname).getroot() + except IOError: import traceback traceback.print_exc() - print 'Bad file' - exit() + print fname, ': Bad file' + continue -notestreams = iv.findall("./streams/stream[@type='ns']") -groups = set([ns.get('group') for ns in notestreams if 'group' in ns.keys()]) -print len(notestreams), 'notestreams' -print len(clients), 'clients' -print len(groups), 'groups' + notestreams = iv.findall("./streams/stream[@type='ns']") + groups = set([ns.get('group') for ns in notestreams if 'group' in ns.keys()]) + print len(notestreams), 'notestreams' + print len(clients), 'clients' + print len(groups), 'groups' -class Route(object): - def __init__(self, fattr, fvalue, group, excl=False): - if fattr == 'U': - self.map = uid_groups - elif fattr == 'T': - self.map = type_groups - else: - raise ValueError('Not a valid attribute specifier: %r'%(fattr,)) - self.value = fvalue - if group is not None and group not in groups: - raise ValueError('Not a present group: %r'%(group,)) - self.group = group - self.excl = excl - @classmethod - def Parse(cls, s): - fspecs, _, grpspecs = map(lambda x: x.strip(), s.partition('=')) - fpairs = [] - ret = [] - for fspec in [i.strip() for i in fspecs.split(',')]: - fattr, _, fvalue = map(lambda x: x.strip(), fspec.partition(':')) - fpairs.append((fattr, fvalue)) - for part in [i.strip() for i in grpspecs.split(',')]: - for fattr, fvalue in fpairs: - if part[0] == '+': - ret.append(Route(fattr, fvalue, part[1:], False)) - elif part[0] == '-': - ret.append(Route(fattr, fvalue, part[1:], True)) - elif part[0] == '0': - ret.append(Route(fattr, fvalue, None, True)) - else: - raise ValueError('Not an exclusivity: %r'%(part[0],)) - return ret - def Apply(self, cli): - return cli in self.map.get(self.value, []) - def __repr__(self): - return ''%(self.group, ('U' if self.map is uid_groups else 'T'), self.value) + class Route(object): + def __init__(self, fattr, fvalue, group, excl=False): + if fattr == 'U': + self.map = uid_groups + elif fattr == 'T': + self.map = type_groups + else: + raise ValueError('Not a valid attribute specifier: %r'%(fattr,)) + self.value = fvalue + if group is not None and group not in groups: + raise ValueError('Not a present group: %r'%(group,)) + self.group = group + self.excl = excl + @classmethod + def Parse(cls, s): + fspecs, _, grpspecs = map(lambda x: x.strip(), s.partition('=')) + fpairs = [] + ret = [] + for fspec in [i.strip() for i in fspecs.split(',')]: + fattr, _, fvalue = map(lambda x: x.strip(), fspec.partition(':')) + fpairs.append((fattr, fvalue)) + for part in [i.strip() for i in grpspecs.split(',')]: + for fattr, fvalue in fpairs: + if part[0] == '+': + ret.append(Route(fattr, fvalue, part[1:], False)) + elif part[0] == '-': + ret.append(Route(fattr, fvalue, part[1:], True)) + elif part[0] == '0': + ret.append(Route(fattr, fvalue, None, True)) + else: + raise ValueError('Not an exclusivity: %r'%(part[0],)) + return ret + def Apply(self, cli): + return cli in self.map.get(self.value, []) + def __repr__(self): + return ''%(self.group, ('U' if self.map is uid_groups else 'T'), self.value) -class RouteSet(object): - def __init__(self, clis=None): - if clis is None: - clis = clients[:] - self.clients = clis - self.routes = [] - def Route(self, stream): - testset = self.clients[:] - grp = stream.get('group', 'ALL') - if options.verbose: - print 'Routing', grp, '...' - excl = False - for route in self.routes: - if route.group == grp: - if options.verbose: - print '\tMatches route', route - excl = excl or route.excl - matches = filter(lambda x, route=route: route.Apply(x), testset) - if matches: + class RouteSet(object): + def __init__(self, clis=None): + if clis is None: + clis = clients[:] + self.clients = clis + self.routes = [] + def Route(self, stream): + testset = self.clients[:] + grp = stream.get('group', 'ALL') + if options.verbose: + print 'Routing', grp, '...' + excl = False + for route in self.routes: + if route.group == grp: + if options.verbose: + print '\tMatches route', route + excl = excl or route.excl + matches = filter(lambda x, route=route: route.Apply(x), testset) + if matches: + if options.verbose: + print '\tUsing client', matches[0] + self.clients.remove(matches[0]) + return matches[0] + if options.verbose: + print '\tNo matches, moving on...' + if route.group is None: if options.verbose: - print '\tUsing client', matches[0] - self.clients.remove(matches[0]) - return matches[0] + print 'Encountered NULL route, removing from search space...' + toremove = [] + for cli in testset: + if route.Apply(cli): + toremove.append(cli) + for cli in toremove: + if options.verbose: + print '\tRemoving', cli, '...' + testset.remove(cli) + if excl: if options.verbose: - print '\tNo matches, moving on...' - if route.group is None: + print '\tExclusively routed, no route matched.' + return None + if not testset: if options.verbose: - print 'Encountered NULL route, removing from search space...' - toremove = [] - for cli in testset: - if route.Apply(cli): - toremove.append(cli) - for cli in toremove: - if options.verbose: - print '\tRemoving', cli, '...' - testset.remove(cli) - if excl: - if options.verbose: - print '\tExclusively routed, no route matched.' - return None - if not testset: + print '\tOut of clients, no route matched.' + return None + cli = testset[0] + self.clients.remove(cli) if options.verbose: - print '\tOut of clients, no route matched.' - return None - cli = testset[0] - self.clients.remove(cli) - if options.verbose: - print '\tDefault route to', cli - return cli + print '\tDefault route to', cli + return cli -routeset = RouteSet() -for rspec in options.routes: - try: - routeset.routes.extend(Route.Parse(rspec)) - except Exception: - import traceback - traceback.print_exc() + routeset = RouteSet() + for rspec in options.routes: + try: + routeset.routes.extend(Route.Parse(rspec)) + except Exception: + import traceback + traceback.print_exc() -if options.verbose: - print 'All routes:' - for route in routeset.routes: - print route + if options.verbose: + print 'All routes:' + for route in routeset.routes: + print route -class NSThread(threading.Thread): + class NSThread(threading.Thread): def wait_for(self, t): if t <= 0: return @@ -333,22 +334,27 @@ class NSThread(threading.Thread): print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) if options.verbose: - print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' + print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel + self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) + if options.verbose: + print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' -threads = [] -for ns in notestreams: - cli = routeset.Route(ns) - if cli: - nsq = ns.findall('note') - threads.append(NSThread(args=(nsq, cli))) + threads = [] + for ns in notestreams: + cli = routeset.Route(ns) + if cli: + nsq = ns.findall('note') + threads.append(NSThread(args=(nsq, cli))) -if options.verbose: - print 'Playback threads:' + if options.verbose: + print 'Playback threads:' + for thr in threads: + print thr._Thread__args[1] + + BASETIME = time.time() + for thr in threads: + thr.start() for thr in threads: - print thr._Thread__args[1] + thr.join() -BASETIME = time.time() -for thr in threads: - thr.start() -for thr in threads: - thr.join() + print fname, ': Done!' diff --git a/piano.py b/piano.py new file mode 100644 index 0000000..8c8b9a7 --- /dev/null +++ b/piano.py @@ -0,0 +1,529 @@ +import socket +import sys +import struct +import time +import xml.etree.ElementTree as ET +import threading +import optparse +import random + +from packet import Packet, CMD, itos + +parser = optparse.OptionParser() +parser.add_option('-t', '--test', dest='test', action='store_true', help='Play a test tone (440, 880) on all clients in sequence (the last overlaps with the first of the next)') +parser.add_option('-T', '--sync-test', dest='sync_test', action='store_true', help='Don\'t wait for clients to play tones properly--have them all test tone at the same time') +parser.add_option('-R', '--random', dest='random', type='float', help='Generate random notes at approximately this period') +parser.add_option('--rand-low', dest='rand_low', type='int', help='Low frequency to randomly sample') +parser.add_option('--rand-high', dest='rand_high', type='int', help='High frequency to randomly sample') +parser.add_option('-l', '--live', dest='live', help='Enter live mode (play from a controller in real time), specifying the port to connect to as "client,port"; use just "," to manually subscribe later') +parser.add_option('-L', '--list-live', dest='list_live', action='store_true', help='List all the clients and ports that can be connected to for live performance') +parser.add_option('-q', '--quit', dest='quit', action='store_true', help='Instruct all clients to quit') +parser.add_option('-p', '--play', dest='play', action='append', help='Play a single tone or chord (specified multiple times) on all listening clients (either "midi pitch" or "@frequency")') +parser.add_option('-P', '--play-async', dest='play_async', action='store_true', help='Don\'t wait for the tone to finish using the local clock') +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='How loud to play this note (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('-f', '--factor', dest='factor', type='float', help='Rescale time by this factor (0 0: + while True: + for cl in clients: + s.sendto(str(Packet(CMD.PLAY, int(options.random), int(1000000*(options.random-int(options.random))), random.randint(options.rand_low, options.rand_high), 255)), cl) + time.sleep(options.random) + +if options.live or options.list_live: + import midi + from midi import sequencer + S = sequencer.S + if options.list_live: + print sequencer.SequencerHardware() + exit() + seq = sequencer.SequencerRead(sequencer_resolution=120) + client_set = set(clients) + active_set = {} # note (pitch) -> client + deferred_set = set() # pitches held due to sustain + sustain_status = False + client, _, port = options.live.partition(',') + if client or port: + seq.subscribe_port(client, port) + seq.start_sequencer() + while True: + ev = S.event_input(seq.client) + event = None + if ev: + if options.verbose: + print 'SEQ:', ev + if ev < 0: + seq._error(ev) + if ev.type == S.SND_SEQ_EVENT_NOTEON: + event = midi.NoteOnEvent(channel = ev.data.note.channel, pitch = ev.data.note.note, velocity = ev.data.note.velocity) + elif ev.type == S.SND_SEQ_EVENT_NOTEOFF: + event = midi.NoteOffEvent(channel = ev.data.note.channel, pitch = ev.data.note.note, velocity = ev.data.note.velocity) + 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) + elif ev.type == S.SND_SEQ_EVENT_PITCHBEND: + event = midi.PitchWheelEvent(channel = ev.data.control.channel, pitch = ev.data.control.value) + elif options.verbose: + print 'WARNING: Unparsed event, type %r'%(ev.type,) + continue + if event is not None: + if isinstance(event, midi.NoteOnEvent) and event.velocity == 0: + event.__class__ = midi.NoteOffEvent + if options.verbose: + print 'EVENT:', event + if isinstance(event, midi.NoteOnEvent): + 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()) + if not inactive_set: + print 'WARNING: Out of clients to do note %r; dropped'%(event.pitch,) + continue + cli = random.choice(list(inactive_set)) + 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 + elif isinstance(event, midi.NoteOffEvent): + if event.pitch not in active_set: + 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] + 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: + print 'WARNING: Attempted deferred removal of inactive note %r'%(pitch,) + continue + s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), active_set[pitch]) + del active_set[pitch] + deferred_set.clear() + + + + + +if options.keyboard: + import pygame + import midi + from pygame import * + pygame.init() + size = width , height = 640,360 + screen = pygame.display.set_mode(size) + picture = pygame.image.load("aaaa.png") + surface = pygame.display.get_surface() + pitch = 60 + velocity = 127 + client_set = set(clients) + active_set = {} # note (pitch) -> client + sustain_status = False + sharp = 0 + while True: + surface.blit(picture,(0,0)) + pygame.display.update() + for event in pygame.event.get(): + if event.type == pygame.QUIT: + pygame.quit() + sys.exit() + elif event.type == pygame.KEYDOWN: + if event.key == K_a: + pitch = 60 + if event.key == K_s: + pitch = 62 + if event.key == K_d: + pitch = 64 + if event.key == K_f: + pitch = 65 + if event.key == K_g: + pitch = 67 + if event.key == K_h: + pitch = 69 + if event.key == K_j: + pitch = 71 + if event.key == K_k: + pitch = 72 + if event.key == K_l: + pitch = 74 + if event.key == K_z: + pitch = 48 + if event.key == K_x: + pitch = 50 + if event.key == K_c: + pitch = 52 + if event.key == K_v: + pitch = 53 + if event.key == K_b: + pitch = 55 + if event.key == K_n: + pitch = 57 + if event.key == K_m: + pitch = 59 + if event.key == K_q: + pitch = 76 + if event.key == K_w: + pitch = 77 + if event.key == K_e: + pitch = 79 + if event.key == K_r: + pitch = 81 + if event.key == K_t: + pitch = 83 + if event.key == K_y: + pitch = 84 + if event.key == K_u: + pitch = 86 + if event.key == K_i: + pitch = 88 + if event.key == K_o: + pitch = 89 + if event.key == K_p: + pitch = 91 + if event.key == K_LSHIFT: + sharp = 1 + continue + pitch = pitch + sharp + mevent = midi.NoteOnEvent(channel = 0, pitch = pitch, velocity = velocity) + if mevent.pitch in active_set: + if sustain_status: + deferred_set.discard(mevent.pitch) + else: + print 'WARNING: Note already activated: %r \n'%(mevent.pitch,), + continue + inactive_set = client_set - set(active_set.values()) + if not inactive_set: + print 'WARNING: Out of clients to do note %r; dropped'%(mevent.pitch,) + continue + cli = random.choice(list(inactive_set)) + s.sendto(str(Packet(CMD.PLAY, 65535, 0, int(440.0 * 2**((mevent.pitch-69)/12.0)), 2*mevent.velocity)), cli) + active_set[mevent.pitch] = cli + elif event.type == pygame.KEYUP: + if event.key == K_a: + pitch = 60 + if event.key == K_s: + pitch = 62 + if event.key == K_d: + pitch = 64 + if event.key == K_f: + pitch = 65 + if event.key == K_g: + pitch = 67 + if event.key == K_h: + pitch = 69 + if event.key == K_j: + pitch = 71 + if event.key == K_k: + pitch = 72 + if event.key == K_l: + pitch = 74 + if event.key == K_z: + pitch = 48 + if event.key == K_x: + pitch = 50 + if event.key == K_c: + pitch = 52 + if event.key == K_v: + pitch = 53 + if event.key == K_b: + pitch = 55 + if event.key == K_n: + pitch = 57 + if event.key == K_m: + pitch = 59 + if event.key == K_q: + pitch = 76 + if event.key == K_w: + pitch = 77 + if event.key == K_e: + pitch = 79 + if event.key == K_r: + pitch = 81 + if event.key == K_t: + pitch = 83 + if event.key == K_y: + pitch = 84 + if event.key == K_u: + pitch = 86 + if event.key == K_i: + pitch = 88 + if event.key == K_o: + pitch = 89 + if event.key == K_p: + pitch = 91 + if event.key == K_LSHIFT: + sharp = 0 + continue + mevent = midi.NoteOffEvent(channel = 0, pitch = pitch, velocity = velocity) + if mevent.pitch not in active_set: + print 'WARNING: Deactivating inactive note %r'%(mevent.pitch,) + continue + if sustain_status: + deferred_set.add(mevent.pitch) + continue + s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), active_set[mevent.pitch]) + del active_set[mevent.pitch] + mevent = midi.NoteOffEvent(channel = 0, pitch = pitch + 1, velocity = velocity) + if mevent.pitch not in active_set: + print 'WARNING: Deactivating inactive note %r'%(mevent.pitch,) + continue + if sustain_status: + deferred_set.add(mevent.pitch) + continue + s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), active_set[mevent.pitch]) + del active_set[mevent.pitch] + +for fname in args: + try: + iv = ET.parse(fname).getroot() + except IOError: + import traceback + traceback.print_exc() + print fname, ': Bad file' + continue + + notestreams = iv.findall("./streams/stream[@type='ns']") + groups = set([ns.get('group') for ns in notestreams if 'group' in ns.keys()]) + print len(notestreams), 'notestreams' + print len(clients), 'clients' + print len(groups), 'groups' + + class Route(object): + def __init__(self, fattr, fvalue, group, excl=False): + if fattr == 'U': + self.map = uid_groups + elif fattr == 'T': + self.map = type_groups + else: + raise ValueError('Not a valid attribute specifier: %r'%(fattr,)) + self.value = fvalue + if group is not None and group not in groups: + raise ValueError('Not a present group: %r'%(group,)) + self.group = group + self.excl = excl + @classmethod + def Parse(cls, s): + fspecs, _, grpspecs = map(lambda x: x.strip(), s.partition('=')) + fpairs = [] + ret = [] + for fspec in [i.strip() for i in fspecs.split(',')]: + fattr, _, fvalue = map(lambda x: x.strip(), fspec.partition(':')) + fpairs.append((fattr, fvalue)) + for part in [i.strip() for i in grpspecs.split(',')]: + for fattr, fvalue in fpairs: + if part[0] == '+': + ret.append(Route(fattr, fvalue, part[1:], False)) + elif part[0] == '-': + ret.append(Route(fattr, fvalue, part[1:], True)) + elif part[0] == '0': + ret.append(Route(fattr, fvalue, None, True)) + else: + raise ValueError('Not an exclusivity: %r'%(part[0],)) + return ret + def Apply(self, cli): + return cli in self.map.get(self.value, []) + def __repr__(self): + return ''%(self.group, ('U' if self.map is uid_groups else 'T'), self.value) + + class RouteSet(object): + def __init__(self, clis=None): + if clis is None: + clis = clients[:] + self.clients = clis + self.routes = [] + def Route(self, stream): + testset = self.clients[:] + grp = stream.get('group', 'ALL') + if options.verbose: + print 'Routing', grp, '...' + excl = False + for route in self.routes: + if route.group == grp: + if options.verbose: + print '\tMatches route', route + excl = excl or route.excl + matches = filter(lambda x, route=route: route.Apply(x), testset) + if matches: + if options.verbose: + print '\tUsing client', matches[0] + self.clients.remove(matches[0]) + return matches[0] + if options.verbose: + print '\tNo matches, moving on...' + if route.group is None: + if options.verbose: + print 'Encountered NULL route, removing from search space...' + toremove = [] + for cli in testset: + if route.Apply(cli): + toremove.append(cli) + for cli in toremove: + if options.verbose: + print '\tRemoving', cli, '...' + testset.remove(cli) + if excl: + if options.verbose: + print '\tExclusively routed, no route matched.' + return None + if not testset: + if options.verbose: + print '\tOut of clients, no route matched.' + return None + cli = testset[0] + self.clients.remove(cli) + if options.verbose: + print '\tDefault route to', cli + return cli + + routeset = RouteSet() + for rspec in options.routes: + try: + routeset.routes.extend(Route.Parse(rspec)) + except Exception: + import traceback + traceback.print_exc() + + if options.verbose: + print 'All routes:' + for route in routeset.routes: + print route + + class NSThread(threading.Thread): + def wait_for(self, t): + if t <= 0: + return + time.sleep(t) + def run(self): + nsq, cl = self._Thread__args + for note in nsq: + ttime = float(note.get('time')) + pitch = int(note.get('pitch')) + vel = int(note.get('vel')) + dur = factor*float(note.get('dur')) + while time.time() - BASETIME < factor*ttime: + self.wait_for(factor*ttime - (time.time() - BASETIME)) + s.sendto(str(Packet(CMD.PLAY, int(dur), int((dur*1000000)%1000000), int(440.0 * 2**((pitch-69)/12.0)), vel*2)), cl) + if options.verbose: + print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel + self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) + if options.verbose: + print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' + + threads = [] + for ns in notestreams: + cli = routeset.Route(ns) + if cli: + nsq = ns.findall('note') + threads.append(NSThread(args=(nsq, cli))) + + if options.verbose: + print 'Playback threads:' + for thr in threads: + print thr._Thread__args[1] + + BASETIME = time.time() + for thr in threads: + thr.start() + for thr in threads: + thr.join() + + print fname, ': Done!' diff --git a/voice.py b/voice.py index eb0b43f..14961c1 100644 --- a/voice.py +++ b/voice.py @@ -107,10 +107,16 @@ class VMeanMixer(Voice): def __init__(self, *voices): self.voices = list(voices) def __call__(self, theta): - return sum([i(theta)/len(self.voices) for i in self.voices]) + return norm_amp(sum([i(theta)/len(self.voices) for i in self.voices])) class VSumMixer(Voice): def __init__(self, *voices): self.voices = list(voices) def __call__(self, theta): - return sum([i(theta) for i in self.voices]) + return norm_amp(sum([i(theta) for i in self.voices])) + +class object(object): + def __init__(self): + this_obj = object() + +foo = object() -- cgit v1.2.3-70-g09d2 From 1d18a4ff6e656b69454f82a42af4203fdfc3a2f0 Mon Sep 17 00:00:00 2001 From: Graham Northup Date: Sat, 3 Oct 2015 02:58:37 -0400 Subject: Some pygame stuff --- piano.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/piano.py b/piano.py index 8c8b9a7..2c8dd22 100644 --- a/piano.py +++ b/piano.py @@ -211,8 +211,8 @@ if options.keyboard: pygame.init() size = width , height = 640,360 screen = pygame.display.set_mode(size) - picture = pygame.image.load("aaaa.png") - surface = pygame.display.get_surface() + #picture = pygame.image.load("aaaa.png") + #surface = pygame.display.get_surface() pitch = 60 velocity = 127 client_set = set(clients) @@ -220,8 +220,8 @@ if options.keyboard: sustain_status = False sharp = 0 while True: - surface.blit(picture,(0,0)) - pygame.display.update() + #surface.blit(picture,(0,0)) + #pygame.display.update() for event in pygame.event.get(): if event.type == pygame.QUIT: pygame.quit() -- cgit v1.2.3-70-g09d2 From c9b9f9d6871c2daa89a7d356160be8d607e77c6f Mon Sep 17 00:00:00 2001 From: Graham Northup Date: Sat, 3 Oct 2015 03:03:52 -0400 Subject: Oops (indentation woes) --- broadcast.py | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/broadcast.py b/broadcast.py index 1e52f14..23fe5c8 100644 --- a/broadcast.py +++ b/broadcast.py @@ -334,10 +334,7 @@ for fname in args: print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) if options.verbose: - print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel - self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) - if options.verbose: - print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' + print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' threads = [] for ns in notestreams: -- cgit v1.2.3-70-g09d2 From 7c9661d892f6145d123d91924b720d9d87b69502 Mon Sep 17 00:00:00 2001 From: Graham Northup Date: Tue, 6 Oct 2015 22:25:26 -0400 Subject: Added some volumes --- broadcast.py | 4 ++-- client.py | 3 ++- 2 files changed, 4 insertions(+), 3 deletions(-) diff --git a/broadcast.py b/broadcast.py index 23fe5c8..5606137 100644 --- a/broadcast.py +++ b/broadcast.py @@ -22,7 +22,7 @@ parser.add_option('-q', '--quit', dest='quit', action='store_true', help='Instru parser.add_option('-p', '--play', dest='play', action='append', help='Play a single tone or chord (specified multiple times) on all listening clients (either "midi pitch" or "@frequency")') parser.add_option('-P', '--play-async', dest='play_async', action='store_true', help='Don\'t wait for the tone to finish using the local clock') 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='How loud to play this note (0-255)') +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('-f', '--factor', dest='factor', type='float', help='Rescale time by this factor (0 Date: Thu, 20 Aug 2015 23:06:11 -0400 Subject: Partial commit (sorry :( ) --- broadcast.py | 4 ++- mkarduino.py | 29 +++++++++++++++ shiv.py | 44 +++++++++++++++++++++++ voice.py | 116 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 4 files changed, 192 insertions(+), 1 deletion(-) create mode 100644 mkarduino.py create mode 100644 shiv.py create mode 100644 voice.py diff --git a/broadcast.py b/broadcast.py index 9388c06..2cff0a3 100644 --- a/broadcast.py +++ b/broadcast.py @@ -146,6 +146,8 @@ if options.live or options.list_live: ev = S.event_input(seq.client) event = None if ev: + if options.verbose: + print 'SEQ:', ev if ev < 0: seq._error(ev) if ev.type == S.SND_SEQ_EVENT_NOTEON: @@ -163,7 +165,7 @@ if options.live or options.list_live: continue if event is not None: if isinstance(event, midi.NoteOnEvent) and event.velocity == 0: - ev.__class__ = midi.NoteOffEvent + event.__class__ = midi.NoteOffEvent if options.verbose: print 'EVENT:', event if isinstance(event, midi.NoteOnEvent): diff --git a/mkarduino.py b/mkarduino.py new file mode 100644 index 0000000..39926c5 --- /dev/null +++ b/mkarduino.py @@ -0,0 +1,29 @@ +#IV to arduino array computer + +import xml.etree.ElementTree as ET +import sys + +iv = ET.parse(sys.argv[1]).getroot() + +streams = iv.findall('./streams/stream[@type="ns"]') +if len(streams) > 3: + print 'WARNING: Too many streams' + +for i in xrange(min(3, len(streams))): + stream = streams[i] + notes = stream.findall('note') + +# First, the header + sys.stdout.write('const uint16_t track%d[] PROGMEM = {\n'%(i,)) + +# For the first note, write out the delay needed to get there + if notes[0].get('time') > 0: + sys.stdout.write('%d, 0,\n'%(int(float(notes[0].get('time'))*1000),)) + + for idx, note in enumerate(notes): + sys.stdout.write('%d, FREQ(%d),\n'%(int(float(note.get('dur'))*1000), int(440.0 * 2**((int(note.get('pitch'))-69)/12.0)))) + if idx < len(notes)-1 and float(note.get('time'))+float(note.get('dur')) < float(notes[idx+1].get('time')): + sys.stdout.write('%d, 0,\n'%(int(1000*(float(notes[idx+1].get('time')) - (float(note.get('time')) + float(note.get('dur'))))),)) + +# Finish up the stream + sys.stdout.write('};\n\n') diff --git a/shiv.py b/shiv.py new file mode 100644 index 0000000..96d8ea2 --- /dev/null +++ b/shiv.py @@ -0,0 +1,44 @@ +# IV file viewer + +import xml.etree.ElementTree as ET +import optparse +import sys + +parser = optparse.OptionParser() +parser.add_option('-n', '--number', dest='number', action='store_true', help='Show number of tracks') +parser.add_option('-g', '--groups', dest='groups', action='store_true', help='Show group names') +parser.add_option('-N', '--notes', dest='notes', action='store_true', help='Show number of notes') +parser.add_option('-m', '--meta', dest='meta', action='store_true', help='Show meta track information') +parser.add_option('-h', '--histogram', dest='histogram', action='store_true', help='Show a histogram distribution of pitches') +parser.add_option('-H', '--histogram-tracks', dest='histogram_tracks', action='store_true', help='Show a histogram distribution of pitches per track') +parser.add_option('-d', '--duration', dest='duration', action='store_true', help='Show the duration of the piece') +parser.add_option('-D', '--duty-cycle', dest='duty_cycle', action='store_true', help='Show the duration of the notes within tracks, and as a percentage of the piece duration') + +parser.add_option('-a', '--almost-all', dest='almost_all', action='store_true', help='Show useful information') +parser.add_option('-A', '--all', dest='all', action='store_true', help='Show everything') + +options, args = parser.parse_args() + +if options.almost_all or options.all: + options.number = True + options.groups = True + options.notes = True + options.histogram = True + options.duration = True + if options.all: + options.meta = True + options.histogram_tracks= True + options.duty_cycle = True + +for fname in args: + try: + iv = ET.parse(fname).getroot() + except IOError: + import traceback + traceback.print_exc() + print 'Bad file :', fname, ', skipping...' + continue + print + print 'File :', fname + print '\t' + diff --git a/voice.py b/voice.py new file mode 100644 index 0000000..eb0b43f --- /dev/null +++ b/voice.py @@ -0,0 +1,116 @@ +''' +voice -- Voices + +A voice is a simple, singular unit of sound generation that encompasses the following +properties: +-A *generator*: some function that generates a waveform. As an input, it receives theta, + the phase of the signal it is to generate (in [0, 2pi)) and, as an output, it produces + the sample at that point, a normalized amplitude value in [-1, 1]. +-An *envelope*: a function that receives a boolean (the status of whether or not a note + is playing now) and the change in time, and outputs a factor in [0, 1] that represents + a modification to the volume of the generator (pre-output mix). +All of these functions may internally store state or other data, usually by being +implemented as a class with a __call__ method. + +Voices are meant to generate audio data. This can be done in a number of ways, least to +most abstracted: +-A sample at a certain phase (theta) may be gotten from the generator; this can be done + by calling the voice outright; +-A set of samples can be generated via the .samples() method, which receives the number + of samples to generate and the phase velocity (a function of the sample rate and the + desired frequency of the waveform's period; this can be calculated using the static + method .phase_vel()); +-Audio data with enveloping can be generated using the .data() method, which calls the + envelope function as if the note is depressed at the given phase velocity; if the + freq is specified as None, then the note is treated as released. Note that + this will often be necessary for envelopes, as many of them are stateful (as they + depend on the first derivative of time). Also, at this level, the Voice will maintain + some state (namely, the phase at the end of generation) which will ensure (C0) smooth + transitions between already smooth generator functions, even if the frequency changes. +-Finally, a pyaudio-compatible stream callback can be provided with .pyaudio_scb(), a + method that returns a function that arranges to call .data() with the appropriate values. + The freq input to .data() will be taken from the .freq member of the voice in a possibly + non-atomic manner. +''' + +import math +import pyaudio +import struct + +def norm_theta(theta): + return theta % (2*math.pi) + +def norm_amp(amp): + return min(1.0, max(-1.0, amp)) + +def theta2lin(theta): + return theta / (2*math.pi) + +def lin2theta(lin): + return lin * 2*math.pi + +class ParamInfo(object): + PT_ANY = 0x0000 + PT_CONST = 0x0001 + PT_SPECIAL = 0x0002 + PT_INT = 0x0100 + PT_FLOAT = 0x0200 + PT_STR = 0x0400 + PT_THETA = 0x0102 + PT_TIME_SEC = 0x0202 + PT_SAMPLES = 0x0302 + PT_REALTIME = 0x0402 + def __init__(self, name, tp=PT_ANY): + self.name = name + self.tp = tp + +class GenInfo(object): + def __init__(self, name, *params): + self.name = name + self.params = list(params) + +class Generator(object): + class __metaclass__(type): + def __init__(self + +class Voice(object): + @classmethod + def register_gen(cls, name, params): + def __init__(self, generator=None, envelope=None): + self.generator = generator or self.DEFAULT_GENERATOR + self.envelope = envelope or self.DEFAULT_ENVELOPE + self.phase = 0 + self.freq = None + def __call__(self, theta): + return norm_amp(self.generator(norm_theta(theta))) + @staticmethod + def phase_vel(freq, samp_rate): + return 2 * math.pi * freq / samp_rate + def samples(self, frames, pvel): + for i in xrange(frames): + yield self(self.phase) + self.phase = norm_theta(self.phase + pvel) + def data(self, frames, freq, samp_rate): + period = 1.0/samp_rate + status = freq is not None + for samp in self.samples(frames, self.phase_vel(freq, samp_rate)): + yield samp * self.envelope(status, period) + def pyaudio_scb(self, rate, fmt=pyaudio.paInt16): + samp_size = pyaudio.get_sample_size(fmt) + maxint = (1 << (8*samp_size)) - 1 + dtype = ['!', 'h', 'i', '!', 'l', '!', '!', '!', 'q'][samp_size] + def __callback(data, frames, time, status, self=self, rate=rate, maxint=maxint, dtype=dtype): + return struct.pack(dtype*frames, *[maxint*int(i) for i in self.data(frames, self.freq, rate)]) + return __callback + +class VMeanMixer(Voice): + def __init__(self, *voices): + self.voices = list(voices) + def __call__(self, theta): + return sum([i(theta)/len(self.voices) for i in self.voices]) + +class VSumMixer(Voice): + def __init__(self, *voices): + self.voices = list(voices) + def __call__(self, theta): + return sum([i(theta) for i in self.voices]) -- cgit v1.2.3-70-g09d2 From 28796422ed09760b8d2ead3f58d8329eb6ea9010 Mon Sep 17 00:00:00 2001 From: Grissess Date: Sun, 23 Aug 2015 23:49:11 -0400 Subject: Partial commit (last from cosi02!) --- broadcast.py | 254 +++++++++++++++------------- piano.py | 529 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ voice.py | 10 +- 3 files changed, 680 insertions(+), 113 deletions(-) create mode 100644 piano.py diff --git a/broadcast.py b/broadcast.py index 2cff0a3..23b4a2f 100644 --- a/broadcast.py +++ b/broadcast.py @@ -205,119 +205,122 @@ if options.live or options.list_live: del active_set[pitch] deferred_set.clear() -try: - iv = ET.parse(args[0]).getroot() -except IOError: +for fname in args: + try: + iv = ET.parse(fname).getroot() + except IOError: import traceback traceback.print_exc() - print 'Bad file' - exit() + print fname, ': Bad file' + continue -notestreams = iv.findall("./streams/stream[@type='ns']") -groups = set([ns.get('group') for ns in notestreams if 'group' in ns.keys()]) -print len(notestreams), 'notestreams' -print len(groups), 'groups' + notestreams = iv.findall("./streams/stream[@type='ns']") + groups = set([ns.get('group') for ns in notestreams if 'group' in ns.keys()]) + print len(notestreams), 'notestreams' + print len(clients), 'clients' + print len(groups), 'groups' -class Route(object): - def __init__(self, fattr, fvalue, group, excl=False): - if fattr == 'U': - self.map = uid_groups - elif fattr == 'T': - self.map = type_groups - else: - raise ValueError('Not a valid attribute specifier: %r'%(fattr,)) - self.value = fvalue - if group is not None and group not in groups: - raise ValueError('Not a present group: %r'%(group,)) - self.group = group - self.excl = excl - @classmethod - def Parse(cls, s): - fspecs, _, grpspecs = map(lambda x: x.strip(), s.partition('=')) - fpairs = [] - ret = [] - for fspec in [i.strip() for i in fspecs.split(',')]: - fattr, _, fvalue = map(lambda x: x.strip(), fspec.partition(':')) - fpairs.append((fattr, fvalue)) - for part in [i.strip() for i in grpspecs.split(',')]: - for fattr, fvalue in fpairs: - if part[0] == '+': - ret.append(Route(fattr, fvalue, part[1:], False)) - elif part[0] == '-': - ret.append(Route(fattr, fvalue, part[1:], True)) - elif part[0] == '0': - ret.append(Route(fattr, fvalue, None, True)) - else: - raise ValueError('Not an exclusivity: %r'%(part[0],)) - return ret - def Apply(self, cli): - return cli in self.map.get(self.value, []) - def __repr__(self): - return ''%(self.group, ('U' if self.map is uid_groups else 'T'), self.value) + class Route(object): + def __init__(self, fattr, fvalue, group, excl=False): + if fattr == 'U': + self.map = uid_groups + elif fattr == 'T': + self.map = type_groups + else: + raise ValueError('Not a valid attribute specifier: %r'%(fattr,)) + self.value = fvalue + if group is not None and group not in groups: + raise ValueError('Not a present group: %r'%(group,)) + self.group = group + self.excl = excl + @classmethod + def Parse(cls, s): + fspecs, _, grpspecs = map(lambda x: x.strip(), s.partition('=')) + fpairs = [] + ret = [] + for fspec in [i.strip() for i in fspecs.split(',')]: + fattr, _, fvalue = map(lambda x: x.strip(), fspec.partition(':')) + fpairs.append((fattr, fvalue)) + for part in [i.strip() for i in grpspecs.split(',')]: + for fattr, fvalue in fpairs: + if part[0] == '+': + ret.append(Route(fattr, fvalue, part[1:], False)) + elif part[0] == '-': + ret.append(Route(fattr, fvalue, part[1:], True)) + elif part[0] == '0': + ret.append(Route(fattr, fvalue, None, True)) + else: + raise ValueError('Not an exclusivity: %r'%(part[0],)) + return ret + def Apply(self, cli): + return cli in self.map.get(self.value, []) + def __repr__(self): + return ''%(self.group, ('U' if self.map is uid_groups else 'T'), self.value) -class RouteSet(object): - def __init__(self, clis=None): - if clis is None: - clis = clients[:] - self.clients = clis - self.routes = [] - def Route(self, stream): - testset = self.clients[:] - grp = stream.get('group', 'ALL') - if options.verbose: - print 'Routing', grp, '...' - excl = False - for route in self.routes: - if route.group == grp: - if options.verbose: - print '\tMatches route', route - excl = excl or route.excl - matches = filter(lambda x, route=route: route.Apply(x), testset) - if matches: + class RouteSet(object): + def __init__(self, clis=None): + if clis is None: + clis = clients[:] + self.clients = clis + self.routes = [] + def Route(self, stream): + testset = self.clients[:] + grp = stream.get('group', 'ALL') + if options.verbose: + print 'Routing', grp, '...' + excl = False + for route in self.routes: + if route.group == grp: if options.verbose: - print '\tUsing client', matches[0] - self.clients.remove(matches[0]) - return matches[0] + print '\tMatches route', route + excl = excl or route.excl + matches = filter(lambda x, route=route: route.Apply(x), testset) + if matches: + if options.verbose: + print '\tUsing client', matches[0] + self.clients.remove(matches[0]) + return matches[0] + if options.verbose: + print '\tNo matches, moving on...' + if route.group is None: + if options.verbose: + print 'Encountered NULL route, removing from search space...' + toremove = [] + for cli in testset: + if route.Apply(cli): + toremove.append(cli) + for cli in toremove: + if options.verbose: + print '\tRemoving', cli, '...' + testset.remove(cli) + if excl: if options.verbose: - print '\tNo matches, moving on...' - if route.group is None: + print '\tExclusively routed, no route matched.' + return None + if not testset: if options.verbose: - print 'Encountered NULL route, removing from search space...' - toremove = [] - for cli in testset: - if route.Apply(cli): - toremove.append(cli) - for cli in toremove: - if options.verbose: - print '\tRemoving', cli, '...' - testset.remove(cli) - if excl: - if options.verbose: - print '\tExclusively routed, no route matched.' - return None - if not testset: + print '\tOut of clients, no route matched.' + return None + cli = testset[0] + self.clients.remove(cli) if options.verbose: - print '\tOut of clients, no route matched.' - return None - cli = testset[0] - self.clients.remove(cli) - if options.verbose: - print '\tDefault route to', cli - return cli + print '\tDefault route to', cli + return cli -routeset = RouteSet() -for rspec in options.routes: - try: - routeset.routes.extend(Route.Parse(rspec)) - except Exception: - import traceback - traceback.print_exc() + routeset = RouteSet() + for rspec in options.routes: + try: + routeset.routes.extend(Route.Parse(rspec)) + except Exception: + import traceback + traceback.print_exc() -if options.verbose: - print 'All routes:' - for route in routeset.routes: - print route + if options.verbose: + print 'All routes:' + for route in routeset.routes: + print route +<<<<<<< b1e54746e876a4ed485e2ed1bbc1d8b302e908ab class NSThread(threading.Thread): def drop_missed(self): nsq, cl = self._Thread__args @@ -328,10 +331,14 @@ class NSThread(threading.Thread): if options.verbose: print self, 'dropped', cnt, 'notes due to miss' self._Thread__args = (nsq, cl) +======= + class NSThread(threading.Thread): +>>>>>>> Partial commit (last from cosi02!) def wait_for(self, t): if t <= 0: return time.sleep(t) +<<<<<<< b1e54746e876a4ed485e2ed1bbc1d8b302e908ab def run(self): nsq, cl = self._Thread__args for note in nsq: @@ -345,21 +352,43 @@ class NSThread(threading.Thread): if options.verbose: print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) +======= + def run(self): + nsq, cl = self._Thread__args + for note in nsq: + ttime = float(note.get('time')) + pitch = int(note.get('pitch')) + vel = int(note.get('vel')) + dur = factor*float(note.get('dur')) + while time.time() - BASETIME < factor*ttime: + self.wait_for(factor*ttime - (time.time() - BASETIME)) + s.sendto(str(Packet(CMD.PLAY, int(dur), int((dur*1000000)%1000000), int(440.0 * 2**((pitch-69)/12.0)), vel*2)), cl) +>>>>>>> Partial commit (last from cosi02!) if options.verbose: - print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' + print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel + self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) + if options.verbose: + print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' -threads = [] -for ns in notestreams: - cli = routeset.Route(ns) - if cli: - nsq = ns.findall('note') - threads.append(NSThread(args=(nsq, cli))) + threads = [] + for ns in notestreams: + cli = routeset.Route(ns) + if cli: + nsq = ns.findall('note') + threads.append(NSThread(args=(nsq, cli))) -if options.verbose: - print 'Playback threads:' + if options.verbose: + print 'Playback threads:' + for thr in threads: + print thr._Thread__args[1] + + BASETIME = time.time() + for thr in threads: + thr.start() for thr in threads: - print thr._Thread__args[1] + thr.join() +<<<<<<< b1e54746e876a4ed485e2ed1bbc1d8b302e908ab BASETIME = time.time() - (options.seek*factor) if options.seek > 0: for thr in threads: @@ -368,3 +397,6 @@ for thr in threads: thr.start() for thr in threads: thr.join() +======= + print fname, ': Done!' +>>>>>>> Partial commit (last from cosi02!) diff --git a/piano.py b/piano.py new file mode 100644 index 0000000..8c8b9a7 --- /dev/null +++ b/piano.py @@ -0,0 +1,529 @@ +import socket +import sys +import struct +import time +import xml.etree.ElementTree as ET +import threading +import optparse +import random + +from packet import Packet, CMD, itos + +parser = optparse.OptionParser() +parser.add_option('-t', '--test', dest='test', action='store_true', help='Play a test tone (440, 880) on all clients in sequence (the last overlaps with the first of the next)') +parser.add_option('-T', '--sync-test', dest='sync_test', action='store_true', help='Don\'t wait for clients to play tones properly--have them all test tone at the same time') +parser.add_option('-R', '--random', dest='random', type='float', help='Generate random notes at approximately this period') +parser.add_option('--rand-low', dest='rand_low', type='int', help='Low frequency to randomly sample') +parser.add_option('--rand-high', dest='rand_high', type='int', help='High frequency to randomly sample') +parser.add_option('-l', '--live', dest='live', help='Enter live mode (play from a controller in real time), specifying the port to connect to as "client,port"; use just "," to manually subscribe later') +parser.add_option('-L', '--list-live', dest='list_live', action='store_true', help='List all the clients and ports that can be connected to for live performance') +parser.add_option('-q', '--quit', dest='quit', action='store_true', help='Instruct all clients to quit') +parser.add_option('-p', '--play', dest='play', action='append', help='Play a single tone or chord (specified multiple times) on all listening clients (either "midi pitch" or "@frequency")') +parser.add_option('-P', '--play-async', dest='play_async', action='store_true', help='Don\'t wait for the tone to finish using the local clock') +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='How loud to play this note (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('-f', '--factor', dest='factor', type='float', help='Rescale time by this factor (0 0: + while True: + for cl in clients: + s.sendto(str(Packet(CMD.PLAY, int(options.random), int(1000000*(options.random-int(options.random))), random.randint(options.rand_low, options.rand_high), 255)), cl) + time.sleep(options.random) + +if options.live or options.list_live: + import midi + from midi import sequencer + S = sequencer.S + if options.list_live: + print sequencer.SequencerHardware() + exit() + seq = sequencer.SequencerRead(sequencer_resolution=120) + client_set = set(clients) + active_set = {} # note (pitch) -> client + deferred_set = set() # pitches held due to sustain + sustain_status = False + client, _, port = options.live.partition(',') + if client or port: + seq.subscribe_port(client, port) + seq.start_sequencer() + while True: + ev = S.event_input(seq.client) + event = None + if ev: + if options.verbose: + print 'SEQ:', ev + if ev < 0: + seq._error(ev) + if ev.type == S.SND_SEQ_EVENT_NOTEON: + event = midi.NoteOnEvent(channel = ev.data.note.channel, pitch = ev.data.note.note, velocity = ev.data.note.velocity) + elif ev.type == S.SND_SEQ_EVENT_NOTEOFF: + event = midi.NoteOffEvent(channel = ev.data.note.channel, pitch = ev.data.note.note, velocity = ev.data.note.velocity) + 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) + elif ev.type == S.SND_SEQ_EVENT_PITCHBEND: + event = midi.PitchWheelEvent(channel = ev.data.control.channel, pitch = ev.data.control.value) + elif options.verbose: + print 'WARNING: Unparsed event, type %r'%(ev.type,) + continue + if event is not None: + if isinstance(event, midi.NoteOnEvent) and event.velocity == 0: + event.__class__ = midi.NoteOffEvent + if options.verbose: + print 'EVENT:', event + if isinstance(event, midi.NoteOnEvent): + 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()) + if not inactive_set: + print 'WARNING: Out of clients to do note %r; dropped'%(event.pitch,) + continue + cli = random.choice(list(inactive_set)) + 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 + elif isinstance(event, midi.NoteOffEvent): + if event.pitch not in active_set: + 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] + 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: + print 'WARNING: Attempted deferred removal of inactive note %r'%(pitch,) + continue + s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), active_set[pitch]) + del active_set[pitch] + deferred_set.clear() + + + + + +if options.keyboard: + import pygame + import midi + from pygame import * + pygame.init() + size = width , height = 640,360 + screen = pygame.display.set_mode(size) + picture = pygame.image.load("aaaa.png") + surface = pygame.display.get_surface() + pitch = 60 + velocity = 127 + client_set = set(clients) + active_set = {} # note (pitch) -> client + sustain_status = False + sharp = 0 + while True: + surface.blit(picture,(0,0)) + pygame.display.update() + for event in pygame.event.get(): + if event.type == pygame.QUIT: + pygame.quit() + sys.exit() + elif event.type == pygame.KEYDOWN: + if event.key == K_a: + pitch = 60 + if event.key == K_s: + pitch = 62 + if event.key == K_d: + pitch = 64 + if event.key == K_f: + pitch = 65 + if event.key == K_g: + pitch = 67 + if event.key == K_h: + pitch = 69 + if event.key == K_j: + pitch = 71 + if event.key == K_k: + pitch = 72 + if event.key == K_l: + pitch = 74 + if event.key == K_z: + pitch = 48 + if event.key == K_x: + pitch = 50 + if event.key == K_c: + pitch = 52 + if event.key == K_v: + pitch = 53 + if event.key == K_b: + pitch = 55 + if event.key == K_n: + pitch = 57 + if event.key == K_m: + pitch = 59 + if event.key == K_q: + pitch = 76 + if event.key == K_w: + pitch = 77 + if event.key == K_e: + pitch = 79 + if event.key == K_r: + pitch = 81 + if event.key == K_t: + pitch = 83 + if event.key == K_y: + pitch = 84 + if event.key == K_u: + pitch = 86 + if event.key == K_i: + pitch = 88 + if event.key == K_o: + pitch = 89 + if event.key == K_p: + pitch = 91 + if event.key == K_LSHIFT: + sharp = 1 + continue + pitch = pitch + sharp + mevent = midi.NoteOnEvent(channel = 0, pitch = pitch, velocity = velocity) + if mevent.pitch in active_set: + if sustain_status: + deferred_set.discard(mevent.pitch) + else: + print 'WARNING: Note already activated: %r \n'%(mevent.pitch,), + continue + inactive_set = client_set - set(active_set.values()) + if not inactive_set: + print 'WARNING: Out of clients to do note %r; dropped'%(mevent.pitch,) + continue + cli = random.choice(list(inactive_set)) + s.sendto(str(Packet(CMD.PLAY, 65535, 0, int(440.0 * 2**((mevent.pitch-69)/12.0)), 2*mevent.velocity)), cli) + active_set[mevent.pitch] = cli + elif event.type == pygame.KEYUP: + if event.key == K_a: + pitch = 60 + if event.key == K_s: + pitch = 62 + if event.key == K_d: + pitch = 64 + if event.key == K_f: + pitch = 65 + if event.key == K_g: + pitch = 67 + if event.key == K_h: + pitch = 69 + if event.key == K_j: + pitch = 71 + if event.key == K_k: + pitch = 72 + if event.key == K_l: + pitch = 74 + if event.key == K_z: + pitch = 48 + if event.key == K_x: + pitch = 50 + if event.key == K_c: + pitch = 52 + if event.key == K_v: + pitch = 53 + if event.key == K_b: + pitch = 55 + if event.key == K_n: + pitch = 57 + if event.key == K_m: + pitch = 59 + if event.key == K_q: + pitch = 76 + if event.key == K_w: + pitch = 77 + if event.key == K_e: + pitch = 79 + if event.key == K_r: + pitch = 81 + if event.key == K_t: + pitch = 83 + if event.key == K_y: + pitch = 84 + if event.key == K_u: + pitch = 86 + if event.key == K_i: + pitch = 88 + if event.key == K_o: + pitch = 89 + if event.key == K_p: + pitch = 91 + if event.key == K_LSHIFT: + sharp = 0 + continue + mevent = midi.NoteOffEvent(channel = 0, pitch = pitch, velocity = velocity) + if mevent.pitch not in active_set: + print 'WARNING: Deactivating inactive note %r'%(mevent.pitch,) + continue + if sustain_status: + deferred_set.add(mevent.pitch) + continue + s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), active_set[mevent.pitch]) + del active_set[mevent.pitch] + mevent = midi.NoteOffEvent(channel = 0, pitch = pitch + 1, velocity = velocity) + if mevent.pitch not in active_set: + print 'WARNING: Deactivating inactive note %r'%(mevent.pitch,) + continue + if sustain_status: + deferred_set.add(mevent.pitch) + continue + s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), active_set[mevent.pitch]) + del active_set[mevent.pitch] + +for fname in args: + try: + iv = ET.parse(fname).getroot() + except IOError: + import traceback + traceback.print_exc() + print fname, ': Bad file' + continue + + notestreams = iv.findall("./streams/stream[@type='ns']") + groups = set([ns.get('group') for ns in notestreams if 'group' in ns.keys()]) + print len(notestreams), 'notestreams' + print len(clients), 'clients' + print len(groups), 'groups' + + class Route(object): + def __init__(self, fattr, fvalue, group, excl=False): + if fattr == 'U': + self.map = uid_groups + elif fattr == 'T': + self.map = type_groups + else: + raise ValueError('Not a valid attribute specifier: %r'%(fattr,)) + self.value = fvalue + if group is not None and group not in groups: + raise ValueError('Not a present group: %r'%(group,)) + self.group = group + self.excl = excl + @classmethod + def Parse(cls, s): + fspecs, _, grpspecs = map(lambda x: x.strip(), s.partition('=')) + fpairs = [] + ret = [] + for fspec in [i.strip() for i in fspecs.split(',')]: + fattr, _, fvalue = map(lambda x: x.strip(), fspec.partition(':')) + fpairs.append((fattr, fvalue)) + for part in [i.strip() for i in grpspecs.split(',')]: + for fattr, fvalue in fpairs: + if part[0] == '+': + ret.append(Route(fattr, fvalue, part[1:], False)) + elif part[0] == '-': + ret.append(Route(fattr, fvalue, part[1:], True)) + elif part[0] == '0': + ret.append(Route(fattr, fvalue, None, True)) + else: + raise ValueError('Not an exclusivity: %r'%(part[0],)) + return ret + def Apply(self, cli): + return cli in self.map.get(self.value, []) + def __repr__(self): + return ''%(self.group, ('U' if self.map is uid_groups else 'T'), self.value) + + class RouteSet(object): + def __init__(self, clis=None): + if clis is None: + clis = clients[:] + self.clients = clis + self.routes = [] + def Route(self, stream): + testset = self.clients[:] + grp = stream.get('group', 'ALL') + if options.verbose: + print 'Routing', grp, '...' + excl = False + for route in self.routes: + if route.group == grp: + if options.verbose: + print '\tMatches route', route + excl = excl or route.excl + matches = filter(lambda x, route=route: route.Apply(x), testset) + if matches: + if options.verbose: + print '\tUsing client', matches[0] + self.clients.remove(matches[0]) + return matches[0] + if options.verbose: + print '\tNo matches, moving on...' + if route.group is None: + if options.verbose: + print 'Encountered NULL route, removing from search space...' + toremove = [] + for cli in testset: + if route.Apply(cli): + toremove.append(cli) + for cli in toremove: + if options.verbose: + print '\tRemoving', cli, '...' + testset.remove(cli) + if excl: + if options.verbose: + print '\tExclusively routed, no route matched.' + return None + if not testset: + if options.verbose: + print '\tOut of clients, no route matched.' + return None + cli = testset[0] + self.clients.remove(cli) + if options.verbose: + print '\tDefault route to', cli + return cli + + routeset = RouteSet() + for rspec in options.routes: + try: + routeset.routes.extend(Route.Parse(rspec)) + except Exception: + import traceback + traceback.print_exc() + + if options.verbose: + print 'All routes:' + for route in routeset.routes: + print route + + class NSThread(threading.Thread): + def wait_for(self, t): + if t <= 0: + return + time.sleep(t) + def run(self): + nsq, cl = self._Thread__args + for note in nsq: + ttime = float(note.get('time')) + pitch = int(note.get('pitch')) + vel = int(note.get('vel')) + dur = factor*float(note.get('dur')) + while time.time() - BASETIME < factor*ttime: + self.wait_for(factor*ttime - (time.time() - BASETIME)) + s.sendto(str(Packet(CMD.PLAY, int(dur), int((dur*1000000)%1000000), int(440.0 * 2**((pitch-69)/12.0)), vel*2)), cl) + if options.verbose: + print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel + self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) + if options.verbose: + print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' + + threads = [] + for ns in notestreams: + cli = routeset.Route(ns) + if cli: + nsq = ns.findall('note') + threads.append(NSThread(args=(nsq, cli))) + + if options.verbose: + print 'Playback threads:' + for thr in threads: + print thr._Thread__args[1] + + BASETIME = time.time() + for thr in threads: + thr.start() + for thr in threads: + thr.join() + + print fname, ': Done!' diff --git a/voice.py b/voice.py index eb0b43f..14961c1 100644 --- a/voice.py +++ b/voice.py @@ -107,10 +107,16 @@ class VMeanMixer(Voice): def __init__(self, *voices): self.voices = list(voices) def __call__(self, theta): - return sum([i(theta)/len(self.voices) for i in self.voices]) + return norm_amp(sum([i(theta)/len(self.voices) for i in self.voices])) class VSumMixer(Voice): def __init__(self, *voices): self.voices = list(voices) def __call__(self, theta): - return sum([i(theta) for i in self.voices]) + return norm_amp(sum([i(theta) for i in self.voices])) + +class object(object): + def __init__(self): + this_obj = object() + +foo = object() -- cgit v1.2.3-70-g09d2 From f98effd66255f3af2aedf922c6e74a149020e736 Mon Sep 17 00:00:00 2001 From: Graham Northup Date: Sat, 3 Oct 2015 02:58:37 -0400 Subject: Some pygame stuff --- piano.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/piano.py b/piano.py index 8c8b9a7..2c8dd22 100644 --- a/piano.py +++ b/piano.py @@ -211,8 +211,8 @@ if options.keyboard: pygame.init() size = width , height = 640,360 screen = pygame.display.set_mode(size) - picture = pygame.image.load("aaaa.png") - surface = pygame.display.get_surface() + #picture = pygame.image.load("aaaa.png") + #surface = pygame.display.get_surface() pitch = 60 velocity = 127 client_set = set(clients) @@ -220,8 +220,8 @@ if options.keyboard: sustain_status = False sharp = 0 while True: - surface.blit(picture,(0,0)) - pygame.display.update() + #surface.blit(picture,(0,0)) + #pygame.display.update() for event in pygame.event.get(): if event.type == pygame.QUIT: pygame.quit() -- cgit v1.2.3-70-g09d2 From 833f48aba6781bf0d07917eba7ebfcbf24de6729 Mon Sep 17 00:00:00 2001 From: Grissess Date: Thu, 20 Aug 2015 23:06:11 -0400 Subject: Partial commit (sorry :( ) --- broadcast.py | 38 ++++++-------------------------------- voice.py | 6 ------ 2 files changed, 6 insertions(+), 38 deletions(-) diff --git a/broadcast.py b/broadcast.py index 23b4a2f..337efaa 100644 --- a/broadcast.py +++ b/broadcast.py @@ -320,7 +320,6 @@ for fname in args: for route in routeset.routes: print route -<<<<<<< b1e54746e876a4ed485e2ed1bbc1d8b302e908ab class NSThread(threading.Thread): def drop_missed(self): nsq, cl = self._Thread__args @@ -331,14 +330,10 @@ class NSThread(threading.Thread): if options.verbose: print self, 'dropped', cnt, 'notes due to miss' self._Thread__args = (nsq, cl) -======= - class NSThread(threading.Thread): ->>>>>>> Partial commit (last from cosi02!) def wait_for(self, t): if t <= 0: return time.sleep(t) -<<<<<<< b1e54746e876a4ed485e2ed1bbc1d8b302e908ab def run(self): nsq, cl = self._Thread__args for note in nsq: @@ -352,18 +347,6 @@ class NSThread(threading.Thread): if options.verbose: print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) -======= - def run(self): - nsq, cl = self._Thread__args - for note in nsq: - ttime = float(note.get('time')) - pitch = int(note.get('pitch')) - vel = int(note.get('vel')) - dur = factor*float(note.get('dur')) - while time.time() - BASETIME < factor*ttime: - self.wait_for(factor*ttime - (time.time() - BASETIME)) - s.sendto(str(Packet(CMD.PLAY, int(dur), int((dur*1000000)%1000000), int(440.0 * 2**((pitch-69)/12.0)), vel*2)), cl) ->>>>>>> Partial commit (last from cosi02!) if options.verbose: print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) @@ -382,21 +365,12 @@ class NSThread(threading.Thread): for thr in threads: print thr._Thread__args[1] - BASETIME = time.time() - for thr in threads: - thr.start() + BASETIME = time.time() - (options.seek*factor) + if options.seek > 0: + for thr in threads: + thr.drop_missed() for thr in threads: - thr.join() - -<<<<<<< b1e54746e876a4ed485e2ed1bbc1d8b302e908ab -BASETIME = time.time() - (options.seek*factor) -if options.seek > 0: + thr.start() for thr in threads: - thr.drop_missed() -for thr in threads: - thr.start() -for thr in threads: - thr.join() -======= + thr.join() print fname, ': Done!' ->>>>>>> Partial commit (last from cosi02!) diff --git a/voice.py b/voice.py index 14961c1..f440a68 100644 --- a/voice.py +++ b/voice.py @@ -114,9 +114,3 @@ class VSumMixer(Voice): self.voices = list(voices) def __call__(self, theta): return norm_amp(sum([i(theta) for i in self.voices])) - -class object(object): - def __init__(self): - this_obj = object() - -foo = object() -- cgit v1.2.3-70-g09d2 From f18ab395dd37c90be120dcedbcd5a2c6c3540f94 Mon Sep 17 00:00:00 2001 From: Grissess Date: Sun, 23 Aug 2015 23:49:11 -0400 Subject: Partial commit (last from cosi02!) --- piano.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/piano.py b/piano.py index 2c8dd22..8c8b9a7 100644 --- a/piano.py +++ b/piano.py @@ -211,8 +211,8 @@ if options.keyboard: pygame.init() size = width , height = 640,360 screen = pygame.display.set_mode(size) - #picture = pygame.image.load("aaaa.png") - #surface = pygame.display.get_surface() + picture = pygame.image.load("aaaa.png") + surface = pygame.display.get_surface() pitch = 60 velocity = 127 client_set = set(clients) @@ -220,8 +220,8 @@ if options.keyboard: sustain_status = False sharp = 0 while True: - #surface.blit(picture,(0,0)) - #pygame.display.update() + surface.blit(picture,(0,0)) + pygame.display.update() for event in pygame.event.get(): if event.type == pygame.QUIT: pygame.quit() -- cgit v1.2.3-70-g09d2 From da2a7795f54a9e5a4d0bb82a3e43a8114b5ac2e3 Mon Sep 17 00:00:00 2001 From: Graham Northup Date: Sat, 3 Oct 2015 02:58:37 -0400 Subject: Some pygame stuff --- piano.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/piano.py b/piano.py index 8c8b9a7..2c8dd22 100644 --- a/piano.py +++ b/piano.py @@ -211,8 +211,8 @@ if options.keyboard: pygame.init() size = width , height = 640,360 screen = pygame.display.set_mode(size) - picture = pygame.image.load("aaaa.png") - surface = pygame.display.get_surface() + #picture = pygame.image.load("aaaa.png") + #surface = pygame.display.get_surface() pitch = 60 velocity = 127 client_set = set(clients) @@ -220,8 +220,8 @@ if options.keyboard: sustain_status = False sharp = 0 while True: - surface.blit(picture,(0,0)) - pygame.display.update() + #surface.blit(picture,(0,0)) + #pygame.display.update() for event in pygame.event.get(): if event.type == pygame.QUIT: pygame.quit() -- cgit v1.2.3-70-g09d2 From 4b4962090222e96c4d3f689c2a501288c579d92c Mon Sep 17 00:00:00 2001 From: Graham Northup Date: Sat, 3 Oct 2015 03:03:52 -0400 Subject: Oops (indentation woes) --- broadcast.py | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/broadcast.py b/broadcast.py index 337efaa..375528c 100644 --- a/broadcast.py +++ b/broadcast.py @@ -348,10 +348,7 @@ class NSThread(threading.Thread): print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) if options.verbose: - print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel - self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) - if options.verbose: - print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' + print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' threads = [] for ns in notestreams: -- cgit v1.2.3-70-g09d2 From 960d044246ca0d2b255b3b81632b0380c14b632f Mon Sep 17 00:00:00 2001 From: Graham Northup Date: Tue, 6 Oct 2015 22:25:26 -0400 Subject: Added some volumes --- broadcast.py | 4 ++-- client.py | 3 ++- 2 files changed, 4 insertions(+), 3 deletions(-) diff --git a/broadcast.py b/broadcast.py index 375528c..82f0e21 100644 --- a/broadcast.py +++ b/broadcast.py @@ -22,7 +22,7 @@ parser.add_option('-q', '--quit', dest='quit', action='store_true', help='Instru parser.add_option('-p', '--play', dest='play', action='append', help='Play a single tone or chord (specified multiple times) on all listening clients (either "midi pitch" or "@frequency")') parser.add_option('-P', '--play-async', dest='play_async', action='store_true', help='Don\'t wait for the tone to finish using the local clock') 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='How loud to play this note (0-255)') +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 Date: Sat, 19 Dec 2015 12:28:26 -0500 Subject: Getting things clean again --- broadcast.py | 37 +++++++++++++++++++++++++++++++------ voice.py | 33 +++++++++------------------------ 2 files changed, 40 insertions(+), 30 deletions(-) diff --git a/broadcast.py b/broadcast.py index 3b163b1..b597268 100644 --- a/broadcast.py +++ b/broadcast.py @@ -320,8 +320,7 @@ for fname in args: for route in routeset.routes: print route -<<<<<<< HEAD -class NSThread(threading.Thread): + class NSThread(threading.Thread): def drop_missed(self): nsq, cl = self._Thread__args cnt = 0 @@ -331,9 +330,6 @@ class NSThread(threading.Thread): 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 @@ -351,8 +347,37 @@ class NSThread(threading.Thread): if options.verbose: print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) +======= + 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 +>>>>>>> Stashed changes if options.verbose: - print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' + print self, 'dropped', cnt, 'notes due to miss' + self._Thread__args = (nsq, cl) + def wait_for(self, t): + if t <= 0: + return + time.sleep(t) + def run(self): + nsq, cl = self._Thread__args + for note in nsq: + ttime = float(note.get('time')) + pitch = int(note.get('pitch')) + options.transpose + vel = int(note.get('vel')) + dur = factor*float(note.get('dur')) + while time.time() - BASETIME < factor*ttime: + self.wait_for(factor*ttime - (time.time() - BASETIME)) + s.sendto(str(Packet(CMD.PLAY, int(dur), int((dur*1000000)%1000000), int(440.0 * 2**((pitch-69)/12.0)), int(vel*2 * options.volume/255.0))), cl) + if options.verbose: + print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel + self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) + if options.verbose: + print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' threads = [] for ns in notestreams: diff --git a/voice.py b/voice.py index f440a68..b31aa84 100644 --- a/voice.py +++ b/voice.py @@ -36,6 +36,7 @@ most abstracted: import math import pyaudio import struct +import time def norm_theta(theta): return theta % (2*math.pi) @@ -49,30 +50,6 @@ def theta2lin(theta): def lin2theta(lin): return lin * 2*math.pi -class ParamInfo(object): - PT_ANY = 0x0000 - PT_CONST = 0x0001 - PT_SPECIAL = 0x0002 - PT_INT = 0x0100 - PT_FLOAT = 0x0200 - PT_STR = 0x0400 - PT_THETA = 0x0102 - PT_TIME_SEC = 0x0202 - PT_SAMPLES = 0x0302 - PT_REALTIME = 0x0402 - def __init__(self, name, tp=PT_ANY): - self.name = name - self.tp = tp - -class GenInfo(object): - def __init__(self, name, *params): - self.name = name - self.params = list(params) - -class Generator(object): - class __metaclass__(type): - def __init__(self - class Voice(object): @classmethod def register_gen(cls, name, params): @@ -114,3 +91,11 @@ class VSumMixer(Voice): self.voices = list(voices) def __call__(self, theta): return norm_amp(sum([i(theta) for i in self.voices])) + +class VLFOMixer(Voice): + def __init__(self, lfo, *voices): + self.lfo = lfo + self.voices = list(voices) + def __call__(self, theta): + i = int(len(self.voices) * self.lfo * (time.time() % (1.0 / self.lfo))) + return self.voices[i](theta) -- cgit v1.2.3-70-g09d2 From c18fea05f21e0b2927a378c545b86837c3a55794 Mon Sep 17 00:00:00 2001 From: Grissess Date: Wed, 17 Feb 2016 16:18:36 -0500 Subject: wat --- broadcast.py | 12 ------------ 1 file changed, 12 deletions(-) diff --git a/broadcast.py b/broadcast.py index b597268..0c2d8c6 100644 --- a/broadcast.py +++ b/broadcast.py @@ -347,7 +347,6 @@ for fname in args: if options.verbose: print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) -======= class NSThread(threading.Thread): def drop_missed(self): nsq, cl = self._Thread__args @@ -355,7 +354,6 @@ for fname in args: while nsq and float(nsq[0].get('time'))*factor < time.time() - BASETIME: nsq.pop(0) cnt += 1 ->>>>>>> Stashed changes if options.verbose: print self, 'dropped', cnt, 'notes due to miss' self._Thread__args = (nsq, cl) @@ -390,7 +388,6 @@ 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: @@ -400,13 +397,4 @@ for fname in args: thr.start() for thr in threads: thr.join() -======= - - BASETIME = time.time() - for thr in threads: - thr.start() - for thr in threads: - thr.join() - ->>>>>>> 7c9661d892f6145d123d91924b720d9d87b69502 print fname, ': Done!' -- cgit v1.2.3-70-g09d2 From 5eb82698723a95d8e8838ad0e09b6f392a535d9e Mon Sep 17 00:00:00 2001 From: Grissess Date: Sat, 2 Apr 2016 00:02:48 -0400 Subject: GUI and bind addresses --- broadcast.py | 8 +++++- client.py | 83 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++- 2 files changed, 89 insertions(+), 2 deletions(-) diff --git a/broadcast.py b/broadcast.py index 0c2d8c6..da0ee00 100644 --- a/broadcast.py +++ b/broadcast.py @@ -29,8 +29,9 @@ parser.add_option('-f', '--factor', dest='factor', type='float', help='Rescale t 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('-B', '--bind-addr', dest='bind_addr', help='The IP address (or IP:port) to bind to (influences the network to send to)') 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, seek=0.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, bind_addr='') options, args = parser.parse_args() if options.help_routes: @@ -56,6 +57,11 @@ print 'Factor:', factor s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) s.setsockopt(socket.SOL_SOCKET, socket.SO_BROADCAST, 1) +if options.bind_addr: + addr, _, port = options.bind_addr.partition(':') + if not port: + port = '12074' + s.bind((addr, int(port))) clients = [] uid_groups = {} diff --git a/client.py b/client.py index c170186..935a7e0 100644 --- a/client.py +++ b/client.py @@ -11,6 +11,7 @@ import socket import optparse import array import random +import threading from packet import Packet, CMD, stoi @@ -22,6 +23,7 @@ parser.add_option('-u', '--uid', dest='uid', default='', help='Set the UID (iden parser.add_option('-p', '--port', dest='port', type='int', default=13676, help='Set the port to listen on') parser.add_option('-r', '--rate', dest='rate', type='int', default=44100, help='Set the sample rate of the audio device') parser.add_option('-V', '--volume', dest='volume', type='float', default=1.0, help='Set the volume factor (>1 distorts, <1 attenuates)') +parser.add_option('-G', '--gui', dest='gui', default='', help='set a GUI to use') options, args = parser.parse_args() @@ -31,6 +33,7 @@ IDENT = 'TONE' UID = options.uid LAST_SAMP = 0 +LAST_SAMPLES = [] FREQ = 0 PHASE = 0 RATE = options.rate @@ -44,6 +47,77 @@ MIN = -0x80000000 def lin_interp(frm, to, p): return p*to + (1-p)*frm +# GUIs + +GUIs = {} + +def GUI(f): + GUIs[f.__name__] = f + return f + +@GUI +def pygame_notes(): + import pygame + import pygame.gfxdraw + pygame.init() + + SAMP_WIDTH = 512 + BGR_WIDTH = 512 + HEIGHT = 1024 + + disp = pygame.display.set_mode((SAMP_WIDTH + BGR_WIDTH, HEIGHT)) + + WIDTH, HEIGHT = disp.get_size() + SAMP_WIDTH = WIDTH / 2 + BGR_WIDTH = WIDTH - SAMP_WIDTH + PFAC = HEIGHT / 128.0 + + sampwin = pygame.Surface((SAMP_WIDTH, HEIGHT)) + lastsy = HEIGHT / 2 + + clock = pygame.time.Clock() + + while True: + if FREQ > 0: + try: + pitch = 12 * math.log(FREQ / 440.0, 2) + 69 + except ValueError: + pitch = 0 + else: + pitch = 0 + col = [int((AMP / MAX) * 255)] * 3 + + disp.fill((0, 0, 0), (BGR_WIDTH, 0, SAMP_WIDTH, HEIGHT)) + disp.scroll(-1, 0) + disp.fill(col, (BGR_WIDTH - 1, HEIGHT - pitch * PFAC - PFAC, 1, PFAC)) + + sampwin.scroll(-len(LAST_SAMPLES), 0) + x = max(0, SAMP_WIDTH - len(LAST_SAMPLES)) + sampwin.fill((0, 0, 0), (x, 0, SAMP_WIDTH - x, HEIGHT)) + for i in LAST_SAMPLES: + sy = int((float(i) / MAX) * (HEIGHT / 2) + (HEIGHT / 2)) + pygame.gfxdraw.line(sampwin, x - 1, lastsy, x, sy, (0, 255, 0)) + x += 1 + lastsy = sy + del LAST_SAMPLES[:] + #w, h = SAMP_WIDTH, HEIGHT + #pts = [(BGR_WIDTH, HEIGHT / 2), (w + BGR_WIDTH, HEIGHT / 2)] + #x = w + BGR_WIDTH + #for i in reversed(LAST_SAMPLES): + # pts.insert(1, (x, int((h / 2) + (float(i) / MAX) * (h / 2)))) + # x -= 1 + # if x < BGR_WIDTH: + # break + #if len(pts) > 2: + # pygame.gfxdraw.aapolygon(disp, pts, [0, 255, 0]) + disp.blit(sampwin, (BGR_WIDTH, 0)) + pygame.display.flip() + + for i in pygame.event.get(): + pass # Todo + + clock.tick(60) + # Generator functions--should be cyclic within [0, 2*math.pi) and return [-1, 1] GENERATORS = [{'name': 'math.sin', 'args': None, 'desc': 'Sine function'}, @@ -179,7 +253,7 @@ def to_data(samps): return struct.pack('i'*len(samps), *samps) def gen_data(data, frames, time, status): - global FREQ, PHASE, Z_SAMP, LAST_SAMP + global FREQ, PHASE, Z_SAMP, LAST_SAMP, LAST_SAMPLES if FREQ == 0: PHASE = 0 if LAST_SAMP == 0: @@ -188,12 +262,19 @@ def gen_data(data, frames, time, status): LAST_SAMP = fdata[-1] return (to_data(fdata), pyaudio.paContinue) fdata, PHASE = samps(FREQ, PHASE, frames) + if options.gui: + LAST_SAMPLES.extend(fdata) LAST_SAMP = fdata[-1] return (to_data(fdata), pyaudio.paContinue) pa = pyaudio.PyAudio() stream = pa.open(rate=RATE, channels=1, format=pyaudio.paInt32, output=True, frames_per_buffer=FPB, stream_callback=gen_data) +if options.gui: + guithread = threading.Thread(target=GUIs[options.gui]) + guithread.setDaemon(True) + guithread.start() + if options.test: FREQ = 440 time.sleep(1) -- cgit v1.2.3-70-g09d2 From 49a979f28bc779f8126015a0749fa93290a07ad3 Mon Sep 17 00:00:00 2001 From: Grissess Date: Sat, 2 Apr 2016 00:49:37 -0400 Subject: Stabilized GUI in client --- client.py | 44 +++++++++++++++++++++++++++++++++++++------- 1 file changed, 37 insertions(+), 7 deletions(-) diff --git a/client.py b/client.py index 935a7e0..87f4437 100644 --- a/client.py +++ b/client.py @@ -12,6 +12,7 @@ import optparse import array import random import threading +import thread from packet import Packet, CMD, stoi @@ -24,6 +25,10 @@ parser.add_option('-p', '--port', dest='port', type='int', default=13676, help=' parser.add_option('-r', '--rate', dest='rate', type='int', default=44100, help='Set the sample rate of the audio device') parser.add_option('-V', '--volume', dest='volume', type='float', default=1.0, help='Set the volume factor (>1 distorts, <1 attenuates)') parser.add_option('-G', '--gui', dest='gui', default='', help='set a GUI to use') +parser.add_option('--pg-fullscreen', dest='fullscreen', action='store_true', help='Use a full-screen video mode') +parser.add_option('--pg-samp-width', dest='samp_width', type='int', help='Set the width of the sample pane (by default display width / 2)') +parser.add_option('--pg-bgr-width', dest='bgr_width', type='int', help='Set the width of the bargraph pane (by default display width / 2)') +parser.add_option('--pg-height', dest='height', type='int', help='Set the height of the window or full-screen video mode') options, args = parser.parse_args() @@ -61,11 +66,28 @@ def pygame_notes(): import pygame.gfxdraw pygame.init() - SAMP_WIDTH = 512 - BGR_WIDTH = 512 - HEIGHT = 1024 - - disp = pygame.display.set_mode((SAMP_WIDTH + BGR_WIDTH, HEIGHT)) + dispinfo = pygame.display.Info() + DISP_WIDTH = 640 + DISP_HEIGHT = 480 + if dispinfo.current_h > 0 and dispinfo.current_w > 0: + DISP_WIDTH = dispinfo.current_w + DISP_HEIGHT = dispinfo.current_h + + SAMP_WIDTH = DISP_WIDTH / 2 + if options.samp_width > 0: + SAMP_WIDTH = options.samp_width + BGR_WIDTH = DISP_WIDTH / 2 + if options.bgr_width > 0: + NGR_WIDTH = options.bgr_width + HEIGHT = DISP_HEIGHT + if options.height > 0: + HEIGHT = options.height + + flags = 0 + if options.fullscreen: + flags |= pygame.FULLSCREEN + + disp = pygame.display.set_mode((SAMP_WIDTH + BGR_WIDTH, HEIGHT), flags) WIDTH, HEIGHT = disp.get_size() SAMP_WIDTH = WIDTH / 2 @@ -113,8 +135,12 @@ def pygame_notes(): disp.blit(sampwin, (BGR_WIDTH, 0)) pygame.display.flip() - for i in pygame.event.get(): - pass # Todo + for ev in pygame.event.get(): + if ev.type == pygame.KEYDOWN: + if ev.key == pygame.K_ESCAPE: + thread.interrupt_main() + pygame.quit() + exit() clock.tick(60) @@ -257,8 +283,12 @@ def gen_data(data, frames, time, status): if FREQ == 0: PHASE = 0 if LAST_SAMP == 0: + if options.gui: + LAST_SAMPLES.extend([0]*frames) return (Z_SAMP*frames, pyaudio.paContinue) fdata = lin_seq(LAST_SAMP, 0, frames) + if options.gui: + LAST_SAMPLES.extend(fdata) LAST_SAMP = fdata[-1] return (to_data(fdata), pyaudio.paContinue) fdata, PHASE = samps(FREQ, PHASE, frames) -- cgit v1.2.3-70-g09d2 From 5192480ed1022f8365fe25245933fd6b553970c4 Mon Sep 17 00:00:00 2001 From: Grissess Date: Sat, 2 Apr 2016 03:35:08 -0400 Subject: Better random support, option to disable sustain --- broadcast.py | 10 ++++++++-- 1 file changed, 8 insertions(+), 2 deletions(-) diff --git a/broadcast.py b/broadcast.py index da0ee00..0731503 100644 --- a/broadcast.py +++ b/broadcast.py @@ -18,6 +18,7 @@ parser.add_option('--rand-low', dest='rand_low', type='int', help='Low frequency parser.add_option('--rand-high', dest='rand_high', type='int', help='High frequency to randomly sample') parser.add_option('-l', '--live', dest='live', help='Enter live mode (play from a controller in real time), specifying the port to connect to as "client,port"; use just "," to manually subscribe later') parser.add_option('-L', '--list-live', dest='list_live', action='store_true', help='List all the clients and ports that can be connected to for live performance') +parser.add_option('--no-sustain', dest='no_sustain', action='store_true', help='Don\'t use sustain hacks in live mode') parser.add_option('-q', '--quit', dest='quit', action='store_true', help='Instruct all clients to quit') parser.add_option('-p', '--play', dest='play', action='append', help='Play a single tone or chord (specified multiple times) on all listening clients (either "midi pitch" or "@frequency")') parser.add_option('-P', '--play-async', dest='play_async', action='store_true', help='Don\'t wait for the tone to finish using the local clock') @@ -129,7 +130,7 @@ if options.test or options.quit or options.silence: if options.random > 0: while True: for cl in clients: - s.sendto(str(Packet(CMD.PLAY, int(options.random), int(1000000*(options.random-int(options.random))), random.randint(options.rand_low, options.rand_high), 255)), cl) + s.sendto(str(Packet(CMD.PLAY, int(options.random), int(1000000*(options.random-int(options.random))), random.randint(options.rand_low, options.rand_high), options.volume)), cl) time.sleep(options.random) if options.live or options.list_live: @@ -148,6 +149,7 @@ if options.live or options.list_live: if client or port: seq.subscribe_port(client, port) seq.start_sequencer() + seq.set_nonblock(False) while True: ev = S.event_input(seq.client) event = None @@ -198,9 +200,13 @@ if options.live or options.list_live: s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), cli) if options.verbose: print 'LIVE:', event.pitch, '- =>', active_set[event.pitch] + if sustain_status: + print '...ignored (sustain on)' elif isinstance(event, midi.ControlChangeEvent): - if event.control == 64: + if event.control == 64 and not options.no_sustain: sustain_status = (event.value >= 64) + if options.verbose: + print 'LIVE: SUSTAIN', ('+' if sustain_status else '-') if not sustain_status: for pitch in deferred_set: if pitch not in active_set or not active_set[pitch]: -- cgit v1.2.3-70-g09d2 From b6ab9fcbec899e345d81554d60111e95cf9ce466 Mon Sep 17 00:00:00 2001 From: Grissess Date: Wed, 20 Apr 2016 04:13:13 -0400 Subject: Bugfixes, graphics, and new generators --- broadcast.py | 99 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++--- client.py | 10 +++++- mkiv.py | 77 ++++++++++++++++++++++++++++++++++------------ shiv.py | 36 ++++++++++++++++++++++ 4 files changed, 197 insertions(+), 25 deletions(-) diff --git a/broadcast.py b/broadcast.py index 0731503..2a7ce3c 100644 --- a/broadcast.py +++ b/broadcast.py @@ -4,6 +4,7 @@ import struct import time import xml.etree.ElementTree as ET import threading +import thread import optparse import random @@ -11,6 +12,7 @@ from packet import Packet, CMD, itos parser = optparse.OptionParser() parser.add_option('-t', '--test', dest='test', action='store_true', help='Play a test tone (440, 880) on all clients in sequence (the last overlaps with the first of the next)') +parser.add_option('--test-delay', dest='test_delay', type='float', help='Time for which to play a test tone') parser.add_option('-T', '--transpose', dest='transpose', type='int', help='Transpose by a set amount of semitones (positive or negative)') parser.add_option('--sync-test', dest='sync_test', action='store_true', help='Don\'t wait for clients to play tones properly--have them all test tone at the same time') parser.add_option('-R', '--random', dest='random', type='float', help='Generate random notes at approximately this period') @@ -31,8 +33,12 @@ parser.add_option('-r', '--route', dest='routes', action='append', help='Add a r 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('-B', '--bind-addr', dest='bind_addr', help='The IP address (or IP:port) to bind to (influences the network to send to)') +parser.add_option('-G', '--gui', dest='gui', default='', help='set a GUI to use') +parser.add_option('--pg-fullscreen', dest='fullscreen', action='store_true', help='Use a full-screen video mode') +parser.add_option('--pg-width', dest='pg_width', type='int', help='Width of the pygame window') +parser.add_option('--pg-height', dest='pg_height', type='int', help='Width of the pygame window') 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, seek=0.0, bind_addr='') +parser.set_defaults(routes=[], test_delay=0.25, 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, bind_addr='', pg_width = 0, pg_height = 0) options, args = parser.parse_args() if options.help_routes: @@ -51,6 +57,66 @@ The syntax for that specification resembles the following: The specifier consists of a comma-separated list of attribute-colon-value pairs, followed by an equal sign. After this is a comma-separated list of exclusivities paired with the name of a stream group as specified in the file. The above example shows that stream groups "bass", "treble", and "beeps" will be routed to clients with UID "bass", "treble", and TYPE "BEEP" respectively. Additionally, TYPE "BEEP" will receive tracks 4 and 6 (indices 3 and 5) of the MIDI file (presumably split with -T), and that these three groups are exclusively to be routed to TYPE "BEEP" clients only (the broadcaster will drop the stream if no more are available), as opposed to the preference of the bass and treble groups, which may be routed onto other stream clients if they are available. Finally, the last route says that all "noise" UID clients should not proceed any further (receiving "null" streams) instead. Order is important; if a "noise" client already received a stream (such as "+beeps"), then it would receive that route with priority.''' exit() +GUIS = {} + +def gui_pygame(): + print 'Starting pygame GUI...' + import pygame, colorsys + pygame.init() + print 'Pygame init' + + dispinfo = pygame.display.Info() + DISP_WIDTH = 640 + DISP_HEIGHT = 480 + if dispinfo.current_h > 0 and dispinfo.current_w > 0: + DISP_WIDTH = dispinfo.current_w + DISP_HEIGHT = dispinfo.current_h + print 'Pygame info' + + WIDTH = DISP_WIDTH + if options.pg_width > 0: + WIDTH = options.pg_width + HEIGHT = DISP_HEIGHT + if options.pg_height > 0: + HEIGHT = options.pg_height + + flags = 0 + if options.fullscreen: + flags |= pygame.FULLSCREEN + + disp = pygame.display.set_mode((WIDTH, HEIGHT), flags) + print 'Disp acquire' + + PFAC = HEIGHT / 128.0 + + clock = pygame.time.Clock() + + print 'Pygame GUI initialized, running...' + + while True: + + disp.scroll(-1, 0) + disp.fill((0, 0, 0), (WIDTH - 1, 0, 1, HEIGHT)) + idx = 0 + for cli, note in sorted(playing_notes.items(), key = lambda pair: pair[0]): + pitch = note[0] + col = colorsys.hls_to_rgb(float(idx) / len(clients), note[1]/512.0, 1.0) + col = [int(i*255) for i in col] + disp.fill(col, (WIDTH - 1, HEIGHT - pitch * PFAC - PFAC, 1, PFAC)) + idx += 1 + pygame.display.flip() + + for ev in pygame.event.get(): + if ev.type == pygame.KEYDOWN: + if ev.key == pygame.K_ESCAPE: + thread.interrupt_main() + pygame.quit() + exit() + + clock.tick(60) + +GUIS['pygame'] = gui_pygame + PORT = 13676 factor = options.factor @@ -78,6 +144,10 @@ try: except socket.timeout: pass +playing_notes = {} +for cli in clients: + playing_notes[cli] = (0, 0) + print len(clients), 'detected clients' print 'Clients:' @@ -96,15 +166,21 @@ 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, options.volume)), cl) + ts, tms = int(options.test_delay), int(options.test_delay * 1000000) % 1000000 + s.sendto(str(Packet(CMD.PLAY, ts, tms, 440, options.volume)), cl) if not options.sync_test: - time.sleep(0.25) - s.sendto(str(Packet(CMD.PLAY, 0, 250000, 880, options.volume)), cl) + time.sleep(options.test_delay) + s.sendto(str(Packet(CMD.PLAY, ts, tms, 880, options.volume)), cl) if options.quit: s.sendto(str(Packet(CMD.QUIT)), cl) if options.silence: s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), cl) +if options.gui: + gui_thr = threading.Thread(target=GUIS[options.gui], args=()) + gui_thr.setDaemon(True) + gui_thr.start() + if options.play: for i, val in enumerate(options.play): if val.startswith('@'): @@ -134,6 +210,9 @@ if options.random > 0: time.sleep(options.random) if options.live or options.list_live: + if options.gui: + print 'Waiting a second for GUI init...' + time.sleep(3.0) import midi from midi import sequencer S = sequencer.S @@ -149,9 +228,12 @@ if options.live or options.list_live: if client or port: seq.subscribe_port(client, port) seq.start_sequencer() - seq.set_nonblock(False) + if not options.gui: # FIXME + seq.set_nonblock(False) while True: ev = S.event_input(seq.client) + if ev is None: + time.sleep(0) event = None if ev: if options.verbose: @@ -187,6 +269,7 @@ if options.live or options.list_live: 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.setdefault(event.pitch, []).append(cli) + playing_notes[cli] = (event.pitch, 2*event.velocity) if options.verbose: print 'LIVE:', event.pitch, '+ =>', active_set[event.pitch] elif isinstance(event, midi.NoteOffEvent): @@ -198,6 +281,7 @@ if options.live or options.list_live: continue cli = active_set[event.pitch].pop() s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), cli) + playing_notes[cli] = (0, 0) if options.verbose: print 'LIVE:', event.pitch, '- =>', active_set[event.pitch] if sustain_status: @@ -214,6 +298,7 @@ if options.live or options.list_live: continue for cli in active_set[pitch]: s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), cli) + playing_notes[cli] = (0, 0) del active_set[pitch] deferred_set.clear() @@ -238,6 +323,8 @@ for fname in args: self.map = uid_groups elif fattr == 'T': self.map = type_groups + elif fattr == '0': + self.map = {} else: raise ValueError('Not a valid attribute specifier: %r'%(fattr,)) self.value = fvalue @@ -385,7 +472,9 @@ for fname in args: s.sendto(str(Packet(CMD.PLAY, int(dur), int((dur*1000000)%1000000), int(440.0 * 2**((pitch-69)/12.0)), int(vel*2 * options.volume/255.0))), cl) if options.verbose: print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel + playing_notes[cl] = (pitch, vel*2) self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) + playing_notes[cl] = (0, 0) if options.verbose: print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' diff --git a/client.py b/client.py index 87f4437..91ef663 100644 --- a/client.py +++ b/client.py @@ -78,7 +78,7 @@ def pygame_notes(): SAMP_WIDTH = options.samp_width BGR_WIDTH = DISP_WIDTH / 2 if options.bgr_width > 0: - NGR_WIDTH = options.bgr_width + BGR_WIDTH = options.bgr_width HEIGHT = DISP_HEIGHT if options.height > 0: HEIGHT = options.height @@ -214,6 +214,14 @@ class harmonic(object): def __call__(self, theta): return max(-1, min(1, sum([amp*self.gen((i+1)*theta % (2*math.pi)) for i, amp in enumerate(self.spectrum)]))) +@generator('General harmonics generator (adds arbitrary overtones)', '(, , , , , ...)') +class genharmonic(object): + def __init__(self, gen, *harmonics): + self.gen = gen + self.harmonics = zip(harmonics[::2], harmonics[1::2]) + def __call__(self, theta): + return max(-1, min(1, sum([amp * self.gen(i * theta % (2*math.pi)) for i, amp in self.harmonics]))) + @generator('Mix generator', '([, ], [[, ], [...]])') class mixer(object): def __init__(self, *specs): diff --git a/mkiv.py b/mkiv.py index 2293861..0d2bfa0 100644 --- a/mkiv.py +++ b/mkiv.py @@ -6,10 +6,8 @@ This simple script (using python-midi) reads a MIDI file and makes an interval (.iv) file (actually XML) that contains non-overlapping notes. TODO: --Reserve channels by track --Reserve channels by MIDI channel --Pitch limits for channels -MIDI Control events +-Percussion ''' import xml.etree.ElementTree as ET @@ -27,10 +25,12 @@ parser.add_option('-c', '--preserve-channels', dest='chanskeep', action='store_t parser.add_option('-T', '--track-split', dest='tracks', action='append_const', const=TRACKS, help='Ensure all tracks are on non-mutual streams') parser.add_option('-t', '--track', dest='tracks', action='append', help='Reserve an exclusive set of streams for certain conditions (try --help-conds)') parser.add_option('--help-conds', dest='help_conds', action='store_true', help='Print help on filter conditions for streams') -parser.add_option('-P', '--no-percussion', dest='no_perc', action='store_true', help='Don\'t try to filter percussion events out') +parser.add_option('-P', '--percussion', dest='perc', help='Which percussion standard to use to automatically filter to "perc" (GM, GM2, or none)') parser.add_option('-f', '--fuckit', dest='fuckit', action='store_true', help='Use the Python Error Steamroller when importing MIDIs (useful for extended formats)') parser.add_option('-n', '--target-num', dest='repeaterNumber', type='int', help='Target count of devices') -parser.set_defaults(tracks=[], repeaterNumber=1) +parser.add_option('-v', '--verbose', dest='verbose', action='store_true', help='Be verbose; show important parts about the MIDI scheduling process') +parser.add_option('-d', '--debug', dest='debug', action='store_true', help='Debugging output; show excessive output about the MIDI scheduling process') +parser.set_defaults(tracks=[], repeaterNumber=1, perc='GM') options, args = parser.parse_args() if options.help_conds: @@ -40,9 +40,14 @@ Every filter is an expression; internally, this expression is evaluated as the b The "ev" object will be a MergeEvent with the following properties: -ev.tidx: the originating track index (starting at 0) -ev.abstime: the real time in seconds of this event relative to the beginning of playback +-ev.bank: the selected bank (all bits) +-ev.prog: the selected program -ev.ev: a midi.NoteOnEvent: -ev.ev.pitch: the MIDI pitch -ev.ev.velocity: the MIDI velocity + -ev.ev.channel: the MIDI channel + +All valid Python expressions are accepted. Take care to observe proper shell escaping. Specifying a -t = will group all streams under a filter; if the part is omitted, no group will be added. For example: @@ -122,31 +127,59 @@ for fname in args: print 'Merging events...' class MergeEvent(object): - __slots__ = ['ev', 'tidx', 'abstime'] - def __init__(self, ev, tidx, abstime): + __slots__ = ['ev', 'tidx', 'abstime', 'bank', 'prog'] + def __init__(self, ev, tidx, abstime, bank, prog): self.ev = ev self.tidx = tidx self.abstime = abstime + self.bank = bank + self.prog = prog def __repr__(self): return ''%(self.ev, self.tidx, self.abstime) events = [] bpm_at = {0: 120} + cur_bank = [[0 for i in range(16)] for j in range(len(pat))] + cur_prog = [[0 for i in range(16)] for j in range(len(pat))] + chg_bank = [[0 for i in range(16)] for j in range(len(pat))] + chg_prog = [[0 for i in range(16)] for j in range(len(pat))] + ev_cnts = [[0 for i in range(16)] for j in range(len(pat))] + tnames = [''] * len(pat) for tidx, track in enumerate(pat): abstime = 0 absticks = 0 for ev in track: + if options.debug: + print ev if isinstance(ev, midi.SetTempoEvent): absticks += ev.tick bpm_at[absticks] = ev.bpm - else: + elif isinstance(ev, midi.ProgramChangeEvent): + cur_prog[tidx][ev.channel] = ev.value + chg_prog[tidx][ev.channel] += 1 + elif isinstance(ev, midi.ControlChangeEvent): + if ev.control == 0: + cur_bank[tidx][ev.channel] = (0x3F80 & cur_bank[tidx][ev.channel]) | ev.value + chg_bank[tidx][ev.channel] += 1 + elif ev.control == 32: + cur_bank[tidx][ev.channel] = (0x3F & cur_bank[tidx][ev.channel]) | (ev.value << 7) + chg_bank[tidx][ev.channel] += 1 + elif isinstance(ev, midi.TrackNameEvent): + tnames[tidx] = ev.text + elif isinstance(ev, midi.Event): if isinstance(ev, midi.NoteOnEvent) and ev.velocity == 0: ev.__class__ = midi.NoteOffEvent #XXX Oww bpm = filter(lambda pair: pair[0] <= absticks, sorted(bpm_at.items(), key=lambda pair: pair[0]))[-1][1] abstime += (60.0 * ev.tick) / (bpm * pat.resolution) absticks += ev.tick - events.append(MergeEvent(ev, tidx, abstime)) + events.append(MergeEvent(ev, tidx, abstime, cur_bank[tidx][ev.channel], cur_prog[tidx][ev.channel])) + ev_cnts[tidx][ev.channel] += 1 + + if options.verbose: + print 'Track name, event count, final banks, bank changes, final programs, program changes:' + for tidx, tname in enumerate(tnames): + print tidx, ':', tname, ',', ','.join(map(str, ev_cnts[tidx])), ',', ','.join(map(str, cur_bank[tidx])), ',', ','.join(map(str, chg_bank[tidx])), ',', ','.join(map(str, cur_prog[tidx])), ',', ','.join(map(str, chg_prog[tidx])) print 'Sorting events...' @@ -158,7 +191,7 @@ for fname in args: class DurationEvent(MergeEvent): __slots__ = ['duration'] def __init__(self, me, dur): - MergeEvent.__init__(self, me.ev, me.tidx, me.abstime) + MergeEvent.__init__(self, me.ev, me.tidx, me.abstime, me.bank, me.prog) self.duration = dur class NoteStream(object): @@ -200,8 +233,13 @@ for fname in args: notegroups = [] auxstream = [] - if not options.no_perc: - notegroups.append(NSGroup(filter = lambda mev: mev.ev.channel == 10, name='perc')) + if options.perc and options.perc != 'none': + if options.perc == 'GM': + notegroups.append(NSGroup(filter = lambda mev: mev.ev.channel == 9, name='perc')) + elif options.perc == 'GM2': + notegroups.append(NSGroup(filter = lambda mev: mev.bank == 15360, name='perc')) + else: + print 'Unrecognized --percussion option %r; should be GM, GM2, or none'%(options.perc,) for spec in options.tracks: if spec is TRACKS: @@ -214,9 +252,10 @@ for fname in args: name = None notegroups.append(NSGroup(filter = eval("lambda ev: "+spec), name = name)) - print 'Initial group mappings:' - for group in notegroups: - print ('' if group.name is None else group.name), '<=', group.filter + if options.verbose: + print 'Initial group mappings:' + for group in notegroups: + print ('' if group.name is None else group.name) for mev in events: if isinstance(mev.ev, midi.NoteOnEvent): @@ -250,9 +289,10 @@ for fname in args: print 'WARNING: Active notes at end of playback.' ns.Deactivate(MergeEvent(ns.active, ns.active.tidx, lastabstime)) - print 'Final group mappings:' - for group in notegroups: - print ('' if group.name is None else group.name), '<=', group.filter, '(', len(group.streams), 'streams)' + if options.verbose: + print 'Final group mappings:' + for group in notegroups: + print ('' if group.name is None else group.name), '<=', '(', len(group.streams), 'streams)' print 'Generated %d streams in %d groups'%(sum(map(lambda x: len(x.streams), notegroups)), len(notegroups)) print 'Playtime:', lastabstime, 'seconds' @@ -289,7 +329,6 @@ for fname in args: ivnote.set('time', str(note.abstime)) ivnote.set('dur', str(note.duration)) x+=1 - print x if(x>=options.repeaterNumber and options.repeaterNumber!=1): break if(x>=options.repeaterNumber and options.repeaterNumber!=1): diff --git a/shiv.py b/shiv.py index 96d8ea2..423bdf7 100644 --- a/shiv.py +++ b/shiv.py @@ -42,3 +42,39 @@ for fname in args: print 'File :', fname print '\t' + if options.meta: + print 'Metatrack:', + meta = iv.find('./meta') + if meta: + print 'exists' + print '\tBPM track:', + bpms = meta.find('./bpms') + if bpms: + print 'exists' + for elem in bpms.iterfind('./bpm'): + print '\t\tAt ticks {}, time {}: {} bpm'.format(elem.get('ticks'), elem.get('time'), elem.get('bpm')) + + if not (options.number or options.groups or options.notes or options.histogram or options.histogram_tracks or options.duration or options.duty_cycle): + continue + + streams = iv.findall('./streams/stream') + notestreams = [s for s in streams if s.get('type') == 'ns'] + if options.number: + print 'Stream count:' + print '\tNotestreams:', len(notestreams) + print '\tTotal:', len(streams) + + if not (options.groups or options.notes or options.histogram or options.histogram_tracks or options.duration or options.duty_cycle): + continue + + if options.groups: + groups = {} + for s in notestreams: + group = s.get('group', ' Date: Fri, 22 Apr 2016 19:15:16 -0400 Subject: Added saw wave --- client.py | 4 ++++ mkiv.py | 46 ++++++++++++++++++++++++++++++++++++++-------- shiv.py | 4 ++-- 3 files changed, 44 insertions(+), 10 deletions(-) diff --git a/client.py b/client.py index 91ef663..01344c1 100644 --- a/client.py +++ b/client.py @@ -166,6 +166,10 @@ def tri_wave(theta): else: return lin_interp(-1, 0, (theta-3*math.pi/2)/(math.pi/2)) +@generator('Saw wave (line from (0, 1) to (2pi, -1))') +def saw_wave(theta): + return lin_interp(1, -1, theta/(math.pi * 2)) + @generator('Simple square wave (piecewise 1 at x= btimes[i] and sev.abstick < btimes[i+1], sorted_events) + print fname, ': BPM partition', i, 'contains', len(fev), 'events' + class MergeEvent(object): __slots__ = ['ev', 'tidx', 'abstime', 'bank', 'prog'] def __init__(self, ev, tidx, abstime, bank, prog): @@ -138,7 +172,6 @@ for fname in args: return ''%(self.ev, self.tidx, self.abstime) events = [] - bpm_at = {0: 120} cur_bank = [[0 for i in range(16)] for j in range(len(pat))] cur_prog = [[0 for i in range(16)] for j in range(len(pat))] chg_bank = [[0 for i in range(16)] for j in range(len(pat))] @@ -150,12 +183,11 @@ for fname in args: abstime = 0 absticks = 0 for ev in track: + bpm = filter(lambda pair: pair[0] <= absticks, sorted(bpm_at.items(), key=lambda pair: pair[0]))[-1][1] if options.debug: - print ev - if isinstance(ev, midi.SetTempoEvent): - absticks += ev.tick - bpm_at[absticks] = ev.bpm - elif isinstance(ev, midi.ProgramChangeEvent): + print ev, ': bpm=', bpm + absticks += ev.tick + if isinstance(ev, midi.ProgramChangeEvent): cur_prog[tidx][ev.channel] = ev.value chg_prog[tidx][ev.channel] += 1 elif isinstance(ev, midi.ControlChangeEvent): @@ -170,9 +202,7 @@ for fname in args: elif isinstance(ev, midi.Event): if isinstance(ev, midi.NoteOnEvent) and ev.velocity == 0: ev.__class__ = midi.NoteOffEvent #XXX Oww - bpm = filter(lambda pair: pair[0] <= absticks, sorted(bpm_at.items(), key=lambda pair: pair[0]))[-1][1] abstime += (60.0 * ev.tick) / (bpm * pat.resolution) - absticks += ev.tick events.append(MergeEvent(ev, tidx, abstime, cur_bank[tidx][ev.channel], cur_prog[tidx][ev.channel])) ev_cnts[tidx][ev.channel] += 1 diff --git a/shiv.py b/shiv.py index 423bdf7..80411e3 100644 --- a/shiv.py +++ b/shiv.py @@ -9,8 +9,8 @@ parser.add_option('-n', '--number', dest='number', action='store_true', help='Sh parser.add_option('-g', '--groups', dest='groups', action='store_true', help='Show group names') parser.add_option('-N', '--notes', dest='notes', action='store_true', help='Show number of notes') parser.add_option('-m', '--meta', dest='meta', action='store_true', help='Show meta track information') -parser.add_option('-h', '--histogram', dest='histogram', action='store_true', help='Show a histogram distribution of pitches') -parser.add_option('-H', '--histogram-tracks', dest='histogram_tracks', action='store_true', help='Show a histogram distribution of pitches per track') +parser.add_option('--histogram', dest='histogram', action='store_true', help='Show a histogram distribution of pitches') +parser.add_option('--histogram-tracks', dest='histogram_tracks', action='store_true', help='Show a histogram distribution of pitches per track') parser.add_option('-d', '--duration', dest='duration', action='store_true', help='Show the duration of the piece') parser.add_option('-D', '--duty-cycle', dest='duty_cycle', action='store_true', help='Show the duration of the notes within tracks, and as a percentage of the piece duration') -- cgit v1.2.3-70-g09d2 From d59e5fdb05afe7a9bc73ab8bbf1f44cf4777238c Mon Sep 17 00:00:00 2001 From: Grissess Date: Sat, 23 Apr 2016 20:52:20 -0400 Subject: Fixed shiv --- shiv.py | 105 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++---- 1 file changed, 99 insertions(+), 6 deletions(-) diff --git a/shiv.py b/shiv.py index 80411e3..07f0bdd 100644 --- a/shiv.py +++ b/shiv.py @@ -8,27 +8,54 @@ parser = optparse.OptionParser() parser.add_option('-n', '--number', dest='number', action='store_true', help='Show number of tracks') parser.add_option('-g', '--groups', dest='groups', action='store_true', help='Show group names') parser.add_option('-N', '--notes', dest='notes', action='store_true', help='Show number of notes') +parser.add_option('-M', '--notes-stream', dest='notes_stream', action='store_true', help='Show notes per stream') parser.add_option('-m', '--meta', dest='meta', action='store_true', help='Show meta track information') parser.add_option('--histogram', dest='histogram', action='store_true', help='Show a histogram distribution of pitches') parser.add_option('--histogram-tracks', dest='histogram_tracks', action='store_true', help='Show a histogram distribution of pitches per track') +parser.add_option('--vel-hist', dest='vel_hist', action='store_true', help='Show a histogram distribution of velocities') +parser.add_option('--vel-hist-tracks', dest='vel_hist_tracks', action='store_true', help='Show a histogram distributions of velocities per track') parser.add_option('-d', '--duration', dest='duration', action='store_true', help='Show the duration of the piece') parser.add_option('-D', '--duty-cycle', dest='duty_cycle', action='store_true', help='Show the duration of the notes within tracks, and as a percentage of the piece duration') +parser.add_option('-H', '--height', dest='height', type='int', help='Height of histograms') parser.add_option('-a', '--almost-all', dest='almost_all', action='store_true', help='Show useful information') parser.add_option('-A', '--all', dest='all', action='store_true', help='Show everything') +parser.set_defaults(height=20) + options, args = parser.parse_args() if options.almost_all or options.all: options.number = True options.groups = True options.notes = True + options.notes_stream = True options.histogram = True + options.vel_hist = True options.duration = True + options.duty_cycle = True + options.meta = True if options.all: - options.meta = True options.histogram_tracks= True - options.duty_cycle = True + options.vel_hist_tracks = True + +def show_hist(values, height=None): + if not values: + print '{empty histogram}' + if height is None: + height = options.height + xs, ys = values.keys(), values.values() + minx, maxx = min(xs), max(xs) + miny, maxy = min(ys), max(ys) + xv = range(minx, maxx + 1) + incs = max((maxy - miny) / height, 1) + print '\t --' + '-' * len(xv) + for ub in range(maxy + incs, miny, -incs): + print '{}\t | {}'.format(ub, ''.join(['#' if values.get(x) > (ub - incs) else ' ' for x in xv])) + print '\t |-' + '-' * len(xv) + xvs = map(str, xv) + for i in range(max(map(len, xvs))): + print '\t ' + ''.join([s[i] if len(s) > i else ' ' for s in xvs]) for fname in args: try: @@ -45,11 +72,11 @@ for fname in args: if options.meta: print 'Metatrack:', meta = iv.find('./meta') - if meta: + if len(meta): print 'exists' print '\tBPM track:', bpms = meta.find('./bpms') - if bpms: + if len(bpms): print 'exists' for elem in bpms.iterfind('./bpm'): print '\t\tAt ticks {}, time {}: {} bpm'.format(elem.get('ticks'), elem.get('time'), elem.get('bpm')) @@ -70,11 +97,77 @@ for fname in args: if options.groups: groups = {} for s in notestreams: - group = s.get('group', '') groups[group] = groups.get(group, 0) + 1 print 'Groups:' for name, cnt in groups.iteritems(): print '\t{} ({} streams)'.format(name, cnt) - if not (options.notes or options.histogram or options.histogram_tracks or options.duration or options.duty_cycle): + if not (options.notes or options.notes_stream or options.histogram or options.histogram_tracks or options.duration or options.duty_cycle): continue + + if options.notes: + note_cnt = 0 + if options.notes_stream: + notes_stream = [0] * len(notestreams) + if options.histogram: + pitches = {} + if options.histogram_tracks: + pitch_tracks = [{} for i in notestreams] + if options.vel_hist: + velocities = {} + if options.vel_hist_tracks: + velocities_tracks = [{} for i in notestreams] + if options.duration or options.duty_cycle: + max_dur = 0 + if options.duty_cycle: + cum_dur = [0.0] * len(notestreams) + + for sidx, stream in enumerate(notestreams): + notes = stream.findall('note') + for note in notes: + pitch = int(note.get('pitch')) + vel = int(note.get('vel')) + time = float(note.get('time')) + dur = float(note.get('dur')) + if options.notes: + note_cnt += 1 + if options.notes_stream: + notes_stream[sidx] += 1 + if options.histogram: + pitches[pitch] = pitches.get(pitch, 0) + 1 + if options.histogram_tracks: + pitch_tracks[sidx][pitch] = pitch_tracks[sidx].get(pitch, 0) + 1 + if options.vel_hist: + velocities[vel] = velocities.get(vel, 0) + 1 + if options.vel_hist_tracks: + velocities_tracks[sidx][vel] = velocities_tracks[sidx].get(vel, 0) + 1 + if (options.duration or options.duty_cycle) and time + dur > max_dur: + max_dur = time + dur + if options.duty_cycle: + cum_dur[sidx] += dur + + if options.histogram_tracks: + for sidx, hist in enumerate(pitch_tracks): + print 'Stream {} (group {}) pitch histogram:'.format(sidx, notestreams[sidx].get('group', '')) + show_hist(hist) + if options.vel_hist_tracks: + for sidx, hist in enumerate(velocities_tracks): + print 'Stream {} (group {}) velocity histogram:'.format(sidx, notestreams[sidx].get('group', '')) + show_hist(hist) + if options.notes_stream: + for sidx, value in enumerate(notes_stream): + print 'Stream {} (group {}) note count: {}'.format(sidx, notestreams[sidx].get('group', ''), value) + if options.duty_cycle: + for sidx, value in enumerate(cum_dur): + print 'Stream {} (group {}) duty cycle: {}'.format(sidx, notestreams[sidx].get('group', ''), value / max_dur) + if options.notes: + print 'Total notes: {}'.format(note_cnt) + if options.histogram: + print 'Pitch histogram:' + show_hist(pitches) + if options.vel_hist: + print 'Velocity histogram:' + show_hist(velocities) + if options.duration: + print 'Playing duration: {}'.format(max_dur) -- cgit v1.2.3-70-g09d2 From 90fe1672d81de5a3a3b077c025f851470891b566 Mon Sep 17 00:00:00 2001 From: Grissess Date: Sun, 24 Apr 2016 23:36:07 -0400 Subject: Features and bugfixes all around --- client.py | 4 ++++ mkiv.py | 20 ++++++++++++++++++-- shiv.py | 63 +++++++++++++++++++++++++++++++++++++++++++++++++++++++-------- 3 files changed, 77 insertions(+), 10 deletions(-) diff --git a/client.py b/client.py index 01344c1..2d1ab40 100644 --- a/client.py +++ b/client.py @@ -141,6 +141,10 @@ def pygame_notes(): thread.interrupt_main() pygame.quit() exit() + elif ev.type == pygame.QUIT: + thread.interrupt_main() + pygame.quit() + exit() clock.tick(60) diff --git a/mkiv.py b/mkiv.py index 3a71847..ff2fdbf 100644 --- a/mkiv.py +++ b/mkiv.py @@ -17,6 +17,7 @@ import os import optparse TRACKS = object() +PROGRAMS = object() parser = optparse.OptionParser() parser.add_option('-s', '--channel-split', dest='chansplit', action='store_true', help='Split MIDI channels into independent tracks (as far as -T is concerned)') @@ -25,6 +26,7 @@ parser.add_option('-c', '--preserve-channels', dest='chanskeep', action='store_t parser.add_option('-T', '--track-split', dest='tracks', action='append_const', const=TRACKS, help='Ensure all tracks are on non-mutual streams') parser.add_option('-t', '--track', dest='tracks', action='append', help='Reserve an exclusive set of streams for certain conditions (try --help-conds)') parser.add_option('--help-conds', dest='help_conds', action='store_true', help='Print help on filter conditions for streams') +parser.add_option('-p', '--program-split', dest='tracks', action='append_const', const=PROGRAMS, help='Ensure all programs are on non-mutual streams (overrides -T presently)') parser.add_option('-P', '--percussion', dest='perc', help='Which percussion standard to use to automatically filter to "perc" (GM, GM2, or none)') parser.add_option('-f', '--fuckit', dest='fuckit', action='store_true', help='Use the Python Error Steamroller when importing MIDIs (useful for extended formats)') parser.add_option('-n', '--target-num', dest='repeaterNumber', type='int', help='Target count of devices') @@ -62,11 +64,19 @@ will cause these groups to be made: As can be seen, order of specification is important. Equally important is the location of -T, which should be at the end. -NoteOffEvents are always matched to the stream which has their corresponding NoteOnEvent (in track and pitch), and so are +NoteOffEvents are always matched to the stream which has their corresponding NoteOnEvent (in track, pitch, and channel), and so are not affected or observed by filters. If the filters specified are not a complete cover, an anonymous group will be created with no filter to contain the rest. If -it is desired to force this group to have a name, use -t =True.''' +it is desired to force this group to have a name, use -t =True. This should be placed at the end. + +-T behaves exactly as if: + -t trk0=ev.tidx==0 -t trk1=ev.tidx==1 -t trk2=ev.tidx==2 [...] +had been specified in its place, though it is automatically sized to the number of tracks. Similarly, -P operates as if + -t prg31=ev.prog==31 -t prg81=ev.prog==81 [...] +had been specified, again containing only the programs that were observed in the piece. + +Groups for which no streams are generated are not written to the resulting file.''' exit() if not args: @@ -178,6 +188,7 @@ for fname in args: chg_prog = [[0 for i in range(16)] for j in range(len(pat))] ev_cnts = [[0 for i in range(16)] for j in range(len(pat))] tnames = [''] * len(pat) + progs = set([0]) for tidx, track in enumerate(pat): abstime = 0 @@ -189,6 +200,7 @@ for fname in args: absticks += ev.tick if isinstance(ev, midi.ProgramChangeEvent): cur_prog[tidx][ev.channel] = ev.value + progs.add(ev.value) chg_prog[tidx][ev.channel] += 1 elif isinstance(ev, midi.ControlChangeEvent): if ev.control == 0: @@ -210,6 +222,7 @@ for fname in args: print 'Track name, event count, final banks, bank changes, final programs, program changes:' for tidx, tname in enumerate(tnames): print tidx, ':', tname, ',', ','.join(map(str, ev_cnts[tidx])), ',', ','.join(map(str, cur_bank[tidx])), ',', ','.join(map(str, chg_bank[tidx])), ',', ','.join(map(str, cur_prog[tidx])), ',', ','.join(map(str, chg_prog[tidx])) + print 'All programs observed:', progs print 'Sorting events...' @@ -275,6 +288,9 @@ for fname in args: if spec is TRACKS: for tidx in xrange(len(pat)): notegroups.append(NSGroup(filter = lambda mev, tidx=tidx: mev.tidx == tidx, name = 'trk%d'%(tidx,))) + elif spec is PROGRAMS: + for prog in progs: + notegroups.append(NSGroup(filter = lambda mev, prog=prog: mev.prog == prog, name = 'prg%d'%(prog,))) else: if '=' in spec: name, _, spec = spec.partition('=') diff --git a/shiv.py b/shiv.py index 07f0bdd..8444a72 100644 --- a/shiv.py +++ b/shiv.py @@ -17,6 +17,8 @@ parser.add_option('--vel-hist-tracks', dest='vel_hist_tracks', action='store_tru parser.add_option('-d', '--duration', dest='duration', action='store_true', help='Show the duration of the piece') parser.add_option('-D', '--duty-cycle', dest='duty_cycle', action='store_true', help='Show the duration of the notes within tracks, and as a percentage of the piece duration') parser.add_option('-H', '--height', dest='height', type='int', help='Height of histograms') +parser.add_option('-C', '--no-color', dest='no_color', action='store_true', help='Don\'t use ANSI color escapes') +parser.add_option('-x', '--aux', dest='aux', action='store_true', help='Show information about the auxiliary streams') parser.add_option('-a', '--almost-all', dest='almost_all', action='store_true', help='Show useful information') parser.add_option('-A', '--all', dest='all', action='store_true', help='Show everything') @@ -34,11 +36,31 @@ if options.almost_all or options.all: options.vel_hist = True options.duration = True options.duty_cycle = True - options.meta = True if options.all: + options.aux = True + options.meta = True options.histogram_tracks= True options.vel_hist_tracks = True +if options.no_color: + class COL: + NONE='' + RED='' + GREEN='' + BLUE='' + YELLOW='' + MAGENTA='' + CYAN='' +else: + class COL: + NONE='\x1b[0m' + RED='\x1b[31m' + GREEN='\x1b[32m' + BLUE='\x1b[34m' + YELLOW='\x1b[33m' + MAGENTA='\x1b[35m' + CYAN='\x1b[36m' + def show_hist(values, height=None): if not values: print '{empty histogram}' @@ -49,13 +71,18 @@ def show_hist(values, height=None): miny, maxy = min(ys), max(ys) xv = range(minx, maxx + 1) incs = max((maxy - miny) / height, 1) - print '\t --' + '-' * len(xv) + print COL.BLUE + '\t --' + '-' * len(xv) + COL.NONE for ub in range(maxy + incs, miny, -incs): - print '{}\t | {}'.format(ub, ''.join(['#' if values.get(x) > (ub - incs) else ' ' for x in xv])) - print '\t |-' + '-' * len(xv) + print '{}{}\t | {}{}{}'.format(COL.BLUE, ub, COL.YELLOW, ''.join(['#' if values.get(x) > (ub - incs) else ' ' for x in xv]), COL.NONE) + print COL.BLUE + '\t |-' + '-' * len(xv) + COL.NONE xvs = map(str, xv) for i in range(max(map(len, xvs))): - print '\t ' + ''.join([s[i] if len(s) > i else ' ' for s in xvs]) + print COL.BLUE + '\t ' + ''.join([s[i] if len(s) > i else ' ' for s in xvs]) + COL.NONE + print + xcs = map(str, [values.get(x, 0) for x in xv]) + for i in range(max(map(len, xcs))): + print COL.YELLOW + '\t ' + ''.join([s[i] if len(s) > i else ' ' for s in xcs]) + COL.NONE + print for fname in args: try: @@ -81,17 +108,18 @@ for fname in args: for elem in bpms.iterfind('./bpm'): print '\t\tAt ticks {}, time {}: {} bpm'.format(elem.get('ticks'), elem.get('time'), elem.get('bpm')) - if not (options.number or options.groups or options.notes or options.histogram or options.histogram_tracks or options.duration or options.duty_cycle): + if not (options.number or options.groups or options.notes or options.histogram or options.histogram_tracks or options.vel_hist or options.vel_hist_tracks or options.duration or options.duty_cycle or options.aux): continue streams = iv.findall('./streams/stream') notestreams = [s for s in streams if s.get('type') == 'ns'] + auxstreams = [s for s in streams if s.get('type') == 'aux'] if options.number: print 'Stream count:' print '\tNotestreams:', len(notestreams) print '\tTotal:', len(streams) - if not (options.groups or options.notes or options.histogram or options.histogram_tracks or options.duration or options.duty_cycle): + if not (options.groups or options.notes or options.histogram or options.histogram_tracks or options.vel_hist or options.vel_hist_tracks or options.duration or options.duty_cycle or options.aux): continue if options.groups: @@ -103,7 +131,26 @@ for fname in args: for name, cnt in groups.iteritems(): print '\t{} ({} streams)'.format(name, cnt) - if not (options.notes or options.notes_stream or options.histogram or options.histogram_tracks or options.duration or options.duty_cycle): + if options.aux: + import midi + fr = midi.FileReader() + fr.RunningStatus = None # XXX Hack + print 'Aux stream data:' + for aidx, astream in enumerate(auxstreams): + evs = astream.findall('ev') + failed = 0 + print '\tFrom stream {}, {} events:'.format(aidx, len(evs)) + for ev in evs: + try: + data = eval(ev.get('data')) + mev = fr.parse_midi_event(iter(data)) + except AssertionError: + failed += 1 + else: + print '\t\tAt time {}: {}'.format(ev.get('time'), mev) + print '\t\t(...and {} others which failed to parse)'.format(failed) + + if not (options.notes or options.notes_stream or options.histogram or options.histogram_tracks or options.vel_hist or options.vel_hist_tracks or options.duration or options.duty_cycle): continue if options.notes: -- cgit v1.2.3-70-g09d2 From 6faeb841a0cc452b9e94b3e9c36ab93623128475 Mon Sep 17 00:00:00 2001 From: Grissess Date: Thu, 9 Jun 2016 21:57:51 -0400 Subject: Minor changes --- mkiv.py | 2 ++ shiv.py | 8 ++++---- 2 files changed, 6 insertions(+), 4 deletions(-) diff --git a/mkiv.py b/mkiv.py index ff2fdbf..d069a93 100644 --- a/mkiv.py +++ b/mkiv.py @@ -155,6 +155,8 @@ for fname in args: sorted_events.sort(key=lambda x: x.abstick) bpm_at = {0: 120} + print 'Computing tempos...' + for sev in sorted_events: if isinstance(sev.ev, midi.SetTempoEvent): if options.debug: diff --git a/shiv.py b/shiv.py index 8444a72..0a8169e 100644 --- a/shiv.py +++ b/shiv.py @@ -71,13 +71,13 @@ def show_hist(values, height=None): miny, maxy = min(ys), max(ys) xv = range(minx, maxx + 1) incs = max((maxy - miny) / height, 1) - print COL.BLUE + '\t --' + '-' * len(xv) + COL.NONE + print COL.CYAN + '\t --' + '-' * len(xv) + COL.NONE for ub in range(maxy + incs, miny, -incs): - print '{}{}\t | {}{}{}'.format(COL.BLUE, ub, COL.YELLOW, ''.join(['#' if values.get(x) > (ub - incs) else ' ' for x in xv]), COL.NONE) - print COL.BLUE + '\t |-' + '-' * len(xv) + COL.NONE + print '{}{}\t | {}{}{}'.format(COL.CYAN, ub, COL.YELLOW, ''.join(['#' if values.get(x) > (ub - incs) else ' ' for x in xv]), COL.NONE) + print COL.CYAN + '\t |-' + '-' * len(xv) + COL.NONE xvs = map(str, xv) for i in range(max(map(len, xvs))): - print COL.BLUE + '\t ' + ''.join([s[i] if len(s) > i else ' ' for s in xvs]) + COL.NONE + print COL.CYAN + '\t ' + ''.join([s[i] if len(s) > i else ' ' for s in xvs]) + COL.NONE print xcs = map(str, [values.get(x, 0) for x in xv]) for i in range(max(map(len, xcs))): -- cgit v1.2.3-70-g09d2 From 5b1881000baec07d5102e360ef4bf232e93bb158 Mon Sep 17 00:00:00 2001 From: csguest Date: Thu, 9 Jun 2016 22:05:29 -0400 Subject: bugfix --- mkiv.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/mkiv.py b/mkiv.py index d069a93..4cd0fea 100644 --- a/mkiv.py +++ b/mkiv.py @@ -335,7 +335,7 @@ for fname in args: for ns in group.streams: if ns.IsActive(): print 'WARNING: Active notes at end of playback.' - ns.Deactivate(MergeEvent(ns.active, ns.active.tidx, lastabstime)) + ns.Deactivate(MergeEvent(ns.active, ns.active.tidx, lastabstime, 0, 0)) if options.verbose: print 'Final group mappings:' -- cgit v1.2.3-70-g09d2 From 0fc951601706982aeedf035dc4c5ae1c40c671cb Mon Sep 17 00:00:00 2001 From: Grissess Date: Fri, 10 Jun 2016 00:24:07 -0400 Subject: Pitch bend support --- broadcast.py | 2 +- mkiv.py | 59 ++++++++++++++++++++++++++++++++++++++++++++++++++--------- shiv.py | 5 +++-- 3 files changed, 54 insertions(+), 12 deletions(-) diff --git a/broadcast.py b/broadcast.py index 2a7ce3c..714533d 100644 --- a/broadcast.py +++ b/broadcast.py @@ -464,7 +464,7 @@ for fname in args: nsq, cl = self._Thread__args for note in nsq: ttime = float(note.get('time')) - pitch = int(note.get('pitch')) + options.transpose + pitch = float(note.get('pitch')) + options.transpose vel = int(note.get('vel')) dur = factor*float(note.get('dur')) while time.time() - BASETIME < factor*ttime: diff --git a/mkiv.py b/mkiv.py index 4cd0fea..dd63130 100644 --- a/mkiv.py +++ b/mkiv.py @@ -32,7 +32,8 @@ parser.add_option('-f', '--fuckit', dest='fuckit', action='store_true', help='Us parser.add_option('-n', '--target-num', dest='repeaterNumber', type='int', help='Target count of devices') parser.add_option('-v', '--verbose', dest='verbose', action='store_true', help='Be verbose; show important parts about the MIDI scheduling process') parser.add_option('-d', '--debug', dest='debug', action='store_true', help='Debugging output; show excessive output about the MIDI scheduling process') -parser.set_defaults(tracks=[], repeaterNumber=1, perc='GM') +parser.add_option('-D', '--deviation', dest='deviation', type='int', help='Amount (in semitones/MIDI pitch units) by which a fully deflected pitchbend modifies the base pitch (0 disables pitchbend processing)') +parser.set_defaults(tracks=[], repeaterNumber=1, perc='GM', deviation=2) options, args = parser.parse_args() if options.help_conds: @@ -180,8 +181,12 @@ for fname in args: self.abstime = abstime self.bank = bank self.prog = prog + def copy(self, **kwargs): + args = {'ev': self.ev, 'tidx': self.tidx, 'abstime': self.abstime, 'bank': self.bank, 'prog': self.prog} + args.update(kwargs) + return MergeEvent(**args) def __repr__(self): - return ''%(self.ev, self.tidx, self.abstime) + return ''%(self.ev, self.tidx, self.bank, self.prog, self.abstime) events = [] cur_bank = [[0 for i in range(16)] for j in range(len(pat))] @@ -234,27 +239,37 @@ for fname in args: print 'Generating streams...' class DurationEvent(MergeEvent): - __slots__ = ['duration'] - def __init__(self, me, dur): + __slots__ = ['duration', 'pitch'] + def __init__(self, me, pitch, dur): MergeEvent.__init__(self, me.ev, me.tidx, me.abstime, me.bank, me.prog) + self.pitch = pitch self.duration = dur class NoteStream(object): - __slots__ = ['history', 'active'] + __slots__ = ['history', 'active', 'realpitch'] def __init__(self): self.history = [] self.active = None + self.realpitch = None def IsActive(self): return self.active is not None - def Activate(self, mev): + def Activate(self, mev, realpitch = None): + if realpitch is None: + realpitch = mev.ev.pitch self.active = mev + self.realpitch = realpitch def Deactivate(self, mev): - self.history.append(DurationEvent(self.active, mev.abstime - self.active.abstime)) + self.history.append(DurationEvent(self.active, self.realpitch, mev.abstime - self.active.abstime)) self.active = None + self.realpitch = None def WouldDeactivate(self, mev): if not self.IsActive(): return False - return mev.ev.pitch == self.active.ev.pitch and mev.tidx == self.active.tidx + if isinstance(mev.ev, midi.NoteOffEvent): + return mev.ev.pitch == self.active.ev.pitch and mev.tidx == self.active.tidx and mev.ev.channel == self.active.ev.channel + if isinstance(mev.ev, midi.PitchWheelEvent): + return mev.tidx == self.active.tidx and mev.ev.channel == self.active.ev.channel + raise TypeError('Tried to deactivate with bad type %r'%(type(mev.ev),)) class NSGroup(object): __slots__ = ['streams', 'filter', 'name'] @@ -326,6 +341,32 @@ for fname in args: break else: print 'WARNING: Did not match %r with any stream deactivation.'%(mev,) + if options.verbose: + print ' Current state:' + for group in notegroups: + print ' Group %r:'%(group.name,) + for stream in group.streams: + print ' Stream: %r'%(stream.active,) + elif options.deviation > 0 and isinstance(mev.ev, midi.PitchWheelEvent): + for group in notegroups: + found = False + for stream in group.streams: + if stream.WouldDeactivate(mev): + base = stream.active.copy(abstime=mev.abstime) + stream.Deactivate(mev) + stream.Activate(base, base.ev.pitch + options.deviation * (mev.ev.pitch / 2000.0)) + found = True + break + if found: + break + else: + print 'WARNING: Did not find any matching active streams for %r'%(mev,) + if options.verbose: + print ' Current state:' + for group in notegroups: + print ' Group %r:'%(group.name,) + for stream in group.streams: + print ' Stream: %r'%(stream.active,) else: auxstream.append(mev) @@ -372,7 +413,7 @@ for fname in args: ivns.set('group', group.name) for note in ns.history: ivnote = ET.SubElement(ivns, 'note') - ivnote.set('pitch', str(note.ev.pitch)) + ivnote.set('pitch', str(note.pitch)) ivnote.set('vel', str(note.ev.velocity)) ivnote.set('time', str(note.abstime)) ivnote.set('dur', str(note.duration)) diff --git a/shiv.py b/shiv.py index 0a8169e..ac6e2b1 100644 --- a/shiv.py +++ b/shiv.py @@ -3,6 +3,7 @@ import xml.etree.ElementTree as ET import optparse import sys +import math parser = optparse.OptionParser() parser.add_option('-n', '--number', dest='number', action='store_true', help='Show number of tracks') @@ -69,7 +70,7 @@ def show_hist(values, height=None): xs, ys = values.keys(), values.values() minx, maxx = min(xs), max(xs) miny, maxy = min(ys), max(ys) - xv = range(minx, maxx + 1) + xv = range(int(math.floor(minx)), int(math.ceil(maxx + 1))) incs = max((maxy - miny) / height, 1) print COL.CYAN + '\t --' + '-' * len(xv) + COL.NONE for ub in range(maxy + incs, miny, -incs): @@ -173,7 +174,7 @@ for fname in args: for sidx, stream in enumerate(notestreams): notes = stream.findall('note') for note in notes: - pitch = int(note.get('pitch')) + pitch = float(note.get('pitch')) vel = int(note.get('vel')) time = float(note.get('time')) dur = float(note.get('dur')) -- cgit v1.2.3-70-g09d2 From 154526e1a56ec78b745a8b76bb8949cf6cb9d298 Mon Sep 17 00:00:00 2001 From: Grissess Date: Fri, 10 Jun 2016 01:16:16 -0400 Subject: Fixed annoying tempo bug --- .gitignore | 1 + mkiv.py | 33 ++++++++++++++++++--------------- 2 files changed, 19 insertions(+), 15 deletions(-) diff --git a/.gitignore b/.gitignore index ee6e3f3..7df3784 100644 --- a/.gitignore +++ b/.gitignore @@ -4,4 +4,5 @@ client *.swp *.swo *.pyc +*.mid *~ diff --git a/mkiv.py b/mkiv.py index dd63130..a55f755 100644 --- a/mkiv.py +++ b/mkiv.py @@ -154,7 +154,7 @@ for fname in args: sorted_events.append(SortEvent(ev, tidx, absticks)) sorted_events.sort(key=lambda x: x.abstick) - bpm_at = {0: 120} + bpm_at = [{0: 120} for i in pat] print 'Computing tempos...' @@ -162,16 +162,18 @@ for fname in args: if isinstance(sev.ev, midi.SetTempoEvent): if options.debug: print fname, ': SetTempo at', sev.abstick, 'to', sev.ev.bpm, ':', sev.ev - bpm_at[sev.abstick] = sev.ev.bpm + bpm_at[sev.tidx][sev.abstick] = sev.ev.bpm if options.verbose: print fname, ': Events:', len(sorted_events) print fname, ': Resolved global BPM:', bpm_at if options.debug: - btimes = bpm_at.keys() - for i in range(len(btimes) - 1): - fev = filter(lambda sev: sev.abstick >= btimes[i] and sev.abstick < btimes[i+1], sorted_events) - print fname, ': BPM partition', i, 'contains', len(fev), 'events' + for tidx, bpms in enumerate(bpm_at): + print fname, ': Tempos in track', tidx + btimes = bpms.keys() + for i in range(len(btimes) - 1): + fev = filter(lambda sev: sev.tidx == tidx and sev.abstick >= btimes[i] and sev.abstick < btimes[i+1], sorted_events) + print fname, ': BPM partition', i, 'contains', len(fev), 'events' class MergeEvent(object): __slots__ = ['ev', 'tidx', 'abstime', 'bank', 'prog'] @@ -201,7 +203,7 @@ for fname in args: abstime = 0 absticks = 0 for ev in track: - bpm = filter(lambda pair: pair[0] <= absticks, sorted(bpm_at.items(), key=lambda pair: pair[0]))[-1][1] + bpm = filter(lambda pair: pair[0] <= absticks, sorted(bpm_at[tidx].items(), key=lambda pair: pair[0]))[-1][1] if options.debug: print ev, ': bpm=', bpm absticks += ev.tick @@ -389,17 +391,18 @@ for fname in args: ##### Write to XML and exit ##### ivmeta = ET.SubElement(iv, 'meta') - ivbpms = ET.SubElement(ivmeta, 'bpms') abstime = 0 prevticks = 0 prev_bpm = 120 - for absticks, bpm in sorted(bpm_at.items(), key = lambda pair: pair[0]): - abstime += ((absticks - prevticks) * 60.0) / (prev_bpm * pat.resolution) - prevticks = absticks - ivbpm = ET.SubElement(ivbpms, 'bpm') - ivbpm.set('bpm', str(bpm)) - ivbpm.set('ticks', str(absticks)) - ivbpm.set('time', str(abstime)) + for tidx, bpms in enumerate(bpm_at): + ivbpms = ET.SubElement(ivmeta, 'bpms', track=str(tidx)) + for absticks, bpm in sorted(bpms.items(), key = lambda pair: pair[0]): + abstime += ((absticks - prevticks) * 60.0) / (prev_bpm * pat.resolution) + prevticks = absticks + ivbpm = ET.SubElement(ivbpms, 'bpm') + ivbpm.set('bpm', str(bpm)) + ivbpm.set('ticks', str(absticks)) + ivbpm.set('time', str(abstime)) ivstreams = ET.SubElement(iv, 'streams') -- cgit v1.2.3-70-g09d2 From 37c9df9fe7d6c9faace591d726266a86f503aab6 Mon Sep 17 00:00:00 2001 From: Grissess Date: Fri, 10 Jun 2016 01:29:59 -0400 Subject: Unfixed tempo bug --- mkiv.py | 16 ++++++++++++---- 1 file changed, 12 insertions(+), 4 deletions(-) diff --git a/mkiv.py b/mkiv.py index a55f755..218ea7b 100644 --- a/mkiv.py +++ b/mkiv.py @@ -33,8 +33,13 @@ parser.add_option('-n', '--target-num', dest='repeaterNumber', type='int', help= parser.add_option('-v', '--verbose', dest='verbose', action='store_true', help='Be verbose; show important parts about the MIDI scheduling process') parser.add_option('-d', '--debug', dest='debug', action='store_true', help='Debugging output; show excessive output about the MIDI scheduling process') parser.add_option('-D', '--deviation', dest='deviation', type='int', help='Amount (in semitones/MIDI pitch units) by which a fully deflected pitchbend modifies the base pitch (0 disables pitchbend processing)') -parser.set_defaults(tracks=[], repeaterNumber=1, perc='GM', deviation=2) +parser.add_option('--tempo', dest='tempo', help='Adjust interpretation of tempo (try "f1"/"global", "f2"/"track")') +parser.set_defaults(tracks=[], repeaterNumber=1, perc='GM', deviation=2, tempo='global') options, args = parser.parse_args() +if options.tempo == 'f1': + options.tempo == 'global' +elif options.tempo == 'f2': + options.tempo == 'track' if options.help_conds: print '''Filter conditions are used to route events to groups of streams. @@ -154,7 +159,10 @@ for fname in args: sorted_events.append(SortEvent(ev, tidx, absticks)) sorted_events.sort(key=lambda x: x.abstick) - bpm_at = [{0: 120} for i in pat] + if options.tempo == 'global': + bpm_at = [{0: 120}] + else: + bpm_at = [{0: 120} for i in pat] print 'Computing tempos...' @@ -162,7 +170,7 @@ for fname in args: if isinstance(sev.ev, midi.SetTempoEvent): if options.debug: print fname, ': SetTempo at', sev.abstick, 'to', sev.ev.bpm, ':', sev.ev - bpm_at[sev.tidx][sev.abstick] = sev.ev.bpm + bpm_at[sev.tidx if options.tempo == 'track' else 0][sev.abstick] = sev.ev.bpm if options.verbose: print fname, ': Events:', len(sorted_events) @@ -203,7 +211,7 @@ for fname in args: abstime = 0 absticks = 0 for ev in track: - bpm = filter(lambda pair: pair[0] <= absticks, sorted(bpm_at[tidx].items(), key=lambda pair: pair[0]))[-1][1] + bpm = filter(lambda pair: pair[0] <= absticks, sorted(bpm_at[tidx if options.tempo == 'track' else 0].items(), key=lambda pair: pair[0]))[-1][1] if options.debug: print ev, ': bpm=', bpm absticks += ev.tick -- cgit v1.2.3-70-g09d2 From f649b5fd77607f1bef8943ff548d4330d9bc4e99 Mon Sep 17 00:00:00 2001 From: Grissess Date: Fri, 10 Jun 2016 01:37:28 -0400 Subject: Fixed constant for pitchbend --- mkiv.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/mkiv.py b/mkiv.py index 218ea7b..7a4c7c7 100644 --- a/mkiv.py +++ b/mkiv.py @@ -364,7 +364,7 @@ for fname in args: if stream.WouldDeactivate(mev): base = stream.active.copy(abstime=mev.abstime) stream.Deactivate(mev) - stream.Activate(base, base.ev.pitch + options.deviation * (mev.ev.pitch / 2000.0)) + stream.Activate(base, base.ev.pitch + options.deviation * (mev.ev.pitch / float(0x2000))) found = True break if found: -- cgit v1.2.3-70-g09d2 From 75662f3a3c525b8417268124d7a2c41a2d5ad5b8 Mon Sep 17 00:00:00 2001 From: Grissess Date: Fri, 10 Jun 2016 03:57:34 -0400 Subject: Begone ye foul tempo demons! --- mkiv.py | 54 +++++++++++++++++++++++++++++++++++++++++------------- 1 file changed, 41 insertions(+), 13 deletions(-) diff --git a/mkiv.py b/mkiv.py index 7a4c7c7..d52ef76 100644 --- a/mkiv.py +++ b/mkiv.py @@ -176,13 +176,44 @@ for fname in args: print fname, ': Events:', len(sorted_events) print fname, ': Resolved global BPM:', bpm_at if options.debug: - for tidx, bpms in enumerate(bpm_at): - print fname, ': Tempos in track', tidx - btimes = bpms.keys() + if options.tempo == 'track': + for tidx, bpms in enumerate(bpm_at): + print fname, ': Tempos in track', tidx + btimes = bpms.keys() + for i in range(len(btimes) - 1): + fev = filter(lambda sev: sev.tidx == tidx and sev.abstick >= btimes[i] and sev.abstick < btimes[i+1], sorted_events) + print fname, ': BPM partition', i, 'contains', len(fev), 'events' + else: + btimes = bpm_at[0].keys() for i in range(len(btimes) - 1): - fev = filter(lambda sev: sev.tidx == tidx and sev.abstick >= btimes[i] and sev.abstick < btimes[i+1], sorted_events) + fev = filter(lambda sev: sev.abstick >= btimes[i] and sev.abstick < btimes[i+1], sorted_events) print fname, ': BPM partition', i, 'contains', len(fev), 'events' + def at2rt(abstick, bpms): + bpm_segs = bpms.items() + bpm_segs.sort(key=lambda pair: pair[0]) + bpm_segs = filter(lambda pair: pair[0] <= abstick, bpm_segs) + rt = 0 + atick = 0 + if not bpm_segs: + rt = 0 + else: + ctick, bpm = bpm_segs[0] + rt = (60.0 * ctick) / (bpm * pat.resolution) + for idx in range(1, len(bpm_segs)): + dt = bpm_segs[idx][0] - bpm_segs[idx-1][0] + bpm = bpm_segs[idx-1][1] + rt += (60.0 * dt) / (bpm * pat.resolution) + if not bpm_segs: + bpm = 120 + ctick = 0 + else: + ctick, bpm = bpm_segs[-1] + if options.debug: + print 'seg through', bpm_segs, 'final seg', (abstick - ctick, bpm) + rt += (60.0 * (abstick - ctick)) / (bpm * pat.resolution) + return rt + class MergeEvent(object): __slots__ = ['ev', 'tidx', 'abstime', 'bank', 'prog'] def __init__(self, ev, tidx, abstime, bank, prog): @@ -210,11 +241,12 @@ for fname in args: for tidx, track in enumerate(pat): abstime = 0 absticks = 0 + lastbpm = 120 for ev in track: - bpm = filter(lambda pair: pair[0] <= absticks, sorted(bpm_at[tidx if options.tempo == 'track' else 0].items(), key=lambda pair: pair[0]))[-1][1] - if options.debug: - print ev, ': bpm=', bpm absticks += ev.tick + abstime = at2rt(absticks, bpm_at[tidx if options.tempo == 'track' else 0]) + if options.debug: + print 'tick', absticks, 'realtime', abstime if isinstance(ev, midi.ProgramChangeEvent): cur_prog[tidx][ev.channel] = ev.value progs.add(ev.value) @@ -231,7 +263,6 @@ for fname in args: elif isinstance(ev, midi.Event): if isinstance(ev, midi.NoteOnEvent) and ev.velocity == 0: ev.__class__ = midi.NoteOffEvent #XXX Oww - abstime += (60.0 * ev.tick) / (bpm * pat.resolution) events.append(MergeEvent(ev, tidx, abstime, cur_bank[tidx][ev.channel], cur_prog[tidx][ev.channel])) ev_cnts[tidx][ev.channel] += 1 @@ -358,18 +389,15 @@ for fname in args: for stream in group.streams: print ' Stream: %r'%(stream.active,) elif options.deviation > 0 and isinstance(mev.ev, midi.PitchWheelEvent): + found = False for group in notegroups: - found = False for stream in group.streams: if stream.WouldDeactivate(mev): base = stream.active.copy(abstime=mev.abstime) stream.Deactivate(mev) stream.Activate(base, base.ev.pitch + options.deviation * (mev.ev.pitch / float(0x2000))) found = True - break - if found: - break - else: + if not found: print 'WARNING: Did not find any matching active streams for %r'%(mev,) if options.verbose: print ' Current state:' -- cgit v1.2.3-70-g09d2 From dc59723180707561e0f6a7e1fa7b1b33b4439104 Mon Sep 17 00:00:00 2001 From: Grissess Date: Fri, 10 Jun 2016 04:24:40 -0400 Subject: Separate text streams --- mkiv.py | 15 ++++++++++++--- 1 file changed, 12 insertions(+), 3 deletions(-) diff --git a/mkiv.py b/mkiv.py index d52ef76..03b3fdf 100644 --- a/mkiv.py +++ b/mkiv.py @@ -247,6 +247,8 @@ for fname in args: abstime = at2rt(absticks, bpm_at[tidx if options.tempo == 'track' else 0]) if options.debug: print 'tick', absticks, 'realtime', abstime + if isinstance(ev, midi.TrackNameEvent): + tnames[tidx] = ev.text if isinstance(ev, midi.ProgramChangeEvent): cur_prog[tidx][ev.channel] = ev.value progs.add(ev.value) @@ -258,8 +260,8 @@ for fname in args: elif ev.control == 32: cur_bank[tidx][ev.channel] = (0x3F & cur_bank[tidx][ev.channel]) | (ev.value << 7) chg_bank[tidx][ev.channel] += 1 - elif isinstance(ev, midi.TrackNameEvent): - tnames[tidx] = ev.text + elif isinstance(ev, midi.MetaEventWithText): + events.append(MergeEvent(ev, tidx, abstime, 0, 0)) elif isinstance(ev, midi.Event): if isinstance(ev, midi.NoteOnEvent) and ev.velocity == 0: ev.__class__ = midi.NoteOffEvent #XXX Oww @@ -333,6 +335,7 @@ for fname in args: notegroups = [] auxstream = [] + textstream = [] if options.perc and options.perc != 'none': if options.perc == 'GM': @@ -362,7 +365,9 @@ for fname in args: print ('' if group.name is None else group.name) for mev in events: - if isinstance(mev.ev, midi.NoteOnEvent): + if isinstance(mev.ev, midi.MetaEventWithText): + textstream.append(mev) + elif isinstance(mev.ev, midi.NoteOnEvent): for group in notegroups: if group.Accept(mev): break @@ -464,6 +469,10 @@ for fname in args: if(x>=options.repeaterNumber and options.repeaterNumber!=1): break + ivtext = ET.SubElement(ivstreams, 'stream', type='text') + for tev in textstream: + ivev = ET.SubElement(ivtext, 'text', time=str(tev.abstime), type=type(tev.ev).__name__, text=tev.ev.text) + ivaux = ET.SubElement(ivstreams, 'stream') ivaux.set('type', 'aux') -- cgit v1.2.3-70-g09d2 From 368b5db51d76c162656abd26c88991f0f7f8a556 Mon Sep 17 00:00:00 2001 From: Grissess Date: Sun, 12 Jun 2016 22:49:51 -0400 Subject: Removed track duplication kludge from mkiv and moved to broadcast --- broadcast.py | 33 +++++++++++++++++++++++---------- mkiv.py | 38 ++++++++++++++------------------------ 2 files changed, 37 insertions(+), 34 deletions(-) diff --git a/broadcast.py b/broadcast.py index 714533d..c7d379d 100644 --- a/broadcast.py +++ b/broadcast.py @@ -7,6 +7,7 @@ import threading import thread import optparse import random +import itertools from packet import Packet, CMD, itos @@ -33,12 +34,14 @@ parser.add_option('-r', '--route', dest='routes', action='append', help='Add a r 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('-B', '--bind-addr', dest='bind_addr', help='The IP address (or IP:port) to bind to (influences the network to send to)') +parser.add_option('--repeat', dest='repeat', action='store_true', help='Repeat the file playlist indefinitely') +parser.add_option('-n', '--number', dest='number', type='int', help='Number of clients to use; if negative (default -1), use the product of stream count and the absolute value of this parameter') parser.add_option('-G', '--gui', dest='gui', default='', help='set a GUI to use') parser.add_option('--pg-fullscreen', dest='fullscreen', action='store_true', help='Use a full-screen video mode') parser.add_option('--pg-width', dest='pg_width', type='int', help='Width of the pygame window') parser.add_option('--pg-height', dest='pg_height', type='int', help='Width of the pygame window') parser.add_option('--help-routes', dest='help_routes', action='store_true', help='Show help about routing directives') -parser.set_defaults(routes=[], test_delay=0.25, 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, bind_addr='', pg_width = 0, pg_height = 0) +parser.set_defaults(routes=[], test_delay=0.25, 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, bind_addr='', pg_width = 0, pg_height = 0, number=-1) options, args = parser.parse_args() if options.help_routes: @@ -302,6 +305,9 @@ if options.live or options.list_live: del active_set[pitch] deferred_set.clear() +if options.repeat: + args = itertools.cycle(args) + for fname in args: try: iv = ET.parse(fname).getroot() @@ -313,9 +319,11 @@ for fname in args: notestreams = iv.findall("./streams/stream[@type='ns']") groups = set([ns.get('group') for ns in notestreams if 'group' in ns.keys()]) + number = (len(notestreams) * abs(options.number) if options.number < 0 else options.number) print len(notestreams), 'notestreams' print len(clients), 'clients' print len(groups), 'groups' + print number, 'clients used (number)' class Route(object): def __init__(self, fattr, fvalue, group, excl=False): @@ -461,7 +469,7 @@ for fname in args: return time.sleep(t) def run(self): - nsq, cl = self._Thread__args + nsq, cls = self._Thread__args for note in nsq: ttime = float(note.get('time')) pitch = float(note.get('pitch')) + options.transpose @@ -469,7 +477,8 @@ for fname in args: dur = factor*float(note.get('dur')) while time.time() - BASETIME < factor*ttime: self.wait_for(factor*ttime - (time.time() - BASETIME)) - s.sendto(str(Packet(CMD.PLAY, int(dur), int((dur*1000000)%1000000), int(440.0 * 2**((pitch-69)/12.0)), int(vel*2 * options.volume/255.0))), cl) + for cl in cls: + s.sendto(str(Packet(CMD.PLAY, int(dur), int((dur*1000000)%1000000), int(440.0 * 2**((pitch-69)/12.0)), int(vel*2 * options.volume/255.0))), cl) if options.verbose: print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel playing_notes[cl] = (pitch, vel*2) @@ -478,24 +487,28 @@ for fname in args: if options.verbose: print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' - threads = [] - for ns in notestreams: + threads = {} + nscycle = itertools.cycle(notestreams) + for idx, ns in zip(xrange(number), nscycle): cli = routeset.Route(ns) if cli: nsq = ns.findall('note') - threads.append(NSThread(args=(nsq, cli))) + if ns in threads: + threads[ns]._Thread__args[1].add(cli) + else: + threads[ns] = NSThread(args=(nsq, set([cli]))) if options.verbose: print 'Playback threads:' - for thr in threads: + for thr in threads.values(): print thr._Thread__args[1] BASETIME = time.time() - (options.seek*factor) if options.seek > 0: - for thr in threads: + for thr in threads.values(): thr.drop_missed() - for thr in threads: + for thr in threads.values(): thr.start() - for thr in threads: + for thr in threads.values(): thr.join() print fname, ': Done!' diff --git a/mkiv.py b/mkiv.py index 03b3fdf..717220c 100644 --- a/mkiv.py +++ b/mkiv.py @@ -29,12 +29,11 @@ parser.add_option('--help-conds', dest='help_conds', action='store_true', help=' parser.add_option('-p', '--program-split', dest='tracks', action='append_const', const=PROGRAMS, help='Ensure all programs are on non-mutual streams (overrides -T presently)') parser.add_option('-P', '--percussion', dest='perc', help='Which percussion standard to use to automatically filter to "perc" (GM, GM2, or none)') parser.add_option('-f', '--fuckit', dest='fuckit', action='store_true', help='Use the Python Error Steamroller when importing MIDIs (useful for extended formats)') -parser.add_option('-n', '--target-num', dest='repeaterNumber', type='int', help='Target count of devices') parser.add_option('-v', '--verbose', dest='verbose', action='store_true', help='Be verbose; show important parts about the MIDI scheduling process') -parser.add_option('-d', '--debug', dest='debug', action='store_true', help='Debugging output; show excessive output about the MIDI scheduling process') +parser.add_option('-d', '--debug', dest='debug', action='store_true', help='Debugging output; show excessive output about the MIDI scheduling process (please use less or write to a file)') parser.add_option('-D', '--deviation', dest='deviation', type='int', help='Amount (in semitones/MIDI pitch units) by which a fully deflected pitchbend modifies the base pitch (0 disables pitchbend processing)') parser.add_option('--tempo', dest='tempo', help='Adjust interpretation of tempo (try "f1"/"global", "f2"/"track")') -parser.set_defaults(tracks=[], repeaterNumber=1, perc='GM', deviation=2, tempo='global') +parser.set_defaults(tracks=[], perc='GM', deviation=2, tempo='global') options, args = parser.parse_args() if options.tempo == 'f1': options.tempo == 'global' @@ -447,27 +446,18 @@ for fname in args: ivstreams = ET.SubElement(iv, 'streams') - x = 0 - while(x=options.repeaterNumber and options.repeaterNumber!=1): - break - if(x>=options.repeaterNumber and options.repeaterNumber!=1): - break - if(x>=options.repeaterNumber and options.repeaterNumber!=1): - break + for group in notegroups: + for ns in group.streams: + ivns = ET.SubElement(ivstreams, 'stream') + ivns.set('type', 'ns') + if group.name is not None: + ivns.set('group', group.name) + for note in ns.history: + ivnote = ET.SubElement(ivns, 'note') + ivnote.set('pitch', str(note.pitch)) + ivnote.set('vel', str(note.ev.velocity)) + ivnote.set('time', str(note.abstime)) + ivnote.set('dur', str(note.duration)) ivtext = ET.SubElement(ivstreams, 'stream', type='text') for tev in textstream: -- cgit v1.2.3-70-g09d2