File: Automated_Testing_Support.txt

package info (click to toggle)
yoshimi 2.3.5.3-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 16,976 kB
  • sloc: cpp: 62,482; sh: 94; xml: 93; python: 45; makefile: 14
file content (217 lines) | stat: -rw-r--r-- 11,888 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217

Ideally, Yoshimi should reliably always behave the same as ever, allowing musicians to learn handling
the subtleties of a given voice. A carefully balanced MIDI score should not be blown out of proportions
due to some development work in the Yoshimi code base. To help keeping that level of compatibility,
since 2021 Yoshimi offers some support for automated testing.

A major obstacle to overcome for acceptance testing is non-determinism: when playing Yoshimi,
the sound will feel vivid, since every note will be subtly different. However, since this randomness
is based on a Pseudo Random Number Generator (PRNG) with very long sequence, in fact all of the
computations are deterministic, once we start from a well defined seed and initial state. One
single isolated note can thus be reproduced up to the last bit. However, as soon as you send
yet another MIDI note while sound is being calculated, because this second note essentially
starts at a random point in time, the calculations proceed from an non-predictable state
and turn over to complete randomness.

Based on these observations, Yoshimi provides a dedicated test invoker component accessible through
the CLI. Once triggered, this test invoker will terminate the regular sound production, similar as
when shutting down the Yoshimi application. The SynthEngine will then be re-seeded and ephemeral
processing state will be cleared out. After these preparations, a sequence of test notes will be
calculated by directly invoking the SynthEngine functions, synchronously. After completion,
the Yoshimi application will terminate, unconditionally discarding any unsaved state.

For a typical automated test case, Yoshimi would be started, possibly with a state file.
A CLI script would then be piped into STDIN, to set up the parts, instruments and settings
to be covered by this test, and finally the test calculation would be launched. By default,
the resulting sound bits will be discarded, just capturing the calculation time. However,
it is possible to dump this calculated sound into a RAW sound file, which can be opened by
many sound processing applications and libraries, when providing the correct parameters
of the produced sound: 32-bit floating point little-endian, two channels interleaved,
and the actual sampling rate (e.g. 48kHz). If we store such a calculated sound as baseline,
and then repeat the calculation and subtract the baseline sample by sample, the result should
be total silence. When detecting any residual sound, we can listen to that sound and determine
if the difference is sonic (=bad) or just noisy (-> maybe acceptable). And we can measure the
volume level (or RMS) of the difference to get a clue how large the musical impact might be.

Within the CLI, there is a special test context that can be entered with "set test".
The following parameters of the test are exposed for set / get / min / max / default commands:

    "NOte [n]"
        MIDI note number of the first test note, default: 60 == C4

    "CHannel [n]"
        MIDI channel number 1-based (1..16), default channel 1
        It depends on the actual part configuration which part(s) will be activated,
        the test invoker just passes this number (minus 1) to the NoteOn() function.

    "VElocity [n]"
        likewise, this value (0..127) is just passed to the NoteOn() function, default 64.

    "DUration [n]"
        Overall duration of the sound in seconds, default 1.0
        Can be fractional, minimum 0.01f (10 milliseconds). From this duration value,
        the exact number of samples is calculated and then rounded up to the next full
        buffersize / "chunksize", to determine how often the MasterAudio() function will
        be invoked. After these calls, SynthEngine::shutUp() will be invoked to clear all
        sounding notes and intermediary buffers.

    "HOldfraction [n]"
        Fraction of the duration time for actually holding the note, default 0.8 (80%).
        The resulting time is likewise rounded up to the next buffersize. After passing
        the corresponding number of MasterAudio() calls, the NoteOff() function is invoked,
        thus turning the active parts into note release state. All other processing, like e.g.
        reverb continues until reaching the full duration time.

    "REpetitions [n]"
        The test invoker can produce several isolated notes in sequence, default is 4.
        After each note, SynthEngine::shutUp() is invoked to avoid carrying the calculation
        state from one note into the following note; this call also clears out the buffers
        of global effects, so the next note starts with pristine state. However, the
        timing measurement only covers the NoteOn, NoteOff and MasterAudio() calls while
        excluding the time spent in shutUp(), since the goal is to capture the performance
        of regular sound processing.

    "SCalestep [n]"
        Move up/down by this number of semi tones before repeating, default is 4 (major third).
        This allows to cover the whole sonic range with a sequence of test tones; if the note
        number reaches the end of the allowed MIDI note range, it "bounces" and the series
        continues in reversed direction.

    "AOffset [n]"
        Launch an additional possibly overlapping note within each repetition; this allows
        to cover legato and portamento effects. The second note is started at a well defined
        reproducible point just before a SynthEngine::MasterAudio() call. The start offset n
        of this additional note is defined as fraction of the total "Duration" parameter and
        by default the hold time will be defined by the current "Holdfraction" setting.

    "AHold [n]"
        use a different hold-fraction for an additional (possibly overlapping) note. Together with
        the AOffset setting it is thus possible to start the second note within the play time of
        the main note, partially overlapping or completely afterwards.

    "SWapWave [n]"
        Provide a new wavetable to the first PADSynth item while test is underway.
        When PADSynth detects the availability of a new wavetable (which is typically built
        within a background task), it will reconfigure currently playing notes and possibly
        perform a cross fade for a smooth transition. New notes started after that point will
        only use the new wavetable. To cover this behaviour, the wavetable of the first enabled
        PADSynth kitItem (typically part 1 and kitItem 1) will be built right at that point when
        this command is given at CLI, and based on the parameters set at that point. Any parameter
        changes made afterwards will be reflected in the second wavetable. Each repetition will then
        start with the first wavetable, and after offset n (given as fraction of the total "Duration")
        has passed, the second wavetable will be swapped in. Both wavetables will be built after
        reseeding the Synth for reproducible test state.

    "BUffersize [n]"
        Actual number of samples to request from MasterAudio(), default is the global buffsize,
        which is defined by the Yoshimi settings, rsp. by the Jack or LV2 server. Note that
        SynthEngine::MasterAudio() will ignore values larger than this current global buffersize.
        Some settings behave different depending on the buffersize, and some sounds can be
        notably different, especially when passing very small values -- which can be
        investigated and documented through this test setting.

    "TArget [s]"
        File path to write sound data, default is empty, which means to discard calculated data.
        Yoshimi attempts to open a RAW sound file at the given location; if missing, the extension
        ".raw" will be added. Warning: an existing file will be overwritten without further notice!
        While calculating, the sound samples are just interleaved and copied into a buffer in memory.
        This buffer will be dumped into the actual file after finishing the actual test and thus
        outside of the timing measurement.

    "EXEcute"
        Shut down regular Yoshimi operation, re-seed the SynthEngine, launch test and then exit.

All these values can be changed by "set <setting> [newval]". If [newval] is omitted,
the default value for this setting will be used instead. This default value can be inspected
by "default <setting>".

Basically the "execute" command also needs to be given as "set execute".
However, as convenience shortcut, it is also possible to append the "execute" at the end
of another "set" command within the test context. Moreover, when in test context, the
"execute" command is also recognised as free standing command (without "set")


After re-seeding, also the PAD-Synth wavetables need to be rebuilt, since they might use harmonic
randomisation. Depending on the PAD size setting and the complexity of the instrument, this may
require a considerable amount of time. These preparations attempt to establish well defined starting
conditions, while retaining loaded instruments and any specific settings done via CLI to prepare
the test. This preparation is done in the function SynthEngine::setReproducibleState(int seed).
This function resets the continuous LFO time and beat settings, and then re-seeds the master
PRNG of the synthEngine with the given value. Moreover, it visits all parts and instructs all
existing OscilGen instances to draw a seed value from the global PRNG to reseed their local
basePrng, and also to draw and store a randseed for the local harmonicPrng. This randseed
is used on each NoteOn to re-seed the harmonicPrnc. Moreover, the setReproducibleState()
also triggers the rebuilding of PAD-Synth wavetables, which will also draw further random
values from the master PRNG.

Output Markers: The test invoker will send some markers and data to the Log,
which typically (depending on the Yoshimi settings) can be captured on STDOUT:

TEST::Prepare
    The regular sound processing has been shut down, and the invoking CLI has waited for
    the SynthEngine to reach muted state. Moreover, Memory for the working buffers has
    been successfully allocated and the output file is open for writing.
    Next step will be invoking SynthEngine::setReproduibleState()

TEST::Launch
    Start of the actual sound calculations to test, which can be a sequence of notes

TEST::Complete runtime ##.# ns speed ##.# ns/Sample
    Test completed successfully, yielding the indicated timings.
    Next step will be possibly to write the sound data file, and then
    the Yoshimi application will terminate, discarding all settings.



Example for using this test feature from the Unix shell:


/path/to/yoshimi <<ENDSCRIPT >mytest.log
set part 1 instrument 36
/
set test target test-out
execute
ENDSCRIPT


This test will overwrite the file test-out.raw with the produced sound
and write something similar to the following example into "mytest.log":

Yoshimi 2.x.y is starting
March is Little Endian
Card Format is Signed Little Endian 32 Bit 18 Channel
Yoshimi 2.x.y
Clientname: yoshimi
Audio: alsa -> 'default'
Midi: alsa -> 'No MIDI sources seen'
Oscilsize: 512
Samplerate: 48000
Period size: 256

Yay! We're up and running :-)
/usr/share/yoshimi/banks

Found 943 instruments in 25 banks
Root 2. Bank set to 42 "Ichthyo"
yoshimi> set part 1 instrument 36
Main Part Number 1
Main Part 1 loaded 0036-Harm.Princip
GUI refreshed
@ p1+, (Multi)
yoshimi> /
@ Top
yoshimi> set test target test-out
Target RAW-filename set to: "test-out.raw"
@ TEST: exec 4 notes start (C4) step 4 up to (C5) on Ch.1 each 1.0s (hold 80%) buffer=256 write "test-out.raw"
yoshimi> execute
Main Sound Stopped
TEST::Prepare
SynthEngine(0): reseeded with 0
TEST::Launch
TEST::Complete runtime 119753064.0 ns speed 622.1 ns/Sample
EXIT

Goodbye - Play again soon?