Skip to content

Latest commit

 

History

History
307 lines (226 loc) · 27.9 KB

README.md

File metadata and controls

307 lines (226 loc) · 27.9 KB

playsmf

If you have additional questions, remarks, comments, ideas, requests - contact bm3_2000@yahoo.com

playsmf is a small, but powerful Windows (32/64bit) commandline standard midi file (SMF) player. Its specifically designed for low CPU and memory consumption to leave enough system recources for other applications such as soft-synths, DAWs, mixer apps, etc. while playing live and running in background.

In addition it comes with intrinsic flow control features based on labels, jumps and interrupts defined by smf marker events. This allows to program loops, breaks, fills, intros, outros, etc. In combination with realtime interrupt (song sequence transition) control, based on incoming midi data with or without chord recognition, the player can turn into an fully equipped arranger/arpeggiator accompaniement software. However unlike typical arpeggiators or style players, the player doesnt do any (more or less intelligent) event data modification such as transpose, volume adjustments, etc. to the smf midi data and plays strongly the raw data exactly as provided by the smf. This gives full transparency to all transmitted midi data and you know exacly what gets played with each individual chord, however the smf needs to provide individual pattern for all required key/scale/inversion combinations which are played during a live session. Therefore its possible to play individual pre-compiled pattern - for instance with randomized timings/volume/controller/sysex events - for each individual chord. On purpose, the player lags for a graphical user interface since everything should be controlled in realtime by external MIDI equipment without taking your hands off the keyboard while playing. This includes direct access to style variations, grouped track mutes/unmutes, start/stop/continue/reset control, etc.

example files

To get started quickly, few example midi files are attached. Some of them are converted from Yamaha style files to demonstrate the players capabilities. Therefore best results will be achieved with an XG compatible sound device. In order to get full realtime performance, it is strongly recommended to use either real midi equipment or softsynths with low latency settings (response lag time <= 10ms). For realtime accompaniement demonstration of course a real midi controller aka. keyboard is strongly recommended as a primary input device.

In case you dont know your midi device IDs or names for the correct player setup, run MidiPorts.bat to list all input and output devices respective their IDs. Chose the right ones as primary input and output devices and adjust command line parameters #3 and #4 accordingly.

The style examples are typically setup with chord recognition left hand across 2 octaves thru keys 36..59 and melody right hand thru keys above 60 (middle C). If required, adjust the transmission midi channel on your primary input device in order to attach to the right smf tracks while playing.

example video link

example video link example video link example video link example video link example video link example video link example video link example video link example video link example video link example video link example video link example video link

console output screen

Generally there is not much to display since the user should more focus on playing and listen rather than watching the screen. However in some situations it comes handy to track and follow requested sequence transitions. Therefore the console output displays one line for each transition request with additional information such as labels and song/sequence positions for playing and target sequences.

output MIDI devices

The player generally allows to play across multiple output devices simultaneously. Typically devices are chosen by SMF Meta-Event 0x20 (port ID select) at the very beginning of each individual track, but in addition the player also allows switching port IDs while playing at any song position. If port-select events are missing, the player uses the default midi output device provided by command line parameter #3. In addition SMF Meta-Text-Event 0x9 is supported as well for name based port selection, however it is recommented to use device IDs rather than port names for better across system portability.

midi smf recording

The player generally supports full live session recording and captures all data across all sources such as smf-, primary- and secondary inputs at once. This provides a merged smf output for either further offline processing or manual editing and inspection in any sequencer software. The output smf structure is organized in track groups and tracks for each individual data source and their functional data splits. This includes the following output tracks:

  • "conductor" track - contains general SMF setup data such as Tempo, Key-Signature, Time-Signature, etc. and Marker Labels for each individual Label transition. Together with the primary chord track, you can easily follow chord changes respective their triggered sequence transitions.
  • "SMF*" tracks contain all input SMF data in the same track structure and order as provided by the input SMF
  • "Primary" - contains all remaining events which are not part of the primary key zones below (mainly all non-key channel, SysEx and system or realtime data)
  • "Pri-Var" - variation key zone
  • "Pri-Mute" - single mute key zone
  • "Pri-Mutes" - mute-set key zone
  • "Pri-Chord" - chord key zone
  • "Pri-Other" - all other key events which are not part of the key zones above
  • "Zone*" - track(s) w/ midi thru data as defined by the zonal key setup per individual thru zone
  • "Secondary*" - track(s) w/ non-channel data from Seondary input port(s) while channel data get mixed directly into corresponding SMF tracks

