return false;
}
+// Send a record (datagram version, accepts all record types, handles encryption and authentication).
+static bool send_record_priv_datagram(sptps_t *s, uint8_t type, const char *data, uint16_t len) {
+ char buffer[len + 23UL];
+
+ // Create header with sequence number, length and record type
+ uint32_t seqno = htonl(s->outseqno++);
+ uint16_t netlen = htons(len);
+
+ memcpy(buffer, &netlen, 2);
+ memcpy(buffer + 2, &seqno, 4);
+ buffer[6] = type;
+
+ // Add plaintext (TODO: avoid unnecessary copy)
+ memcpy(buffer + 7, data, len);
+
+ if(s->outstate) {
+ // If first handshake has finished, encrypt and HMAC
+ cipher_set_counter(&s->outcipher, &seqno, sizeof seqno);
+ if(!cipher_counter_xor(&s->outcipher, buffer + 6, len + 1UL, buffer + 6))
+ return false;
+
+ if(!digest_create(&s->outdigest, buffer, len + 7UL, buffer + 7UL + len))
+ return false;
+
+ return s->send_data(s->handle, buffer + 2, len + 21UL);
+ } else {
+ // Otherwise send as plaintext
+ return s->send_data(s->handle, buffer + 2, len + 5UL);
+ }
+}
// Send a record (private version, accepts all record types, handles encryption and authentication).
static bool send_record_priv(sptps_t *s, uint8_t type, const char *data, uint16_t len) {
+ if(s->datagram)
+ return send_record_priv_datagram(s, type, data, len);
+
char buffer[len + 23UL];
// Create header with sequence number, length and record type
size_t keylen = ECDH_SIZE;
// Make room for our KEX message, which we will keep around since send_sig() needs it.
+ if(s->mykex)
+ abort();
s->mykex = realloc(s->mykex, 1 + 32 + keylen);
if(!s->mykex)
return error(s, errno, strerror(errno));
// Ignore version number for now.
// Make a copy of the KEX message, send_sig() and receive_sig() need it
+ if(s->hiskex)
+ abort();
s->hiskex = realloc(s->hiskex, len);
if(!s->hiskex)
return error(s, errno, strerror(errno));
// If we already sent our secondary public ECDH key, we expect the peer to send his.
if(!receive_sig(s, data, len))
return false;
- // s->state = SPTPS_ACK;
s->state = SPTPS_ACK;
return true;
case SPTPS_ACK:
}
}
+// Receive incoming data, datagram version.
+static bool sptps_receive_data_datagram(sptps_t *s, const char *data, size_t len) {
+ if(len < (s->instate ? 21 : 5))
+ return error(s, EIO, "Received short packet");
+
+ uint32_t seqno;
+ memcpy(&seqno, data, 4);
+ seqno = ntohl(seqno);
+
+ if(!s->instate) {
+ if(seqno != s->inseqno) {
+ fprintf(stderr, "Received invalid packet seqno: %d != %d\n", seqno, s->inseqno);
+ return error(s, EIO, "Invalid packet seqno");
+ }
+
+ s->inseqno = seqno + 1;
+
+ uint8_t type = data[4];
+
+ if(type != SPTPS_HANDSHAKE)
+ return error(s, EIO, "Application record received before handshake finished");
+
+ return receive_handshake(s, data + 5, len - 5);
+ }
+
+ if(seqno < s->inseqno) {
+ fprintf(stderr, "Received late or replayed packet: %d < %d\n", seqno, s->inseqno);
+ return true;
+ }
+
+ if(seqno > s->inseqno)
+ fprintf(stderr, "Missed %d packets\n", seqno - s->inseqno);
+
+ s->inseqno = seqno + 1;
+
+ uint16_t netlen = htons(len - 21);
+
+ char buffer[len + 23];
+
+ memcpy(buffer, &netlen, 2);
+ memcpy(buffer + 2, data, len);
+
+ memcpy(&seqno, buffer + 2, 4);
+
+ // Check HMAC and decrypt.
+ if(!digest_verify(&s->indigest, buffer, len - 14, buffer + len - 14))
+ return error(s, EIO, "Invalid HMAC");
+
+ cipher_set_counter(&s->incipher, &seqno, sizeof seqno);
+ if(!cipher_counter_xor(&s->incipher, buffer + 6, len - 4, buffer + 6))
+ return false;
+
+ // Append a NULL byte for safety.
+ buffer[len - 14] = 0;
+
+ uint8_t type = buffer[6];
+
+ if(type < SPTPS_HANDSHAKE) {
+ if(!s->instate)
+ return error(s, EIO, "Application record received before handshake finished");
+ if(!s->receive_record(s->handle, type, buffer + 7, len - 21))
+ return false;
+ } else {
+ return error(s, EIO, "Invalid record type");
+ }
+
+ return true;
+}
// Receive incoming data. Check if it contains a complete record, if so, handle it.
bool sptps_receive_data(sptps_t *s, const char *data, size_t len) {
+ if(s->datagram)
+ return sptps_receive_data_datagram(s, data, len);
+
while(len) {
// First read the 2 length bytes.
if(s->buflen < 6) {
}
// Start a SPTPS session.
-bool sptps_start(sptps_t *s, void *handle, bool initiator, ecdsa_t mykey, ecdsa_t hiskey, const char *label, size_t labellen, send_data_t send_data, receive_record_t receive_record) {
+bool sptps_start(sptps_t *s, void *handle, bool initiator, bool datagram, ecdsa_t mykey, ecdsa_t hiskey, const char *label, size_t labellen, send_data_t send_data, receive_record_t receive_record) {
// Initialise struct sptps
memset(s, 0, sizeof *s);
s->handle = handle;
s->initiator = initiator;
+ s->datagram = datagram;
s->mykey = mykey;
s->hiskey = hiskey;
if(!s->label)
return error(s, errno, strerror(errno));
- s->inbuf = malloc(7);
- if(!s->inbuf)
- return error(s, errno, strerror(errno));
- s->buflen = 4;
- memset(s->inbuf, 0, 4);
+ if(!datagram) {
+ s->inbuf = malloc(7);
+ if(!s->inbuf)
+ return error(s, errno, strerror(errno));
+ s->buflen = 4;
+ memset(s->inbuf, 0, 4);
+ }
memcpy(s->label, label, labellen);
s->labellen = labellen;