forked from adafruit/RadioHead
-
Notifications
You must be signed in to change notification settings - Fork 0
/
RHEncryptedDriver.cpp
155 lines (140 loc) · 4.81 KB
/
RHEncryptedDriver.cpp
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
// RHEncryptedDriver.cpp
//
// Author: Philippe.Rochat'at'gmail.com
// Contributed to the RadioHead project by the author
// $Id: RHEncryptedDriver.cpp,v 1.3 2018/02/11 23:57:18 mikem Exp $
#include <RHEncryptedDriver.h>
#ifdef RH_ENABLE_ENCRYPTION_MODULE
RHEncryptedDriver::RHEncryptedDriver(RHGenericDriver& driver, BlockCipher& blockcipher)
: _driver(driver),
_blockcipher(blockcipher)
{
_buffer = (uint8_t *)calloc(_driver.maxMessageLength(), sizeof(uint8_t));
}
bool RHEncryptedDriver::recv(uint8_t* buf, uint8_t* len)
{
int h = 0; // Index of output _buffer
bool status = _driver.recv(_buffer, len);
if (status && buf && len)
{
int blockSize = _blockcipher.blockSize(); // Size of blocks used by encryption
int nbBlocks = *len / blockSize; // Number of blocks in that message
if (nbBlocks * blockSize == *len)
{
// Or we have a missmatch ... this is probably not symetrically encrypted
for (int k = 0; k < nbBlocks; k++)
{
// Decrypt each block
_blockcipher.decryptBlock(&buf[h], &_buffer[k*blockSize]); // Decrypt that block into buf
h += blockSize;
#ifdef STRICT_CONTENT_LEN
if (k == 0)
{
// if (buf[0] > *len - 1)
// return false; // Bogus payload length
*len = buf[0]; // First byte contains length
h--; // First block is of length--
memmove(buf, buf+1, blockSize - 1);
}
#endif
}
}
}
return status;
}
bool RHEncryptedDriver::send(const uint8_t* data, uint8_t len)
{
if (len > maxMessageLength())
return false;
bool status = true;
int blockSize = _blockcipher.blockSize(); // Size of blocks used by encryption
if (len == 0) // PassThru
return _driver.send(data, len);
if (_cipheringBlocks.blockSize != blockSize)
{
// Cipher has changed it's block size
_cipheringBlocks.inputBlock = (uint8_t *)realloc(_cipheringBlocks.inputBlock, blockSize);
_cipheringBlocks.blockSize = blockSize;
}
int max_message_length = maxMessageLength();
#ifdef STRICT_CONTENT_LEN
uint8_t nbBlocks = len / blockSize + 1; // How many blocks do we need for that message
uint8_t nbBpM = max_message_length + 1 / blockSize; // Max number of blocks per message
#else
uint8_t nbBlocks = (len - 1) / blockSize + 1; // How many blocks do we need for that message
uint8_t nbBpM = max_message_length / blockSize; // Max number of blocks per message
#endif
int k = 0, j = 0; // k is block index, j is original message index
#ifndef ALLOW_MULTIPLE_MSG
#ifdef STRICT_CONTENT_LEN
for (k = 0; k < nbBpM && k * blockSize < len + 1; k++)
#else
for (k = 0; k < nbBpM && k * blockSize < len ; k++)
#endif
{
// k blocks in that message
int h = 0; // h is block content index
#ifdef STRICT_CONTENT_LEN
if (k == 0)
_cipheringBlocks.inputBlock[h++] = len; // put in first byte of first block the message length
#endif
while (h < blockSize)
{
// Copy each msg byte into inputBlock, and trail with 0 if necessary
if (j < len)
_cipheringBlocks.inputBlock[h++] = data[j++];
else
_cipheringBlocks.inputBlock[h++] = 0; // Completing with trailing 0
}
_blockcipher.encryptBlock(&_buffer[k * blockSize], _cipheringBlocks.inputBlock); // Cipher that message into _buffer
}
// Serial.println(max_message_length);
// Serial.println(nbBlocks);
// Serial.println(nbBpM);
// Serial.println(k);
// Serial.println(blockSize);
// printBuffer("single send", _buffer, k * blockSize);
if (!_driver.send(_buffer, k*blockSize)) // We now send that message with it's new length
status = false;
#else
uint8_t nbMsg = (nbBlocks * blockSize) / max_message_length + 1; // How many message do we need
for (int i = 0; i < nbMsg; i++)
{
// i is message index
for (k = 0; k < nbBpM && k * blockSize < len ; k++)
{
// k blocks in that message
int h = 0;
#ifdef STRICT_CONTENT_LEN
if (k == 0 && i == 0)
_cipheringBlocks.inputBlock[h++] = len; // put in first byte of first block of first message the message length
#endif
while (h < blockSize)
{
// Copy each msg byte into inputBlock, and trail with 0 if necessary
if (j < len)
_cipheringBlocks.inputBlock[h++] = data[j++];
else
_cipheringBlocks.inputBlock[h++] = 0;
}
_blockcipher.encryptBlock(&_buffer[k * blockSize], _cipheringBlocks.inputBlock); // Cipher that message into buffer
}
// printBuffer("multiple send", _buffer, k * blockSize);
if (!_driver.send(_buffer, k * blockSize)) // We now send that message with it's new length
status = false;
}
#endif
return status;
}
uint8_t RHEncryptedDriver::maxMessageLength()
{
int driver_len = _driver.maxMessageLength();
#ifndef ALLOW_MULTIPLE_MSG
driver_len = ((int)(driver_len/_blockcipher.blockSize()) ) * _blockcipher.blockSize();
#endif
#ifdef STRICT_CONTENT_LEN
driver_len--;
#endif
return driver_len;
}
#endif