Adds options DROPPED PACKETS and LOSS INTERVALS to receiver. In this patch is added the mechanism of gathering information about loss intervals and storing it, for later construction of these two options. Changes: - Adds tfrc_loss_data and tfrc_loss_data_entry, structures that register loss intervals info - Adds dccp_skb_is_ecn_ect0 and dccp_skb_is_ecn_ect1 as necessary, so ecn can be verified and used in loss intervals option, that reports ecn nonce sum - Adds tfrc_sp_update_li_data that updates information about loss intervals - Adds tfrc_sp_ld_prepare_data, that fills fields on tfrc_loss_data with current options values - And adds a field of type struct tfrc_loss_data to struct tfrc_hc_rx_sock Signed-off-by: Ivo Calado <ivocalado@xxxxxxxxxxxxxxxxxxxx> Signed-off-by: Erivaldo Xavier <desadoc@xxxxxxxxx> Signed-off-by: Leandro Sales <leandroal@xxxxxxxxx> Index: dccp_tree_work03/net/dccp/ccids/lib/packet_history_sp.c =================================================================== --- dccp_tree_work03.orig/net/dccp/ccids/lib/packet_history_sp.c 2009-10-08 22:59:07.442411383 -0300 +++ dccp_tree_work03/net/dccp/ccids/lib/packet_history_sp.c 2009-10-08 22:59:20.526408512 -0300 @@ -356,11 +356,12 @@ */ bool tfrc_sp_rx_congestion_event(struct tfrc_rx_hist *h, struct tfrc_loss_hist *lh, + struct tfrc_loss_data *ld, struct sk_buff *skb, const u64 ndp, u32 (*first_li)(struct sock *), struct sock *sk) { - bool new_event = false; + bool new_loss = false, new_event = false; if (tfrc_sp_rx_hist_duplicate(h, skb)) return 0; @@ -377,6 +378,7 @@ /* * Update Loss Interval database and recycle RX records */ + new_loss = true; new_event = tfrc_sp_lh_interval_add(lh, h, first_li, sk); __three_after_loss(h); @@ -403,6 +405,11 @@ } /* + * Update Loss Interval data used for options + */ + tfrc_sp_update_li_data(ld, h, skb, new_loss, new_event); + + /* * Update moving-average of `s' and the sum of received payload bytes. */ if (dccp_data_packet(skb)) { Index: dccp_tree_work03/net/dccp/ccids/lib/loss_interval_sp.c =================================================================== --- dccp_tree_work03.orig/net/dccp/ccids/lib/loss_interval_sp.c 2009-10-08 22:59:14.214408089 -0300 +++ dccp_tree_work03/net/dccp/ccids/lib/loss_interval_sp.c 2009-10-08 22:59:20.526408512 -0300 @@ -14,6 +14,7 @@ #include "tfrc_sp.h" static struct kmem_cache *tfrc_lh_slab __read_mostly; +static struct kmem_cache *tfrc_ld_slab __read_mostly; /* Loss Interval weights from [RFC 3448, 5.4], scaled by 10 */ static const int tfrc_lh_weights[NINTERVAL] = { 10, 10, 10, 10, 8, 6, 4, 2 }; @@ -82,6 +83,230 @@ } } +/* + * Allocation routine for new entries of loss interval data + */ +static struct tfrc_loss_data_entry *tfrc_ld_add_new(struct tfrc_loss_data *ld) +{ + struct tfrc_loss_data_entry *new = + kmem_cache_alloc(tfrc_ld_slab, GFP_ATOMIC); + + if (new == NULL) + return NULL; + + memset(new, 0, sizeof(struct tfrc_loss_data_entry)); + + new->next = ld->head; + ld->head = new; + ld->counter++; + + return new; +} + +void tfrc_sp_ld_cleanup(struct tfrc_loss_data *ld) +{ + struct tfrc_loss_data_entry *next, *h = ld->head; + + while (h) { + next = h->next; + kmem_cache_free(tfrc_ld_slab, h); + h = next; + } + + ld->head = NULL; + ld->counter = 0; +} + +/* + * tfrc_sp_ld_prepare_data - updates arrays on tfrc_loss_data + * so they can be sent as options + * @loss_count: current loss count (packets after hole on transmission), + * used to determine skip length for loss intervals option + * @ld: loss intervals data being updated + */ +void tfrc_sp_ld_prepare_data(u8 loss_count, struct tfrc_loss_data *ld) +{ + u8 *li_ofs, *d_ofs; + struct tfrc_loss_data_entry *e; + u16 count; + + li_ofs = &ld->loss_intervals_opts[0]; + d_ofs = &ld->drop_opts[0]; + + count = 0; + e = ld->head; + + *li_ofs = loss_count + 1; + li_ofs++; + + while (e != NULL) { + + if (count < TFRC_LOSS_INTERVALS_OPT_MAX_LENGTH) { + *li_ofs = ((htonl(e->lossless_length) & 0xFFFFFF)<<8); + li_ofs += 3; + *li_ofs = ((e->ecn_nonce_sum&0x1) << 31) | + (htonl((e->loss_length & 0x7FFFFF))<<8); + li_ofs += 3; + *li_ofs = ((htonl(e->data_length) & 0xFFFFFF)<<8); + li_ofs += 3; + } else + break; + + if (count < TFRC_DROP_OPT_MAX_LENGTH) { + *d_ofs = (htonl(e->drop_count) & 0xFFFFFF)<<8; + d_ofs += 3; + } else + break; + + count++; + e = e->next; + } +} + +/* + * tfrc_sp_update_li_data - Update tfrc_loss_data upon + * packet receiving or loss detection + * @ld: tfrc_loss_data being updated + * @rh: loss event record + * @skb: received packet + * @new_loss: dictates if new loss was detected + * upon receiving current packet + * @new_event: ...and if the loss starts new loss interval + */ +void tfrc_sp_update_li_data(struct tfrc_loss_data *ld, + struct tfrc_rx_hist *rh, + struct sk_buff *skb, + bool new_loss, bool new_event) +{ + struct tfrc_loss_data_entry *new, *h; + + if (!dccp_data_packet(skb)) + return; + + if (ld->head == NULL) { + new = tfrc_ld_add_new(ld); + if (unlikely(new == NULL)) { + DCCP_CRIT("Cannot allocate new loss data registry."); + return; + } + + if (new_loss) { + new->drop_count = rh->num_losses; + new->lossless_length = 1; + new->loss_length = rh->num_losses; + + new->data_length = 1; + + if (dccp_skb_is_ecn_ect1(skb)) + new->ecn_nonce_sum = 1; + else + new->ecn_nonce_sum = 0; + } else { + new->drop_count = 0; + new->lossless_length = 1; + new->loss_length = 0; + + new->data_length = 1; + + if (dccp_skb_is_ecn_ect1(skb)) + new->ecn_nonce_sum = 1; + else + new->ecn_nonce_sum = 0; + } + + return; + } + + if (new_event) { + new = tfrc_ld_add_new(ld); + if (unlikely(new == NULL)) { + DCCP_CRIT("Cannot allocate new loss data registry. \ + Cleaning up."); + tfrc_sp_ld_cleanup(ld); + return; + } + + new->drop_count = rh->num_losses; + new->lossless_length = (ld->last_loss_count - rh->loss_count); + new->loss_length = rh->num_losses; + + new->ecn_nonce_sum = 0; + new->data_length = 0; + + while (ld->last_loss_count > rh->loss_count) { + ld->last_loss_count--; + + if (ld->sto_is_data & (1 << (ld->last_loss_count))) { + new->data_length++; + + if (ld->sto_ecn & (1 << (ld->last_loss_count))) + new->ecn_nonce_sum = + !new->ecn_nonce_sum; + } + } + + return; + } + + h = ld->head; + + if (rh->loss_count > ld->last_loss_count) { + ld->last_loss_count = rh->loss_count; + + ld->sto_is_data |= (1 << (ld->last_loss_count - 1)); + + if (dccp_skb_is_ecn_ect1(skb)) + ld->sto_ecn |= (1 << (ld->last_loss_count - 1)); + + return; + } + + if (new_loss) { + h->drop_count += rh->num_losses; + h->lossless_length = (ld->last_loss_count - rh->loss_count); + h->loss_length += h->lossless_length + rh->num_losses; + + h->ecn_nonce_sum = 0; + h->data_length = 0; + + while (ld->last_loss_count > rh->loss_count) { + ld->last_loss_count--; + + if (ld->sto_is_data&(1 << (ld->last_loss_count))) { + h->data_length++; + + if (ld->sto_ecn & (1 << (ld->last_loss_count))) + h->ecn_nonce_sum = !h->ecn_nonce_sum; + } + } + + return; + } + + if (ld->last_loss_count > rh->loss_count) { + while (ld->last_loss_count > rh->loss_count) { + ld->last_loss_count--; + + h->lossless_length++; + + if (ld->sto_is_data & (1 << (ld->last_loss_count))) { + h->data_length++; + + if (ld->sto_ecn & (1 << (ld->last_loss_count))) + h->ecn_nonce_sum = !h->ecn_nonce_sum; + } + } + + return; + } + + h->lossless_length++; + h->data_length++; + + if (dccp_skb_is_ecn_ect1(skb)) + h->ecn_nonce_sum = !h->ecn_nonce_sum; +} + static void tfrc_sp_lh_calc_i_mean(struct tfrc_loss_hist *lh, __u8 curr_ccval) { u32 i_i, i_tot0 = 0, i_tot1 = 0, w_tot = 0; @@ -265,7 +490,24 @@ tfrc_lh_slab = kmem_cache_create("tfrc_sp_li_hist", sizeof(struct tfrc_loss_interval), 0, SLAB_HWCACHE_ALIGN, NULL); - return tfrc_lh_slab == NULL ? -ENOBUFS : 0; + tfrc_ld_slab = kmem_cache_create("tfrc_sp_li_data", + sizeof(struct tfrc_loss_data_entry), 0, + SLAB_HWCACHE_ALIGN, NULL); + + if ((tfrc_lh_slab != NULL) && (tfrc_ld_slab != NULL)) + return 0; + + if (tfrc_lh_slab != NULL) { + kmem_cache_destroy(tfrc_lh_slab); + tfrc_lh_slab = NULL; + } + + if (tfrc_ld_slab != NULL) { + kmem_cache_destroy(tfrc_ld_slab); + tfrc_ld_slab = NULL; + } + + return -ENOBUFS; } void tfrc_sp_li_exit(void) @@ -274,4 +516,9 @@ kmem_cache_destroy(tfrc_lh_slab); tfrc_lh_slab = NULL; } + + if (tfrc_ld_slab != NULL) { + kmem_cache_destroy(tfrc_ld_slab); + tfrc_ld_slab = NULL; + } } Index: dccp_tree_work03/net/dccp/ccids/lib/loss_interval_sp.h =================================================================== --- dccp_tree_work03.orig/net/dccp/ccids/lib/loss_interval_sp.h 2009-10-08 22:59:14.214408089 -0300 +++ dccp_tree_work03/net/dccp/ccids/lib/loss_interval_sp.h 2009-10-08 22:59:20.526408512 -0300 @@ -72,13 +72,77 @@ struct tfrc_rx_hist; #endif +/* + * tfrc_loss_data_entry - Holds info about one loss interval + * @next: next entry on this linked list + * @lossless_length: length of lossless sequence + * @ecn_nonce_sum: ecn nonce sum for this interval + * @loss_length: length of lossy part + * @data_length: data length on lossless part + * @drop_count: count of dopped packets + */ +struct tfrc_loss_data_entry { + struct tfrc_loss_data_entry *next; + u32 lossless_length:24; + u8 ecn_nonce_sum:1; + u32 loss_length:23; + u32 data_length:24; + u32 drop_count:24; +}; + +/* As defined at section 8.6.1. of RFC 4342 */ +#define TFRC_LOSS_INTERVALS_OPT_MAX_LENGTH 28 +/* Specified on section 8.7. of CCID4 draft */ +#define TFRC_DROP_OPT_MAX_LENGTH 84 +#define TFRC_LI_OPT_SZ \ + (2 + TFRC_LOSS_INTERVALS_OPT_MAX_LENGTH*9) +#define TFRC_DROPPED_OPT_SZ \ + (1 + TFRC_DROP_OPT_MAX_LENGTH*3) + +/* + * tfrc_loss_data - loss interval data + * used by loss intervals and dropped packets options + * @head: linked list containing loss interval data + * @counter: number of entries + * @loss_intervals_opts: space necessary for writing temporary option + * data for loss intervals option + * @drop_opts: same for dropped packets option + * @last_loss_count: last loss count (num. of packets + * after hole on transmission) observed + * @sto_ecn: ecn's observed while waiting for hole + * to be filled or accepted as missing + * @sto_is_data: flags about if packets saw were data packets + */ +struct tfrc_loss_data { + struct tfrc_loss_data_entry *head; + u16 counter; + u8 loss_intervals_opts[TFRC_LI_OPT_SZ]; + u8 drop_opts[TFRC_DROPPED_OPT_SZ]; + u8 last_loss_count; + u8 sto_ecn; + u8 sto_is_data; +}; + +static inline void tfrc_ld_init(struct tfrc_loss_data *ld) +{ + memset(ld, 0, sizeof(*ld)); +} + +struct tfrc_rx_hist; + extern bool tfrc_sp_lh_interval_add(struct tfrc_loss_hist *, struct tfrc_rx_hist *, u32 (*first_li)(struct sock *), struct sock *); +extern void tfrc_sp_update_li_data(struct tfrc_loss_data *, + struct tfrc_rx_hist *, + struct sk_buff *, + bool new_loss, bool new_event); extern void tfrc_sp_lh_update_i_mean(struct tfrc_loss_hist *lh, struct sk_buff *); extern void tfrc_sp_lh_cleanup(struct tfrc_loss_hist *lh); +extern void tfrc_sp_ld_cleanup(struct tfrc_loss_data *ld); +extern void tfrc_sp_ld_prepare_data(u8 loss_count, struct tfrc_loss_data *ld); #endif /* _DCCP_LI_HIST_SP_ */ Index: dccp_tree_work03/net/dccp/dccp.h =================================================================== --- dccp_tree_work03.orig/net/dccp/dccp.h 2009-10-08 22:59:07.442411383 -0300 +++ dccp_tree_work03/net/dccp/dccp.h 2009-10-08 22:59:20.526408512 -0300 @@ -401,6 +401,16 @@ return (DCCP_SKB_CB(skb)->dccpd_ecn & INET_ECN_MASK) == INET_ECN_CE; } +static inline bool dccp_skb_is_ecn_ect0(const struct sk_buff *skb) +{ + return (DCCP_SKB_CB(skb)->dccpd_ecn & INET_ECN_MASK) == INET_ECN_ECT_0; +} + +static inline bool dccp_skb_is_ecn_ect1(const struct sk_buff *skb) +{ + return (DCCP_SKB_CB(skb)->dccpd_ecn & INET_ECN_MASK) == INET_ECN_ECT_1; +} + /* RFC 4340, sec. 7.7 */ static inline int dccp_non_data_packet(const struct sk_buff *skb) { Index: dccp_tree_work03/net/dccp/ccids/lib/packet_history_sp.h =================================================================== --- dccp_tree_work03.orig/net/dccp/ccids/lib/packet_history_sp.h 2009-10-08 22:59:07.439908552 -0300 +++ dccp_tree_work03/net/dccp/ccids/lib/packet_history_sp.h 2009-10-08 22:59:20.526408512 -0300 @@ -202,6 +202,7 @@ extern bool tfrc_sp_rx_congestion_event(struct tfrc_rx_hist *h, struct tfrc_loss_hist *lh, + struct tfrc_loss_data *ld, struct sk_buff *skb, const u64 ndp, u32 (*first_li)(struct sock *sk), struct sock *sk); Index: dccp_tree_work03/net/dccp/ccids/lib/tfrc_ccids_sp.h =================================================================== --- dccp_tree_work03.orig/net/dccp/ccids/lib/tfrc_ccids_sp.h 2009-10-08 22:54:16.671408004 -0300 +++ dccp_tree_work03/net/dccp/ccids/lib/tfrc_ccids_sp.h 2009-10-08 22:59:20.526408512 -0300 @@ -133,6 +133,7 @@ * @hist - Packet history (loss detection + RTT sampling) * @li_hist - Loss Interval database * @p_inverse - Inverse of Loss Event Rate (RFC 4342, sec. 8.5) + * @li_data - loss interval data for options */ struct tfrc_hc_rx_sock { u8 last_counter:4; @@ -142,6 +143,7 @@ struct tfrc_rx_hist hist; struct tfrc_loss_hist li_hist; #define p_inverse li_hist.i_mean + struct tfrc_loss_data li_data; }; static inline struct tfrc_hc_rx_sock *tfrc_hc_rx_sk(const struct sock *sk) -- To unsubscribe from this list: send the line "unsubscribe dccp" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html