In addition, PortID and PortName information is stored for each track in case multi port setups are used. So if you replay your recording using playsmf, the right midi output ports should be selected automatically.

Recorded data will be stored into a smf named MyMidRecordTimeStamp.mid within the current working directory. This way you can record as many takes as required without deleting older ones for further processing.

Recording is controlled by command line parameter #8 (0x0ff = off, else enabled). In addition, the parameter controls internal smf recording using a message mask filter scheme in the following binary/hexadecimal format:

binary format: 0b_0mmmmmmm_MMMMMMMM_e_fffffff_FFFFFFFF
0mmmmmmm: data mask (7 bit)
MMMMMMMM: status mask (8 bit)
0fffffff: data filter (7 bit)
FFFFFFFF: status filter (8 bit)
e: disable/enable event playing (allows to record internal events w/o playing them)

example argument settings (hex values):

  • 0x0ff - recording off
  • 0x8000 - record all events across all input sources
  • 0x1 - record only external primary input events
  • 0x7ff040b0 - external input + internal damper pedal events across all channels w/o playing them
  • 0x7ff0c0b0 - external input + internal damper pedal events across all channels
  • 0x7fff06ff - external input + internal marker text events (record meta events)
  • 0xff80f0 - external input + internal sysex events (might be used for sysex data/dump requests)

supported midi event types (playing, recording)

Basically all types of midi events including system common (sysex, time code, song select/position, etc.) and system realtime (start, stop, continue, active sensing) events are supported. Since system common (except sysex) and realtime events are not part of the smf specification, they are typically embedded in escape meta events (0xf7). The player generally supports such events and therfore it is possible to include for instance time code, active sensing or start/stop/continue transport control events into the song sequence for playback.

On the recording side, the player records everything including sysex, system common and system realtime events except midi timecode, timing clock and active sensing. Similar as on the player side, the recorder packs system common and realtime events (which are not part of the smf spec.) into escape meta events. This way you can also record something like start/stop/continue transport control data into the smf.

SMF text event support

SMF text events are typically ignored unless they are explicitly enbaled using argument 0x5xxxx where bits [15:1] represent a mask filter for SMF text meta event types 1 to 15. If bit 0 is disbaled, text messages within the 1st bar measure are ignored, else everthing gets displayed on the console output including track names, device names, etc. Text messages are simply displayed as they are on the console output screen at the time when they appear in the sequence. This can be used either for simple lyric printing or in combination with style type pattern to display messages on certain timestamps or in combination with marker labels for style accompaniement tracking.

Specifically on Windows10 and 11, the std. console text output supports VT100 terminal ansi escape codes according to the Microsoft VT100 implementation. This allows printing colored text output, changing terminal background colors or user defined console window titles in realtime while playing live.

MIDI-Thru and track-follow mode

The player generally supports MIDI-Thru functionality with split and multi-layer modes for live sessions. However instead assigning fixed devices/channels to play on, you can assign tracks to follow their current device/channel combinations while playing. This enables dynamic MIDI-Thru (re)assignments during a live session.

general system integration

The player runs as a standalone console application more or less in background and is mainly controlled by the primary midi input device/controller. In addition, the computer keyboard controls sequence restart (CTRL+C) and sequence exit (CTRL+PAUSE/BREAK) flow control jumps.

For seamless system integration along with your faforite DAWs, software synthesizers, virtual midi controllers, etc. it is strongly recommented to install virtual midi routers/cables to connect playsmf with other applications. Especially since MIDI devices are typically blocked when opened by one client, it is possible to route additional (secondary slaved) input devices thru playsmf to all open output devices. This allows to hook real or virtual midi controllers (mixer applications, etc.) thru playsmf to all open outputs.

Labels

Labels are basically address markers defining entry-points for either jumps or interrupts into the smf sequence. Internally they represent numerical addresses pointing to the next midi event within the smf data stream. Address labels can have any (positive) numerical value as long as they get only used in combination with intrinsic flow control using jumps. However in combination with external interrupts, they have to follow a certain structure with rules (see interrupt section). Numerical values can be presented in decimal or hexadecimal (0x) notation.

Sequence entry-points can share multiple Labels simultaneously. For example different chord-inversions for the same chord-type and chord-key can get assigned to the same physical sequence unless you want to split them on purpose to build individual variations. Label auto-expansion makes use of this feature generating additional internal chord/variation Labels if they are not already defined.

Two Labels are always defined: "sequence-start" and "sequence-exit". "sequence-start" represents a label pointing to the very first event in your sequence while "sequence-exit" presents a label right after the very last event. Those labels can be used to restart or exit your sequence and can be triggered by either <KeyStart>/<KeyExit> MIDI-keys or alternatively by <CTRL-C> or <CTRL-PAUSE/BREAK>.

Auto-Labels (Label-1)

One specific type of labels are Auto-Labels. Essentially they represent automatically generated internal labels and can only be used in combination with relative jumps since their absolute addresses are not puplic.

Label auto-expansion

If the smf doesnt provide individual pattern with dedicated labels across all chord-scales/inversions and variations, the player expands existing base-labels automatically.

Immediate Labels ('i')

Immediate labels define sequence sections which allow to interrupt immediately without waiting for the next sync point if an interrupt request is pending. Pending notes will be interrupted and the sequencer jumps immediately to the requested sequence entry point.

Re-trigger Labels ('r')

Typically looped sequences will not get retriggered on consecutive subsequent requests for the very same label unless the label is marked as re-trigger type.

Jumps

Jumps are basically branches jumping immediately to target address labels. The target label can either be a (positive) absolute address or a (negative) relative number of labels for relative backward jumps or looping. Numerical values can be presented in decimal or hexadecimal (0x) notation.

Interrupts (sequence transitions)

Interrupts are basically breaks stopping the current midi flow by jumping to defined target address labels. They are typically requested by external MIDI (NoteOn/Off) events either directly by single keys or thru the chord recognition module. Since both NoteOn and -Off events can request individual interrupts with different target address labels, its possible to trigger different sequences by either pressing keys down or releasing them.

Interrupt vectors (target Labels in binary / hexadecimal notation):

general binary vector structure: 0bvvvvttttUiiirrrr
 vvvv:variation
 tttt:key/chord-type
  0x0:direct key w/o chord recognition (0x00..0x7f)
  0x1:5th or single key thru chord recognition
  0x2:major triad or 7th
  0x3:minor triad or 7th
  0x4:sus2 triad
  0x5:diminished triad
  0x6:augmented triad
  0x7:maj7
  0x8:m7
  0x9:7
  0xa:mmaj7
  0xb..0xe: unused (reserved for extensions)
  0xf:internally used for auto-labels
 U: 0:key down; 1: key up
 iii:inversion (0: root down)
 rrrr:root (0x0..0xb; C..B)

#decoded value examples
0x000..0x07f single key pressed down w/o chord recognition
0x100..0x10b 5th root C..B no inversion
0x110..0x11b 5th root C..B inverted
0x120..0x12b single key root C..B
0x200..0x20b major triad root C..B no inversion
0x210..0x21b major triad root C..B 1st inversion
0x220..0x22b major triad root C..B 2nd inversion
0x300..0x30b minor triad root C..B no inversion
0x310..0x31b minor triad root C..B 1st inversion
0x320..0x32b minor triad root C..B 2nd inversion

Interrupt sequence flow transition syncronisation

In order to guarantee smooth song sequence flow transitions, interrupts are only taken at either jump marker positions or certain user-defined midi events such as metronome clicks, controller data, meta events, etc. within the song sequence. This approach decouples interrupt syncronisation completely from fixed time signatures or grids in terms of bars, beats, ticks and refers only to so called interrupt (INT) positions - basically song positions where interrupt transtions are allowed.

Interrupts are controlled by command line parameter #9 (0x0ff = off, else enabled) using a similar event mask filter scheme as used for recording in the following binary/hexadecimal format:

binary format: 0b_0mmmmmmm_MMMMMMMM_e_fffffff_FFFFFFFF
0mmmmmmm: data mask (7 bit)
MMMMMMMM: status mask (8 bit)
0fffffff: data filter (7 bit)
FFFFFFFF: status filter (8 bit)
e: disable/enable event playing (0: use event only for interrupt sync w/o playing)

example argument settings (hex values):

  • 0x00008000 - trigger on all events (used for regular smf playing)
  • 0x7fefa189 - trigger on NoteOn events from channel 9, key 0x21 (metronome klick - e.g. use for styles)

Please note that interrupts are still triggered on Jump position markers even though the interrupt argument might be turned off (0x0ff).

Special Jumps - return from interrupt (Jump-1/Jump-3)

One specific jump type is return-from-interrupt. Actually there are two basic types of sequences: "non-return" (e.g. ending in a infinite loop) or "return" ending with an "return-from-interrupt" jump. Return-type sequences terminate with an immediate jump back to the caller sequence bei either restarting the caller sequence from its interrupt entry point (Jump-1) or from the latest label playing when interrupted (Jump-3).

Conditional Jumps/Interrupts (transitions)

Typically jumps and interrupts are taken unconditional and immediate unless they are marked as conditional ones.

Jump continue to next sync ('>')

This is a regular Jump, however the sequence doesnt branch if a non-return interrupt is pending and continues beyond the jump marker without clearing the interrupt flag. Therefore the interrupt is not taken immediately and the sequence will continue until the next interrupt sync point (INT) is reached. This allows to run intermediate transitional sequences before jumping to the target sequence.

Jump continue ('>>')

This is a regular Jump, however the sequence doesnt branch if a retrigger interrupt is pending and continues as normal beyond the jump marker. This behavior can be used to transition into a new variation without explicitly requesting a variation change.

Variations

A set of Labels in the range from 0x000-0xfff is called a variation. To include multiple variations, the player takes the most significant Label digits as a variation number. So for instance Labels between 0x0000-0x0fff belong to variation 0 while Labels between 0x1000-0x1fff belong to variation 1. Variations can get switched by the keys right below the Mute-zone. If you switch from a non-return (looping) sequence to a return (one-shot) sequence variation, the variation will return back once finished, else you'll stay in the new variation. This allows to implement variations with fills or breaks in contrast to regular main (looping) variations.

Mute sets (groups)

A mute-set represents a list of simultaneously muted tracks. So with multiple distinct mute-sets, it is possible to mute and unmute multiple tracks simultaneously. For smooth mute/unmute transitions, a requested mute-set change is only taken at interrupt sync points. Mute-sets are primarily used to build pattern variations. They can get switched in realtime with the keys below the chord recognition zone. Two default mute-sets are always defined: mute-all and unmute-all.

single track mute

In addition to mute-sets, it is possible to toggle individual mutes per track while playing. Single mutes are accessed with the keys below the mute-set zone. However this feature is currently limited to the first track only since there is not much use of single mutes/unmutes.

Midi thru and track follow mode

For live performance, midi thru functionality with multi-zone (split) and multi-layer functionality is implemented. However instead of specifying fixed target output port/channel combinations, midi thru gets hooked to tracks and follows their port/channel combination while playing. This provides the flexibility to change the midi thru output with the individual pattern played. For instance one can assign all major chords to one output while all minor chords get routed to another one.

Typical "style" smf structure

A style-like smf structure starts typically with an initialization sequence at the very beginning. This part contains all events (sysex, controller, local control off, etc.) to setup the midi equipment. Once finished, it can immediately transition into a "silent-loop" just waiting for input. In order to hear if the player is alive, you can additionally put a quiet metronome click into the silent-loop. After that, the data section starts containing all data for the various chord types (major, minor, sus2, etc.). At the very end, you can have an exit-sequence which gets only transmitted when the player finishes (e.g. turn local control on, etc).

exit codes

The player exits with different return codes based on the exit scenario.

  • 0 regular exit: the player normally reached the sequence end
  • 1 error: e.g. file not found or argument error
  • 2 timeout: the player stopped due to midi timeout (lost midi equipment connection)
  • 3 CTRL+PAUSE/BREAK: player was forced to exit from computer keyboard
  • 4 exit: the player was forced to exit from primary midi input (exit label key)
  • 5 last: the player stopped normally after reaching the last sequence label

The player comes with specific features such as:

  • intrinsic realtime player flow control using jump/branch commands in combination with target address labels (primarily used for loops)
  • realtime midi controlled jumps (interrupts) in combination with target address labels
  • chord recognition for midi realtime interrupt/break/jump control
  • multiport capability controlled by SMF Port metaevents per track
  • SysEx in/out support
  • multi-layer/multi-zone midi thru functionality based on "track follow mode" (thru port/channel follows selected tracks)
  • realtime midi controlled mutes/unmutes based on "mute-sets" (mute/unmute multiple tracks simultaneously at interrupt break points)
  • record/save all incoming midi and sysex events into standard midi files

usage:

