aboutsummaryrefslogtreecommitdiffstats
path: root/net/dccp/ackvec.c
diff options
context:
space:
mode:
Diffstat (limited to 'net/dccp/ackvec.c')
-rw-r--r--net/dccp/ackvec.c31
1 files changed, 30 insertions, 1 deletions
diff --git a/net/dccp/ackvec.c b/net/dccp/ackvec.c
index af976fca407a..abaf241c7353 100644
--- a/net/dccp/ackvec.c
+++ b/net/dccp/ackvec.c
@@ -29,7 +29,7 @@ struct dccp_ackvec *dccp_ackvec_alloc(const gfp_t priority)
29 struct dccp_ackvec *av = kmem_cache_zalloc(dccp_ackvec_slab, priority); 29 struct dccp_ackvec *av = kmem_cache_zalloc(dccp_ackvec_slab, priority);
30 30
31 if (av != NULL) { 31 if (av != NULL) {
32 av->av_buf_head = DCCPAV_MAX_ACKVEC_LEN - 1; 32 av->av_buf_head = av->av_buf_tail = DCCPAV_MAX_ACKVEC_LEN - 1;
33 INIT_LIST_HEAD(&av->av_records); 33 INIT_LIST_HEAD(&av->av_records);
34 } 34 }
35 return av; 35 return av;
@@ -72,6 +72,14 @@ int dccp_ackvec_update_records(struct dccp_ackvec *av, u64 seqno, u8 nonce_sum)
72 avr->avr_ack_nonce = nonce_sum; 72 avr->avr_ack_nonce = nonce_sum;
73 avr->avr_ack_runlen = dccp_ackvec_runlen(av->av_buf + av->av_buf_head); 73 avr->avr_ack_runlen = dccp_ackvec_runlen(av->av_buf + av->av_buf_head);
74 /* 74 /*
75 * When the buffer overflows, we keep no more than one record. This is
76 * the simplest way of disambiguating sender-Acks dating from before the
77 * overflow from sender-Acks which refer to after the overflow; a simple
78 * solution is preferable here since we are handling an exception.
79 */
80 if (av->av_overflow)
81 dccp_ackvec_purge_records(av);
82 /*
75 * Since GSS is incremented for each packet, the list is automatically 83 * Since GSS is incremented for each packet, the list is automatically
76 * arranged in descending order of @ack_seqno. 84 * arranged in descending order of @ack_seqno.
77 */ 85 */
@@ -85,6 +93,27 @@ int dccp_ackvec_update_records(struct dccp_ackvec *av, u64 seqno, u8 nonce_sum)
85} 93}
86 94
87/* 95/*
96 * Buffer index and length computation using modulo-buffersize arithmetic.
97 * Note that, as pointers move from right to left, head is `before' tail.
98 */
99static inline u16 __ackvec_idx_add(const u16 a, const u16 b)
100{
101 return (a + b) % DCCPAV_MAX_ACKVEC_LEN;
102}
103
104static inline u16 __ackvec_idx_sub(const u16 a, const u16 b)
105{
106 return __ackvec_idx_add(a, DCCPAV_MAX_ACKVEC_LEN - b);
107}
108
109u16 dccp_ackvec_buflen(const struct dccp_ackvec *av)
110{
111 if (unlikely(av->av_overflow))
112 return DCCPAV_MAX_ACKVEC_LEN;
113 return __ackvec_idx_sub(av->av_buf_tail, av->av_buf_head);
114}
115
116/*
88 * If several packets are missing, the HC-Receiver may prefer to enter multiple 117 * If several packets are missing, the HC-Receiver may prefer to enter multiple
89 * bytes with run length 0, rather than a single byte with a larger run length; 118 * bytes with run length 0, rather than a single byte with a larger run length;
90 * this simplifies table updates if one of the missing packets arrives. 119 * this simplifies table updates if one of the missing packets arrives.