aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorGraham Northup <grissess@nexusg.org>2018-09-24 16:08:35 -0400
committerGraham Northup <grissess@nexusg.org>2018-09-24 16:08:35 -0400
commitec177bb81a2a6ab54e05118066364281ebccc318 (patch)
tree18c025d023b6fe06bdca8a641619c8bdc8e95518
parent5c205fd148900dc787c99a4ce1898a49abfd901d (diff)
oof
-rw-r--r--client.py58
-rw-r--r--mkiv.py10
2 files changed, 52 insertions, 16 deletions
diff --git a/client.py b/client.py
index cb4c030..2c21a9e 100644
--- a/client.py
+++ b/client.py
@@ -32,6 +32,7 @@ parser.add_option('-c', '--clamp', dest='clamp', action='store_true', help='Clam
parser.add_option('-C', '--chorus', dest='chorus', default=0.0, type='float', help='Apply uniform random offsets (in MIDI pitch space)')
parser.add_option('--vibrato', dest='vibrato', default=0.0, type='float', help='Apply periodic perturbances in pitch space by this amplitude (in MIDI pitches)')
parser.add_option('--vibrato-freq', dest='vibrato_freq', default=6.0, type='float', help='Frequency of the vibrato perturbances in Hz')
+parser.add_option('--fmul', dest='fmul', default=1.0, type='float', help='Multiply requested frequencies by this amount')
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)')
@@ -72,6 +73,9 @@ DRIFT_FACTOR = 1.0
DRIFT_ERROR = 0.0
LAST_SYN = None
+CUR_PERIODS = [0] * STREAMS
+CUR_PERIOD = 0.0
+
def lin_interp(frm, to, p):
return p*to + (1-p)*frm
@@ -245,23 +249,41 @@ def square_wave(theta):
def noise(theta):
return random.random() * 2 - 1
-@generator('File generator', '(<file>[, <bits=8>[, <signed=True>[, <0=linear interp (default), 1=nearest>[, <swapbytes=False>]]]])')
+@generator('File generator', '(<file>[, <bits=8>[, <signed=True>[, <0=linear interp (default), 1=nearest>[, <swapbytes=False>[, <loop=(fraction to loop, 0.0 is all, 1.0 is end, or False to not loop)>[, <loopend=1.0>[, periods=1 (periods in wave file)/freq=None (base frequency)/pitch=None (base MIDI pitch)]]]]]]])')
class file_samp(object):
LINEAR = 0
NEAREST = 1
TYPES = {8: 'B', 16: 'H', 32: 'L'}
- def __init__(self, fname, bits=8, signed=True, samp=LINEAR, swab=False):
+ def __init__(self, fname, bits=8, signed=True, samp=LINEAR, swab=False, loop=0.0, loopend=1.0, periods=1.0, freq=None, pitch=None):
tp = self.TYPES[bits]
if signed:
tp = tp.lower()
self.max = float((2 << bits) - 1)
+ if signed:
+ self.max /= 2.0
self.buffer = array.array(tp)
self.buffer.fromstring(open(fname, 'rb').read())
if swab:
self.buffer.byteswap()
self.samp = samp
+ self.loop = loop
+ self.loopend = loopend
+ self.periods = periods
+ if pitch is not None:
+ freq = 440.0 * 2 ** ((pitch - 69) / 12.0)
+ if freq is not None:
+ self.periods = freq * len(self.buffer) / RATE
+ print 'file_samp periods:', self.periods, 'freq:', freq, 'pitch:', pitch
def __call__(self, theta):
- norm = theta / (2*math.pi)
+ full_norm = CUR_PERIOD / (2*self.periods*math.pi)
+ if full_norm > 1.0:
+ if self.loop is False:
+ return self.buffer[0]
+ else:
+ norm = (full_norm - 1.0) / (self.loopend - self.loop) % 1.0 * (self.loopend - self.loop) + self.loop
+ else:
+ norm = full_norm
+ norm %= 1.0
if self.samp == self.LINEAR:
v = norm*len(self.buffer)
l = int(math.floor(v))
@@ -411,13 +433,14 @@ if options.numpy:
return numpy.linspace(frm, to, cnt, dtype=numpy.int32)
def samps(freq, amp, phase, cnt):
+ global CUR_PERIOD
samps = numpy.ndarray((cnt,), numpy.int32)
pvel = 2 * math.pi * freq / RATE
fac = options.volume * amp / float(STREAMS)
for i in xrange(cnt):
- samps[i] = fac * max(-1, min(1, generator(phase)))
- phase = (phase + pvel) % (2 * math.pi)
- return samps, phase
+ samps[i] = fac * max(-1, min(1, generator((phase + i * pvel) % (2*math.pi))))
+ CUR_PERIOD += pvel
+ return samps, phase + pvel * cnt
def to_data(samps):
return samps.tobytes()
@@ -439,11 +462,13 @@ else:
return samps
def samps(freq, amp, phase, cnt):
- global RATE
+ global RATE, CUR_PERIOD
samps = [0]*cnt
for i in xrange(cnt):
- samps[i] = int(2*amp / float(STREAMS) * max(-1, min(1, options.volume*generator((phase + 2 * math.pi * freq * i / RATE) % (2*math.pi)))))
- return samps, (phase + 2 * math.pi * freq * cnt / RATE) % (2*math.pi)
+ samps[i] = int(amp / float(STREAMS) * max(-1, min(1, options.volume*generator((phase + 2 * math.pi * freq * i / RATE) % (2*math.pi)))))
+ CUR_PERIOD += 2 * math.pi * freq / RATE
+ next_phase = (phase + 2 * math.pi * freq * cnt / RATE)
+ return samps, next_phase
def to_data(samps):
return struct.pack('i'*len(samps), *samps)
@@ -468,7 +493,7 @@ else:
return struct.pack(str(amt)+'i', *out)
def gen_data(data, frames, tm, status):
- global FREQS, PHASE, Z_SAMP, LAST_SAMP, LAST_SAMPLES, QUEUED_PCM, DRIFT_FACTOR, DRIFT_ERROR
+ global FREQS, PHASE, Z_SAMP, LAST_SAMP, LAST_SAMPLES, QUEUED_PCM, DRIFT_FACTOR, DRIFT_ERROR, CUR_PERIOD
if len(QUEUED_PCM) >= frames*4:
desired_frames = DRIFT_FACTOR * frames
err_frames = desired_frames - int(desired_frames)
@@ -497,6 +522,7 @@ def gen_data(data, frames, tm, status):
AMP = AMPS[i]
EXPIRATION = EXPIRATIONS[i]
PHASE = PHASES[i]
+ CUR_PERIOD = CUR_PERIODS[i]
if FREQ != 0:
if time.time() > EXPIRATION:
FREQ = 0
@@ -507,9 +533,11 @@ def gen_data(data, frames, tm, status):
fdata = mix(fdata, vdata)
LAST_SAMPS[i] = vdata[-1]
else:
- vdata, PHASE = samps(FREQ, AMP, PHASE, frames)
+ vdata, CUR_PERIOD = samps(FREQ, AMP, CUR_PERIOD, frames)
+ PHASE = (PHASE + CUR_PERIOD) % (2 * math.pi)
fdata = mix(fdata, vdata)
PHASES[i] = PHASE
+ CUR_PERIODS[i] = CUR_PERIOD
LAST_SAMPS[i] = vdata[-1]
if options.gui:
LAST_SAMPLES.extend(fdata)
@@ -526,14 +554,17 @@ if options.gui:
if options.test:
FREQS[0] = 440
EXPIRATIONS[0] = time.time() + 1
+ CUR_PERIODS[0] = 0.0
time.sleep(1)
FREQS[0] = 0
time.sleep(1)
FREQS[0] = 880
EXPIRATIONS[0] = time.time() + 1
+ CUR_PERIODS[0] = 0.0
time.sleep(1)
FREQS[0] = 440
EXPIRATIONS[0] = time.time() + 2
+ CUR_PERIODS[0] = 0.0
time.sleep(2)
exit()
@@ -567,12 +598,13 @@ while True:
elif pkt.cmd == CMD.PLAY:
voice = pkt.data[4]
dur = pkt.data[0]+pkt.data[1]/1000000.0
- freq = pkt.data[2]
+ freq = pkt.data[2] * options.fmul
if options.chorus > 0:
midi = 12 * math.log(freq / 440.0, 2) + 69
midi += (random.random() * 2 - 1) * options.chorus
freq = 440.0 * 2 ** ((midi - 69) / 12)
FREQS[voice] = freq
+ CUR_PERIODS[voice] = 0.0
amp = pkt.as_float(3)
if options.clamp:
amp = max(min(amp, 1.0), 0.0)
@@ -588,7 +620,7 @@ while True:
if pkt.data[5] & PLF.SAMEPHASE:
print '\x1b[1;37mSAMEPHASE',
if pkt.data[0] == 0 and pkt.data[1] == 0:
- print '\x1b[1;35mSTOP!!!'
+ print '\x1b[1;31mSTOP!!!'
else:
print '\x1b[1;36mDUR', '%08.6f'%dur
#signal.setitimer(signal.ITIMER_REAL, dur)
diff --git a/mkiv.py b/mkiv.py
index fb11d50..51eb103 100644
--- a/mkiv.py
+++ b/mkiv.py
@@ -27,6 +27,7 @@ parser.add_option('-p', '--program-split', dest='tracks', action='append_const',
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('-v', '--verbose', dest='verbose', action='store_true', help='Be verbose; show important parts about the MIDI scheduling process')
+parser.add_option('-q', '--quiet', dest='quiet', action='store_true', help='Be quiet; don\'t log certain high-volume outputs')
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('-M', '--modwheel-freq-dev', dest='modfdev', type='float', help='Amount (in semitones/MIDI pitch unites) by which a fully-activated modwheel modifies the base pitch')
@@ -485,7 +486,8 @@ for fname in args:
if found:
break
else:
- print 'WARNING: Did not match %r with any stream deactivation.'%(mev,)
+ if not options.quiet:
+ print 'WARNING: Did not match %r with any stream deactivation.'%(mev,)
if options.verbose:
print ' Current state:'
for group in notegroups:
@@ -503,7 +505,8 @@ for fname in args:
stream.Activate(base, base.ev.pitch + options.deviation * (mev.ev.pitch / float(0x2000)), parent=old)
found = True
if not found:
- print 'WARNING: Did not find any matching active streams for %r'%(mev,)
+ if not options.quiet:
+ print 'WARNING: Did not find any matching active streams for %r'%(mev,)
if options.verbose:
print ' Current state:'
for group in notegroups:
@@ -521,7 +524,8 @@ for fname in args:
stream.Activate(base, stream.bentpitch, mev.mw, parent=old)
found = True
if not found:
- print 'WARNING: Did not find any matching active streams for %r'%(mev,)
+ if not options.quiet:
+ print 'WARNING: Did not find any matching active streams for %r'%(mev,)
if options.verbose:
print ' Current state:'
for group in notegroups: