vII MidiOut Help/Documentation (version 1d)

Contents
1.Introduction/Purpose
2.Basic Usage
   2.1 Getting started
   2.3 Menu Commands
   2.2 Option Settings
3.Advanced Usage
   3.1 The different methods
   3.2 Delay and effect slides
   3.3 The Cut/Retrigger command
   3.4.Effects/MIDI controllers
   3.5 Some explanations about MIDI data
   3.6 NRP for AWE32/64
   3.7 Timing/synchronizing
4.History
5.Plans/Future/Missing Things
Contacting the author

1.Introduction/Purpose
This is a first implementation of a MidiOut Machine for BUZZ. It is to regard as beta and will maybe heavily changed in the future, although I already tried to prepare all needed parameters. I guess a MIDI out machine generally isn't that suited for distributed songs, because it will behave and sound different on every computer. The main purposes may be using BUZZ to make music with all your MIDI-equipment (and mixing all together by other means than BUZZ itself) and the possiblility to sync another device or programm with BUZZ.

Some remarks:
Synchronizing with a MIDI sequencer and external devices should work fine now, also it may be possible to share a DirectX wavedevice with softsynths like Rebirth. I doubt the Wavein machine already is adequate for the mixing task (to get any external sounds back into a final mix), but haven't tried it yet. I don't know how the VirtualAudioCable fits in yet, but Hubis MidiLoopback Device and MidiCable sure help a lot.
More exotic usages would be controlling BUZZ parameters via MIDI, either by the MidiOut machine itself or with a (synchronized) sequencer, the latter would finally allow to change parameters live and record this with the sequencer ;)
Concerning the MIDI conversion and merging options of m2buzz there is now a possibility to output a single starting MIDI note with BUZZ, record this with the sequencer together with any live played material, which should make synchronizing of the converted voice with the rest of the song much easier :) Since m2buzz v3e you can also just convert MIDI files for use with the MIDIout machine, although controllers and tempo changes aren't processed yet.

Some more remarks:
I was asked about MIDI recording for BUZZ machines. For BUZZ 1.0 I see no way to implement this for a machine, this should be dealt with in the main program. But I guess it is quite easy to implement (monophonic at least) MIDI playability for any generator, although I know in general this are more than my 2 lines of code for doing it in the MIDI machine ;)


2.Basic Usage
The machine is designed to be fit for everything else than SysEx, for all the missing things look below. To get it working, put it into the Gear/Generators directory and choose the devices you want to use in the BUZZ Preferences/Midioutput, they will be opened by BUZZ and stay this until you terminate BUZZ or uncheck them. Set the device for a vMidiOut machine in the options dialog, since version 1d you can just do this by name without remembering the device number (which will nonetheless be stored as normal BUZZ attribute of the machine). To adress several devices you need more MidiOut machines (since version 1b). You also have to connect every machine to the master and set a volume greater zero, although it doesn't produce any sound. Don't worry, CPU times seem to be very low.

I'm aware that this machine seems a bit complicated, but keep in mind that in difference to other generators it is used to control very different kinds of MIDI equipment, so it has to have this lot of parameters. And BUZZ 1 doesn't seem best adapted for MIDI integration, so I had to go around several corners to get some things done, but finally I really can use this machine. And if you just start with the simple things in the next chapter, you will be able to use this as well ;)

2.1 Getting Started
After you have set up the machine and BUZZ for using one of your MIDI devices, you can use the notes in the first column like you would do with other BUZZ generators. If you're using a General MIDI device (like intern synthesizers of most sound cards) chances are great that you're listening to some piano sounds. You can change these sounds in the 5th column (MIDI patch) or with the slider in the parameters window.

The second column (velocity) is what volume is for other generators, only that the range is from 0...127(0x7f) with default 64(0x40) instead of double these values, sorry for this, I just used the MIDI standard here.

If you only want to let BUZZ play some melody, you don't need to deal with the channel value in the 4th column. But for using several voices or effects this is necessary: MIDI associates most events with one of 16 (logical) channels: most effects apply to all notes on one channel, and every channel has always only one active instrument. This instrument can be switched with every note you play, but it is more convenient to use different channels for different instruments.

This can be handled very easy with vMidiOut by also using a different channel for every BUZZ track. In this case you have to put the values for the channel and the instrument (=MIDI patch or MIDI program) only once: at the start of your song.

With General MIDI the channel number 10 (0x0a) has a special function: it's the drum channel, and the notes you play on this channel don't change the pitch of one instrument but trigger different drums.

The most useful effects to start with are probably the MIDI controllers 0x000a (volume) and 0x0007 (pan) as the BUZZ sliders don't work for this machine :( The pitch wheel (0x0101) and modulation wheel (0x0001) come handy as well. Of the things I implemented additionally to the MIDI standard I'm most excited about the cut/retrigger command (0x0120), because you won't find this in any sequencer. But with this we are already in the advanced usage topics I'll deal with below. The same goes for the delay, effect slides and synchronizing.

2.2 Menu Commands
The menu just has some reset functions (where all note-off functions can also be configured to sent with the BUZZ pause button) and the option settings.

If you've got a lot of hanging notes (especially if working without checked "monophone tracks" option), you can kill them with the following commands:
All notes off just sends MIDI controller 123 to the device on channel 1, which is very fast but doesn't work with some devices,
Stop all notes really sends a Note-Off for all 128 notes on all 16 channels, this is the PANIC button, but may several secondes on some devices.
If neither this doesn't help, you could try to disable the nasty device for a short time in the BUZZ preferences.
Reset controllers sends MIDI controller 121 to the device. It depends on the MIDI device if it really resets everything.

2.3 Option settings
In the options dialog you can change the following parameters:

3.Advanced usage information
The machine allows 16 tracks, default for every track is MIDI channel 1 at startup. You can change the channels in the corresponding column of the track as you like, several tracks can access one MIDI channel, so it's possible to apply different controllers together. Drum channel generally is 10 (0x0a). This has to be set only once (at the start of the song) and will be remembered later.

If you choose an instrument number a program change message will be sent to the actual channel, so you also only should do this at the start of the song or if you want to change the actual patch. Now every note will be transmitted, default velocity is 64 (0x40), highest possible 127 (0x7f). If you use the default of "monophone tracks", an already running note will be canceled. If not and you only have one running note, a note-off or just a zero in the velocity column will stop it, otherwise only the last one stops and all other have to explicitely switched off with the note-number and a zero velocity.

To reduce the amount of typing into the BUZZ track columns, you have to set only one of the three effect columns. The others are replaced by the last value you've put in there or some defaults (for the effect-slide byte this is to do no sliding at all, and the sync commands generally behave different with or without data values). So if you're working with one effect per track only (during some rows or patters at least), you have to set the effect kind only once together with a starting value for the data. Later only set the data or just the "Slide-To value". After a slide this value will be used as default data instead of the last data value.

3.1 The different methods
Here is a summary of the advantages and disadvantages of the different methods you can use as attribute for the machine.

The obsolete methods 1 and 2 are removed from the machine because they weren't usable or had no advantages over method 3.

I hope to have fixed most of the reasons method 3 crashed before, but casually I still got some crashes while closing a song. Currently I'm just using a delay of a quarter second to avoid these, but this of course takes some time if you've got a song with several MIDI machines. Sorry for this, as soon I've found out the exact reason, I'll make a fixed version.

3.2 Delays and Effect slides
The delay and effect-change bytes are used only with the method attribute of the machine set to 4 or 5. The delay byte can be in the range from 0...0xff (255), the latter gives approximately a delay of a whole row. Currently it may happen, that high delays prevent the event from being triggered at all. If you've choosen a general mididelay for the machine, values are just added together.
The time resolution for the delay and the effect slides of course isn't 255/row, but system, driver and BPM dependent. The actual value is shown in the description field for the effect slide byte, or as minimal difference for the delays in the description for the delay columns. On my system (P200,AWE32) for instance it's about 30/row for 125 BPM with the DirectX driver at a resolution of 4 and a latency of 100ms and gets lower with a higher BPM rate or less latency.
Effect slides are currently implemented for continuous MIDI controllers 0x0001...0x0061, all NRP's and the pitchwheel. You just put the MSB (most significant byte = higher byte) of the effect data value you wish to slide to until the next row into the effect change column (the last one). This would look like

  Note  Vel Del Chan  Patch  Effect   EData  Slide 
C-4 .. .. .. 10 0007 fffe 80
.. .. .. .. .. .. 8000 01
.. .. .. .. .. .. 0000 ..
for a volume slide to zero in rows. The last zero just makes sure the effect data is really at the target, the slide itself may stop a bit earlier. The second data value (0x8000) can be omitted.
For 7Bit controller and NRP data you can slide trough the whole possible range this way, as a slide value of 01 would be exactly the same as an effect value of 0000.
For 14Bit controllers (not yet implemented) and the pitchwheel this is different: the slide value of 0x01...0xff really just gives effect values in the range 0x0100...0xff00, so you not only get just 8Bit resolution, you also miss a small range at the start and the end. This will probably not fixed without releasing a completely new version 2 with a 16Bit effect slide value (and a normal volume range instead of the velocity) although I already had plans to abuse the (only rarely needed) channel byte for the LSB...

3.3 Cut and Retrigger
The lower byte of the data for the cut/retrigger command (0x0120) is a delay in the same format as the mididelay or the delay column. The higher byte gives a counter for the events to happen, that means 0 just toggles note off after the delay(=cut), 1 starts it again immediately after the cut (=Retrigger), 2 cuts again after another delay and so on.
This way every even number let's you with the note cut off, every odd one with a note-on. If there's an event on the next row the process of retriggering is stopped after reaching it and again the delay values near 0xff aren't reached for sure. If there's no new event, the retrigger process may last about several rows.

3.4 Effects/MIDI controllers
You can apply a lot of effects, which generally depend on your MIDI-equipment. If you just use the effect data column the last applied effect is used. You can try to reset all controllers with the command in the machines menu, it just sends controller 121. Since I wanted to have a full 16Bit range for some of the effects, but also wanted to allow using the BUZZ parameter sliders, I expanded the data ranges of continuous controllers to 16Bit (with version 1b). Some mesages need exact values, so I tried to keep them (controller 0 and 98-127 currently). There probably will be more changes here in the future :]

Effects MIDI events MIDI Data Range BUZZ Data Range (if different)
0x00xx (7Bit) Midi Controller xx (0-0x7f)
(Ctrl 0-31 are MSB and 32-63 LSB of 14Bit ctrl)
0-0x7f
ctrl 1-97: 0-0xfffe (shifted 9Bit left)
ctrl 0,98-127: 0-0x007f (=MIDI)
00:Bank change
01:Modulation wheel
02:Breath controller
04:Foot Pedal
05:Portamento Time
07:Volume
08:Balance??
0x0a (10):Pan
0x0b (11):Expression
0x20-0x3f (32-63):least significant bytes of
controllers 0-31 for 14Bit precision
0x40-0x5f (64-69):Switches 0:off 0x7f:on
0x40 (64):Sustain
0x41 (65):Portamento
0x42 (66):Sustenuto
0x43 (67):Soft
0x44 (68):Legato
0x45 (69):Hold 2 Pedal
0x5b (91):Reverb
0x5c (92):Tremolo
0x5d (93):Chorus
0x5e (94):Celeste Level
0x5f (95):Phaser Level
0x60 (96):Data Button increment
0x61 (97):Data Button decrement
0x78(120):All sound off (??) 0
0x79(121):Reset all controllers 0
0x79(122):Local Mode off/on 0 or 0x7f
0x7b(123):All notes off 0
0x26/62/63 (38/98/99):NRP (see below)
0x06/64/65 (6/100/101):RP (pitchbend range, see below)
0x0080-0x00ff 14 Bit Midi Controller 0x80+xx (not implemented yet) 0-0x03fff 0-0xfffe (shifted 2Bit left)
0x0101 Pitchwheel 0-0x3fff (default:0x2000) 0-0xfffe (shifted 2Bit left)
(default:0x8000)
0x0102 Channel pressure (Aftertouch) 0-0x7f 0-0x7f
0x0103 Key pressure (Aftertouch) 2 Byte(note+pressure)
0x011x Sync macros
0x0110: syncing ....(no value): start
0000: refresh sync
0x0111: stop syncing ....(no value): sends sync stop message 0xfc
0000: dont send 0xfc
0x0112: wave sync (!obsolete)
0x0113: start time sync ....(no value): get timerrate from BPM
0000: refresh sync
xxxx: set timer rate (in ms)
0x0120 cut/retrigger 0xAABB: A=count,B=delay
even A cuts, odd A retriggers
0x01fx System message fx
f0:sysex (no way to implement this here)
f2:song position (in beats) 0-0x03fff no 14 bit yet
f3:song select 0-0x7f
realtime messages: none
f8:timing clock (to be sent 6 times per Beat)
fa:start
fb:continue
fc:stop
fe:active sense
ff:system reset
0x02xx 7Bit NRP xx (short cut, see NRP ) 0-0x07f 0-0xfffe (shift 9Bit left)
00: LFO1 delay
01: LFO1 frequency
02: LFO2 delay
03: LFO2 frequency
04: Envelope1 delay
05: Envelope1 attack
06: Envelope1 hold
07: Envelope1 decay
08: Envelope1 sustain
09: Envelope1 release
0x0a (10) Envelope2 delay
0x0b (11) Envelope2 attack
0x0c (12) Envelope2 hold
0x0d (13) Envelope2 decay
0x0e (14) Envelope2 sustain
0x0f (15) Envelope2 release
0x10 (16) Initial pitch
0x11 (17) LFO1 to pitch
0x12 (18) LFO2 to pitch
0x13 (19) Envelope 1 to pitch
0x14 (20) LFO1 to volume
0x15 (21) Initial Filter Cutoff
0x16 (22) Initial Resonance (Filter-Q)
0x17 (23) LFO1 to Filter Cutoff
0x18 (24) Envelope 1 to Filter
0x19 (25) Chorus
0x1a (26) Reverb

3.5 Some explanations about MIDI data
I'm not that MIDI expert and just tried to apply the information I collected from several sources and bring them into an useful form for the MidiOut machine. MIDI is a protocoll for communication between musical devices, which basically consists of a stream of bytes (8Bit), where the single bits are transmitted sequentially, like serial communication (mouse,modem) does, but with a higher bitrate. If the highest bit is set, the byte is a so called status byte, which means some kind of command, while without the highest bit there are only 7 bits remaining for data, that's just the range 0-0x7f (0-127) in the table. To complicate things, different commands/statusbytes need/allow different data, some combining two (7Bit) data bytes to a 14Bit value (range 0-0x03fff). For controllers 0-31 this get's even worse: you can decide for yourself, if 7Bit resolution are enough and just send one data byte, or if you need some finer adjustments sending the second (least significant byte:LSB) as controller xx+32 (32-63). Also the meaning of the controllers is only rarely standard with General MIDI just giving a start. I don't know much about GS and XG.

3.6 NRP for AWE32/64 (and RP)
The NRP (non registered parameter) stuff is especially useful for the AWE, at least with this values (don't know if there's some kind of standard). For instance if you use an effect 0x215, a value of 127 for controller 99 and a value 0x15 (21) for controller 98 will be sent for the current channel. After this the active controller will be set to 0x26 (38) to transmit the effectdata, sou you best should use only the effectdata after. If you want to apply several NRP to the same channel, you have to use the effect value everywhere, but it still seems to fail in this case.

Registered Parameter (RP) as part of the General MIDI standard
I've not added a shortcut for the RP's yet, but I also have found only few used ones:
To set the pitch bend range for a channel, set controller 0x64 (100) (RP LSB) to a value of 0, and controller 0x65 (101) (RP MSB) to a value of 0, then set controller 6 to the desired range in semitones (up to 12).
RP's 1 and 2 are fine and coarse tune, no idea about the parameters.

3.7 Timing/Synchronizing
To sync a sequencer with BUZZ, you would select some option like "MIDI Sync" there, connect BUZZ and the sequencer with Hubi's LoopBack device (or any other of that kind) and BUZZ would have to send a sync start message (=effect 0x01fa), a stop (= 0x1fc) and between a lot of sync timer messages (=0x1f8).
But only one sync message per row isn't enough, you need 6 of them, and this with as equal distance as possible. This was the thing, which should be easy to achieve (BUZZ manages several times 44100 samples the second, shouldn't it manage these 6 syncs in a row?), but nonetheless was rather hard to implement, and even now has (still?) some drawbacks.

One simple solution is to set the TBP rate to 6*4=24 and use only every 6th line for your song (but you don't have to) and every line for a sync message 0x01f8. This should get you going, but the resulting patterns are hard to deal with.

To get timing more correct I used the event mechanism of the BUZZ machine interface, so at least a synchronisation between Wave and MIDI output can be achieved (method attribute set to 3 or 5).

Further I experimented with an own timer, which is based on the BUZZ BPM and TBP settings, but else works independent from the BUZZ timing (sync command 0x0113). This came close, but had a lot of disadvantages:
-limited timer resources
-distortion of BUZZ timer ?
-even best resolution of 1ms matches only distinct BPM rates, for instance 119 BPM <-> 21ms, 125 BPM <-> 20ms, 131 BPM <-> 19ms, the formula is

             time_ms=60000/ BeatsPerMin/6/TicksPerBeat;

(This is also of concern for doing the sync messages with an extern programm, like Vellocets vmidisync, and BUZZ sending only start and stop messages (effects 0x01fa and 0x01fc): get vmidisync together with other useful programs (vmidijoy and c2nrp) at Vellocets homepage .
Starting the timer with effect 0x0113 and a non-zero data just does the same as Vellocets vmidisync, with the advantage to let the BUZZ pause button send a configurable stop message and pausing the sync messages. So 0x0113 0x00014 just would give the BPM rate of 125 (unfortunately not exact), without any further adjustments (=not synced to BUZZ).
Starting the timer with no data-value (0x0113 ....) calculates the rate nearest to the actual BPM setting of BUZZ and starts the timer. Now you have to put a command 0x0113 0x0000 (or just .... 0x0000) in EVERY row. This does two things: it corrects the mismatch between the BUZZ BPM setting and the timer rate by sending additional or inhibiting sync messages if necessary, and further it allows to restart the timer after you pressed the pause button in BUZZ (I really found no other way, strange). This gives an ALMOST usable sync feature, but ALMOST is not exactly what would be the idea of synchronizing :].

The last implemented method is based on the wave based timing of the methods 4 or 5 and basically just tries to always send 6 sync messages between two BUZZ rows. Even succeeding with this there was no perfect sync, so I added the sync fine tune option, which adds or removes a sync message after several rows. This finally gave kind of satisfying results over several minutes, but still may break due to computer activity or ???

To use this you would make 3 patterns for vMidiOut. In the start pattern place a command 0x0110 .... in the first row, followed by all other lines set to .... 0000. The second pattern, which just refreshs the sync and has to be repeated the whole sequence, just consists of .... 0000 (or maybe a 0x0110 0000 in the first row). The last pattern to stop the sync messages has a single 0x0111 in it (either a 0x0111 .... to send a final stop message or a 0x0111 0000 to leave this out).


4. History
v1a:
      -Menu commands for Note-off, Controller reset
v1b:
     -Data incompatibilities! (at least it should not crash with old ones)
     -MIDI device is now machine attribute
     -delay byte on place of former device byte and new effect change byte as last byte (both still unused)
      -changed parameter ranges for continuous controllers, NRP and pitchwheel to 0...0xfffe (ctrl 0, 98-127 stay 0..0x7f)
      -assumed 7Bit for all controllers (14Bit maybe come later as extra effects)
      -Bugfix with NRP causing System reset (0xff) message
      -implemented event method by Jeskola with correct timing (Attribute method=1), still causes some crashs :(
      -process program change, controllers and notes in this order (seems more logical)
      -two new event based methods (2 and 3), but you should use method 0 or 3 only
      -two new experimental sync macros:
         -0x0110 sends a start message and begins a series of event based sync messages (set effectdata to 1),
         (these are still too slow)
         -0x0112 bases it's timing on the BUZZ wave routines, but very imperfect currently :(
         you'll get a BPM independent, very fast and inconstant rate for effect data 1,
         you can divide the rate by raising the data value.
         -Effect 0x0111 stops these syncmessages for both and sends a stop message (don't use 0x01fc only!).
         -Unfortunately all these messages don't stop with BUZZ pausing :(
v1c:
     -added option for playing machine with Midi (very useful for a MIDI machine :]
     -BUZZ stop now ends sync messages and sends ctrl. 123 :)
     -timer sync command 0x0113 :)
     -configuration dialog
     -stop command 0x0111 0x0000 doesn't send stop message
v1d:
     -option for BUZZ-stop to send all note-offs (finally getting rid of all hanging notes :)
     -removed some obsololete things: event sync (0x0110) and methods 1 and 2
     -timer sync must be enabled in options dialog (this way you won't have several timers without thinking about it ;)
     -option to have monophon BUZZ tracks
     -fixed event based timing of method 3
     -some people say, my machine is rather hard to understand: it will get worse ;)
     -added method 4 with wave based timing (delta time =method 0)--> slides and delays should now be possible
     -added method 5 with event + wave based timing (delta time =method 3)
     -added general MIDIout delay, works only with methods 4+5 (with method 5 this should usually set to 0,
       with method 4 this shall be used to correct the gap between wave and MIDI)
     -added event delay for methods 4+5 (still in "work"-units)
     -changed code to use several (2) event cues to have correct delay for method 5
     -added effect slides (still system/bpm dependent) for 7Bit Ctrl, NRPN and pitch wheel
     -added a lot of descriptions for the pattern and parameter view, for the idea and instruments thanks to Rout
     -all time units are now (more or less) system independent (delay 0...0xff for one row)
     -added new sync command based on methods 4 or 5 as 0x0110 (it's as poor as 0x0113 :( but without timer :)
     -added cut/retrigger command :)
     -changed defaults to method 5 and monophone tracks
     -changed option dialog a lot, especially choose device by name now
     -fixed some bugs with synchronizing
     -use 3 event cues now (to access old data for sync and may be other purposes)
     -fixed crashes during machine startup and avoided them for shutdown
     -added sync finetune: finally some positive results with this
     -use slide-end value as new default data value
possible plans/future (and still missing things):


 

contact me: ( vII) or look at my Homepage