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