A decoder that translates Morse code audio to text.
Copyright (C) 2020-2024 Rabbe Fogelholm
Version | What |
---|---|
1.3 | Cygwin/Windows version |
1.4 | Linux support |
1.5 | Optional phase angle plot |
1.6 | Spikes suppression, selectable plot interval |
1.7 | Gaussian blur, frequency plot |
1.8 | Dropouts and spikes removal, improved plot |
2.0 | Selectable decoders and filtering |
2.0.1 | Corrections |
2.0.2 | Corrected default dips-merge limit parameter |
2.0.2.1 | Least-squares decoding, first version |
2.0.2.2 | Include decoded signal in plot |
2.0.2.3 | Support option -t with decoder -D4 |
2.0.2.4 | Candidate for 2.1.0 on master branch |
2.1.0 | Least-squares decoding |
2.1.1 | Various corrections |
2.1.1.1 | Refactoring |
3.0.0 | Adapting to frequency drift and fading |
3.0.1 | Guard against all-zeroes segments in .wav file |
3.0.2 | Clipping depth as hidden parameter, default 0.05 |
3.0.3 | Fix coding fault, improve performance |
3.1.0 | Add tone/space lengths histogram |
3.1.1 | Support .mp3 input |
3.1.2 | Correct Maven archive URL |
3.1.3 | Break up oversized dashes |
3.1.4 | Release content tidyup |
3.1.5 | Fix gnuplot behaviour for Cygwin/X |
3.1.6 | Add some missing code table entries |
3.1.7 | Adjust build scripts |
3.1.8 | Reporting number of dots and dashes (certain decoders only) |
3.2.0 | Introduce integrating decoder |
3.2.1 | Internal consistency fix, tweak parameters |
3.2.2 | Fix Makefile issue |
3.2.3 | Take tone separation from tone midpoints |
3.2.4 | Handle 24-bit 2-channel WAV, decode ..--. to '!' |
3.2.5 | Handle 24-bit 1-channel WAV |
3.2.6 | Build with Java 8 compatibility |
3.2.7 | Improve timings summary |
- Linux
- Windows with Cygwin/X
OpenJDK version 11 or higher is recommended. Binaries are built with Java 8 compatibility.
With this conversion utility installed, audio files in .mp3 format can be decoded.
The -S, -A, -P, -Y and -M options, for plotting various aspects of the recorded signal, invoke the 'gnuplot' program. On major Linux distributions, if 'gnuplot' is not already installed, use the provided package manager to install it.
On Windows with Cygwin/X the gnuplot-base and gnuplot-X11 packages are needed.
Building requires GNU Make. On Windows with Cygwin/X, GNU Make is provided in the 'make' package.
The build process downloads the 'iirj' filter package. For licensing information, read here: https://github.com/berndporr/iirj/blob/master/LICENSE.txt
Installing gerke-decoder requires of the order 50 MB of disk space.
To build executables, type
make
The build procedure will install a dedicated copy of Maven in the local directory. This installation will not interfere with possible other use of Maven on the same host. For licensing info, read here: http://maven.apache.org/#
NOTE: It is recommended that the working directory is on a path that doesn't have embedded spaces. Likewise the Java installation should be on a space-free path. This may be a problem when using Oracle JDK on Windows, which typically is installed under C:\Program Files\ - when using Cygwin this can be overcome by setting up a symlink to the C:\Program Files\Java\ directory.
Building an executable jar (optional) can be done with
make gerke-decoder.jar
Running a quick test can be done with
make test
Removing downloaded and built objects can be done with
make clean
If the source tree was downloaded using a "git clone" command, upgrading to a more recent version can be done with
git pull --rebase
make clean && make
Version 2.1.1 is not the latest; it is maintained on a branch. To build it, do
git checkout 2.1.1-m1
make
Invoke the program as follows:
bin/gerke-decoder -h for built-in help
bin/gerke-decoder WAV_FILE to decode a .wav file
bin/gerke-decoder MP3_FILE to decode a .mp3 file
The executable jar is similarly invoked:
java -jar gerke-decoder.jar -h for built-in help
java -jar gerke-decoder.jar WAV_FILE to decode a .wav file
Note: The executable jar requires a .wav input file.
The relation between WPM (words per minute) and TU (length of a dot in ms) is taken to be
TU = 1200/WPM
as described here: https://en.wikipedia.org/wiki/Morse_code#Timing (the "PARIS" formula).
The decoder expects 3 TU of inter-character silence and 7 TU of inter-word silence. Deviations may cause garbled characters or false word breaks.
Seven different decoding methods are provided, selectable with the -D option:
1: Tone/silence: Dots and dashes are recognized as the signal amplitude raises above and falls below a threshold level. The level is adjustable with the -u option.
2: Patterns: The signal is multiplied with a rectangular wave function representing a character. The product is integrated over the extent of the character and the best matching character is chosen. The beginning and end of a character is still based on tone/silence crossings.
3: Dips: The extent of a character is determined from level crossings, as in method 1. Within the character a search is made for dips, by matching against a negative gaussian function with a width of about 1 TU. Dashes and dots are then identified from the positions of the dips.
4: Prototype segments: Dashes are recognized by least-squares fitting a straight line to dash-length time intervals. Dots are then recognized by trying dot-length time intervals that don't clash with the dashes. This decoding method is of a prototype nature.
5: Sliding segments: Line segments are fitted to the signal data, thereby locating dashes and dots.
6: Adaptive segments: Similar to the no. 5 method, plus it adapts to frequency drift and fading.
7: Integrating: Sums over short segments of the signal are examined, locating probable dots and dashes. This method does not require detection of tone/silence crossings, which tends to be very sensitive to noise. This is the default decoding method.
Add the -v option to have some diagnostics printed. To get even more diagnostics use -v -v or -v -v -v.
Diagnostics are printed to the standard error stream.
By default a search for tone signals in the range 400..1200 Hz is made. The decoder will choose the best frequency in that range. A non-default search range may be set with the -F option, e.g.:
-F 16900,17500
Add the -S option to get a plot of average signal amplitude versus frequency. There should be a maximum somewhere in the frequency range. If no maximum is seen, try a different search range.
Once the tone frequency has been determined it can be passed to the decoder with the -f option, e.g.:
-f 690
If the frequency is given with the -f option, the somewhat time-consuming frequency search is skipped.
The decoder assumes by default that the WPM speed is 15, which implies a dot length of 1200/15 = 80 ms. If the WPM speed of the .wav file is higher or lower than 15, the -w option must be used. For example, if the speed is believed to be 22 WPM, then use
-w 22
A too high value of the -w option will cause the decoder to interpret many tones as dashes. A too low value will cause tones to be interpreted as dots. When a reasonable setting has been found the decoder will distinguish dashes and dots properly.
The effective WPM, as calculated from the timing of dots and dashes in decoded characters, will be reported when the -v option is used. Re-running with the -w option set to this value may result in somewhat improved decoding.
(This option is only effective with the "dips" decoder.)
If silent spaces between characters and words are exaggerated, this can be compensated for by specifying e.g:
-W 1.2
(This option is ignored by the "adaptive line segments" decoder)
By default the decoder will apply a small degree of clipping, with intention to reduce the impact of spiky noise. With the -v option the default level of clipping is reported. More agressive clipping can be specified with the -c option, by giving a smaller value than the reported default. For example, if the default level of clipping is reported as 6655, then try e.g.:
-c 3000
The clipping level is applied to the recording as a whole. In the presence of fading it may thus be that clipping is not effective in faded parts of the recording. To process different parts of the recording differently the -o and -l options can be used.
The signal is estimated at points that are of the order 0.1 TU apart. This time slice period is decoder dependent. To decrease the time slice period to 80% of the default value, use
-q 0.8
To further reduce signal fluctuations a Gaussian average of samples is taken. A sigma value of about 0.18 TU is used by default. To specify a non-default sigma value, use
-s 0.35
The threshold between tone and silence is determined automatically. The threshold level will follow the signal level, so modest fading will not be an issue.
In noisy conditions some improvement may be obtained by setting the threshold slightly lower or higher. The -u option can be used for this; the default value is 1.0, so to set the threshold slightly higher try e.g.:
-u 1.05
When experimenting with clipping it may be necessary to also adjust the threshold.
The signal as a function of time can be studied grapically. Request plotting with the -A option:
-A
To get a good view it may be necessary to select a sufficiently short time interval. For WPM around 15 a 10 s interval may be convenient. To specify an interval starting at 55 s into the recording and lasting for 10 s, use
-Z 55,10
When studying a decoding case that includes some garbled words it may be interesting to plot the signal at that point. Add the -t option to get timestamps inserted in the decoded text. This gives guidance on how to set the -Z option.
The relative phase angle of the signal may be plotted with the -P option. If the decoder frequency is set correctly, the phase angle should wander only slowly.
Signal content versus frequency can be plotted with the -S option. This option cannot be used in combination with the -f option, and not in combination with the "adaptive segments" decoder.
The signal vs. frequency graph can be made more peaked by increasing the -q option value (try -h to see what the default is). Note however that this may cause decoding to be impaired, so do this only for finding the frequency.
This feature is enabled with option -Y, and is available with decoding methods 4, 5 and 6. The signal frequency is plotted as a function of time. A time interval can be specified with the -Z option.
This feature is enabled with option -M:
-M 1 tone lengths
-M 0 space lengths
-M 1,0 tone lengths and space lengths
The -o option specifies an offset, in seconds, into the .wav file. The -l option specifies a length in seconds. When processing a large recording these options can be used to limit the amount of data to be held in RAM. For example, assuming that a segment starting at 4200 s and extending for 180 s is to be decoded, then specify
-o 4200 -l 180
All options, except for -l, refer to the same time axis, starting at 0 seconds at the beginning of the .wav file. To plot a 10 s interval within the 180 s segment, add e.g.
-o 4200 -l 180 -A -Z 4300,10
The -t option causes a timestamp in seconds to be inserted after every decoded word.
The -T option controls casification and line length:
-T U upper case
-T L lower case (the default)
-T C capitalized
The default line length is 72; a non-default value may be specified:
-T U,40
A number of experimental parameters can be set with the -H option. To see the default values, use -h. Values must be given as a comma-separated space-free string, with no values omitted. The parameters are:
DIP Threshold for dips removal
SPIKE Threshold for spikes removal
BREAK_LONG_DASH Break very long dashes in two: 0=disabled, 1=enabled
FILTER Low-pass filter
CUTOFF Low-pass filter cutoff frequency relative to 1/TU
ORDER Filter order
PHASELOCKED Phase locking
PLWIDTH Phase locking width
DIP_MERGE_LIM Dips merge limit
DIP_STRENGTH_MIN Dips strength minimum
CLIP_DEPTH Clipping depth (only with "adaptive segments" decoder)
HALF_WIDTH Sliding line half-width
SPIKE_WIDTH_MAX Maximum width of a spike (unit is TU)
CRACK_WIDTH_MAX Maximum width of a crack (unit is TU)
Valid values for FILTER are:
b Butterworth
cI Chebyshev type I
w Sliding window (ignoring ORDER)
n No filter (ignoring CUTOFF and ORDER)
The -V option causes the program version to be displayed.
The -h options causes a brief summary of the options to be displayed.
The following audio file is a partial recording of the early 2019-06-30 SAQ transmission:
http://privat.bahnhof.se/wb748077/alexanderson-day/grimeton-clip.wav
This file can be decoded with default settings.
The spectrum for VLF radio transmissions overlaps at the lower end with the frequency range of computer soundcards. With an antenna connected to the soundcard input port it is possible to record the signal directly to a .wav file, using a sound recorder program. This program, with a suitably chosen -F option, can then be used for decoding the transmission.
The decoder is named in memory of an early contributor to telecommunications: https://en.wikipedia.org/wiki/Friedrich_Clemens_Gerke