Commit 8e1ee18c authored by Vlad Yasevich's avatar Vlad Yasevich Committed by David S. Miller

sctp: Rework the tsn map to use generic bitmap.

The tsn map currently use is 4K large and is stuck inside
the sctp_association structure making memory references REALLY
expensive.  What we really need is at most 4K worth of bits
so the biggest map we would have is 512 bytes.   Also, the
map is only really usefull when we have gaps to store and
report.  As such, starting with minimal map of say 32 TSNs (bits)
should be enough for normal low-loss operations.  We can grow
the map by some multiple of 32 along with some extra room any
time we receive the TSN which would put us outside of the map
boundry.  As we close gaps, we can shift the map to rebase
it on the latest TSN we've seen.  This saves 4088 bytes per
association just in the map alone along savings from the now
unnecessary structure members.
Signed-off-by: default avatarVlad Yasevich <vladislav.yasevich@hp.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 3c689b73
...@@ -261,7 +261,9 @@ enum { SCTP_ARBITRARY_COOKIE_ECHO_LEN = 200 }; ...@@ -261,7 +261,9 @@ enum { SCTP_ARBITRARY_COOKIE_ECHO_LEN = 200 };
* must be less than 65535 (2^16 - 1), or we will have overflow * must be less than 65535 (2^16 - 1), or we will have overflow
* problems creating SACK's. * problems creating SACK's.
*/ */
#define SCTP_TSN_MAP_SIZE 2048 #define SCTP_TSN_MAP_INITIAL BITS_PER_LONG
#define SCTP_TSN_MAP_INCREMENT SCTP_TSN_MAP_INITIAL
#define SCTP_TSN_MAP_SIZE 4096
#define SCTP_TSN_MAX_GAP 65535 #define SCTP_TSN_MAX_GAP 65535
/* We will not record more than this many duplicate TSNs between two /* We will not record more than this many duplicate TSNs between two
......
...@@ -1545,7 +1545,6 @@ struct sctp_association { ...@@ -1545,7 +1545,6 @@ struct sctp_association {
* in tsn_map--we get it by calling sctp_tsnmap_get_ctsn(). * in tsn_map--we get it by calling sctp_tsnmap_get_ctsn().
*/ */
struct sctp_tsnmap tsn_map; struct sctp_tsnmap tsn_map;
__u8 _map[sctp_tsnmap_storage_size(SCTP_TSN_MAP_SIZE)];
/* Ack State : This flag indicates if the next received /* Ack State : This flag indicates if the next received
* : packet is to be responded to with a * : packet is to be responded to with a
......
...@@ -60,18 +60,7 @@ struct sctp_tsnmap { ...@@ -60,18 +60,7 @@ struct sctp_tsnmap {
* It points at one of the two buffers with which we will * It points at one of the two buffers with which we will
* ping-pong between. * ping-pong between.
*/ */
__u8 *tsn_map; unsigned long *tsn_map;
/* This marks the tsn which overflows the tsn_map, when the
* cumulative ack point reaches this point we know we can switch
* maps (tsn_map and overflow_map swap).
*/
__u32 overflow_tsn;
/* This is the overflow array for tsn_map.
* It points at one of the other ping-pong buffers.
*/
__u8 *overflow_map;
/* This is the TSN at tsn_map[0]. */ /* This is the TSN at tsn_map[0]. */
__u32 base_tsn; __u32 base_tsn;
...@@ -89,15 +78,15 @@ struct sctp_tsnmap { ...@@ -89,15 +78,15 @@ struct sctp_tsnmap {
*/ */
__u32 cumulative_tsn_ack_point; __u32 cumulative_tsn_ack_point;
/* This is the highest TSN we've marked. */
__u32 max_tsn_seen;
/* This is the minimum number of TSNs we can track. This corresponds /* This is the minimum number of TSNs we can track. This corresponds
* to the size of tsn_map. Note: the overflow_map allows us to * to the size of tsn_map. Note: the overflow_map allows us to
* potentially track more than this quantity. * potentially track more than this quantity.
*/ */
__u16 len; __u16 len;
/* This is the highest TSN we've marked. */
__u32 max_tsn_seen;
/* Data chunks pending receipt. used by SCTP_STATUS sockopt */ /* Data chunks pending receipt. used by SCTP_STATUS sockopt */
__u16 pending_data; __u16 pending_data;
...@@ -110,24 +99,17 @@ struct sctp_tsnmap { ...@@ -110,24 +99,17 @@ struct sctp_tsnmap {
/* Record gap ack block information here. */ /* Record gap ack block information here. */
struct sctp_gap_ack_block gabs[SCTP_MAX_GABS]; struct sctp_gap_ack_block gabs[SCTP_MAX_GABS];
int malloced;
__u8 raw_map[0];
}; };
struct sctp_tsnmap_iter { struct sctp_tsnmap_iter {
__u32 start; __u32 start;
}; };
/* This macro assists in creation of external storage for variable length
* internal buffers. We double allocate so the overflow map works.
*/
#define sctp_tsnmap_storage_size(count) (sizeof(__u8) * (count) * 2)
/* Initialize a block of memory as a tsnmap. */ /* Initialize a block of memory as a tsnmap. */
struct sctp_tsnmap *sctp_tsnmap_init(struct sctp_tsnmap *, __u16 len, struct sctp_tsnmap *sctp_tsnmap_init(struct sctp_tsnmap *, __u16 len,
__u32 initial_tsn); __u32 initial_tsn, gfp_t gfp);
void sctp_tsnmap_free(struct sctp_tsnmap *map);
/* Test the tracking state of this TSN. /* Test the tracking state of this TSN.
* Returns: * Returns:
...@@ -138,7 +120,7 @@ struct sctp_tsnmap *sctp_tsnmap_init(struct sctp_tsnmap *, __u16 len, ...@@ -138,7 +120,7 @@ struct sctp_tsnmap *sctp_tsnmap_init(struct sctp_tsnmap *, __u16 len,
int sctp_tsnmap_check(const struct sctp_tsnmap *, __u32 tsn); int sctp_tsnmap_check(const struct sctp_tsnmap *, __u32 tsn);
/* Mark this TSN as seen. */ /* Mark this TSN as seen. */
void sctp_tsnmap_mark(struct sctp_tsnmap *, __u32 tsn); int sctp_tsnmap_mark(struct sctp_tsnmap *, __u32 tsn);
/* Mark this TSN and all lower as seen. */ /* Mark this TSN and all lower as seen. */
void sctp_tsnmap_skip(struct sctp_tsnmap *map, __u32 tsn); void sctp_tsnmap_skip(struct sctp_tsnmap *map, __u32 tsn);
...@@ -183,10 +165,7 @@ static inline struct sctp_gap_ack_block *sctp_tsnmap_get_gabs(struct sctp_tsnmap ...@@ -183,10 +165,7 @@ static inline struct sctp_gap_ack_block *sctp_tsnmap_get_gabs(struct sctp_tsnmap
/* Is there a gap in the TSN map? */ /* Is there a gap in the TSN map? */
static inline int sctp_tsnmap_has_gap(const struct sctp_tsnmap *map) static inline int sctp_tsnmap_has_gap(const struct sctp_tsnmap *map)
{ {
int has_gap; return (map->cumulative_tsn_ack_point != map->max_tsn_seen);
has_gap = (map->cumulative_tsn_ack_point != map->max_tsn_seen);
return has_gap;
} }
/* Mark a duplicate TSN. Note: limit the storage of duplicate TSN /* Mark a duplicate TSN. Note: limit the storage of duplicate TSN
......
...@@ -283,8 +283,7 @@ static struct sctp_association *sctp_association_init(struct sctp_association *a ...@@ -283,8 +283,7 @@ static struct sctp_association *sctp_association_init(struct sctp_association *a
if (!sctp_ulpq_init(&asoc->ulpq, asoc)) if (!sctp_ulpq_init(&asoc->ulpq, asoc))
goto fail_init; goto fail_init;
/* Set up the tsn tracking. */ memset(&asoc->peer.tsn_map, 0, sizeof(struct sctp_tsnmap));
sctp_tsnmap_init(&asoc->peer.tsn_map, SCTP_TSN_MAP_SIZE, 0);
asoc->need_ecne = 0; asoc->need_ecne = 0;
...@@ -402,6 +401,8 @@ void sctp_association_free(struct sctp_association *asoc) ...@@ -402,6 +401,8 @@ void sctp_association_free(struct sctp_association *asoc)
/* Dispose of any pending chunks on the inqueue. */ /* Dispose of any pending chunks on the inqueue. */
sctp_inq_free(&asoc->base.inqueue); sctp_inq_free(&asoc->base.inqueue);
sctp_tsnmap_free(&asoc->peer.tsn_map);
/* Free ssnmap storage. */ /* Free ssnmap storage. */
sctp_ssnmap_free(asoc->ssnmap); sctp_ssnmap_free(asoc->ssnmap);
...@@ -1122,8 +1123,8 @@ void sctp_assoc_update(struct sctp_association *asoc, ...@@ -1122,8 +1123,8 @@ void sctp_assoc_update(struct sctp_association *asoc,
asoc->peer.rwnd = new->peer.rwnd; asoc->peer.rwnd = new->peer.rwnd;
asoc->peer.sack_needed = new->peer.sack_needed; asoc->peer.sack_needed = new->peer.sack_needed;
asoc->peer.i = new->peer.i; asoc->peer.i = new->peer.i;
sctp_tsnmap_init(&asoc->peer.tsn_map, SCTP_TSN_MAP_SIZE, sctp_tsnmap_init(&asoc->peer.tsn_map, SCTP_TSN_MAP_INITIAL,
asoc->peer.i.initial_tsn); asoc->peer.i.initial_tsn, GFP_ATOMIC);
/* Remove any peer addresses not present in the new association. */ /* Remove any peer addresses not present in the new association. */
list_for_each_safe(pos, temp, &asoc->peer.transport_addr_list) { list_for_each_safe(pos, temp, &asoc->peer.transport_addr_list) {
......
...@@ -2288,8 +2288,9 @@ int sctp_process_init(struct sctp_association *asoc, sctp_cid_t cid, ...@@ -2288,8 +2288,9 @@ int sctp_process_init(struct sctp_association *asoc, sctp_cid_t cid,
} }
/* Set up the TSN tracking pieces. */ /* Set up the TSN tracking pieces. */
sctp_tsnmap_init(&asoc->peer.tsn_map, SCTP_TSN_MAP_SIZE, if (!sctp_tsnmap_init(&asoc->peer.tsn_map, SCTP_TSN_MAP_INITIAL,
asoc->peer.i.initial_tsn); asoc->peer.i.initial_tsn, gfp))
goto clean_up;
/* RFC 2960 6.5 Stream Identifier and Stream Sequence Number /* RFC 2960 6.5 Stream Identifier and Stream Sequence Number
* *
......
...@@ -1152,7 +1152,8 @@ static int sctp_cmd_interpreter(sctp_event_t event_type, ...@@ -1152,7 +1152,8 @@ static int sctp_cmd_interpreter(sctp_event_t event_type,
case SCTP_CMD_REPORT_TSN: case SCTP_CMD_REPORT_TSN:
/* Record the arrival of a TSN. */ /* Record the arrival of a TSN. */
sctp_tsnmap_mark(&asoc->peer.tsn_map, cmd->obj.u32); error = sctp_tsnmap_mark(&asoc->peer.tsn_map,
cmd->obj.u32);
break; break;
case SCTP_CMD_REPORT_FWDTSN: case SCTP_CMD_REPORT_FWDTSN:
......
...@@ -43,37 +43,44 @@ ...@@ -43,37 +43,44 @@
*/ */
#include <linux/types.h> #include <linux/types.h>
#include <linux/bitmap.h>
#include <net/sctp/sctp.h> #include <net/sctp/sctp.h>
#include <net/sctp/sm.h> #include <net/sctp/sm.h>
static void sctp_tsnmap_update(struct sctp_tsnmap *map); static void sctp_tsnmap_update(struct sctp_tsnmap *map);
static void sctp_tsnmap_find_gap_ack(__u8 *map, __u16 off, static void sctp_tsnmap_find_gap_ack(unsigned long *map, __u16 off,
__u16 len, __u16 base, __u16 len, __u16 *start, __u16 *end);
int *started, __u16 *start, static int sctp_tsnmap_grow(struct sctp_tsnmap *map, u16 gap);
int *ended, __u16 *end);
/* Initialize a block of memory as a tsnmap. */ /* Initialize a block of memory as a tsnmap. */
struct sctp_tsnmap *sctp_tsnmap_init(struct sctp_tsnmap *map, __u16 len, struct sctp_tsnmap *sctp_tsnmap_init(struct sctp_tsnmap *map, __u16 len,
__u32 initial_tsn) __u32 initial_tsn, gfp_t gfp)
{ {
map->tsn_map = map->raw_map; if (!map->tsn_map) {
map->overflow_map = map->tsn_map + len; map->tsn_map = kzalloc(len>>3, gfp);
map->len = len; if (map->tsn_map == NULL)
return NULL;
/* Clear out a TSN ack status. */ map->len = len;
memset(map->tsn_map, 0x00, map->len + map->len); } else {
bitmap_zero(map->tsn_map, map->len);
}
/* Keep track of TSNs represented by tsn_map. */ /* Keep track of TSNs represented by tsn_map. */
map->base_tsn = initial_tsn; map->base_tsn = initial_tsn;
map->overflow_tsn = initial_tsn + map->len;
map->cumulative_tsn_ack_point = initial_tsn - 1; map->cumulative_tsn_ack_point = initial_tsn - 1;
map->max_tsn_seen = map->cumulative_tsn_ack_point; map->max_tsn_seen = map->cumulative_tsn_ack_point;
map->malloced = 0;
map->num_dup_tsns = 0; map->num_dup_tsns = 0;
return map; return map;
} }
void sctp_tsnmap_free(struct sctp_tsnmap *map)
{
map->len = 0;
kfree(map->tsn_map);
}
/* Test the tracking state of this TSN. /* Test the tracking state of this TSN.
* Returns: * Returns:
* 0 if the TSN has not yet been seen * 0 if the TSN has not yet been seen
...@@ -82,66 +89,69 @@ struct sctp_tsnmap *sctp_tsnmap_init(struct sctp_tsnmap *map, __u16 len, ...@@ -82,66 +89,69 @@ struct sctp_tsnmap *sctp_tsnmap_init(struct sctp_tsnmap *map, __u16 len,
*/ */
int sctp_tsnmap_check(const struct sctp_tsnmap *map, __u32 tsn) int sctp_tsnmap_check(const struct sctp_tsnmap *map, __u32 tsn)
{ {
__s32 gap; u32 gap;
int dup;
/* Check to see if this is an old TSN */
if (TSN_lte(tsn, map->cumulative_tsn_ack_point))
return 1;
/* Verify that we can hold this TSN and that it will not
* overlfow our map
*/
if (!TSN_lt(tsn, map->base_tsn + SCTP_TSN_MAP_SIZE))
return -1;
/* Calculate the index into the mapping arrays. */ /* Calculate the index into the mapping arrays. */
gap = tsn - map->base_tsn; gap = tsn - map->base_tsn;
/* Verify that we can hold this TSN. */ /* Check to see if TSN has already been recorded. */
if (gap >= (/* base */ map->len + /* overflow */ map->len)) { if (gap < map->len && test_bit(gap, map->tsn_map))
dup = -1; return 1;
goto out;
}
/* Honk if we've already seen this TSN.
* We have three cases:
* 1. The TSN is ancient or belongs to a previous tsn_map.
* 2. The TSN is already marked in the tsn_map.
* 3. The TSN is already marked in the tsn_map_overflow.
*/
if (gap < 0 ||
(gap < map->len && map->tsn_map[gap]) ||
(gap >= map->len && map->overflow_map[gap - map->len]))
dup = 1;
else else
dup = 0; return 0;
out:
return dup;
} }
/* Mark this TSN as seen. */ /* Mark this TSN as seen. */
void sctp_tsnmap_mark(struct sctp_tsnmap *map, __u32 tsn) int sctp_tsnmap_mark(struct sctp_tsnmap *map, __u32 tsn)
{ {
__s32 gap; u16 gap;
/* Vacuously mark any TSN which precedes the map base or
* exceeds the end of the map.
*/
if (TSN_lt(tsn, map->base_tsn)) if (TSN_lt(tsn, map->base_tsn))
return; return 0;
if (!TSN_lt(tsn, map->base_tsn + map->len + map->len))
return;
/* Bump the max. */ gap = tsn - map->base_tsn;
if (gap >= map->len && !sctp_tsnmap_grow(map, gap))
return -ENOMEM;
if (!sctp_tsnmap_has_gap(map) && gap == 0) {
/* In this case the map has no gaps and the tsn we are
* recording is the next expected tsn. We don't touch
* the map but simply bump the values.
*/
map->max_tsn_seen++;
map->cumulative_tsn_ack_point++;
map->base_tsn++;
} else {
/* Either we already have a gap, or about to record a gap, so
* have work to do.
*
* Bump the max.
*/
if (TSN_lt(map->max_tsn_seen, tsn)) if (TSN_lt(map->max_tsn_seen, tsn))
map->max_tsn_seen = tsn; map->max_tsn_seen = tsn;
/* Assert: TSN is in range. */
gap = tsn - map->base_tsn;
/* Mark the TSN as received. */ /* Mark the TSN as received. */
if (gap < map->len) set_bit(gap, map->tsn_map);
map->tsn_map[gap]++;
else
map->overflow_map[gap - map->len]++;
/* Go fixup any internal TSN mapping variables including /* Go fixup any internal TSN mapping variables including
* cumulative_tsn_ack_point. * cumulative_tsn_ack_point.
*/ */
sctp_tsnmap_update(map); sctp_tsnmap_update(map);
}
return 0;
} }
...@@ -160,66 +170,34 @@ SCTP_STATIC int sctp_tsnmap_next_gap_ack(const struct sctp_tsnmap *map, ...@@ -160,66 +170,34 @@ SCTP_STATIC int sctp_tsnmap_next_gap_ack(const struct sctp_tsnmap *map,
struct sctp_tsnmap_iter *iter, struct sctp_tsnmap_iter *iter,
__u16 *start, __u16 *end) __u16 *start, __u16 *end)
{ {
int started, ended; int ended = 0;
__u16 start_, end_, offset; __u16 start_ = 0, end_ = 0, offset;
/* We haven't found a gap yet. */
started = ended = 0;
/* If there are no more gap acks possible, get out fast. */ /* If there are no more gap acks possible, get out fast. */
if (TSN_lte(map->max_tsn_seen, iter->start)) if (TSN_lte(map->max_tsn_seen, iter->start))
return 0; return 0;
/* Search the first mapping array. */
if (iter->start - map->base_tsn < map->len) {
offset = iter->start - map->base_tsn; offset = iter->start - map->base_tsn;
sctp_tsnmap_find_gap_ack(map->tsn_map, offset, map->len, 0, sctp_tsnmap_find_gap_ack(map->tsn_map, offset, map->len,
&started, &start_, &ended, &end_); &start_, &end_);
}
/* Do we need to check the overflow map? */
if (!ended) {
/* Fix up where we'd like to start searching in the
* overflow map.
*/
if (iter->start - map->base_tsn < map->len)
offset = 0;
else
offset = iter->start - map->base_tsn - map->len;
/* Search the overflow map. */
sctp_tsnmap_find_gap_ack(map->overflow_map,
offset,
map->len,
map->len,
&started, &start_,
&ended, &end_);
}
/* The Gap Ack Block happens to end at the end of the /* The Gap Ack Block happens to end at the end of the map. */
* overflow map. if (start_ && !end_)
*/ end_ = map->len - 1;
if (started && !ended) {
ended++;
end_ = map->len + map->len - 1;
}
/* If we found a Gap Ack Block, return the start and end and /* If we found a Gap Ack Block, return the start and end and
* bump the iterator forward. * bump the iterator forward.
*/ */
if (ended) { if (end_) {
/* Fix up the start and end based on the /* Fix up the start and end based on the
* Cumulative TSN Ack offset into the map. * Cumulative TSN Ack which is always 1 behind base.
*/ */
int gap = map->cumulative_tsn_ack_point - *start = start_ + 1;
map->base_tsn; *end = end_ + 1;
*start = start_ - gap;
*end = end_ - gap;
/* Move the iterator forward. */ /* Move the iterator forward. */
iter->start = map->cumulative_tsn_ack_point + *end + 1; iter->start = map->cumulative_tsn_ack_point + *end + 1;
ended = 1;
} }
return ended; return ended;
...@@ -228,35 +206,33 @@ SCTP_STATIC int sctp_tsnmap_next_gap_ack(const struct sctp_tsnmap *map, ...@@ -228,35 +206,33 @@ SCTP_STATIC int sctp_tsnmap_next_gap_ack(const struct sctp_tsnmap *map,
/* Mark this and any lower TSN as seen. */ /* Mark this and any lower TSN as seen. */
void sctp_tsnmap_skip(struct sctp_tsnmap *map, __u32 tsn) void sctp_tsnmap_skip(struct sctp_tsnmap *map, __u32 tsn)
{ {
__s32 gap; u32 gap;
/* Vacuously mark any TSN which precedes the map base or
* exceeds the end of the map.
*/
if (TSN_lt(tsn, map->base_tsn)) if (TSN_lt(tsn, map->base_tsn))
return; return;
if (!TSN_lt(tsn, map->base_tsn + map->len + map->len)) if (!TSN_lt(tsn, map->base_tsn + SCTP_TSN_MAP_SIZE))
return; return;
/* Bump the max. */ /* Bump the max. */
if (TSN_lt(map->max_tsn_seen, tsn)) if (TSN_lt(map->max_tsn_seen, tsn))
map->max_tsn_seen = tsn; map->max_tsn_seen = tsn;
/* Assert: TSN is in range. */
gap = tsn - map->base_tsn + 1; gap = tsn - map->base_tsn + 1;
/* Mark the TSNs as received. */ map->base_tsn += gap;
if (gap <= map->len) map->cumulative_tsn_ack_point += gap;
memset(map->tsn_map, 0x01, gap); if (gap >= map->len) {
else { /* If our gap is larger then the map size, just
memset(map->tsn_map, 0x01, map->len); * zero out the map.
memset(map->overflow_map, 0x01, (gap - map->len)); */
} bitmap_zero(map->tsn_map, map->len);
} else {
/* Go fixup any internal TSN mapping variables including /* If the gap is smaller then the map size,
* cumulative_tsn_ack_point. * shift the map by 'gap' bits and update further.
*/ */
bitmap_shift_right(map->tsn_map, map->tsn_map, gap, map->len);
sctp_tsnmap_update(map); sctp_tsnmap_update(map);
}
} }
/******************************************************************** /********************************************************************
...@@ -268,27 +244,19 @@ void sctp_tsnmap_skip(struct sctp_tsnmap *map, __u32 tsn) ...@@ -268,27 +244,19 @@ void sctp_tsnmap_skip(struct sctp_tsnmap *map, __u32 tsn)
*/ */
static void sctp_tsnmap_update(struct sctp_tsnmap *map) static void sctp_tsnmap_update(struct sctp_tsnmap *map)
{ {
__u32 ctsn; u16 len;
unsigned long zero_bit;
ctsn = map->cumulative_tsn_ack_point;
do {
ctsn++; len = map->max_tsn_seen - map->cumulative_tsn_ack_point;
if (ctsn == map->overflow_tsn) { zero_bit = find_first_zero_bit(map->tsn_map, len);
/* Now tsn_map must have been all '1's, if (!zero_bit)
* so we swap the map and check the overflow table return; /* The first 0-bit is bit 0. nothing to do */
*/
__u8 *tmp = map->tsn_map;
memset(tmp, 0, map->len);
map->tsn_map = map->overflow_map;
map->overflow_map = tmp;
/* Update the tsn_map boundaries. */
map->base_tsn += map->len;
map->overflow_tsn += map->len;
}
} while (map->tsn_map[ctsn - map->base_tsn]);
map->cumulative_tsn_ack_point = ctsn - 1; /* Back up one. */ map->base_tsn += zero_bit;
map->cumulative_tsn_ack_point += zero_bit;
bitmap_shift_right(map->tsn_map, map->tsn_map, zero_bit, map->len);
} }
/* How many data chunks are we missing from our peer? /* How many data chunks are we missing from our peer?
...@@ -299,31 +267,19 @@ __u16 sctp_tsnmap_pending(struct sctp_tsnmap *map) ...@@ -299,31 +267,19 @@ __u16 sctp_tsnmap_pending(struct sctp_tsnmap *map)
__u32 max_tsn = map->max_tsn_seen; __u32 max_tsn = map->max_tsn_seen;
__u32 base_tsn = map->base_tsn; __u32 base_tsn = map->base_tsn;
__u16 pending_data; __u16 pending_data;
__s32 gap, start, end, i; u32 gap, i;
pending_data = max_tsn - cum_tsn; pending_data = max_tsn - cum_tsn;
gap = max_tsn - base_tsn; gap = max_tsn - base_tsn;
if (gap <= 0 || gap >= (map->len + map->len)) if (gap == 0 || gap >= map->len)
goto out; goto out;
start = ((cum_tsn >= base_tsn) ? (cum_tsn - base_tsn + 1) : 0); for (i = 0; i < gap+1; i++) {
end = ((gap > map->len ) ? map->len : gap + 1); if (test_bit(i, map->tsn_map))
for (i = start; i < end; i++) {
if (map->tsn_map[i])
pending_data--; pending_data--;
} }
if (gap >= map->len) {
start = 0;
end = gap - map->len + 1;
for (i = start; i < end; i++) {
if (map->overflow_map[i])
pending_data--;
}
}
out: out:
return pending_data; return pending_data;
} }
...@@ -334,10 +290,8 @@ __u16 sctp_tsnmap_pending(struct sctp_tsnmap *map) ...@@ -334,10 +290,8 @@ __u16 sctp_tsnmap_pending(struct sctp_tsnmap *map)
* The flags "started" and "ended" tell is if we found the beginning * The flags "started" and "ended" tell is if we found the beginning
* or (respectively) the end of a Gap Ack Block. * or (respectively) the end of a Gap Ack Block.
*/ */
static void sctp_tsnmap_find_gap_ack(__u8 *map, __u16 off, static void sctp_tsnmap_find_gap_ack(unsigned long *map, __u16 off,
__u16 len, __u16 base, __u16 len, __u16 *start, __u16 *end)
int *started, __u16 *start,
int *ended, __u16 *end)
{ {
int i = off; int i = off;
...@@ -348,49 +302,36 @@ static void sctp_tsnmap_find_gap_ack(__u8 *map, __u16 off, ...@@ -348,49 +302,36 @@ static void sctp_tsnmap_find_gap_ack(__u8 *map, __u16 off,
/* Also, stop looking past the maximum TSN seen. */ /* Also, stop looking past the maximum TSN seen. */
/* Look for the start. */ /* Look for the start. */
if (!(*started)) { i = find_next_bit(map, len, off);
for (; i < len; i++) { if (i < len)
if (map[i]) { *start = i;
(*started)++;
*start = base + i;
break;
}
}
}
/* Look for the end. */ /* Look for the end. */
if (*started) { if (*start) {
/* We have found the start, let's find the /* We have found the start, let's find the
* end. If we find the end, break out. * end. If we find the end, break out.
*/ */
for (; i < len; i++) { i = find_next_zero_bit(map, len, i);
if (!map[i]) { if (i < len)
(*ended)++; *end = i - 1;
*end = base + i - 1;
break;
}
}
} }
} }
/* Renege that we have seen a TSN. */ /* Renege that we have seen a TSN. */
void sctp_tsnmap_renege(struct sctp_tsnmap *map, __u32 tsn) void sctp_tsnmap_renege(struct sctp_tsnmap *map, __u32 tsn)
{ {
__s32 gap; u32 gap;
if (TSN_lt(tsn, map->base_tsn)) if (TSN_lt(tsn, map->base_tsn))
return; return;
if (!TSN_lt(tsn, map->base_tsn + map->len + map->len)) /* Assert: TSN is in range. */
if (!TSN_lt(tsn, map->base_tsn + map->len))
return; return;
/* Assert: TSN is in range. */
gap = tsn - map->base_tsn; gap = tsn - map->base_tsn;
/* Pretend we never saw the TSN. */ /* Pretend we never saw the TSN. */
if (gap < map->len) clear_bit(gap, map->tsn_map);
map->tsn_map[gap] = 0;
else
map->overflow_map[gap - map->len] = 0;
} }
/* How many gap ack blocks do we have recorded? */ /* How many gap ack blocks do we have recorded? */
...@@ -416,3 +357,27 @@ __u16 sctp_tsnmap_num_gabs(struct sctp_tsnmap *map) ...@@ -416,3 +357,27 @@ __u16 sctp_tsnmap_num_gabs(struct sctp_tsnmap *map)
} }
return gabs; return gabs;
} }
static int sctp_tsnmap_grow(struct sctp_tsnmap *map, u16 gap)
{
unsigned long *new;
unsigned long inc;
u16 len;
if (gap >= SCTP_TSN_MAP_SIZE)
return 0;
inc = ALIGN((gap - map->len),BITS_PER_LONG) + SCTP_TSN_MAP_INCREMENT;
len = min_t(u16, map->len + inc, SCTP_TSN_MAP_SIZE);
new = kzalloc(len>>3, GFP_ATOMIC);
if (!new)
return 0;
bitmap_copy(new, map->tsn_map, map->max_tsn_seen - map->base_tsn);
kfree(map->tsn_map);
map->tsn_map = new;
map->len = len;
return 1;
}
...@@ -713,7 +713,9 @@ struct sctp_ulpevent *sctp_ulpevent_make_rcvmsg(struct sctp_association *asoc, ...@@ -713,7 +713,9 @@ struct sctp_ulpevent *sctp_ulpevent_make_rcvmsg(struct sctp_association *asoc,
/* Now that all memory allocations for this chunk succeeded, we /* Now that all memory allocations for this chunk succeeded, we
* can mark it as received so the tsn_map is updated correctly. * can mark it as received so the tsn_map is updated correctly.
*/ */
sctp_tsnmap_mark(&asoc->peer.tsn_map, ntohl(chunk->subh.data_hdr->tsn)); if (sctp_tsnmap_mark(&asoc->peer.tsn_map,
ntohl(chunk->subh.data_hdr->tsn)))
goto fail_mark;
/* First calculate the padding, so we don't inadvertently /* First calculate the padding, so we don't inadvertently
* pass up the wrong length to the user. * pass up the wrong length to the user.
...@@ -755,8 +757,12 @@ struct sctp_ulpevent *sctp_ulpevent_make_rcvmsg(struct sctp_association *asoc, ...@@ -755,8 +757,12 @@ struct sctp_ulpevent *sctp_ulpevent_make_rcvmsg(struct sctp_association *asoc,
event->msg_flags |= chunk->chunk_hdr->flags; event->msg_flags |= chunk->chunk_hdr->flags;
event->iif = sctp_chunk_iif(chunk); event->iif = sctp_chunk_iif(chunk);
fail:
return event; return event;
fail_mark:
kfree_skb(skb);
fail:
return NULL;
} }
/* Create a partial delivery related event. /* Create a partial delivery related event.
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment