# MeshCore KISS Modem Protocol Serial protocol for the KISS modem firmware. Enables sending/receiving MeshCore packets over LoRa and cryptographic operations using the modem's identity. ## Serial Configuration 115200 baud, 8N1, no flow control. ## Frame Format Standard KISS framing with byte stuffing. | Byte | Name | Description | |------|------|-------------| | `0xC0` | FEND | Frame delimiter | | `0xDB` | FESC | Escape character | | `0xDC` | TFEND | Escaped FEND (FESC + TFEND = 0xC0) | | `0xDD` | TFESC | Escaped FESC (FESC + TFESC = 0xDB) | ``` ┌──────┬─────────┬──────────────┬──────┐ │ FEND │ Command │ Data (escaped)│ FEND │ │ 0xC0 │ 1 byte │ 0-510 bytes │ 0xC0 │ └──────┴─────────┴──────────────┴──────┘ ``` Maximum unescaped frame size: 512 bytes. ## Commands ### Request Commands (Host → Modem) | Command | Value | Data | |---------|-------|------| | `CMD_DATA` | `0x00` | Packet (2-255 bytes) | | `CMD_GET_IDENTITY` | `0x01` | - | | `CMD_GET_RANDOM` | `0x02` | Length (1 byte, 1-64) | | `CMD_VERIFY_SIGNATURE` | `0x03` | PubKey (32) + Signature (64) + Data | | `CMD_SIGN_DATA` | `0x04` | Data to sign | | `CMD_ENCRYPT_DATA` | `0x05` | Key (32) + Plaintext | | `CMD_DECRYPT_DATA` | `0x06` | Key (32) + MAC (2) + Ciphertext | | `CMD_KEY_EXCHANGE` | `0x07` | Remote PubKey (32) | | `CMD_HASH` | `0x08` | Data to hash | | `CMD_SET_RADIO` | `0x09` | Freq (4) + BW (4) + SF (1) + CR (1) | | `CMD_SET_TX_POWER` | `0x0A` | Power dBm (1) | | *reserved* | `0x0B` | *(not implemented)* | | `CMD_GET_RADIO` | `0x0C` | - | | `CMD_GET_TX_POWER` | `0x0D` | - | | *reserved* | `0x0E` | *(not implemented)* | | `CMD_GET_VERSION` | `0x0F` | - | | `CMD_GET_CURRENT_RSSI` | `0x10` | - | | `CMD_IS_CHANNEL_BUSY` | `0x11` | - | | `CMD_GET_AIRTIME` | `0x12` | Packet length (1) | | `CMD_GET_NOISE_FLOOR` | `0x13` | - | | `CMD_GET_STATS` | `0x14` | - | | `CMD_GET_BATTERY` | `0x15` | - | | `CMD_PING` | `0x16` | - | | `CMD_GET_SENSORS` | `0x17` | Permissions (1) | ### Response Commands (Modem → Host) | Command | Value | Data | |---------|-------|------| | `CMD_DATA` | `0x00` | SNR (1) + RSSI (1) + Packet | | `RESP_IDENTITY` | `0x21` | PubKey (32) | | `RESP_RANDOM` | `0x22` | Random bytes (1-64) | | `RESP_VERIFY` | `0x23` | Result (1): 0x00=invalid, 0x01=valid | | `RESP_SIGNATURE` | `0x24` | Signature (64) | | `RESP_ENCRYPTED` | `0x25` | MAC (2) + Ciphertext | | `RESP_DECRYPTED` | `0x26` | Plaintext | | `RESP_SHARED_SECRET` | `0x27` | Shared secret (32) | | `RESP_HASH` | `0x28` | SHA-256 hash (32) | | `RESP_OK` | `0x29` | - | | `RESP_RADIO` | `0x2A` | Freq (4) + BW (4) + SF (1) + CR (1) | | `RESP_TX_POWER` | `0x2B` | Power dBm (1) | | *reserved* | `0x2C` | *(not implemented)* | | `RESP_VERSION` | `0x2D` | Version (1) + Reserved (1) | | `RESP_ERROR` | `0x2E` | Error code (1) | | `RESP_TX_DONE` | `0x2F` | Result (1): 0x00=failed, 0x01=success | | `RESP_CURRENT_RSSI` | `0x30` | RSSI dBm (1, signed) | | `RESP_CHANNEL_BUSY` | `0x31` | Result (1): 0x00=clear, 0x01=busy | | `RESP_AIRTIME` | `0x32` | Milliseconds (4) | | `RESP_NOISE_FLOOR` | `0x33` | dBm (2, signed) | | `RESP_STATS` | `0x34` | RX (4) + TX (4) + Errors (4) | | `RESP_BATTERY` | `0x35` | Millivolts (2) | | `RESP_PONG` | `0x36` | - | | `RESP_SENSORS` | `0x37` | CayenneLPP payload | ## Error Codes | Code | Value | Description | |------|-------|-------------| | `ERR_INVALID_LENGTH` | `0x01` | Request data too short | | `ERR_INVALID_PARAM` | `0x02` | Invalid parameter value | | `ERR_NO_CALLBACK` | `0x03` | Feature not available | | `ERR_MAC_FAILED` | `0x04` | MAC verification failed | | `ERR_UNKNOWN_CMD` | `0x05` | Unknown command | | `ERR_ENCRYPT_FAILED` | `0x06` | Encryption failed | | `ERR_TX_PENDING` | `0x07` | TX already pending | ## Data Formats ### Radio Parameters (CMD_SET_RADIO / RESP_RADIO) All values little-endian. | Field | Size | Description | |-------|------|-------------| | Frequency | 4 bytes | Hz (e.g., 869618000) | | Bandwidth | 4 bytes | Hz (e.g., 62500) | | SF | 1 byte | Spreading factor (5-12) | | CR | 1 byte | Coding rate (5-8) | ### Received Packet (CMD_DATA response) | Field | Size | Description | |-------|------|-------------| | SNR | 1 byte | Signal-to-noise × 4, signed | | RSSI | 1 byte | Signal strength dBm, signed | | Packet | variable | Raw MeshCore packet | ### Noise Floor (RESP_NOISE_FLOOR) Response to `CMD_GET_NOISE_FLOOR` (0x13). Little-endian. | Field | Size | Description | |--------------|------|--------------------------------| | Noise floor | 2 | int16_t, dBm (signed), e.g. -120 | The modem recalibrates the noise floor every two seconds with an AGC reset every 30 seconds. ### Stats (RESP_STATS) Response to `CMD_GET_STATS` (0x14). All values little-endian. | Field | Size | Description | |-------|------|-------------| | RX | 4 bytes | Packets received | | TX | 4 bytes | Packets transmitted | | Errors | 4 bytes | Receive errors | ### Sensor Permissions (CMD_GET_SENSORS) | Bit | Value | Description | |-----|-------|-------------| | 0 | `0x01` | Base (battery) | | 1 | `0x02` | Location (GPS) | | 2 | `0x04` | Environment (temp, humidity, pressure) | Use `0x07` for all permissions. ### Sensor Data (RESP_SENSORS) Data returned in CayenneLPP format. See [CayenneLPP documentation](https://docs.mydevices.com/docs/lorawan/cayenne-lpp) for parsing. ## Notes - Modem generates identity on first boot (stored in flash) - SNR values multiplied by 4 for 0.25 dB precision - Wait for `RESP_TX_DONE` before sending next packet - Sending `CMD_DATA` while TX is pending returns `ERR_TX_PENDING` - See [packet_structure.md](./packet_structure.md) for packet format