Commit f86677226ddd5c69be73ded03b89c43cc86a28a1
1 parent
6141bbcc
QoS message queue is a list, for easy ordering
Showing
2 changed files
with
37 additions
and
15 deletions
session.cpp
| @@ -47,7 +47,10 @@ void Session::writePacket(const MqttPacket &packet, char qos_arg) | @@ -47,7 +47,10 @@ void Session::writePacket(const MqttPacket &packet, char qos_arg) | ||
| 47 | } | 47 | } |
| 48 | const uint16_t pid = nextPacketId++; | 48 | const uint16_t pid = nextPacketId++; |
| 49 | copyPacket->setPacketId(pid); | 49 | copyPacket->setPacketId(pid); |
| 50 | - qosPacketQueue[pid] = copyPacket; | 50 | + QueuedQosPacket p; |
| 51 | + p.packet = copyPacket; | ||
| 52 | + p.id = pid; | ||
| 53 | + qosPacketQueue.push_back(p); | ||
| 51 | qosQueueBytes += copyPacket->getTotalMemoryFootprint(); | 54 | qosQueueBytes += copyPacket->getTotalMemoryFootprint(); |
| 52 | locker.unlock(); | 55 | locker.unlock(); |
| 53 | 56 | ||
| @@ -60,18 +63,31 @@ void Session::writePacket(const MqttPacket &packet, char qos_arg) | @@ -60,18 +63,31 @@ void Session::writePacket(const MqttPacket &packet, char qos_arg) | ||
| 60 | } | 63 | } |
| 61 | } | 64 | } |
| 62 | 65 | ||
| 66 | +// Normatively, this while loop will break on the first element, because all messages are sent out in order and | ||
| 67 | +// should be acked in order. | ||
| 63 | void Session::clearQosMessage(uint16_t packet_id) | 68 | void Session::clearQosMessage(uint16_t packet_id) |
| 64 | { | 69 | { |
| 65 | std::lock_guard<std::mutex> locker(qosQueueMutex); | 70 | std::lock_guard<std::mutex> locker(qosQueueMutex); |
| 66 | - auto it = qosPacketQueue.find(packet_id); | ||
| 67 | - if (it != qosPacketQueue.end()) | 71 | + |
| 72 | + auto it = qosPacketQueue.begin(); | ||
| 73 | + auto end = qosPacketQueue.end(); | ||
| 74 | + while (it != end) | ||
| 68 | { | 75 | { |
| 69 | - std::shared_ptr<MqttPacket> packet = it->second; | ||
| 70 | - qosPacketQueue.erase(it); | ||
| 71 | - qosQueueBytes -= packet->getTotalMemoryFootprint(); | ||
| 72 | - assert(qosQueueBytes >= 0); | ||
| 73 | - if (qosQueueBytes < 0) // Should not happen, but correcting a hypothetical bug is fine for this purpose. | ||
| 74 | - qosQueueBytes = 0; | 76 | + QueuedQosPacket &p = *it; |
| 77 | + if (p.id == packet_id) | ||
| 78 | + { | ||
| 79 | + size_t mem = p.packet->getTotalMemoryFootprint(); | ||
| 80 | + qosQueueBytes -= mem; | ||
| 81 | + assert(qosQueueBytes >= 0); | ||
| 82 | + if (qosQueueBytes < 0) // Should not happen, but correcting a hypothetical bug is fine for this purpose. | ||
| 83 | + qosQueueBytes = 0; | ||
| 84 | + | ||
| 85 | + qosPacketQueue.erase(it); | ||
| 86 | + | ||
| 87 | + break; | ||
| 88 | + } | ||
| 89 | + | ||
| 90 | + it++; | ||
| 75 | } | 91 | } |
| 76 | } | 92 | } |
| 77 | 93 | ||
| @@ -87,10 +103,10 @@ void Session::sendPendingQosMessages() | @@ -87,10 +103,10 @@ void Session::sendPendingQosMessages() | ||
| 87 | { | 103 | { |
| 88 | Client_p c = makeSharedClient(); | 104 | Client_p c = makeSharedClient(); |
| 89 | std::lock_guard<std::mutex> locker(qosQueueMutex); | 105 | std::lock_guard<std::mutex> locker(qosQueueMutex); |
| 90 | - for (auto &qosMessage : qosPacketQueue) // TODO: wrong: the order must be maintained. Combine the fix with that vector idea | 106 | + for (QueuedQosPacket &qosMessage : qosPacketQueue) |
| 91 | { | 107 | { |
| 92 | - c->writeMqttPacketAndBlameThisClient(*qosMessage.second.get()); | ||
| 93 | - qosMessage.second->setDuplicate(); // Any dealings with this packet from here will be a duplicate. | 108 | + c->writeMqttPacketAndBlameThisClient(*qosMessage.packet.get()); |
| 109 | + qosMessage.packet->setDuplicate(); // Any dealings with this packet from here will be a duplicate. | ||
| 94 | } | 110 | } |
| 95 | } | 111 | } |
| 96 | } | 112 | } |
session.h
| @@ -2,21 +2,27 @@ | @@ -2,21 +2,27 @@ | ||
| 2 | #define SESSION_H | 2 | #define SESSION_H |
| 3 | 3 | ||
| 4 | #include <memory> | 4 | #include <memory> |
| 5 | -#include <unordered_map> | 5 | +#include <list> |
| 6 | #include <mutex> | 6 | #include <mutex> |
| 7 | 7 | ||
| 8 | #include "forward_declarations.h" | 8 | #include "forward_declarations.h" |
| 9 | #include "logger.h" | 9 | #include "logger.h" |
| 10 | 10 | ||
| 11 | -// TODO make settings | 11 | +// TODO make settings. But, num of packets can't exceed 65536, because the counter is 16 bit. |
| 12 | #define MAX_QOS_MSG_PENDING_PER_CLIENT 32 | 12 | #define MAX_QOS_MSG_PENDING_PER_CLIENT 32 |
| 13 | #define MAX_QOS_BYTES_PENDING_PER_CLIENT 4096 | 13 | #define MAX_QOS_BYTES_PENDING_PER_CLIENT 4096 |
| 14 | 14 | ||
| 15 | +struct QueuedQosPacket | ||
| 16 | +{ | ||
| 17 | + uint16_t id; | ||
| 18 | + std::shared_ptr<MqttPacket> packet; | ||
| 19 | +}; | ||
| 20 | + | ||
| 15 | class Session | 21 | class Session |
| 16 | { | 22 | { |
| 17 | std::weak_ptr<Client> client; | 23 | std::weak_ptr<Client> client; |
| 18 | std::string client_id; | 24 | std::string client_id; |
| 19 | - std::unordered_map<uint16_t, std::shared_ptr<MqttPacket>> qosPacketQueue; // TODO: because the max queue length should remain low-ish, perhaps a vector is better here. | 25 | + std::list<QueuedQosPacket> qosPacketQueue; // Using list because it's easiest to maintain order [MQTT-4.6.0-6] |
| 20 | std::mutex qosQueueMutex; | 26 | std::mutex qosQueueMutex; |
| 21 | uint16_t nextPacketId = 0; | 27 | uint16_t nextPacketId = 0; |
| 22 | ssize_t qosQueueBytes = 0; | 28 | ssize_t qosQueueBytes = 0; |