Skip to content

Commit 6ea0fed

Browse files
eraydcaveman99
authored andcommitted
If a packet is heard multiple times, rebroadcast using the highest hop limit
Sometimes a packet will be in the TX queue waiting to be transmitted, when it is overheard being rebroadcast by another node, with a higher hop limit remaining. When this occurs, modify the pending packet in the TX queue to avoid unnecessarily wasting hops.
1 parent 7dd3629 commit 6ea0fed

6 files changed

+40
-0
lines changed

src/mesh/FloodingRouter.cpp

+4
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,10 @@ bool FloodingRouter::shouldFilterReceived(const meshtastic_MeshPacket *p)
2828
// cancel rebroadcast of this message *if* there was already one, unless we're a router/repeater!
2929
if (Router::cancelSending(p->from, p->id))
3030
txRelayCanceled++;
31+
} else if (iface && p->hop_limit > 0) {
32+
// If we overhear a duplicate copy of the packet with more hops left than the one we are waiting to
33+
// rebroadcast, then update the hop limit of the packet currently sitting in the TX queue.
34+
iface->clampHopsToMax(getFrom(p), p->id, p->hop_limit - 1);
3135
}
3236

3337
/* If the original transmitter is doing retransmissions (hopStart equals hopLimit) for a reliable transmission, e.g., when

src/mesh/MeshPacketQueue.cpp

+13
Original file line numberDiff line numberDiff line change
@@ -93,6 +93,19 @@ meshtastic_MeshPacket *MeshPacketQueue::getFront()
9393
return p;
9494
}
9595

96+
/** Attempt to find a packet in this queue. Returns a pointer to the found packet, or NULL if not found. */
97+
meshtastic_MeshPacket *MeshPacketQueue::find(NodeNum from, PacketId id)
98+
{
99+
for (auto it = queue.begin(); it != queue.end(); it++) {
100+
auto p = (*it);
101+
if (getFrom(p) == from && p->id == id) {
102+
return p;
103+
}
104+
}
105+
106+
return NULL;
107+
}
108+
96109
/** Attempt to find and remove a packet from this queue. Returns a pointer to the removed packet, or NULL if not found */
97110
meshtastic_MeshPacket *MeshPacketQueue::remove(NodeNum from, PacketId id)
98111
{

src/mesh/MeshPacketQueue.h

+3
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,9 @@ class MeshPacketQueue
3535

3636
meshtastic_MeshPacket *getFront();
3737

38+
/** Attempt to find a packet in this queue. Returns a pointer to the found packet, or NULL if not found. */
39+
meshtastic_MeshPacket *find(NodeNum from, PacketId id);
40+
3841
/** Attempt to find and remove a packet from this queue. Returns the packet which was removed from the queue */
3942
meshtastic_MeshPacket *remove(NodeNum from, PacketId id);
4043
};

src/mesh/RadioInterface.h

+3
Original file line numberDiff line numberDiff line change
@@ -176,6 +176,9 @@ class RadioInterface
176176
/** The delay to use when we want to flood a message. Use a weighted scale based on SNR */
177177
uint32_t getTxDelayMsecWeighted(float snr);
178178

179+
/** Clamp the hop limit to the greater of the hop count provided, or the hop count in the queue */
180+
virtual void clampHopsToMax(NodeNum from, PacketId id, uint32_t hop_limit) { return; }
181+
179182
/**
180183
* Calculate airtime per
181184
* https://www.rs-online.com/designspark/rel-assets/ds-assets/uploads/knowledge-items/application-notes-for-the-internet-of-things/LoRa%20Design%20Guide.pdf

src/mesh/RadioLibInterface.cpp

+12
Original file line numberDiff line numberDiff line change
@@ -336,6 +336,18 @@ void RadioLibInterface::startTransmitTimerSNR(float snr)
336336
}
337337
}
338338

339+
/**
340+
* Clamp the hop limit to the greater of the hop count provided, or the hop count in the queue
341+
*/
342+
void RadioLibInterface::clampHopsToMax(NodeNum from, PacketId id, uint32_t hop_limit)
343+
{
344+
meshtastic_MeshPacket *p = txQueue.find(from, id);
345+
if (p && p->hop_limit < hop_limit) {
346+
LOG_DEBUG("Increasing hop limit for packet from %d to %d", p->hop_limit, hop_limit);
347+
p->hop_limit = hop_limit;
348+
}
349+
}
350+
339351
void RadioLibInterface::handleTransmitInterrupt()
340352
{
341353
// This can be null if we forced the device to enter standby mode. In that case

src/mesh/RadioLibInterface.h

+5
Original file line numberDiff line numberDiff line change
@@ -200,4 +200,9 @@ class RadioLibInterface : public RadioInterface, protected concurrency::Notified
200200
virtual void setStandby();
201201

202202
const char *radioLibErr = "RadioLib err=";
203+
204+
/**
205+
* Clamp the hop limit to the greater of the hop count provided, or the hop count in the queue
206+
*/
207+
void clampHopsToMax(NodeNum from, PacketId id, uint32_t hop_limit);
203208
};

0 commit comments

Comments
 (0)