ShannonConnection.cpp 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. #include "ShannonConnection.h"
  2. #include "Logger.h"
  3. ShannonConnection::ShannonConnection()
  4. {
  5. }
  6. ShannonConnection::~ShannonConnection()
  7. {
  8. }
  9. void ShannonConnection::wrapConnection(std::shared_ptr<PlainConnection> conn, std::vector<uint8_t> &sendKey, std::vector<uint8_t> &recvKey)
  10. {
  11. this->conn = conn;
  12. this->sendCipher = std::make_unique<Shannon>();
  13. this->recvCipher = std::make_unique<Shannon>();
  14. // Set keys
  15. this->sendCipher->key(sendKey);
  16. this->recvCipher->key(recvKey);
  17. // Set initial nonce
  18. this->sendCipher->nonce(pack<uint32_t>(htonl(0)));
  19. this->recvCipher->nonce(pack<uint32_t>(htonl(0)));
  20. }
  21. void ShannonConnection::sendPacket(uint8_t cmd, std::vector<uint8_t> &data)
  22. {
  23. this->writeMutex.lock();
  24. auto rawPacket = this->cipherPacket(cmd, data);
  25. // Shannon encrypt the packet and write it to sock
  26. this->sendCipher->encrypt(rawPacket);
  27. this->conn->writeBlock(rawPacket);
  28. // Generate mac
  29. std::vector<uint8_t> mac(MAC_SIZE);
  30. this->sendCipher->finish(mac);
  31. // Update the nonce
  32. this->sendNonce += 1;
  33. this->sendCipher->nonce(pack<uint32_t>(htonl(this->sendNonce)));
  34. // Write the mac to sock
  35. this->conn->writeBlock(mac);
  36. this->writeMutex.unlock();
  37. }
  38. std::unique_ptr<Packet> ShannonConnection::recvPacket()
  39. {
  40. this->readMutex.lock();
  41. // Receive 3 bytes, cmd + int16 size
  42. auto data = this->conn->readBlock(3);
  43. this->recvCipher->decrypt(data);
  44. auto packetData = std::vector<uint8_t>();
  45. auto readSize = ntohs(extract<uint16_t>(data, 1));
  46. // Read and decode if the packet has an actual body
  47. if (readSize > 0)
  48. {
  49. packetData = this->conn->readBlock(readSize);
  50. this->recvCipher->decrypt(packetData);
  51. }
  52. // Read mac
  53. auto mac = this->conn->readBlock(MAC_SIZE);
  54. // Generate mac
  55. std::vector<uint8_t> mac2(MAC_SIZE);
  56. this->recvCipher->finish(mac2);
  57. if (mac != mac2)
  58. {
  59. CSPOT_LOG(error, "Shannon read: Mac doesn't match");
  60. }
  61. // Update the nonce
  62. this->recvNonce += 1;
  63. this->recvCipher->nonce(pack<uint32_t>(htonl(this->recvNonce)));
  64. // Unlock the mutex
  65. this->readMutex.unlock();
  66. // data[0] == cmd
  67. return std::make_unique<Packet>(data[0], packetData);
  68. }
  69. std::vector<uint8_t> ShannonConnection::cipherPacket(uint8_t cmd, std::vector<uint8_t> &data)
  70. {
  71. // Generate packet structure, [Command] [Size] [Raw data]
  72. auto sizeRaw = pack<uint16_t>(htons(uint16_t(data.size())));
  73. sizeRaw.insert(sizeRaw.begin(), cmd);
  74. sizeRaw.insert(sizeRaw.end(), data.begin(), data.end());
  75. return sizeRaw;
  76. }