Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
linux
Commits
ed1deb70
Commit
ed1deb70
authored
Nov 11, 2010
by
David S. Miller
Browse files
Options
Browse Files
Download
Plain Diff
Merge branch 'dccp' of
git://eden-feed.erg.abdn.ac.uk/net-next-2.6
parents
72cdd1d9
b3d14bff
Changes
6
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
225 additions
and
236 deletions
+225
-236
net/dccp/ackvec.c
net/dccp/ackvec.c
+83
-168
net/dccp/ackvec.h
net/dccp/ackvec.h
+65
-50
net/dccp/ccids/ccid2.c
net/dccp/ccids/ccid2.c
+5
-8
net/dccp/dccp.h
net/dccp/dccp.h
+7
-4
net/dccp/input.c
net/dccp/input.c
+2
-4
net/dccp/options.c
net/dccp/options.c
+63
-2
No files found.
net/dccp/ackvec.c
View file @
ed1deb70
/*
/*
* net/dccp/ackvec.c
* net/dccp/ackvec.c
*
*
* An implementation of the DCCP protocol
* An implementation of Ack Vectors for the DCCP protocol
* Copyright (c) 2007 University of Aberdeen, Scotland, UK
* Copyright (c) 2005 Arnaldo Carvalho de Melo <acme@ghostprotocols.net>
* Copyright (c) 2005 Arnaldo Carvalho de Melo <acme@ghostprotocols.net>
*
*
* This program is free software; you can redistribute it and/or modify it
* This program is free software; you can redistribute it and/or modify it
...
@@ -23,173 +24,93 @@
...
@@ -23,173 +24,93 @@
static
struct
kmem_cache
*
dccp_ackvec_slab
;
static
struct
kmem_cache
*
dccp_ackvec_slab
;
static
struct
kmem_cache
*
dccp_ackvec_record_slab
;
static
struct
kmem_cache
*
dccp_ackvec_record_slab
;
st
atic
struct
dccp_ackvec_record
*
dccp_ackvec_record_new
(
void
)
st
ruct
dccp_ackvec
*
dccp_ackvec_alloc
(
const
gfp_t
priority
)
{
{
struct
dccp_ackvec_record
*
avr
=
struct
dccp_ackvec
*
av
=
kmem_cache_zalloc
(
dccp_ackvec_slab
,
priority
);
kmem_cache_alloc
(
dccp_ackvec_record_slab
,
GFP_ATOMIC
);
if
(
avr
!=
NULL
)
INIT_LIST_HEAD
(
&
avr
->
avr_node
);
return
avr
;
if
(
av
!=
NULL
)
{
av
->
av_buf_head
=
av
->
av_buf_tail
=
DCCPAV_MAX_ACKVEC_LEN
-
1
;
INIT_LIST_HEAD
(
&
av
->
av_records
);
}
return
av
;
}
}
static
void
dccp_ackvec_
record_delete
(
struct
dccp_ackvec_record
*
avr
)
static
void
dccp_ackvec_
purge_records
(
struct
dccp_ackvec
*
av
)
{
{
if
(
unlikely
(
avr
==
NULL
))
struct
dccp_ackvec_record
*
cur
,
*
next
;
return
;
/* Check if deleting a linked record */
list_for_each_entry_safe
(
cur
,
next
,
&
av
->
av_records
,
avr_node
)
WARN_ON
(
!
list_empty
(
&
avr
->
avr_node
)
);
kmem_cache_free
(
dccp_ackvec_record_slab
,
cur
);
kmem_cache_free
(
dccp_ackvec_record_slab
,
avr
);
INIT_LIST_HEAD
(
&
av
->
av_records
);
}
}
static
void
dccp_ackvec_insert_avr
(
struct
dccp_ackvec
*
av
,
void
dccp_ackvec_free
(
struct
dccp_ackvec
*
av
)
struct
dccp_ackvec_record
*
avr
)
{
{
/*
if
(
likely
(
av
!=
NULL
))
{
* AVRs are sorted by seqno. Since we are sending them in order, we
dccp_ackvec_purge_records
(
av
);
* just add the AVR at the head of the list.
kmem_cache_free
(
dccp_ackvec_slab
,
av
);
* -sorbo.
*/
if
(
!
list_empty
(
&
av
->
av_records
))
{
const
struct
dccp_ackvec_record
*
head
=
list_entry
(
av
->
av_records
.
next
,
struct
dccp_ackvec_record
,
avr_node
);
BUG_ON
(
before48
(
avr
->
avr_ack_seqno
,
head
->
avr_ack_seqno
));
}
}
list_add
(
&
avr
->
avr_node
,
&
av
->
av_records
);
}
}
int
dccp_insert_option_ackvec
(
struct
sock
*
sk
,
struct
sk_buff
*
skb
)
/**
* dccp_ackvec_update_records - Record information about sent Ack Vectors
* @av: Ack Vector records to update
* @seqno: Sequence number of the packet carrying the Ack Vector just sent
* @nonce_sum: The sum of all buffer nonces contained in the Ack Vector
*/
int
dccp_ackvec_update_records
(
struct
dccp_ackvec
*
av
,
u64
seqno
,
u8
nonce_sum
)
{
{
struct
dccp_sock
*
dp
=
dccp_sk
(
sk
);
struct
dccp_ackvec
*
av
=
dp
->
dccps_hc_rx_ackvec
;
/* Figure out how many options do we need to represent the ackvec */
const
u8
nr_opts
=
DIV_ROUND_UP
(
av
->
av_vec_len
,
DCCP_SINGLE_OPT_MAXLEN
);
u16
len
=
av
->
av_vec_len
+
2
*
nr_opts
,
i
;
u32
elapsed_time
;
const
unsigned
char
*
tail
,
*
from
;
unsigned
char
*
to
;
struct
dccp_ackvec_record
*
avr
;
struct
dccp_ackvec_record
*
avr
;
suseconds_t
delta
;
if
(
DCCP_SKB_CB
(
skb
)
->
dccpd_opt_len
+
len
>
DCCP_MAX_OPT_LEN
)
return
-
1
;
delta
=
ktime_us_delta
(
ktime_get_real
(),
av
->
av_time
);
elapsed_time
=
delta
/
10
;
if
(
elapsed_time
!=
0
&&
avr
=
kmem_cache_alloc
(
dccp_ackvec_record_slab
,
GFP_ATOMIC
);
dccp_insert_option_elapsed_time
(
skb
,
elapsed_time
))
return
-
1
;
avr
=
dccp_ackvec_record_new
();
if
(
avr
==
NULL
)
if
(
avr
==
NULL
)
return
-
1
;
return
-
ENOBUFS
;
DCCP_SKB_CB
(
skb
)
->
dccpd_opt_len
+=
len
;
to
=
skb_push
(
skb
,
len
);
len
=
av
->
av_vec_len
;
from
=
av
->
av_buf
+
av
->
av_buf_head
;
tail
=
av
->
av_buf
+
DCCP_MAX_ACKVEC_LEN
;
for
(
i
=
0
;
i
<
nr_opts
;
++
i
)
{
int
copylen
=
len
;
if
(
len
>
DCCP_SINGLE_OPT_MAXLEN
)
copylen
=
DCCP_SINGLE_OPT_MAXLEN
;
*
to
++
=
DCCPO_ACK_VECTOR_0
;
*
to
++
=
copylen
+
2
;
/* Check if buf_head wraps */
if
(
from
+
copylen
>
tail
)
{
const
u16
tailsize
=
tail
-
from
;
memcpy
(
to
,
from
,
tailsize
);
to
+=
tailsize
;
len
-=
tailsize
;
copylen
-=
tailsize
;
from
=
av
->
av_buf
;
}
memcpy
(
to
,
from
,
copylen
);
from
+=
copylen
;
to
+=
copylen
;
len
-=
copylen
;
}
avr
->
avr_ack_seqno
=
seqno
;
avr
->
avr_ack_ptr
=
av
->
av_buf_head
;
avr
->
avr_ack_ackno
=
av
->
av_buf_ackno
;
avr
->
avr_ack_nonce
=
nonce_sum
;
avr
->
avr_ack_runlen
=
dccp_ackvec_runlen
(
av
->
av_buf
+
av
->
av_buf_head
);
/*
/*
* From RFC 4340, A.2:
* When the buffer overflows, we keep no more than one record. This is
*
* the simplest way of disambiguating sender-Acks dating from before the
* For each acknowledgement it sends, the HC-Receiver will add an
* overflow from sender-Acks which refer to after the overflow; a simple
* acknowledgement record. ack_seqno will equal the HC-Receiver
* solution is preferable here since we are handling an exception.
* sequence number it used for the ack packet; ack_ptr will equal
* buf_head; ack_ackno will equal buf_ackno; and ack_nonce will
* equal buf_nonce.
*/
*/
avr
->
avr_ack_seqno
=
DCCP_SKB_CB
(
skb
)
->
dccpd_seq
;
if
(
av
->
av_overflow
)
avr
->
avr_ack_ptr
=
av
->
av_buf_head
;
dccp_ackvec_purge_records
(
av
)
;
avr
->
avr_ack_ackno
=
av
->
av_buf_ackno
;
/*
avr
->
avr_ack_nonce
=
av
->
av_buf_nonce
;
* Since GSS is incremented for each packet, the list is automatically
avr
->
avr_sent_len
=
av
->
av_vec_len
;
* arranged in descending order of @ack_seqno.
*/
dccp_ackvec_insert_avr
(
av
,
avr
);
list_add
(
&
avr
->
avr_node
,
&
av
->
av_records
);
dccp_pr_debug
(
"%s ACK Vector 0, len=%d, ack_seqno=%llu, "
dccp_pr_debug
(
"Added Vector, ack_seqno=%llu, ack_ackno=%llu (rl=%u)
\n
"
,
"ack_ackno=%llu
\n
"
,
dccp_role
(
sk
),
avr
->
avr_sent_len
,
(
unsigned
long
long
)
avr
->
avr_ack_seqno
,
(
unsigned
long
long
)
avr
->
avr_ack_seqno
,
(
unsigned
long
long
)
avr
->
avr_ack_ackno
);
(
unsigned
long
long
)
avr
->
avr_ack_ackno
,
avr
->
avr_ack_runlen
);
return
0
;
return
0
;
}
}
struct
dccp_ackvec
*
dccp_ackvec_alloc
(
const
gfp_t
priority
)
/*
{
* Buffer index and length computation using modulo-buffersize arithmetic.
struct
dccp_ackvec
*
av
=
kmem_cache_alloc
(
dccp_ackvec_slab
,
priority
);
* Note that, as pointers move from right to left, head is `before' tail.
*/
if
(
av
!=
NULL
)
{
static
inline
u16
__ackvec_idx_add
(
const
u16
a
,
const
u16
b
)
av
->
av_buf_head
=
DCCP_MAX_ACKVEC_LEN
-
1
;
av
->
av_buf_ackno
=
UINT48_MAX
+
1
;
av
->
av_buf_nonce
=
0
;
av
->
av_time
=
ktime_set
(
0
,
0
);
av
->
av_vec_len
=
0
;
INIT_LIST_HEAD
(
&
av
->
av_records
);
}
return
av
;
}
void
dccp_ackvec_free
(
struct
dccp_ackvec
*
av
)
{
{
if
(
unlikely
(
av
==
NULL
))
return
(
a
+
b
)
%
DCCPAV_MAX_ACKVEC_LEN
;
return
;
if
(
!
list_empty
(
&
av
->
av_records
))
{
struct
dccp_ackvec_record
*
avr
,
*
next
;
list_for_each_entry_safe
(
avr
,
next
,
&
av
->
av_records
,
avr_node
)
{
list_del_init
(
&
avr
->
avr_node
);
dccp_ackvec_record_delete
(
avr
);
}
}
kmem_cache_free
(
dccp_ackvec_slab
,
av
);
}
}
static
inline
u8
dccp_ackvec_state
(
const
struct
dccp_ackvec
*
av
,
static
inline
u16
__ackvec_idx_sub
(
const
u16
a
,
const
u16
b
)
const
u32
index
)
{
{
return
av
->
av_buf
[
index
]
&
DCCP_ACKVEC_STATE_MASK
;
return
__ackvec_idx_add
(
a
,
DCCPAV_MAX_ACKVEC_LEN
-
b
)
;
}
}
static
inline
u8
dccp_ackvec_len
(
const
struct
dccp_ackvec
*
av
,
u16
dccp_ackvec_buflen
(
const
struct
dccp_ackvec
*
av
)
const
u32
index
)
{
{
return
av
->
av_buf
[
index
]
&
DCCP_ACKVEC_LEN_MASK
;
if
(
unlikely
(
av
->
av_overflow
))
return
DCCPAV_MAX_ACKVEC_LEN
;
return
__ackvec_idx_sub
(
av
->
av_buf_tail
,
av
->
av_buf_head
);
}
}
/*
/*
...
@@ -204,7 +125,7 @@ static inline int dccp_ackvec_set_buf_head_state(struct dccp_ackvec *av,
...
@@ -204,7 +125,7 @@ static inline int dccp_ackvec_set_buf_head_state(struct dccp_ackvec *av,
long
gap
;
long
gap
;
long
new_head
;
long
new_head
;
if
(
av
->
av_vec_len
+
packets
>
DCCP_MAX_ACKVEC_LEN
)
if
(
av
->
av_vec_len
+
packets
>
DCCP
AV
_MAX_ACKVEC_LEN
)
return
-
ENOBUFS
;
return
-
ENOBUFS
;
gap
=
packets
-
1
;
gap
=
packets
-
1
;
...
@@ -212,18 +133,18 @@ static inline int dccp_ackvec_set_buf_head_state(struct dccp_ackvec *av,
...
@@ -212,18 +133,18 @@ static inline int dccp_ackvec_set_buf_head_state(struct dccp_ackvec *av,
if
(
new_head
<
0
)
{
if
(
new_head
<
0
)
{
if
(
gap
>
0
)
{
if
(
gap
>
0
)
{
memset
(
av
->
av_buf
,
DCCP
_ACKVEC_STATE
_NOT_RECEIVED
,
memset
(
av
->
av_buf
,
DCCP
AV
_NOT_RECEIVED
,
gap
+
new_head
+
1
);
gap
+
new_head
+
1
);
gap
=
-
new_head
;
gap
=
-
new_head
;
}
}
new_head
+=
DCCP_MAX_ACKVEC_LEN
;
new_head
+=
DCCP
AV
_MAX_ACKVEC_LEN
;
}
}
av
->
av_buf_head
=
new_head
;
av
->
av_buf_head
=
new_head
;
if
(
gap
>
0
)
if
(
gap
>
0
)
memset
(
av
->
av_buf
+
av
->
av_buf_head
+
1
,
memset
(
av
->
av_buf
+
av
->
av_buf_head
+
1
,
DCCP
_ACKVEC_STATE
_NOT_RECEIVED
,
gap
);
DCCP
AV
_NOT_RECEIVED
,
gap
);
av
->
av_buf
[
av
->
av_buf_head
]
=
state
;
av
->
av_buf
[
av
->
av_buf_head
]
=
state
;
av
->
av_vec_len
+=
packets
;
av
->
av_vec_len
+=
packets
;
...
@@ -236,6 +157,8 @@ static inline int dccp_ackvec_set_buf_head_state(struct dccp_ackvec *av,
...
@@ -236,6 +157,8 @@ static inline int dccp_ackvec_set_buf_head_state(struct dccp_ackvec *av,
int
dccp_ackvec_add
(
struct
dccp_ackvec
*
av
,
const
struct
sock
*
sk
,
int
dccp_ackvec_add
(
struct
dccp_ackvec
*
av
,
const
struct
sock
*
sk
,
const
u64
ackno
,
const
u8
state
)
const
u64
ackno
,
const
u8
state
)
{
{
u8
*
cur_head
=
av
->
av_buf
+
av
->
av_buf_head
,
*
buf_end
=
av
->
av_buf
+
DCCPAV_MAX_ACKVEC_LEN
;
/*
/*
* Check at the right places if the buffer is full, if it is, tell the
* Check at the right places if the buffer is full, if it is, tell the
* caller to start dropping packets till the HC-Sender acks our ACK
* caller to start dropping packets till the HC-Sender acks our ACK
...
@@ -260,7 +183,7 @@ int dccp_ackvec_add(struct dccp_ackvec *av, const struct sock *sk,
...
@@ -260,7 +183,7 @@ int dccp_ackvec_add(struct dccp_ackvec *av, const struct sock *sk,
/* See if this is the first ackno being inserted */
/* See if this is the first ackno being inserted */
if
(
av
->
av_vec_len
==
0
)
{
if
(
av
->
av_vec_len
==
0
)
{
av
->
av_buf
[
av
->
av_buf_head
]
=
state
;
*
cur_head
=
state
;
av
->
av_vec_len
=
1
;
av
->
av_vec_len
=
1
;
}
else
if
(
after48
(
ackno
,
av
->
av_buf_ackno
))
{
}
else
if
(
after48
(
ackno
,
av
->
av_buf_ackno
))
{
const
u64
delta
=
dccp_delta_seqno
(
av
->
av_buf_ackno
,
ackno
);
const
u64
delta
=
dccp_delta_seqno
(
av
->
av_buf_ackno
,
ackno
);
...
@@ -269,10 +192,9 @@ int dccp_ackvec_add(struct dccp_ackvec *av, const struct sock *sk,
...
@@ -269,10 +192,9 @@ int dccp_ackvec_add(struct dccp_ackvec *av, const struct sock *sk,
* Look if the state of this packet is the same as the
* Look if the state of this packet is the same as the
* previous ackno and if so if we can bump the head len.
* previous ackno and if so if we can bump the head len.
*/
*/
if
(
delta
==
1
&&
if
(
delta
==
1
&&
dccp_ackvec_state
(
cur_head
)
==
state
&&
dccp_ackvec_state
(
av
,
av
->
av_buf_head
)
==
state
&&
dccp_ackvec_runlen
(
cur_head
)
<
DCCPAV_MAX_RUNLEN
)
dccp_ackvec_len
(
av
,
av
->
av_buf_head
)
<
DCCP_ACKVEC_LEN_MASK
)
*
cur_head
+=
1
;
av
->
av_buf
[
av
->
av_buf_head
]
++
;
else
if
(
dccp_ackvec_set_buf_head_state
(
av
,
delta
,
state
))
else
if
(
dccp_ackvec_set_buf_head_state
(
av
,
delta
,
state
))
return
-
ENOBUFS
;
return
-
ENOBUFS
;
}
else
{
}
else
{
...
@@ -285,21 +207,17 @@ int dccp_ackvec_add(struct dccp_ackvec *av, const struct sock *sk,
...
@@ -285,21 +207,17 @@ int dccp_ackvec_add(struct dccp_ackvec *av, const struct sock *sk,
* could reduce the complexity of this scan.)
* could reduce the complexity of this scan.)
*/
*/
u64
delta
=
dccp_delta_seqno
(
ackno
,
av
->
av_buf_ackno
);
u64
delta
=
dccp_delta_seqno
(
ackno
,
av
->
av_buf_ackno
);
u32
index
=
av
->
av_buf_head
;
while
(
1
)
{
while
(
1
)
{
const
u8
len
=
dccp_ackvec_len
(
av
,
index
);
const
u8
len
=
dccp_ackvec_runlen
(
cur_head
);
const
u8
av_state
=
dccp_ackvec_state
(
av
,
index
);
/*
/*
* valid packets not yet in av_buf have a reserved
* valid packets not yet in av_buf have a reserved
* entry, with a len equal to 0.
* entry, with a len equal to 0.
*/
*/
if
(
av_state
==
DCCP_ACKVEC_STATE_NOT_RECEIVED
&&
if
(
*
cur_head
==
DCCPAV_NOT_RECEIVED
&&
delta
==
0
)
{
len
==
0
&&
delta
==
0
)
{
/* Found our
reserved seat! */
dccp_pr_debug
(
"Found %llu reserved seat!
\n
"
,
dccp_pr_debug
(
"Found %llu reserved seat!
\n
"
,
(
unsigned
long
long
)
ackno
);
(
unsigned
long
long
)
ackno
);
av
->
av_buf
[
index
]
=
state
;
*
cur_head
=
state
;
goto
out
;
goto
out
;
}
}
/* len == 0 means one packet */
/* len == 0 means one packet */
...
@@ -307,13 +225,12 @@ int dccp_ackvec_add(struct dccp_ackvec *av, const struct sock *sk,
...
@@ -307,13 +225,12 @@ int dccp_ackvec_add(struct dccp_ackvec *av, const struct sock *sk,
goto
out_duplicate
;
goto
out_duplicate
;
delta
-=
len
+
1
;
delta
-=
len
+
1
;
if
(
++
index
==
DCCP_MAX_ACKVEC_LEN
)
if
(
++
cur_head
==
buf_end
)
index
=
0
;
cur_head
=
av
->
av_buf
;
}
}
}
}
av
->
av_buf_ackno
=
ackno
;
av
->
av_buf_ackno
=
ackno
;
av
->
av_time
=
ktime_get_real
();
out:
out:
return
0
;
return
0
;
...
@@ -333,13 +250,13 @@ static void dccp_ackvec_throw_record(struct dccp_ackvec *av,
...
@@ -333,13 +250,13 @@ static void dccp_ackvec_throw_record(struct dccp_ackvec *av,
if
(
av
->
av_buf_head
<=
avr
->
avr_ack_ptr
)
if
(
av
->
av_buf_head
<=
avr
->
avr_ack_ptr
)
av
->
av_vec_len
=
avr
->
avr_ack_ptr
-
av
->
av_buf_head
;
av
->
av_vec_len
=
avr
->
avr_ack_ptr
-
av
->
av_buf_head
;
else
else
av
->
av_vec_len
=
DCCP_MAX_ACKVEC_LEN
-
1
-
av
->
av_vec_len
=
DCCP
AV
_MAX_ACKVEC_LEN
-
1
-
av
->
av_buf_head
+
avr
->
avr_ack_ptr
;
av
->
av_buf_head
+
avr
->
avr_ack_ptr
;
/* free records */
/* free records */
list_for_each_entry_safe_from
(
avr
,
next
,
&
av
->
av_records
,
avr_node
)
{
list_for_each_entry_safe_from
(
avr
,
next
,
&
av
->
av_records
,
avr_node
)
{
list_del
_init
(
&
avr
->
avr_node
);
list_del
(
&
avr
->
avr_node
);
dccp_ackvec_record_delete
(
avr
);
kmem_cache_free
(
dccp_ackvec_record_slab
,
avr
);
}
}
}
}
...
@@ -357,7 +274,7 @@ void dccp_ackvec_check_rcv_ackno(struct dccp_ackvec *av, struct sock *sk,
...
@@ -357,7 +274,7 @@ void dccp_ackvec_check_rcv_ackno(struct dccp_ackvec *av, struct sock *sk,
if
(
ackno
==
avr
->
avr_ack_seqno
)
{
if
(
ackno
==
avr
->
avr_ack_seqno
)
{
dccp_pr_debug
(
"%s ACK packet 0, len=%d, ack_seqno=%llu, "
dccp_pr_debug
(
"%s ACK packet 0, len=%d, ack_seqno=%llu, "
"ack_ackno=%llu, ACKED!
\n
"
,
"ack_ackno=%llu, ACKED!
\n
"
,
dccp_role
(
sk
),
1
,
dccp_role
(
sk
),
avr
->
avr_ack_runlen
,
(
unsigned
long
long
)
avr
->
avr_ack_seqno
,
(
unsigned
long
long
)
avr
->
avr_ack_seqno
,
(
unsigned
long
long
)
avr
->
avr_ack_ackno
);
(
unsigned
long
long
)
avr
->
avr_ack_ackno
);
dccp_ackvec_throw_record
(
av
,
avr
);
dccp_ackvec_throw_record
(
av
,
avr
);
...
@@ -387,7 +304,7 @@ static void dccp_ackvec_check_rcv_ackvector(struct dccp_ackvec *av,
...
@@ -387,7 +304,7 @@ static void dccp_ackvec_check_rcv_ackvector(struct dccp_ackvec *av,
*/
*/
avr
=
list_entry
(
av
->
av_records
.
next
,
struct
dccp_ackvec_record
,
avr_node
);
avr
=
list_entry
(
av
->
av_records
.
next
,
struct
dccp_ackvec_record
,
avr_node
);
while
(
i
--
)
{
while
(
i
--
)
{
const
u8
rl
=
*
vector
&
DCCP_ACKVEC_LEN_MASK
;
const
u8
rl
=
dccp_ackvec_runlen
(
vector
)
;
u64
ackno_end_rl
;
u64
ackno_end_rl
;
dccp_set_seqno
(
&
ackno_end_rl
,
*
ackno
-
rl
);
dccp_set_seqno
(
&
ackno_end_rl
,
*
ackno
-
rl
);
...
@@ -404,8 +321,7 @@ static void dccp_ackvec_check_rcv_ackvector(struct dccp_ackvec *av,
...
@@ -404,8 +321,7 @@ static void dccp_ackvec_check_rcv_ackvector(struct dccp_ackvec *av,
break
;
break
;
found:
found:
if
(
between48
(
avr
->
avr_ack_seqno
,
ackno_end_rl
,
*
ackno
))
{
if
(
between48
(
avr
->
avr_ack_seqno
,
ackno_end_rl
,
*
ackno
))
{
const
u8
state
=
*
vector
&
DCCP_ACKVEC_STATE_MASK
;
if
(
dccp_ackvec_state
(
vector
)
!=
DCCPAV_NOT_RECEIVED
)
{
if
(
state
!=
DCCP_ACKVEC_STATE_NOT_RECEIVED
)
{
dccp_pr_debug
(
"%s ACK vector 0, len=%d, "
dccp_pr_debug
(
"%s ACK vector 0, len=%d, "
"ack_seqno=%llu, ack_ackno=%llu, "
"ack_seqno=%llu, ack_ackno=%llu, "
"ACKED!
\n
"
,
"ACKED!
\n
"
,
...
@@ -448,10 +364,9 @@ int __init dccp_ackvec_init(void)
...
@@ -448,10 +364,9 @@ int __init dccp_ackvec_init(void)
if
(
dccp_ackvec_slab
==
NULL
)
if
(
dccp_ackvec_slab
==
NULL
)
goto
out_err
;
goto
out_err
;
dccp_ackvec_record_slab
=
dccp_ackvec_record_slab
=
kmem_cache_create
(
"dccp_ackvec_record"
,
kmem_cache_create
(
"dccp_ackvec_record"
,
sizeof
(
struct
dccp_ackvec_record
),
sizeof
(
struct
dccp_ackvec_record
),
0
,
SLAB_HWCACHE_ALIGN
,
NULL
);
0
,
SLAB_HWCACHE_ALIGN
,
NULL
);
if
(
dccp_ackvec_record_slab
==
NULL
)
if
(
dccp_ackvec_record_slab
==
NULL
)
goto
out_destroy_slab
;
goto
out_destroy_slab
;
...
...
net/dccp/ackvec.h
View file @
ed1deb70
...
@@ -3,9 +3,9 @@
...
@@ -3,9 +3,9 @@
/*
/*
* net/dccp/ackvec.h
* net/dccp/ackvec.h
*
*
* An implementation of the DCCP protocol
* An implementation of Ack Vectors for the DCCP protocol
* Copyright (c) 2007 University of Aberdeen, Scotland, UK
* Copyright (c) 2005 Arnaldo Carvalho de Melo <acme@mandriva.com>
* Copyright (c) 2005 Arnaldo Carvalho de Melo <acme@mandriva.com>
*
* This program is free software; you can redistribute it and/or modify it
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 as
* under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
* published by the Free Software Foundation.
...
@@ -13,75 +13,89 @@
...
@@ -13,75 +13,89 @@
#include <linux/dccp.h>
#include <linux/dccp.h>
#include <linux/compiler.h>
#include <linux/compiler.h>
#include <linux/ktime.h>
#include <linux/list.h>
#include <linux/list.h>
#include <linux/types.h>
#include <linux/types.h>
/* We can spread an ack vector across multiple options */
/*
#define DCCP_MAX_ACKVEC_LEN (DCCP_SINGLE_OPT_MAXLEN * 2)
* Ack Vector buffer space is static, in multiples of %DCCP_SINGLE_OPT_MAXLEN,
* the maximum size of a single Ack Vector. Setting %DCCPAV_NUM_ACKVECS to 1
* will be sufficient for most cases of low Ack Ratios, using a value of 2 gives
* more headroom if Ack Ratio is higher or when the sender acknowledges slowly.
* The maximum value is bounded by the u16 types for indices and functions.
*/
#define DCCPAV_NUM_ACKVECS 2
#define DCCPAV_MAX_ACKVEC_LEN (DCCP_SINGLE_OPT_MAXLEN * DCCPAV_NUM_ACKVECS)
/* Estimated minimum average Ack Vector length - used for updating MPS */
/* Estimated minimum average Ack Vector length - used for updating MPS */
#define DCCPAV_MIN_OPTLEN 16
#define DCCPAV_MIN_OPTLEN 16
#define DCCP_ACKVEC_STATE_RECEIVED 0
enum
dccp_ackvec_states
{
#define DCCP_ACKVEC_STATE_ECN_MARKED (1 << 6)
DCCPAV_RECEIVED
=
0x00
,
#define DCCP_ACKVEC_STATE_NOT_RECEIVED (3 << 6)
DCCPAV_ECN_MARKED
=
0x40
,
DCCPAV_RESERVED
=
0x80
,
DCCPAV_NOT_RECEIVED
=
0xC0
};
#define DCCPAV_MAX_RUNLEN 0x3F
#define DCCP_ACKVEC_STATE_MASK 0xC0
/* 11000000 */
static
inline
u8
dccp_ackvec_runlen
(
const
u8
*
cell
)
#define DCCP_ACKVEC_LEN_MASK 0x3F
/* 00111111 */
{
return
*
cell
&
DCCPAV_MAX_RUNLEN
;
}
/** struct dccp_ackvec - ack vector
static
inline
u8
dccp_ackvec_state
(
const
u8
*
cell
)
*
{
* This data structure is the one defined in RFC 4340, Appendix A.
return
*
cell
&
~
DCCPAV_MAX_RUNLEN
;
*
}
* @av_buf_head - circular buffer head
* @av_buf_tail - circular buffer tail
/** struct dccp_ackvec - Ack Vector main data structure
* @av_buf_ackno - ack # of the most recent packet acknowledgeable in the
* buffer (i.e. %av_buf_head)
* @av_buf_nonce - the one-bit sum of the ECN Nonces on all packets acked
* by the buffer with State 0
*
* Additionally, the HC-Receiver must keep some information about the
* Ack Vectors it has recently sent. For each packet sent carrying an
* Ack Vector, it remembers four variables:
*
*
*
@av_records - list of dccp_ackvec_record
*
This implements a fixed-size circular buffer within an array and is largely
*
@av_ack_nonce - the one-bit sum of the ECN Nonces for all State
0.
*
based on Appendix A of RFC 434
0.
*
*
* @av_time - the time in usecs
* @av_buf: circular buffer storage area
* @av_buf - circular buffer of acknowledgeable packets
* @av_buf_head: head index; begin of live portion in @av_buf
* @av_buf_tail: tail index; first index _after_ the live portion in @av_buf
* @av_buf_ackno: highest seqno of acknowledgeable packet recorded in @av_buf
* @av_tail_ackno: lowest seqno of acknowledgeable packet recorded in @av_buf
* @av_buf_nonce: ECN nonce sums, each covering subsequent segments of up to
* %DCCP_SINGLE_OPT_MAXLEN cells in the live portion of @av_buf
* @av_overflow: if 1 then buf_head == buf_tail indicates buffer wraparound
* @av_records: list of %dccp_ackvec_record (Ack Vectors sent previously)
* @av_veclen: length of the live portion of @av_buf
*/
*/
struct
dccp_ackvec
{
struct
dccp_ackvec
{
u64
av_buf_ackno
;
u8
av_buf
[
DCCPAV_MAX_ACKVEC_LEN
];
struct
list_head
av_records
;
ktime_t
av_time
;
u16
av_buf_head
;
u16
av_buf_head
;
u16
av_buf_tail
;
u64
av_buf_ackno
:
48
;
u64
av_tail_ackno
:
48
;
bool
av_buf_nonce
[
DCCPAV_NUM_ACKVECS
];
u8
av_overflow
:
1
;
struct
list_head
av_records
;
u16
av_vec_len
;
u16
av_vec_len
;
u8
av_buf_nonce
;
u8
av_ack_nonce
;
u8
av_buf
[
DCCP_MAX_ACKVEC_LEN
];
};
};
/** struct dccp_ackvec_record -
ack vector record
/** struct dccp_ackvec_record -
Records information about sent Ack Vectors
*
*
* ACK vector record as defined in Appendix A of spec.
* These list entries define the additional information which the HC-Receiver
* keeps about recently-sent Ack Vectors; again refer to RFC 4340, Appendix A.
*
*
* The list is sorted by avr_ack_seqno
* @avr_node: the list node in @av_records
* @avr_ack_seqno: sequence number of the packet the Ack Vector was sent on
* @avr_ack_ackno: the Ack number that this record/Ack Vector refers to
* @avr_ack_ptr: pointer into @av_buf where this record starts
* @avr_ack_runlen: run length of @avr_ack_ptr at the time of sending
* @avr_ack_nonce: the sum of @av_buf_nonce's at the time this record was sent
*
*
* @avr_node - node in av_records
* The list as a whole is sorted in descending order by @avr_ack_seqno.
* @avr_ack_seqno - sequence number of the packet this record was sent on
* @avr_ack_ackno - sequence number being acknowledged
* @avr_ack_ptr - pointer into av_buf where this record starts
* @avr_ack_nonce - av_ack_nonce at the time this record was sent
* @avr_sent_len - lenght of the record in av_buf
*/
*/
struct
dccp_ackvec_record
{
struct
dccp_ackvec_record
{
struct
list_head
avr_node
;
struct
list_head
avr_node
;
u64
avr_ack_seqno
;
u64
avr_ack_seqno
:
48
;
u64
avr_ack_ackno
;
u64
avr_ack_ackno
:
48
;
u16
avr_ack_ptr
;
u16
avr_ack_ptr
;
u
16
avr_sent_
len
;
u
8
avr_ack_run
len
;
u8
avr_ack_nonce
;
u8
avr_ack_nonce
:
1
;
};
};
struct
sock
;
struct
sock
;
...
@@ -102,10 +116,11 @@ extern int dccp_ackvec_parse(struct sock *sk, const struct sk_buff *skb,
...
@@ -102,10 +116,11 @@ extern int dccp_ackvec_parse(struct sock *sk, const struct sk_buff *skb,
u64
*
ackno
,
const
u8
opt
,
u64
*
ackno
,
const
u8
opt
,
const
u8
*
value
,
const
u8
len
);
const
u8
*
value
,
const
u8
len
);
extern
int
dccp_insert_option_ackvec
(
struct
sock
*
sk
,
struct
sk_buff
*
skb
);
extern
int
dccp_ackvec_update_records
(
struct
dccp_ackvec
*
av
,
u64
seq
,
u8
sum
);
extern
u16
dccp_ackvec_buflen
(
const
struct
dccp_ackvec
*
av
);
static
inline
int
dccp_ackvec_pending
(
const
struct
dccp_ackvec
*
av
)
static
inline
bool
dccp_ackvec_is_empty
(
const
struct
dccp_ackvec
*
av
)
{
{
return
av
->
av_
vec_len
;
return
av
->
av_
overflow
==
0
&&
av
->
av_buf_head
==
av
->
av_buf_tail
;
}
}
#endif
/* _ACKVEC_H */
#endif
/* _ACKVEC_H */
net/dccp/ccids/ccid2.c
View file @
ed1deb70
...
@@ -513,8 +513,7 @@ static void ccid2_hc_tx_packet_recv(struct sock *sk, struct sk_buff *skb)
...
@@ -513,8 +513,7 @@ static void ccid2_hc_tx_packet_recv(struct sock *sk, struct sk_buff *skb)
&
vector
,
&
veclen
))
!=
-
1
)
{
&
vector
,
&
veclen
))
!=
-
1
)
{
/* go through this ack vector */
/* go through this ack vector */
while
(
veclen
--
)
{
while
(
veclen
--
)
{
const
u8
rl
=
*
vector
&
DCCP_ACKVEC_LEN_MASK
;
u64
ackno_end_rl
=
SUB48
(
ackno
,
dccp_ackvec_runlen
(
vector
));
u64
ackno_end_rl
=
SUB48
(
ackno
,
rl
);
ccid2_pr_debug
(
"ackvec start:%llu end:%llu
\n
"
,
ccid2_pr_debug
(
"ackvec start:%llu end:%llu
\n
"
,
(
unsigned
long
long
)
ackno
,
(
unsigned
long
long
)
ackno
,
...
@@ -537,17 +536,15 @@ static void ccid2_hc_tx_packet_recv(struct sock *sk, struct sk_buff *skb)
...
@@ -537,17 +536,15 @@ static void ccid2_hc_tx_packet_recv(struct sock *sk, struct sk_buff *skb)
* run length
* run length
*/
*/
while
(
between48
(
seqp
->
ccid2s_seq
,
ackno_end_rl
,
ackno
))
{
while
(
between48
(
seqp
->
ccid2s_seq
,
ackno_end_rl
,
ackno
))
{
const
u8
state
=
*
vector
&
const
u8
state
=
dccp_ackvec_state
(
vector
);
DCCP_ACKVEC_STATE_MASK
;
/* new packet received or marked */
/* new packet received or marked */
if
(
state
!=
DCCP
_ACKVEC_STATE
_NOT_RECEIVED
&&
if
(
state
!=
DCCP
AV
_NOT_RECEIVED
&&
!
seqp
->
ccid2s_acked
)
{
!
seqp
->
ccid2s_acked
)
{
if
(
state
==
if
(
state
==
DCCPAV_ECN_MARKED
)
DCCP_ACKVEC_STATE_ECN_MARKED
)
{
ccid2_congestion_event
(
sk
,
ccid2_congestion_event
(
sk
,
seqp
);
seqp
);
}
else
else
ccid2_new_ack
(
sk
,
seqp
,
ccid2_new_ack
(
sk
,
seqp
,
&
maxincr
);
&
maxincr
);
...
...
net/dccp/dccp.h
View file @
ed1deb70
...
@@ -457,12 +457,15 @@ static inline void dccp_update_gss(struct sock *sk, u64 seq)
...
@@ -457,12 +457,15 @@ static inline void dccp_update_gss(struct sock *sk, u64 seq)
dp
->
dccps_awh
=
dp
->
dccps_gss
;
dp
->
dccps_awh
=
dp
->
dccps_gss
;
}
}
static
inline
int
dccp_ackvec_pending
(
const
struct
sock
*
sk
)
{
return
dccp_sk
(
sk
)
->
dccps_hc_rx_ackvec
!=
NULL
&&
!
dccp_ackvec_is_empty
(
dccp_sk
(
sk
)
->
dccps_hc_rx_ackvec
);
}
static
inline
int
dccp_ack_pending
(
const
struct
sock
*
sk
)
static
inline
int
dccp_ack_pending
(
const
struct
sock
*
sk
)
{
{
const
struct
dccp_sock
*
dp
=
dccp_sk
(
sk
);
return
dccp_ackvec_pending
(
sk
)
||
inet_csk_ack_scheduled
(
sk
);
return
(
dp
->
dccps_hc_rx_ackvec
!=
NULL
&&
dccp_ackvec_pending
(
dp
->
dccps_hc_rx_ackvec
))
||
inet_csk_ack_scheduled
(
sk
);
}
}
extern
int
dccp_feat_finalise_settings
(
struct
dccp_sock
*
dp
);
extern
int
dccp_feat_finalise_settings
(
struct
dccp_sock
*
dp
);
...
...
net/dccp/input.c
View file @
ed1deb70
...
@@ -378,8 +378,7 @@ int dccp_rcv_established(struct sock *sk, struct sk_buff *skb,
...
@@ -378,8 +378,7 @@ int dccp_rcv_established(struct sock *sk, struct sk_buff *skb,
if
(
dp
->
dccps_hc_rx_ackvec
!=
NULL
&&
if
(
dp
->
dccps_hc_rx_ackvec
!=
NULL
&&
dccp_ackvec_add
(
dp
->
dccps_hc_rx_ackvec
,
sk
,
dccp_ackvec_add
(
dp
->
dccps_hc_rx_ackvec
,
sk
,
DCCP_SKB_CB
(
skb
)
->
dccpd_seq
,
DCCP_SKB_CB
(
skb
)
->
dccpd_seq
,
DCCPAV_RECEIVED
))
DCCP_ACKVEC_STATE_RECEIVED
))
goto
discard
;
goto
discard
;
dccp_deliver_input_to_ccids
(
sk
,
skb
);
dccp_deliver_input_to_ccids
(
sk
,
skb
);
...
@@ -637,8 +636,7 @@ int dccp_rcv_state_process(struct sock *sk, struct sk_buff *skb,
...
@@ -637,8 +636,7 @@ int dccp_rcv_state_process(struct sock *sk, struct sk_buff *skb,
if
(
dp
->
dccps_hc_rx_ackvec
!=
NULL
&&
if
(
dp
->
dccps_hc_rx_ackvec
!=
NULL
&&
dccp_ackvec_add
(
dp
->
dccps_hc_rx_ackvec
,
sk
,
dccp_ackvec_add
(
dp
->
dccps_hc_rx_ackvec
,
sk
,
DCCP_SKB_CB
(
skb
)
->
dccpd_seq
,
DCCP_SKB_CB
(
skb
)
->
dccpd_seq
,
DCCPAV_RECEIVED
))
DCCP_ACKVEC_STATE_RECEIVED
))
goto
discard
;
goto
discard
;
dccp_deliver_input_to_ccids
(
sk
,
skb
);
dccp_deliver_input_to_ccids
(
sk
,
skb
);
...
...
net/dccp/options.c
View file @
ed1deb70
...
@@ -340,6 +340,7 @@ static inline int dccp_elapsed_time_len(const u32 elapsed_time)
...
@@ -340,6 +340,7 @@ static inline int dccp_elapsed_time_len(const u32 elapsed_time)
return
elapsed_time
==
0
?
0
:
elapsed_time
<=
0xFFFF
?
2
:
4
;
return
elapsed_time
==
0
?
0
:
elapsed_time
<=
0xFFFF
?
2
:
4
;
}
}
/* FIXME: This function is currently not used anywhere */
int
dccp_insert_option_elapsed_time
(
struct
sk_buff
*
skb
,
u32
elapsed_time
)
int
dccp_insert_option_elapsed_time
(
struct
sk_buff
*
skb
,
u32
elapsed_time
)
{
{
const
int
elapsed_time_len
=
dccp_elapsed_time_len
(
elapsed_time
);
const
int
elapsed_time_len
=
dccp_elapsed_time_len
(
elapsed_time
);
...
@@ -424,6 +425,67 @@ static int dccp_insert_option_timestamp_echo(struct dccp_sock *dp,
...
@@ -424,6 +425,67 @@ static int dccp_insert_option_timestamp_echo(struct dccp_sock *dp,
return
0
;
return
0
;
}
}
static
int
dccp_insert_option_ackvec
(
struct
sock
*
sk
,
struct
sk_buff
*
skb
)
{
struct
dccp_sock
*
dp
=
dccp_sk
(
sk
);
struct
dccp_ackvec
*
av
=
dp
->
dccps_hc_rx_ackvec
;
const
u16
buflen
=
dccp_ackvec_buflen
(
av
);
/* Figure out how many options do we need to represent the ackvec */
const
u8
nr_opts
=
DIV_ROUND_UP
(
buflen
,
DCCP_SINGLE_OPT_MAXLEN
);
u16
len
=
buflen
+
2
*
nr_opts
;
u8
i
,
nonce
=
0
;
const
unsigned
char
*
tail
,
*
from
;
unsigned
char
*
to
;
if
(
DCCP_SKB_CB
(
skb
)
->
dccpd_opt_len
+
len
>
DCCP_MAX_OPT_LEN
)
return
-
1
;
DCCP_SKB_CB
(
skb
)
->
dccpd_opt_len
+=
len
;
to
=
skb_push
(
skb
,
len
);
len
=
buflen
;
from
=
av
->
av_buf
+
av
->
av_buf_head
;
tail
=
av
->
av_buf
+
DCCPAV_MAX_ACKVEC_LEN
;
for
(
i
=
0
;
i
<
nr_opts
;
++
i
)
{
int
copylen
=
len
;
if
(
len
>
DCCP_SINGLE_OPT_MAXLEN
)
copylen
=
DCCP_SINGLE_OPT_MAXLEN
;
/*
* RFC 4340, 12.2: Encode the Nonce Echo for this Ack Vector via
* its type; ack_nonce is the sum of all individual buf_nonce's.
*/
nonce
^=
av
->
av_buf_nonce
[
i
];
*
to
++
=
DCCPO_ACK_VECTOR_0
+
av
->
av_buf_nonce
[
i
];
*
to
++
=
copylen
+
2
;
/* Check if buf_head wraps */
if
(
from
+
copylen
>
tail
)
{
const
u16
tailsize
=
tail
-
from
;
memcpy
(
to
,
from
,
tailsize
);
to
+=
tailsize
;
len
-=
tailsize
;
copylen
-=
tailsize
;
from
=
av
->
av_buf
;
}
memcpy
(
to
,
from
,
copylen
);
from
+=
copylen
;
to
+=
copylen
;
len
-=
copylen
;
}
/*
* Each sent Ack Vector is recorded in the list, as per A.2 of RFC 4340.
*/
if
(
dccp_ackvec_update_records
(
av
,
DCCP_SKB_CB
(
skb
)
->
dccpd_seq
,
nonce
))
return
-
ENOBUFS
;
return
0
;
}
/**
/**
* dccp_insert_option_mandatory - Mandatory option (5.8.2)
* dccp_insert_option_mandatory - Mandatory option (5.8.2)
* Note that since we are using skb_push, this function needs to be called
* Note that since we are using skb_push, this function needs to be called
...
@@ -519,8 +581,7 @@ int dccp_insert_options(struct sock *sk, struct sk_buff *skb)
...
@@ -519,8 +581,7 @@ int dccp_insert_options(struct sock *sk, struct sk_buff *skb)
if
(
dccp_insert_option_timestamp
(
skb
))
if
(
dccp_insert_option_timestamp
(
skb
))
return
-
1
;
return
-
1
;
}
else
if
(
dp
->
dccps_hc_rx_ackvec
!=
NULL
&&
}
else
if
(
dccp_ackvec_pending
(
sk
)
&&
dccp_ackvec_pending
(
dp
->
dccps_hc_rx_ackvec
)
&&
dccp_insert_option_ackvec
(
sk
,
skb
))
{
dccp_insert_option_ackvec
(
sk
,
skb
))
{
return
-
1
;
return
-
1
;
}
}
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment