VARA HF's Huffman Encoding algorithm, updated to C# and VB.NET.
- Document the Huffman Encoding (compressed) format of messages (or packets) sent via the VARA protocol, which is used with WinLink on amateur radio.
- Create a reference implementation which runs on modern operating systems and compiles on modern compilers.
- VARA does not stand for anything.
- It is proprietary software developed by EA5HVK.
- VARA HF also refers to the soundcard modem modulation scheme. This document and project only cover the data which is sent by it. It does not cover the modulation (such as FSK, BPSK, 4-8PSK, or 16-32QAM at various symbol rates and bandwidths).
- WinLink might disable VARA's compression and use its own FBB B1 compression instead. (Not sure / untested)
- VARA's high speed protocol might be different. (Not sure / untested)
- Documentation of VARA's Huffman encoding scheme first appeared on EA5HVK's website in Feb 2021. It appears on the site with the link text "VARA Huffman compression" pointing to source code written in an unknown version of VisualBasic inside a pdf of a word document on filesharing site mega.nz. (Link to first apperance on the Wayback Machine)
There are two message formats, which I'm calling "Plain Bytes" and "Huffman Encoded".
-
Plain bytes messages begins with a four byte header:
48 45 30 0D
"HE0\r" followed by the message text or arbitrary bytes.-
There's no compression with the format
-
Empty messages are made up of just the four header bytes and nothing else.
-
Unlike Huffman encoded messages, there is no parity byte, nor message length in the header.
-
If you're curious, here's why last character of the header is
\r
, also known a carriage return (CR). It appears to be a neat little hack. The\r
was used by MacOS 9 as a line separator, but Winlink uses it as half of the\r\n
line separator found on Windows and DOS. In a DOS terminal, the\r
moves the text cursor back to the start of the line, while the\n
starts a new line. By leaving off the new line, if the whole packet is printed directly to a Windows terminal, the\r
returns the cursor to the start of the header, so the message following it will overwrite and hide the "HE0" of the header. The user will then only see the message and not the header. I don't know if this hack was ever useful in practice. -
Plain Bytes mode is used when Huffman compression does not decrease the message size, or that's the intention. VARA's provided source code appears to be suboptimial in this regard. When considering the size of the compressed message, it doesn't consider the size of the Huffman tables, so it underestimates the actual size of the Huffman compressed message. This means for smaller messages it will sometimes choose to compress the message even when this adds more bytes. (Is this the case with the release version too? Can we double check this behavior?)
-
-
Bytes 0 to 4: begin with the 4 byte header
48 45 33 0D
"HE3\r"- Presumably stands for "Huffman Encoding 3". (Anyone know which version of which VARA products this was introduced or is compatible with?)
- Only byte 2 differs from Plain Bytes message header (ascii 3 vs 0)
- Messages which don't contain either header ("HE0\r" nor "HE3\r") are rejected
-
Byte 5: Parity byte
- It's calculated by XORing all bytes of original message (uncompressed without the header); Wikipedia's page on longitudinal parity check points out that "with this checksum, any transmission error which flips a single bit of the message, or an odd number of bits, will be detected as an incorrect checksum."
- Note that the message must be fully decoded before parity is checked.
- It is variously referred in the source code as a CRC or Checksum. Note that it is not CRC-8.
-
Bytes 6,7,8,9: Integer with length of decoded message in bytes (without header). Little endian like the rest of integers, I guess.
-
Bytes 10 and 11: SymbolCount: The number of unique (distinct) symbols (8-bit characters) found in the original message
- Same as the number of entries in the Huffman Table 1 (or double the number for the table's size in bytes)
- Which is also the number of leaf nodes in the Huffman tree
- Typical values are 1 to 256 (hex:
01 00
to00 01
). Zero-symbol or empty messages are typically sent as an empty Plain Bytes message and not Huffman encoded, so this field does not have a 0 value in normal operation, unless you use theForceHuffman
option. - For example, a message with only the text
VARA
will give a value of 3.03 00
because it contains three symbols:A
,R
, andV
. - Bytes 10 and 11 together make up a little-endian 16-bit integer. Unless the message contains all 256 characters, byte 11 has a value of 0. (TODO: confirm this)
- All symbols are 8-bit bytes (e.g. typically ascii characters). There's no support for other sized symbols or sequences.
-
Huffman Table 1 (starting byte 12)
- 2-byte structure repeated SymbolCount times.
- 1 byte: an 8-bit symbol (character) found in the message
- 1 byte: integer length of its prefix code in number of bits
- Example: "A\2"
65 02
(the letter A will be represented with a bit sequence two bits long) - The prefix code (bit sequence) for the letter is stored at the end of the message in Huffman Table 2
- Ordered by ascii value (0 to 255) in the original source code but any order would probably work. (If you were to make a robust VARA decoder, you'd want to test other orders in your unit tests)
- see "Huffman Table 2" below, which comes after the encoded message for the rest of the data needed for Huffman decoding.
-
Encoded Message Body
- A message body with a length of any number of bytes, made up of variable length Huffman prefix codes.
- Each Huffman prefix code represents one 8-bit symbol (such as an ascii character)
- It's zero padded: If the encoded message ends with a partial byte, the remaining bits are set to 0
-
Huffman Table 2: List of prefix codes
- Each prefix code listed one after another.
- It's zero padded: If Table 2 ends with a partial byte, the remaining bits are set to 0
- A prefix code has a length 1 to 255 (theoretically at least?)
- A prefix code may also be called a bit sequence, Huffman code, prefix-free binary code or codeword
- Huffman Table 1 (above) gives the length of each prefix code, and what symbol (e.g. ascii character) it represeents
-
Notes:
- Partial decoding or streaming of Huffman Encoded messages cannot be supported: The entire Huffman encoded message must be received before any of it can be practically decoded (because Huffman Table 2, needed to decode the message body, comes at the end, after the encoded message body)
“But the plans were on display…”
“On display? I eventually had to go down to the cellar to find them.”
“That’s the display department.”
“With a flashlight.”
“Ah, well, the lights had probably gone.”
“So had the stairs.”
“But look, you found the notice, didn’t you?”
“Yes,” said Arthur, “yes I did. It was on display in the bottom of a locked filing cabinet stuck in a disused lavatory with a sign on the door saying ‘Beware of the Leopard.”
― Douglas Adams, The Hitchhiker’s Guide to the Galaxy
- The Federal Communications Commission (FCC) encourages the publication of information about novel amateur radio protocols.
- The only information about VARA's protocols from the author is virtually commentless source code.
- It is not published directly on the author's website, nor on any code repository website like github, gitlab or sourceforge, but published as a link labeled "VARA Huffman compression" which goes to file hosting service mega.nz, a site notorious for having paywalls and giving hour-long delays to free-tier users to download files.
- The source code is not a plain text file but instead inside a PDF.
- The code is invalid when copied out of the PDF, requiring manual reformatting.
- There are no instructions on compiling or running the code, how it's intended to be used, or even what language it is in.
- It appears to be a form of Classic VisualBasic (circa 1991 to 1998) such as VB6, though there's no documentation on which version of Basic or VisualBasic it's best suited for.
- The code has many non-trivial incompatibilities with modern VisualBasic (VB.Net), for example it evokes memory copying commands directly from kernel32.dll, which could have possibly been the style back in 1998, but is unnecessary and broken today.
- Variable names are sometimes inconsistant or misleading, though probably not maliciously so; the code doesn't otherwise appear deliberately obfuscated.
- Unlike modern VB.NET, the IDE, compiler and runtime environments for running Classic VB scripts are not free or open source.
- The VB Classic code does not run easily on modern systems.
- There is no documentation on which versions of VARA HF the code is compatible with, or if it works with the current version of VARA.
- Verify VaraHuffmanNet gives identical output to the original VB6 code:
- encoding of binary data
- todo: legacy mode to emulate vb6 heuristics of when to choose HE0 over HE3, and auto mode to prefer smallest size
- input size limitations,
- one unique symbol message encoding/decoding, and 256 unique symbol encoding/decoding)
- What's the preferred/default character encoding used by the original? Does it vary with Windows configuration? (though it's capable of sending any arbitrary byte sequences anyway). Appears to use Windows' locale ANSI mode if encoding from a string.
- Memory copying issues?
- Does it support reading 0-length Huffman Encoding messages? (It checks for them but not sure what original code returns. It never generates them). The original Huffman encoder will avoid encoding an empty message, but it looks like:
48 45 33 0D 00 00 00 00 00 00 00
- To compare with current Vara software:
- Which modes use Huffman Encoding and when?
- Which bugs still occur? (e.g. has heuristics for choosing between HE0 and HE3 improved?)
- when are text or binary modes assumed?
- Compare efficency of gz or bzip2 (in a small number of tests they appear much better for anything of a non-trivial length, such as short or long email messages including email headers)
- What compression is used by other Winlink related protocols and clients? (ardop, pat)
- What error correction is done in practice? (other than the single parity byte) Is it done on another layer? (There's some kind of ECC in the related ROS project, used for Weak Signal Radio Chat)
- Was the provided source code actually used in production? How long ago? Have there been changes and bugfixes? Has efficiency been improved? The inclusion of progression events throughout the code (to give updates to the client about ongoing processing progress) implies it was actually used in production.
- Does the decoder still work if order of the Huffman Tables are rearranged?
- Does WinLink software directly interface with the Huffman encoder/decoder at all?
- Emulating VARA HF modulation (if that's needed or of interest)
- This project may contain publicly-available propritory code that is otherwise difficult to access or use. The intention is to eventually replace any propritory code that remains in this project, or to inspire someone to make an open source reference implementation, e.g. from the reverse engineered spec.
- Documentation (this readme) is dual licensed CC-BY (any) and MIT (take your pick).
- VARA HF on Signal Identification Wiki (sigidwik)
- VARA modes: List of VARA HF 4.0 modem modes VARA FM v3.0.5 modes (via ardop.groups.io)
- VARA Protocol Native TNC Commands (via Pat-Vara project)
- VARA news on Winlink
- Winlink "Open B2F" (Winlink Message Structure and B2 Forwarding Protocol):
- Documentation: FBB B1 compression, using LZH or LZHUF compression algorithm
- VB.Net "actual source code used in Winlink programs"
- FBB for Pat (GO source)