Skip to content

ChipTuneEngine File Format

Rasmus Anthin edited this page May 19, 2024 · 15 revisions

The crown of this header-only library is the ChipTuneEngine class. The ChipTuneEngine class plays a tune defined in a text file with file extension *.ct (for Chip Tune). This page outlines the format of such a tune file.

It is important to note that all commands such as instrument, adsr, filter and params must have all their operands on a single line for them to be properly parsed.

Instrument Definitions

Syntax

instrument <instrument_name> <waveform> ["params:"<params_id>]
 ["ffx:"<freq_preset_effect>] [afx:<ampl_preset_effect>] ["pfx:"<phase_preset_effect>] 
 ["adsr:"<adsr_id>] ["flt:"<filter_id>] ["vol:"<volume>]

instrument <instrument_name> "ring_mod_A:"<ring_mod_instr_A> "ring_mod_B:"<ring_mod_instr_B>
 ["adsr:"<adsr_id>] ["flt:"<filter_id>] ["vol:"<volume>]

instrument <instrument_name> "conv_A:"<conv_instr_A> "conv_B:"<conv_instr_B>
 ["adsr:"<adsr_id>] ["flt:"<filter_id>] ["vol:"<volume>]

instrument <instrument_name> "((" <w0> ", " <instrument_name0> "), (" <w1> ", " <instrument_name1> "), " ... ")"
 ["adsr:"<adsr_id>] ["flt:"<filter_id>] ["vol:"<volume>]

instrument <instrument_name> "&"<lib_instrument_name>
 ["ffx:"<freq_preset_effect>] ["afx:"<ampl_preset_effect>] ["pfx:"<phase_preset_effect>]
 ["adsr:"<adsr_id>] ["flt:"<filter_id>] ["vol:"<volume>]

Where

<instrument_name> := [A-Za-z][A-Za-z0-9]*
<waveform> := "sine"|"square"|"triangle"|"sawtooth"|"noise" ; square, triangle and sawtooth uses the duty_cycle attribute. Default values are 0.5 for square wave and triangle wave and 1 for sawtooth wave.
In the case with the triangle wave, the duty cycle goes from reverse sawtooth wave at 0 to pure triangle wave at 0.5 and regular sawtooth wave at 1.
<duty_cycle> := 0..1
<volume> := 0..1
<lib_instrument_name> := "PIANO"|"VIOLIN"|"ORGAN"|"TRUMPET"|"FLUTE"|"GUITAR"|"KICKDRUM"|"SNAREDRUM"|"HIHAT"|"ANVIL" ; Predefined instruments.
<ffx> := "CONSTANT"|"JET_ENGINE_POWERUP"|"CHIRP_0"|"CHIRP_1"|"CHIRP_2" ; Frequency effect.
<afx> := "CONSTANT"|"JET_ENGINE_POWERUP"|"VIBRATO_0" ; Amplitude effect.
<pfx> := "ZERO" ; Phase effect.

Examples

instrument PIANO ring_mod_A:I0 ring_mod_B:I1 adsr:0
instrument I0 square adsr:1 flt:0 afx:VIBRATO_0
instrument I1 square params:0
instrument ORGAN ((0.5, PIANO), (0.2, I0)) flt:0
instrument FLUTE_VIB &FLUTE ffx:CONSTANT afx:VIBRATO_0 pfx:ZERO vol:0.15
instrument TRUMPET &TRUMPET ffx:CONSTANT afx:CONSTANT pfx:ZERO vol:0.6
instrument TRUMPET_VIB &TRUMPET ffx:CHIRP_2 afx:VIBRATO_0 pfx:ZERO vol:0.6
instrument KDRUM &KICKDRUM
instrument SDRUM &SNAREDRUM
instrument HIHAT &HIHAT
instrument STEEL_PAN conv_A:FLUTE_VIB conv_B:SDRUM adsr:1

Modulation Envelopes

Syntax

adsr <adsr_nr> "["<attack_mode> <attack_ms> [<level_begin>] [<level_end>]"]"
               "["<decay_mode> <decay_ms> [<level_begin>] [<level_end>]"]"
               <sustain_level>
               "["<release_mode> <release_ms> [<level_begin>] [<level_end>]"]"
adsr <adsr_nr> "&"<lib_adsr_name>

Where

<adsr_nr> := [0-9]+
<attack_mode> := <adsr_mode>
<decay_mode> := <adsr_mode>
<release_mode> := <adsr_mode>
<adsr_mode> := "LIN"|"EXP"|"LOG"
<attack_ms> := [0-9]+
<decay_ms> := [0-9]+
<release_ms> := [0-9]+
<sustain_level> := 0..100
<level_begin> := 0..100
<level_end> := 0..100

Charts

+ 100
|     d0
|    .\        s
|    . \   ________  
|    .  \ .        . 
|   /    \.        \ r0
|a1/   d1           \
| /                  \
|/a0.                 \ r1
+-——-+-——-+--------+—--+---> t
|    |    |        |   |
ta   tad  tds      tsr tr

Where

  • Attack : t ∈ [ta = 0, tad].
  • Decay : t ∈ (tad, tds].
  • Sustain : t ∈ (tds, tsr].
  • Release : t ∈ (tsr, tr].

Here we can see that the pairs (a1,d0), (d1,s) and (s,r0) internally actually have separate values and they don’t have to be equal, although they normally are. E.g. Normally a1 = d0 but they can be different, in which case creating a discontinuous jump.

By default the levels are set to:

a0 = 0
a1 = d0 = 100
d1 = s = r0 = 80
r1 = 0

That would instead look like a normal ADSR envelope:

+ 100
|     a1=d0
|       /\   
|      /  \   
|     /    \   d1=s=r0
|    /      \___________           
|   /                   \
|  /                     \
| /                       \ r1
|/ a0                      \
+—————--+-—-+---——-——--+---+——---> t
|       |   |          |   |
ta      tad tds        tsr tr

What's worth to note is that these levels are actually arbitrary, but a normal ADSR is:

a0 = 0
a1 = d0 = 100
d0 > d1
d1 = s = r0
r0 > r1
r1 = 0

Examples

; Linear attack during 15ms starting at level 0 ending at level 50 (vol: 0.5).
; Exponential decay during 10ms starting at level 100 (vol: 1) thus making a discontinuous jump from the attack.
; Sustain at 50 (vol: 0.5).
; Logarithmic release during 100ms.
adsr 0 [LIN 15 0 50] [EXP 10 100] 50 [LOG 100]

adsr 1 [EXP 16] [EXP 20] 80 [LIN 70]
adsr 2 &ORGAN_2
adsr 3 [LOG 60] [LOG 30] 50 [LOG 10]

Filters

Syntax

filter <filter_nr> <type> <op_type> <order> <cutoff_frq_mult> <bandwidth_frq_mult> <ripple> <normalize>

Where

<filter_nr> := [0-9]+
<type> := "Butterworth"|"ChebyshevTypeI"|"ChebyshevTypeII"
<op_type> := "LowPass"|"HighPass"|"BandPass"|"BandStop"
<order> := 1..
<normalize> := "false"|"true"

If bandwidth_frq_mult is not to be used, then set it to a value <= 0.

Examples

filter 0 Butterworth LowPass 2 1.5 0 0.1 false

Waveform Generation Params

Syntax

params <params_nr> ["sample_min:"<sample_range_min>] ["sample_max:"<sample_range_max>]
                   ["duty_cycle:"<duty_cycle>] ["duty_cycle_sweep:"<duty_cycle_sweep>]
                   ["min_freq_limit:"<min_frequency_limit>]
                   ["freq_slide_vel:"<freq_slide_vel>] ["freq_slide_acc:"<freq_slide_acc>]
                   ["vib_depth:"<vibrato_depth>] ["vib_freq:"<vibrato_freq>]
                   ["vib_freq_vel:"<vibrato_freq_vel>] ["vib_freq_acc:"<vibrato_freq_acc>]
                   ["vib_freq_acc_max_vel_lim:"<vibrato_freq_acc_max_vel_limit>]
                   ["noise_flt_order:"<noise_filter_order>] ["noise_flt_rel_bw:"<noise_filter_rel_bw>]
                   ["noise_flt_slot_dur_s:"<noise_filter_slot_dur_s>]
                   ["arpeggio:"<arpeggio>]

Where

<arpeggio> := "(("<t0_ms>, <freq_mult0>"), ("<t1_ms>, <freq_mult1>")," ...")"

Score

Notes

Syntax

<pitch> <duration_ms> <instrument>

Where

<pitch> := [A-G](b|#)?[0-8]

Examples

A3 33 MyInstrument_3
Gb4 250 PIANO
E#5 30 trumpet ; Same pitch as F5.

Modifiers

Syntax

NUM_VOICES <num_voices>

Sets the number of voices. Can only be set once and before the score begins.

TIME_STEP_MS <time_step>

Time step (tempo) of the song. A shorter value will make the duration of each note row after it shorter. This can be set multiple times throughout the music.

VOLUME <volume>

Globally sets the (accumulated) volume. This volume is multiplied with every volume level of each note row that comes after it. This can be set multiple times throughout the music.

Branching

Syntax

LABEL <label>

Destination label for jump with GOTO or GOTO_TIMES.

GOTO <label>

When execution arrives at this command, it will jump to the LABEL that matches <label>.

GOTO_TIMES <label> <count>

Same as GOTO but will only jump <count> times.

ENDING <jump_no>

Allows for structures like this:

LABEL my_label
... ; Part A. Will be played every repeat.
ENDING 0
... ; Part B. Will be played the first time (0th jump).
ENDING 1
... ; Part C. Will be played the second time (1st jump).
ENDING 2
... ; Part D. Will be played the third time (2nd jump).
GOTO_TIMES my_label 2

In this example, it will play: A, B, A, C, A, D and then move on to the notes below the GOTO_TIMES command.


CODA

This is a muscial notation that denotes a passage that leads the music to an end. It works like a specialized label.


SEGNO

Segno means sign. It is a kind of a musical "goto label".

Example:

TIME_STEP_MS 50
NUM_VOICES 3


#TAB | A4 400 STEEL_PAN |
#END
VOLUME 1
LABEL lbl0
TAB | A2 50 KDRUM | C4 250 PIANO adsr:3 |
TAB -
TAB -
TAB -
TAB | A6 20 HIHAT | - | G4 250 ORGAN adsr:3 |
TAB -
TAB -
TAB -
TAB | B4 50 SDRUM |
TAB -
TAB -
TAB -
ENDING 0
TAB | A6 20 HIHAT | E#4 250 PIANO | A4 600 FLUTE_VIB |
TAB -
TAB -
TAB -
TAB | A2 50 KDRUM |
TAB -
TAB -
TAB -
TAB | A6 20 HIHAT |
TAB -
TAB -
TAB -
ENDING 1
TAB | A6 20 HIHAT | F#4 250 PIANO | Db4 600 FLUTE_VIB vol:3.5 |
TAB -
TAB -
TAB -
TAB | A2 50 KDRUM |
TAB -
TAB -
TAB -
TAB | A6 20 HIHAT |
TAB -
TAB -
TAB -
GOTO_TIMES lbl0 1
TAB | B4 50 SDRUM | G4 500 PIANO |
TAB -
TAB -
TAB -
TAB | A6 20 HIHAT | - | Gb4 250 ORGAN |
TAB -
TAB -
TAB -
TAB | A2 50 KDRUM |
TAB -
TAB -
TAB -
SEGNO
TAB | A6 20 HIHAT | D#4 360 TRUMPET |
TAB -
TAB -
TAB -
TAB | B4 50 SDRUM | - | C4 200 TRUMPET |
TAB -
TAB -
TAB -
TAB | A6 20 HIHAT | - | A3 200 TRUMPET |
TAB -
TAB -
TAB | - | - | - |
TAB | A2 50 KDRUM | - | C4 200 TRUMPET |
TAB -
TAB -
TAB -
TAB | A6 20 HIHAT | C4 200 TRUMPET |
TAB -
TAB -
TAB -
TAB | B4 50 SDRUM | Eb4 650 TRUMPET_VIB |
TAB -
TAB -
TAB -
FINE
TO_CODA
TAB | A6 20 HIHAT |
TAB -
TAB -
TAB -
TAB | A2 50 KDRUM |
TAB -
TAB -
TAB -
TIME_STEP_MS 70
TAB | A6 20 HIHAT |
TAB -
TAB -
TAB -
TAB | B4 50 SDRUM |
TAB -
TAB -
TAB -
TIME_STEP_MS 80
TAB | A6 20 HIHAT |
TAB -
TAB -
TAB -
TAB | A2 50 KDRUM |
TAB -
TAB -
TAB -
TIME_STEP_MS 90
TAB | A6 20 HIHAT |
TAB -
TAB -
TAB -
TAB | B4 50 SDRUM |
TAB -
TAB -
TAB -
TIME_STEP_MS 100
TAB | A6 20 HIHAT |
TAB -
TAB -
TAB -
TAB | A2 50 KDRUM |
TAB -
TAB -
TAB -
#DA_CAPO_AL_CODA
#DAL_SEGNO_AL_CODA
VOLUME 0.8
TAB | A6 20 HIHAT | F#4 120 STEEL_PAN adsr:2 |
TAB -
TAB -
TAB -
VOLUME 0.6
TAB | B4 50 SDRUM |
TAB -
TAB -
TAB -
VOLUME 0.4
TAB | A6 20 HIHAT | G5 350 STEEL_PAN adsr:3 |
TAB -
TAB -
TAB -
CODA
TIME_STEP_MS 150
VOLUME 0.2
TAB | A2 50 KDRUM |
TAB -
TAB -
TAB -
TAB | A6 20 HIHAT |
TAB -
TAB -
TAB -
TAB | B4 50 SDRUM |
TAB -
TAB -
TAB -
TAB | A6 20 HIHAT | F4 350 STEEL_PAN |
TAB -
TAB -
TAB -
DA_CAPO_AL_FINE
#DAL_SEGNO_AL_FINE
Clone this wiki locally