2005-09-18 00:17:51 -07:00
# ifndef _ACKVEC_H
# define _ACKVEC_H
/*
* net / dccp / ackvec . h
*
* An implementation of the DCCP protocol
* Copyright ( c ) 2005 Arnaldo Carvalho de Melo < acme @ mandriva . com >
*
* 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
* published by the Free Software Foundation .
*/
2008-11-23 16:02:31 -08:00
# include <linux/dccp.h>
2005-09-18 00:17:51 -07:00
# include <linux/compiler.h>
2007-08-19 17:17:25 -07:00
# include <linux/ktime.h>
2006-03-20 17:19:55 -08:00
# include <linux/list.h>
2005-09-18 00:17:51 -07:00
# include <linux/types.h>
2006-11-24 13:02:42 -02:00
/* We can spread an ack vector across multiple options */
2008-11-23 16:02:31 -08:00
# define DCCP_MAX_ACKVEC_LEN (DCCP_SINGLE_OPT_MAXLEN * 2)
2005-09-18 00:17:51 -07:00
# define DCCP_ACKVEC_STATE_RECEIVED 0
# define DCCP_ACKVEC_STATE_ECN_MARKED (1 << 6)
# define DCCP_ACKVEC_STATE_NOT_RECEIVED (3 << 6)
# define DCCP_ACKVEC_STATE_MASK 0xC0 /* 11000000 */
# define DCCP_ACKVEC_LEN_MASK 0x3F /* 00111111 */
/** struct dccp_ackvec - ack vector
*
2006-10-24 16:17:51 -07:00
* This data structure is the one defined in RFC 4340 , Appendix A .
2005-09-18 00:17:51 -07:00
*
2007-12-30 04:19:31 -08:00
* @ av_buf_head - circular buffer head
* @ av_buf_tail - circular buffer tail
* @ 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
2005-09-18 00:17:51 -07:00
* 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 :
*
2007-12-30 04:19:31 -08:00
* @ av_records - list of dccp_ackvec_record
* @ av_ack_nonce - the one - bit sum of the ECN Nonces for all State 0.
2005-09-18 00:17:51 -07:00
*
2007-12-30 04:19:31 -08:00
* @ av_time - the time in usecs
* @ av_buf - circular buffer of acknowledgeable packets
2005-09-18 00:17:51 -07:00
*/
struct dccp_ackvec {
2007-12-30 04:19:31 -08:00
u64 av_buf_ackno ;
struct list_head av_records ;
ktime_t av_time ;
u16 av_buf_head ;
u16 av_vec_len ;
u8 av_buf_nonce ;
u8 av_ack_nonce ;
u8 av_buf [ DCCP_MAX_ACKVEC_LEN ] ;
2005-09-18 00:17:51 -07:00
} ;
2006-03-20 17:19:55 -08:00
/** struct dccp_ackvec_record - ack vector record
*
* ACK vector record as defined in Appendix A of spec .
*
2007-12-30 04:19:31 -08:00
* The list is sorted by avr_ack_seqno
2006-03-20 17:19:55 -08:00
*
2007-12-30 04:19:31 -08:00
* @ avr_node - node in av_records
* @ 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
2006-03-20 17:19:55 -08:00
*/
struct dccp_ackvec_record {
2007-12-30 04:19:31 -08:00
struct list_head avr_node ;
u64 avr_ack_seqno ;
u64 avr_ack_ackno ;
u16 avr_ack_ptr ;
u16 avr_sent_len ;
u8 avr_ack_nonce ;
2006-03-20 17:19:55 -08:00
} ;
2005-09-18 00:17:51 -07:00
struct sock ;
struct sk_buff ;
# ifdef CONFIG_IP_DCCP_ACKVEC
2006-03-20 17:16:17 -08:00
extern int dccp_ackvec_init ( void ) ;
extern void dccp_ackvec_exit ( void ) ;
2006-03-20 17:15:42 -08:00
extern struct dccp_ackvec * dccp_ackvec_alloc ( const gfp_t priority ) ;
2005-09-18 00:17:51 -07:00
extern void dccp_ackvec_free ( struct dccp_ackvec * av ) ;
extern int dccp_ackvec_add ( struct dccp_ackvec * av , const struct sock * sk ,
const u64 ackno , const u8 state ) ;
extern void dccp_ackvec_check_rcv_ackno ( struct dccp_ackvec * av ,
struct sock * sk , const u64 ackno ) ;
extern int dccp_ackvec_parse ( struct sock * sk , const struct sk_buff * skb ,
2006-11-24 13:02:42 -02:00
u64 * ackno , const u8 opt ,
const u8 * value , const u8 len ) ;
2005-09-18 00:17:51 -07:00
extern int dccp_insert_option_ackvec ( struct sock * sk , struct sk_buff * skb ) ;
static inline int dccp_ackvec_pending ( const struct dccp_ackvec * av )
{
2007-12-30 04:19:31 -08:00
return av - > av_vec_len ;
2005-09-18 00:17:51 -07:00
}
# else /* CONFIG_IP_DCCP_ACKVEC */
2006-03-20 17:16:17 -08:00
static inline int dccp_ackvec_init ( void )
{
return 0 ;
}
static inline void dccp_ackvec_exit ( void )
{
}
2006-03-20 17:15:42 -08:00
static inline struct dccp_ackvec * dccp_ackvec_alloc ( const gfp_t priority )
2005-09-18 00:17:51 -07:00
{
return NULL ;
}
static inline void dccp_ackvec_free ( struct dccp_ackvec * av )
{
}
static inline int dccp_ackvec_add ( struct dccp_ackvec * av , const struct sock * sk ,
const u64 ackno , const u8 state )
{
return - 1 ;
}
static inline void dccp_ackvec_check_rcv_ackno ( struct dccp_ackvec * av ,
struct sock * sk , const u64 ackno )
{
}
static inline int dccp_ackvec_parse ( struct sock * sk , const struct sk_buff * skb ,
2006-11-24 13:02:42 -02:00
const u64 * ackno , const u8 opt ,
const u8 * value , const u8 len )
2005-09-18 00:17:51 -07:00
{
return - 1 ;
}
static inline int dccp_insert_option_ackvec ( const struct sock * sk ,
const struct sk_buff * skb )
{
return - 1 ;
}
static inline int dccp_ackvec_pending ( const struct dccp_ackvec * av )
{
return 0 ;
}
# endif /* CONFIG_IP_DCCP_ACKVEC */
# endif /* _ACKVEC_H */