playsmf.exe <MidiFile> <TimerPeriod> <DefaultMidiOutputDevice> <DefaultMidiInputDevice> <TimeOut> <Channel> <Ofs> <REC> <INT> <KeyStart> <KeyExit>

  • 0 playsmf.exe - application
  • 1 <MidiFile> - inputfile *.mid
  • 2 <TimerPeriod> - windows timer resolution in ms or -1 if not used
  • 3 <DefaultMidiOutputDevice> - default midi output device ID or device name if quoted
  • 4 <DefaultMidiInputDevice> - default primary midi input device ID or device name if quoted
  • 5 <TimeOut> - primary midi-input timeout in ms or -1 if not used (player will close if there is no midi activity)
  • 6 <Channel> - primary midi-input channel filter/router used for midi-thru functionality
  • 7 <Ofs> - primary midi in key offset [-127:0:127]
  • 8 <REC> - midi recording filter (default 0xff - off)
  • 9 <INT> - interrupt message control filter (default 0x00008000 - all messages)
  • 10 <KeyStart> - start key (EntryLabel)
  • 11 <KeyExit> - exit key (ExitLabel)

additional options beyond argument address >= 12:

  • ThruZone ::= (<LowKey> <HighKey> <Track> <Delay> <KeyOffset> <Von> <Voff>)

Defines a midi-thru zone with the following mantadory attributes/parameters:

  • <LowKey> lowest key in zone [0:127] (if <=-2 use from previous zone; if -1 use +1 from previous zone)
  • <HighKey> highest key in zone [0:127] or if negative
  • <Track> assigned track [-1,0:n] (-1: no assignment)
  • <Delay> delay im ms [0:n] (experimental feature - use in rare cases with only small delay values to achive flanger-type effects)
  • <KeyOffset> transpose [-127:0:127] or fixed key [128:255]
  • <Von> note-on velocity modulator <0xssoo>: 0=1,([1:5:255]-1)*.25 -> [1,0:1:63.5] scale factor; [-127:0:127] offset
  • <Voff> note-off velocity modulator <0xssoo>: 0=1,([1:5:255]-1)*.25 -> [1,0:1:63.5] scale factor; [-127:0:127] offset

notes/comments/terminologies:

  • multiple zones can be defined consecutively
  • chord recognizion key range is always derrived from the 1st defined zone in argument list (mandatory zone - even though chord recognition is not used)
  • midi-thru works generally in 'track-follow-mode' with the primary midi input attached to individual tracks following their output device/channel combinations while playing
  • each individual key allows having up to 8 zones/tracks (layers) attached
  • 'active' zones/tracks are all zones (tracks) belonging to the latest pressed key
  • incoming channel messages including controller, pitch-bend, program-change, etc. are generally routed to active zones/tracks only. this allows to change selectively patches, volumes, pannings, etc. for active zones only while playing
  • exceptions are foot/pedal controller such as soft, sostenuto and sustain which are generally sent across all defined zones/tracks simultaneously
  • tracks can get dynamically re-assigned while playing by changing the incoming midi channel. to enable this option, <Channel> needs to be <= -2. in this case, the received channel is added to the target tracks.

port/device mapping

If the smf contains port-select meta-events where port numbers doesnt match to target device-IDs, you can apply additional port->device mappings.

  • <0x0001ppDD> maps smf-port <pp> to device <DD>
  • <0x0003ppDD> same as above, but devices count reverse starting with max-1 device number

secondary slaved midi-input devices

In case you want to hook additional real or virtual midi-equipment such as controller, mixer, keyboards, etc. to your output devices while playing, you can route them thru playsmf attaching directly to midi-tracks in track-follow mode. This might be required for devices using single client midi drivers. Typically those drivers dont allow to open multiple clients in parallel. Since this option works in track-follow-mode, each incoming channel is routed to one corresponding track starting from track <TT>.

  • <0x0002TTDD> opens secondary slaved midi input device <DD> attaching to track <TT>
  • <0x0004TTDD> same as above, but devices count reverse starting with max-1 device number

midi device reset options

In order to reset midi equipment upon player start and/or exit, you can optionally define individual midi messages (typically controller reset messages) sent across all devices/channels before the player starts and after the player exists.

  • <0x01mmmmm0> start midi message (play given message across all channels before smf sequence starts)
  • <0x02mmmmm0> exit midi message (play given message across all channels after smf sequence stops)
  • <0x03mmmmm0> start/exit midi message (play given message across all channels before and after smf sequence starts/stops)

smf intrinsic arguments

In order to store command line arguments with the smf, the player supports sequencer specific meta messages to set and/or override command line arguments. Argument data is stored in 32-bit integer values <DD> starting from argument address <AA> followed by one or more arguments.

  • <0xff> <length> <0x7f> <0x00> <0xab> <0xcd> <0x00> <AA[31:24]> <AA[23:16]> <AA[15:8]> <AA[7:0]> (<DD[31:24]> <DD[23:16]> <DD[15:8]> <DD[7:0]>)*