summaryrefslogtreecommitdiff
path: root/accel-pppd/ctrl/l2tp/l2tp.c
diff options
context:
space:
mode:
authorGuillaume Nault <g.nault@alphalink.fr>2013-02-13 18:27:45 +0100
committerKozlov Dmitry <xeb@mail.ru>2013-02-13 21:28:40 +0400
commitbfc338f17ebbadf377d59c51895f46f02a926fc8 (patch)
tree38b31ffa8d438edb47ad95ef8be3b6e52e3e3d02 /accel-pppd/ctrl/l2tp/l2tp.c
parente9a5661c750da593186f45fe2ffe6324297c4da1 (diff)
downloadaccel-ppp-xebd-bfc338f17ebbadf377d59c51895f46f02a926fc8.tar.gz
accel-ppp-xebd-bfc338f17ebbadf377d59c51895f46f02a926fc8.zip
l2tp: Flush ack-ed messages from retransmission queue
When several messages are sent consecutively, the peer can reply with a single ZLB to acknowledge all messages. In this case, removing only the first message from the retransmission queue isn't enough (acknowledged messages left in the queue will be retransmitted). This case can happen when replying to an outgoing call request, wich is handled by sending two messages in a row (an OCRP and an OCCN). This patch avoids useless retransmissions by removing all messages acknowledged by peer from the retransmission queue. Signed-off-by: Guillaume Nault <g.nault@alphalink.fr>
Diffstat (limited to 'accel-pppd/ctrl/l2tp/l2tp.c')
-rw-r--r--accel-pppd/ctrl/l2tp/l2tp.c10
1 files changed, 8 insertions, 2 deletions
diff --git a/accel-pppd/ctrl/l2tp/l2tp.c b/accel-pppd/ctrl/l2tp/l2tp.c
index d1f0f3c..69e5d14 100644
--- a/accel-pppd/ctrl/l2tp/l2tp.c
+++ b/accel-pppd/ctrl/l2tp/l2tp.c
@@ -2463,8 +2463,14 @@ static int l2tp_conn_read(struct triton_md_handler_t *h)
} else {
if (!list_empty(&pack->attrs))
conn->Nr++;
- if (!list_empty(&conn->send_queue)) {
- p = list_entry(conn->send_queue.next, typeof(*pack), entry);
+ while (!list_empty(&conn->send_queue)) {
+ /* Flush retransmission queue up
+ to the last acknowledged message */
+ p = list_entry(conn->send_queue.next,
+ typeof(*pack), entry);
+ if (nsnr_cmp(ntohs(p->hdr.Ns),
+ ntohs(pack->hdr.Nr)) >= 0)
+ break;
list_del(&p->entry);
l2tp_packet_free(p);
conn->retransmit = 0;