sctp_scope_t, int priority, int flags);
-/*
- * sctp_socket.c
+/*
+ * sctp_socket.c
*/
extern int sctp_backlog_rcv(struct sock *sk, struct sk_buff *skb);
extern int sctp_inet_listen(struct socket *sock, int backlog);
extern int sctp_get_port(struct sock *sk, unsigned short snum);
extern void sctp_write_space(struct sock *sk);
-extern unsigned int sctp_poll(struct file *file, struct socket *sock,
+extern unsigned int sctp_poll(struct file *file, struct socket *sock,
poll_table *wait);
-/*
- * sctp_primitive.c
+/*
+ * sctp_primitive.c
*/
extern int sctp_primitive_ASSOCIATE(sctp_association_t *, void *arg);
extern int sctp_primitive_SHUTDOWN(sctp_association_t *, void *arg);
extern int sctp_primitive_SEND(sctp_association_t *, void *arg);
-/*
+/*
* sctp_crc32c.c
*/
-extern uint32_t count_crc(uint8_t *ptr, uint16_t count);
+extern __u32 count_crc(__u8 *ptr, __u16 count);
-/*
- * sctp_input.c
+/*
+ * sctp_input.c
*/
extern int sctp_rcv(struct sk_buff *skb);
extern void sctp_v4_err(struct sk_buff *skb, u32 info);
extern void sctp_unhash_endpoint(sctp_endpoint_t *);
extern void __sctp_unhash_endpoint(sctp_endpoint_t *);
-/*
+/*
* sctp_hashdriver.c
*/
extern void sctp_hash_digest(const char *secret, const int secret_len,
const char *text, const int text_len,
- uint8_t *digest);
+ __u8 *digest);
/*
* Section: Macros, externs, and inlines
#define SCTP_SOCK_SLEEP_POST(sk) SOCK_SLEEP_POST(sk)
-/* Determine if this is a valid kernel address.
- */
-static inline int
-sctp_is_valid_kaddr(unsigned long addr)
+/* Determine if this is a valid kernel address. */
+static inline int sctp_is_valid_kaddr(unsigned long addr)
{
struct page *page;
/* Make sure the address is not in the user address space. */
- if (addr < PAGE_OFFSET) {
+ if (addr < PAGE_OFFSET)
return 0;
- }
page = virt_to_page(addr);
/* Is this page valid? */
- if (!virt_addr_valid(addr) || PageReserved(page)) {
+ if (!virt_addr_valid(addr) || PageReserved(page))
return 0;
- }
return 1;
-
-} /* sctp_is_valid_kaddr() */
-
+}
#endif /* !TEST_FRAME */
#endif /* SCTP_DEBUG */
-/*
- * Macros for keeping a global reference of object allocations.
+/*
+ * Macros for keeping a global reference of object allocations.
*/
#ifdef CONFIG_SCTP_DBG_OBJCNT
extern atomic_t sctp_dbg_objcnt_bind_addr;
extern atomic_t sctp_dbg_objcnt_addr;
-/* Macros to atomically increment/decrement objcnt counters.
-*/
+/* Macros to atomically increment/decrement objcnt counters. */
#define SCTP_DBG_OBJCNT_INC(name) \
atomic_inc(&sctp_dbg_objcnt_## name)
#define SCTP_DBG_OBJCNT_DEC(name) \
atomic_t sctp_dbg_objcnt_## name = ATOMIC_INIT(0)
/* Macro to help create new entries in in the global array of
- * objcnt counters.
- */
+ * objcnt counters.
+ */
#define SCTP_DBG_OBJCNT_ENTRY(name) \
{.label= #name, .counter= &sctp_dbg_objcnt_## name}
#define SCTP_DBG_OBJCNT_INC(name)
#define SCTP_DBG_OBJCNT_DEC(name)
-static inline void sctp_dbg_objcnt_init(void) { return; }
+static inline void sctp_dbg_objcnt_init(void) { return; }
static inline void sctp_dbg_objcnt_exit(void) { return; }
#endif /* CONFIG_SCTP_DBG_OBJCOUNT */
#if defined CONFIG_SYSCTL
extern void sctp_sysctl_register(void);
-extern void sctp_sysctl_unregister(void);
+extern void sctp_sysctl_unregister(void);
#else
static inline void sctp_sysctl_register(void) { return; }
static inline void sctp_sysctl_unregister(void) { return; }
extern int sctp_v6_init(void);
extern void sctp_v6_exit(void);
-static inline int
-sctp_ipv6_addr_type(const struct in6_addr* addr)
+static inline int sctp_ipv6_addr_type(const struct in6_addr *addr)
{
- return(ipv6_addr_type((struct in6_addr*)addr));
+ return ipv6_addr_type((struct in6_addr*) addr);
}
-#define SCTP_SAT_LEN (sizeof(sctp_paramhdr_t) + 2 * sizeof(uint16_t))
+#define SCTP_SAT_LEN (sizeof(sctp_paramhdr_t) + 2 * sizeof(__u16))
/* Note: These V6 macros are obsolescent. */
/* Use this macro to enclose code fragments which are V6-dependent. */
#define SCTP_V6(m...) m
-#define SCTP_V6_SUPPORT 1
+#define SCTP_V6_SUPPORT 1
#else /* #ifdef defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE) */
#define sctp_ipv6_addr_type(a) 0
-#define SCTP_SAT_LEN (sizeof(sctp_paramhdr_t) + 1 * sizeof(uint16_t))
-#define SCTP_V6(m...) /* Do nothing. */
-#undef SCTP_V6_SUPPORT
+#define SCTP_SAT_LEN (sizeof(sctp_paramhdr_t) + 1 * sizeof(__u16))
+#define SCTP_V6(m...) /* Do nothing. */
+#undef SCTP_V6_SUPPORT
static inline int sctp_v6_init(void) { return 0; }
-static inline void sctp_v6_exit(void) {return; }
+static inline void sctp_v6_exit(void) { return; }
#endif /* #ifdef defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE) */
/* Map an association to an assoc_id. */
-static inline sctp_assoc_t
-sctp_assoc2id(const sctp_association_t *asoc)
+static inline sctp_assoc_t sctp_assoc2id(const sctp_association_t *asoc)
{
- return ((sctp_assoc_t)asoc);
-
-} /* sctp_assoc2id() */
+ return (sctp_assoc_t) asoc;
+}
+/* Look up the association by its id. */
+static inline sctp_association_t *sctp_id2assoc(const struct sock *sk, sctp_assoc_t id)
+{
+ sctp_association_t *asoc = NULL;
-/* Look up the association by its id.
- */
-static inline sctp_association_t *
-sctp_id2assoc(const struct sock *sk, sctp_assoc_t id)
-{
- sctp_association_t *asoc = NULL;
-
/* First, verify that this is a kernel address. */
- if (sctp_is_valid_kaddr((unsigned long)id)) {
- sctp_association_t *temp;
- temp = (sctp_association_t *)id;
-
+ if (sctp_is_valid_kaddr((unsigned long) id)) {
+ sctp_association_t *temp = (sctp_association_t *) id;
+
/* Verify that this _is_ an sctp_association_t
* data structure and if so, that the socket matches.
*/
- if ((SCTP_ASSOC_EYECATCHER == temp->eyecatcher)
- && (temp->base.sk == sk)) {
+ if ((SCTP_ASSOC_EYECATCHER == temp->eyecatcher) &&
+ (temp->base.sk == sk))
asoc = temp;
- }
}
-
- return(asoc);
-} /* sctp_id2assoc() */
+ return asoc;
+}
/* A macro to walk a list of skbs. */
#define sctp_skb_for_each(pos, head, tmp) \
/* A helper to append an entire skb list (list) to another (head). */
-static inline void
-sctp_skb_list_tail(struct sk_buff_head *list,
- struct sk_buff_head *head)
+static inline void sctp_skb_list_tail(struct sk_buff_head *list,
+ struct sk_buff_head *head)
{
int flags __attribute__ ((unused));
- sctp_spin_lock_irqsave(&head->lock, flags);
+ sctp_spin_lock_irqsave(&head->lock, flags);
sctp_spin_lock(&list->lock);
list_splice((struct list_head *)list, (struct list_head *)head->prev);
-
+
head->qlen += list->qlen;
list->qlen = 0;
sctp_spin_unlock(&list->lock);
sctp_spin_unlock_irqrestore(&head->lock, flags);
-
-} /* sctp_skb_list_tail() */
+}
/**
* sctp_list_dequeue - remove from the head of the queue
* returned or %NULL if the list is empty.
*/
-static inline struct list_head *
-sctp_list_dequeue(struct list_head *list)
-{
- struct list_head *result;
-
- result = (void *) 0;
- if ( list->next != list ) {
- result = list->next;
- list->next = result->next;
- list->next->prev = list;
- INIT_LIST_HEAD(result);
- }
- return result;
-
-} /* sctp_list_dequeue() */
+static inline struct list_head *sctp_list_dequeue(struct list_head *list)
+{
+ struct list_head *result = NULL;
+ if (list->next != list) {
+ result = list->next;
+ list->next = result->next;
+ list->next->prev = list;
+ INIT_LIST_HEAD(result);
+ }
+ return result;
+}
/* Calculate the size (in bytes) occupied by the data of an iovec. */
-static inline size_t
-get_user_iov_size(struct iovec *iov, int iovlen) {
- size_t retval;
-
- retval = 0;
-
- for (; iovlen > 0; --iovlen) {
- retval += iov->iov_len;
- iov++;
- }
+static inline size_t get_user_iov_size(struct iovec *iov, int iovlen)
+{
+ size_t retval = 0;
- return(retval);
+ for (; iovlen > 0; --iovlen) {
+ retval += iov->iov_len;
+ iov++;
+ }
-} /* get_user_iov_size() */
+ return retval;
+}
/* Round an int up to the next multiple of 4. */
/* Stolen from net/profile.h. Using it from there is more grief than
* it is worth.
*/
-static inline void
-tv_add(const struct timeval *entered, struct timeval *leaved)
+static inline void tv_add(const struct timeval *entered, struct timeval *leaved)
{
time_t usecs = leaved->tv_usec + entered->tv_usec;
time_t secs = leaved->tv_sec + entered->tv_sec;
}
leaved->tv_sec = secs;
leaved->tv_usec = usecs;
-
-} /* tv_add() */
+}
/* External references. */
/* Return the SCTP protocol structure. */
static inline sctp_protocol_t *sctp_get_protocol(void)
{
- return(&sctp_proto);
-
-} /* sctp_get_protocol() */
-
+ return &sctp_proto;
+}
/* Warning: The following hash functions assume a power of two 'size'. */
/* This is the hash function for the SCTP port hash table. */
-static inline int sctp_phashfn(uint16_t lport)
+static inline int sctp_phashfn(__u16 lport)
{
sctp_protocol_t *sctp_proto = sctp_get_protocol();
- return (lport & (sctp_proto->port_hashsize - 1));
-
-} /* sctp_phashfn() */
+ return (lport & (sctp_proto->port_hashsize - 1));
+}
/* This is the hash function for the endpoint hash table. */
-static inline int sctp_ep_hashfn(uint16_t lport)
+static inline int sctp_ep_hashfn(__u16 lport)
{
sctp_protocol_t *sctp_proto = sctp_get_protocol();
- return (lport & (sctp_proto->ep_hashsize - 1));
-
-} /* sctp_ep_hashfn() */
+ return (lport & (sctp_proto->ep_hashsize - 1));
+}
/* This is the hash function for the association hash table. */
-static inline int sctp_assoc_hashfn(uint16_t lport, uint16_t rport)
+static inline int sctp_assoc_hashfn(__u16 lport, __u16 rport)
{
sctp_protocol_t *sctp_proto = sctp_get_protocol();
int h = (lport << 16) + rport;
h ^= h>>8;
- return (h & (sctp_proto->assoc_hashsize-1));
-
-} /* sctp_ep_hashfn() */
+ return (h & (sctp_proto->assoc_hashsize - 1));
+}
-/* This is the hash function for the association hash table. This is
+/* This is the hash function for the association hash table. This is
* not used yet, but could be used as a better hash function when
- * we have a vtag.
-*/
-static inline int sctp_vtag_hashfn(uint16_t lport, uint16_t rport,
- uint32_t vtag)
+ * we have a vtag.
+ */
+static inline int sctp_vtag_hashfn(__u16 lport, __u16 rport, __u32 vtag)
{
sctp_protocol_t *sctp_proto = sctp_get_protocol();
int h = (lport << 16) + rport;
h ^= vtag;
return (h & (sctp_proto->assoc_hashsize-1));
-
-} /* sctp_vtag_hashfn() */
+}
/* WARNING: Do not change the layout of the members in sctp_sock! */
struct sctp_sock {
SCTP_CMD_SET_BIND_ADDR, /* Set the association bind_addr. */
SCTP_CMD_STRIKE, /* Mark a strike against a transport. */
SCTP_CMD_TRANSMIT, /* Transmit the outqueue. */
- SCTP_CMD_HB_TIMERS_START, /* Start the heartbeat timers. */
+ SCTP_CMD_HB_TIMERS_START, /* Start the heartbeat timers. */
SCTP_CMD_TRANSPORT_RESET, /* Reset the status of a transport. */
- SCTP_CMD_TRANSPORT_ON, /* Mark the transport as active. */
+ SCTP_CMD_TRANSPORT_ON, /* Mark the transport as active. */
SCTP_CMD_REPORT_ERROR, /* Pass this error back out of the sm. */
SCTP_CMD_REPORT_BAD_TAG, /* Verification tags didn't match. */
SCTP_CMD_PROCESS_CTSN, /* Sideeffect from shutdown. */
- SCTP_CMD_ASSOC_FAILED, /* Handle association failure. */
+ SCTP_CMD_ASSOC_FAILED, /* Handle association failure. */
SCTP_CMD_DISCARD_PACKET, /* Discard the whole packet. */
- SCTP_CMD_GEN_SHUTDOWN, /* Generate a SHUTDOWN chunk. */
+ SCTP_CMD_GEN_SHUTDOWN, /* Generate a SHUTDOWN chunk. */
SCTP_CMD_UPDATE_ASSOC, /* Update association information. */
SCTP_CMD_PURGE_OUTQUEUE, /* Purge all data waiting to be sent. */
SCTP_CMD_SETUP_T2, /* Hi-level, setup T2-shutdown parms. */
SCTP_CMD_LAST
-} sctp_verb_t; /* enum */
+} sctp_verb_t;
#define SCTP_CMD_MAX (SCTP_CMD_LAST - 1)
#define SCTP_CMD_NUM_VERBS (SCTP_CMD_MAX + 1)
#define SCTP_MAX_NUM_COMMANDS 14
typedef union {
- int32_t i32;
- uint32_t u32;
- uint16_t u16;
- uint8_t u8;
+ __s32 i32;
+ __u32 u32;
+ __u16 u16;
+ __u8 u8;
int error;
sctp_state_t state;
sctp_event_timeout_t to;
} sctp_arg_t;
/* We are simulating ML type constructors here.
- *
+ *
* SCTP_ARG_CONSTRUCTOR(NAME, TYPE, ELT) builds a function called
* SCTP_NAME() which takes an argument of type TYPE and returns an
* sctp_arg_t. It does this by inserting the sole argument into the
* ELT union element of a local sctp_arg_t.
*
- * E.g., SCTP_ARG_CONSTRUCTOR(I32, int32_t, i32) builds SCTP_I32(arg),
- * which takes an int32_t and returns a sctp_arg_t containing the
- * int32_t. So, after foo = SCTP_I32(arg), foo.i32 == arg.
+ * E.g., SCTP_ARG_CONSTRUCTOR(I32, __s32, i32) builds SCTP_I32(arg),
+ * which takes an __s32 and returns a sctp_arg_t containing the
+ * __s32. So, after foo = SCTP_I32(arg), foo.i32 == arg.
*/
-static inline sctp_arg_t
-SCTP_NULL(void)
+static inline sctp_arg_t SCTP_NULL(void)
{
- sctp_arg_t retval; retval.ptr = NULL; return(retval);
+ sctp_arg_t retval; retval.ptr = NULL; return retval;
}
-static inline sctp_arg_t
-SCTP_NOFORCE(void)
+static inline sctp_arg_t SCTP_NOFORCE(void)
{
- sctp_arg_t retval; retval.i32 = 0; return(retval);
+ sctp_arg_t retval; retval.i32 = 0; return retval;
}
-static inline sctp_arg_t
-SCTP_FORCE(void)
+static inline sctp_arg_t SCTP_FORCE(void)
{
- sctp_arg_t retval; retval.i32 = 1; return(retval);
+ sctp_arg_t retval; retval.i32 = 1; return retval;
}
#define SCTP_ARG_CONSTRUCTOR(name, type, elt) \
static inline sctp_arg_t \
SCTP_## name (type arg) \
-{ sctp_arg_t retval; retval.elt = arg; return(retval); }
+{ sctp_arg_t retval; retval.elt = arg; return retval; }
-SCTP_ARG_CONSTRUCTOR(I32, int32_t, i32)
-SCTP_ARG_CONSTRUCTOR(U32, int32_t, u32)
-SCTP_ARG_CONSTRUCTOR(U16, int32_t, u16)
-SCTP_ARG_CONSTRUCTOR(U8, int32_t, u8)
+SCTP_ARG_CONSTRUCTOR(I32, __s32, i32)
+SCTP_ARG_CONSTRUCTOR(U32, __u32, u32)
+SCTP_ARG_CONSTRUCTOR(U16, __u16, u16)
+SCTP_ARG_CONSTRUCTOR(U8, __u8, u8)
SCTP_ARG_CONSTRUCTOR(ERROR, int, error)
SCTP_ARG_CONSTRUCTOR(STATE, sctp_state_t, state)
SCTP_ARG_CONSTRUCTOR(COUNTER, sctp_counter_t, counter)
typedef struct {
sctp_cmd_t cmds[SCTP_MAX_NUM_COMMANDS];
- uint8_t next_free_slot;
- uint8_t next_cmd;
+ __u8 next_free_slot;
+ __u8 next_cmd;
} sctp_cmd_seq_t;
-/* Create a new sctp_command_sequence.
+/* Create a new sctp_command_sequence.
* Return NULL if creating a new sequence fails.
*/
sctp_cmd_seq_t *sctp_new_cmd_seq(int priority);
/* What a hack! Jiminy Cricket! */
enum { SCTP_MAX_STREAM = 10 };
-/* Define the amount of space to reserve for SCTP, IP, LL.
- * There is a little bit of waste that we are always allocating
+/* Define the amount of space to reserve for SCTP, IP, LL.
+ * There is a little bit of waste that we are always allocating
* for ipv6 headers, but this seems worth the simplicity.
*/
+ sizeof(struct ipv6hdr)\
+ MAX_HEADER))
-/* Define the amount of space to reserve for SCTP, IP, LL.
- * There is a little bit of waste that we are always allocating
+/* Define the amount of space to reserve for SCTP, IP, LL.
+ * There is a little bit of waste that we are always allocating
* for ipv6 headers, but this seems worth the simplicity.
*/
/* These are the different flavours of event. */
typedef enum {
- SCTP_EVENT_T_CHUNK = 1,
- SCTP_EVENT_T_TIMEOUT,
- SCTP_EVENT_T_OTHER,
- SCTP_EVENT_T_PRIMITIVE
+ SCTP_EVENT_T_CHUNK = 1,
+ SCTP_EVENT_T_TIMEOUT,
+ SCTP_EVENT_T_OTHER,
+ SCTP_EVENT_T_PRIMITIVE
} sctp_event_t;
typedef enum {
- SCTP_EVENT_NO_PENDING_TSN = 0,
- SCTP_EVENT_ICMP_UNREACHFRAG,
+ SCTP_EVENT_NO_PENDING_TSN = 0,
+ SCTP_EVENT_ICMP_UNREACHFRAG,
} sctp_event_other_t;
/* These are primitive requests from the ULP. */
typedef enum {
- SCTP_PRIMITIVE_INITIALIZE = 0,
- SCTP_PRIMITIVE_ASSOCIATE,
- SCTP_PRIMITIVE_SHUTDOWN,
- SCTP_PRIMITIVE_ABORT,
- SCTP_PRIMITIVE_SEND,
- SCTP_PRIMITIVE_SETPRIMARY,
- SCTP_PRIMITIVE_RECEIVE,
- SCTP_PRIMITIVE_STATUS,
- SCTP_PRIMITIVE_CHANGEHEARTBEAT,
- SCTP_PRIMITIVE_REQUESTHEARTBEAT,
- SCTP_PRIMITIVE_GETSRTTREPORT,
- SCTP_PRIMITIVE_SETFAILURETHRESHOLD,
- SCTP_PRIMITIVE_SETPROTOPARAMETERS,
- SCTP_PRIMITIVE_RECEIVE_UNSENT,
- SCTP_PRIMITIVE_RECEIVE_UNACKED,
- SCTP_PRIMITIVE_DESTROY,
+ SCTP_PRIMITIVE_INITIALIZE = 0,
+ SCTP_PRIMITIVE_ASSOCIATE,
+ SCTP_PRIMITIVE_SHUTDOWN,
+ SCTP_PRIMITIVE_ABORT,
+ SCTP_PRIMITIVE_SEND,
+ SCTP_PRIMITIVE_SETPRIMARY,
+ SCTP_PRIMITIVE_RECEIVE,
+ SCTP_PRIMITIVE_STATUS,
+ SCTP_PRIMITIVE_CHANGEHEARTBEAT,
+ SCTP_PRIMITIVE_REQUESTHEARTBEAT,
+ SCTP_PRIMITIVE_GETSRTTREPORT,
+ SCTP_PRIMITIVE_SETFAILURETHRESHOLD,
+ SCTP_PRIMITIVE_SETPROTOPARAMETERS,
+ SCTP_PRIMITIVE_RECEIVE_UNSENT,
+ SCTP_PRIMITIVE_RECEIVE_UNACKED,
+ SCTP_PRIMITIVE_DESTROY,
} sctp_event_primitive_t;
*/
typedef union {
-
+
sctp_cid_t chunk;
sctp_event_timeout_t timeout;
sctp_event_other_t other;
#define SCTP_SUBTYPE_CONSTRUCTOR(_name, _type, _elt) \
static inline sctp_subtype_t \
SCTP_ST_## _name (_type _arg) \
-{ sctp_subtype_t _retval; _retval._elt = _arg; return(_retval); }
+{ sctp_subtype_t _retval; _retval._elt = _arg; return _retval; }
SCTP_SUBTYPE_CONSTRUCTOR(CHUNK, sctp_cid_t, chunk)
SCTP_SUBTYPE_CONSTRUCTOR(TIMEOUT, sctp_event_timeout_t, timeout)
/* Internal error codes */
typedef enum {
- SCTP_IERROR_NO_ERROR = 0,
- SCTP_IERROR_BASE = 1000,
- SCTP_IERROR_NO_COOKIE,
- SCTP_IERROR_BAD_SIG,
- SCTP_IERROR_STALE_COOKIE,
- SCTP_IERROR_NOMEM,
- SCTP_IERROR_MALFORMED,
- SCTP_IERROR_BAD_TAG,
- SCTP_IERROR_BIG_GAP,
- SCTP_IERROR_DUP_TSN,
+ SCTP_IERROR_NO_ERROR = 0,
+ SCTP_IERROR_BASE = 1000,
+ SCTP_IERROR_NO_COOKIE,
+ SCTP_IERROR_BAD_SIG,
+ SCTP_IERROR_STALE_COOKIE,
+ SCTP_IERROR_NOMEM,
+ SCTP_IERROR_MALFORMED,
+ SCTP_IERROR_BAD_TAG,
+ SCTP_IERROR_BIG_GAP,
+ SCTP_IERROR_DUP_TSN,
-} sctp_ierror_t; /* enum */
+} sctp_ierror_t;
SCTP_STATE_SHUTDOWN_RECEIVED = 7,
SCTP_STATE_SHUTDOWN_ACK_SENT = 8,
-} sctp_state_t; /* enum */
+} sctp_state_t;
#define SCTP_STATE_MAX SCTP_STATE_SHUTDOWN_ACK_SENT
#define SCTP_STATE_NUM_STATES (SCTP_STATE_MAX + 1)
const char *sctp_tname(const sctp_subtype_t); /* timeouts */
const char *sctp_pname(const sctp_subtype_t); /* primitives */
-/* This is a table of printable names of sctp_state_t's. */
+/* This is a table of printable names of sctp_state_t's. */
extern const char *sctp_state_tbl[], *sctp_evttype_tbl[], *sctp_status_tbl[];
/* SCTP reachability state for each address */
* NEVER make this more than 32767 (2^15-1). The Gap Ack Blocks in a
* SACK (see section 3.3.4) are only 16 bits, so 2*SCTP_TSN_MAP_SIZE
* 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_MAX_GAP 65535
/* We will not record more than this many duplicate TSNs between two
* SACKs. The minimum PMTU is 576. Remove all the headers and there
* is enough room for 131 duplicate reports. Round down to the
- * nearest power of 2.
+ * nearest power of 2.
*/
#define SCTP_MAX_DUP_TSNS 128
typedef enum {
SCTP_COUNTER_INIT_ERROR,
-} sctp_counter_t;
+} sctp_counter_t;
/* How many counters does an association need? */
#define SCTP_NUMBER_COUNTERS 5
-/* Here we define the default timers.
- */
+/* Here we define the default timers. */
/* cookie timer def = ? seconds */
#define SCTP_DEFAULT_TIMEOUT_T1_COOKIE (3 * HZ)
*/
#define SCTP_DEFAULT_MINSEGMENT 512 /* MTU size ... if no mtu disc */
#define SCTP_HOW_MANY_SECRETS 2 /* How many secrets I keep */
-#define SCTP_HOW_LONG_COOKIE_LIVE 3600 /* How many seconds the current
+#define SCTP_HOW_LONG_COOKIE_LIVE 3600 /* How many seconds the current
* secret will live?
*/
#define SCTP_SECRET_SIZE 32 /* Number of octets in a 256 bits. */
* routines which form the lower interface to SCTP_outqueue.
*/
typedef enum {
- SCTP_XMIT_OK,
- SCTP_XMIT_PMTU_FULL,
- SCTP_XMIT_RWND_FULL,
- SCTP_XMIT_MUST_FRAG,
+ SCTP_XMIT_OK,
+ SCTP_XMIT_PMTU_FULL,
+ SCTP_XMIT_RWND_FULL,
+ SCTP_XMIT_MUST_FRAG,
} sctp_xmit_t;
/* These are the commands for manipulating transports. */
} sctp_transport_cmd_t;
/* These are the address scopes defined mainly for IPv4 addresses
- * based on draft of SCTP IPv4 scoping <draft-stewart-tsvwg-sctp-ipv4-00.txt>.
- * These scopes are hopefully generic enough to be used on scoping both
- * IPv4 and IPv6 addresses in SCTP.
+ * based on draft of SCTP IPv4 scoping <draft-stewart-tsvwg-sctp-ipv4-00.txt>.
+ * These scopes are hopefully generic enough to be used on scoping both
+ * IPv4 and IPv6 addresses in SCTP.
* At this point, the IPv6 scopes will be mapped to these internal scopes
* as much as possible.
*/
SCTP_SCOPE_UNUSABLE, /* IPv4 unusable addresses */
} sctp_scope_t;
-/* Based on IPv4 scoping <draft-stewart-tsvwg-sctp-ipv4-00.txt>,
- * SCTP IPv4 unusable addresses: 0.0.0.0/8, 224.0.0.0/4, 198.18.0.0/24,
+/* Based on IPv4 scoping <draft-stewart-tsvwg-sctp-ipv4-00.txt>,
+ * SCTP IPv4 unusable addresses: 0.0.0.0/8, 224.0.0.0/4, 198.18.0.0/24,
* 192.88.99.0/24.
- * Also, RFC 8.4, non-unicast addresses are not considered valid SCTP
+ * Also, RFC 8.4, non-unicast addresses are not considered valid SCTP
* addresses.
*/
#define IS_IPV4_UNUSABLE_ADDRESS(a) \
/* Flags used for the bind address copy functions. */
#define SCTP_ADDR6_ALLOWED 0x00000001 /* IPv6 address is allowed by
- local sock family */
+ local sock family */
#define SCTP_ADDR4_PEERSUPP 0x00000002 /* IPv4 address is supported by
peer */
#define SCTP_ADDR6_PEERSUPP 0x00000004 /* IPv6 address is supported by
#ifndef __SLA1_h__
#define __SLA1_h__
-#ifdef __cplusplus
-extern "C" {
-#endif
+struct SLA_1_Context {
+ unsigned int A;
+ unsigned int B;
+ unsigned int C;
+ unsigned int D;
+ unsigned int E;
+ unsigned int H0;
+ unsigned int H1;
+ unsigned int H2;
+ unsigned int H3;
+ unsigned int H4;
+ unsigned int words[80];
+ unsigned int TEMP;
+
+ /* block I am collecting to process */
+ char SLAblock[64];
-struct SLA_1_Context{
- unsigned int A;
- unsigned int B;
- unsigned int C;
- unsigned int D;
- unsigned int E;
- unsigned int H0;
- unsigned int H1;
- unsigned int H2;
- unsigned int H3;
- unsigned int H4;
- unsigned int words[80];
- unsigned int TEMP;
- /* block I am collecting to process */
- char SLAblock[64];
- /* collected so far */
- int howManyInBlock;
- unsigned int runningTotal;
+ /* collected so far */
+ int howManyInBlock;
+ unsigned int runningTotal;
};
#define F3(B,C,D) ((B & C) | (B & D) | (C & D)) /* 40 <= t <= 59 */
#define F4(B,C,D) (B ^ C ^ D) /*600 <= t <= 79 */
/* circular shift */
-
+
#define CSHIFT(A,B) ((B << A) | (B >> (32-A)))
#define K1 0x5a827999 /* 0 <= t <= 19 */
extern void SLA1_Process(struct SLA_1_Context *, const unsigned char *, int);
extern void SLA1_Final(struct SLA_1_Context *, unsigned char *);
-
-#ifdef __cplusplus
-}
-#endif
-
#endif
*/
-#include <linux/config.h>
#include <linux/types.h>
#include <linux/compiler.h>
#include <linux/sched.h>
sctp_cmd_seq_t *);
typedef void (sctp_timer_event_t) (unsigned long);
typedef struct {
- sctp_state_fn_t *fn;
- char *name;
+ sctp_state_fn_t *fn;
+ char *name;
} sctp_sm_table_entry_t;
-/* A naming convention of "sctp_sf_xxx" applies to all the state functions
+/* A naming convention of "sctp_sf_xxx" applies to all the state functions
* currently in use.
*/
sctp_state_fn_t sctp_sf_do_5_1B_init;
sctp_state_fn_t sctp_sf_do_5_1C_ack;
sctp_state_fn_t sctp_sf_do_5_1D_ce;
-sctp_state_fn_t sctp_sf_do_5_1E_ca ;
+sctp_state_fn_t sctp_sf_do_5_1E_ca;
sctp_state_fn_t sctp_sf_do_4_C;
sctp_state_fn_t sctp_sf_eat_data_6_2;
sctp_state_fn_t sctp_sf_eat_data_fast_4_4;
sctp_state_fn_t sctp_sf_autoclose_timer_expire;
-/* These are state functions which are either obsolete or not in use yet.
+/* These are state functions which are either obsolete or not in use yet.
* If any of these functions needs to be revived, it should be renamed with
* the "sctp_sf_xxx" prefix, and be moved to the above prototype groups.
- */
+ */
/* Prototypes for chunk state functions. Not in use. */
sctp_state_fn_t sctp_sf_do_5_2_6_stale;
sctp_state_fn_t sctp_addip_do_asconf_ack;
/* Prototypes for utility support functions. */
-uint8_t sctp_get_chunk_type(sctp_chunk_t *chunk);
+__u8 sctp_get_chunk_type(sctp_chunk_t *chunk);
sctp_sm_table_entry_t *sctp_sm_lookup_event(sctp_event_t event_type,
sctp_state_t state,
sctp_subtype_t event_subtype);
sctp_association_t *sctp_make_temp_asoc(const sctp_endpoint_t *,
sctp_chunk_t *,
const int priority);
-uint32_t sctp_generate_verification_tag(void);
+__u32 sctp_generate_verification_tag(void);
sctpParam_t sctp_get_my_addrs_raw(const sctp_association_t *,
const int priority, int *addrs_len);
-void sctp_populate_tie_tags(uint8_t *cookie, uint32_t curTag, uint32_t hisTag);
-
-
+void sctp_populate_tie_tags(__u8 *cookie, __u32 curTag, __u32 hisTag);
/* Prototypes for chunk-building functions. */
sctp_chunk_t *sctp_make_init(const sctp_association_t *,
sctp_chunk_t *sctp_make_cookie_ack(const sctp_association_t *,
const sctp_chunk_t *);
sctp_chunk_t *sctp_make_cwr(const sctp_association_t *,
- const uint32_t lowest_tsn,
+ const __u32 lowest_tsn,
const sctp_chunk_t *);
sctp_chunk_t *sctp_make_datafrag(sctp_association_t *,
const struct sctp_sndrcvinfo *sinfo,
- int len, const uint8_t *data,
- uint8_t flags, uint16_t ssn);
+ int len, const __u8 *data,
+ __u8 flags, __u16 ssn);
sctp_chunk_t * sctp_make_datafrag_empty(sctp_association_t *,
const struct sctp_sndrcvinfo *sinfo,
- int len, const uint8_t flags,
- uint16_t ssn);
+ int len, const __u8 flags,
+ __u16 ssn);
sctp_chunk_t *sctp_make_data(sctp_association_t *,
const struct sctp_sndrcvinfo *sinfo,
- int len, const uint8_t *data);
+ int len, const __u8 *data);
sctp_chunk_t *sctp_make_data_empty(sctp_association_t *,
const struct sctp_sndrcvinfo *, int len);
sctp_chunk_t *sctp_make_ecne(const sctp_association_t *,
- const uint32_t);
+ const __u32);
sctp_chunk_t *sctp_make_sack(const sctp_association_t *);
sctp_chunk_t *sctp_make_shutdown(const sctp_association_t *asoc);
sctp_chunk_t *sctp_make_shutdown_ack(const sctp_association_t *asoc,
const sctp_chunk_t *);
sctp_chunk_t *sctp_make_shutdown_complete(const sctp_association_t *,
const sctp_chunk_t *);
-void sctp_init_cause(sctp_chunk_t *, uint16_t cause, const void *, size_t);
+void sctp_init_cause(sctp_chunk_t *, __u16 cause, const void *, size_t);
sctp_chunk_t *sctp_make_abort(const sctp_association_t *,
const sctp_chunk_t *,
const size_t hint);
sctp_chunk_t *sctp_make_abort_no_data(const sctp_association_t *,
const sctp_chunk_t *,
- uint32_t tsn);
+ __u32 tsn);
sctp_chunk_t *sctp_make_heartbeat(const sctp_association_t *,
const sctp_transport_t *,
const void *payload,
const size_t paylen);
sctp_chunk_t *sctp_make_op_error(const sctp_association_t *,
const sctp_chunk_t *chunk,
- uint16_t cause_code,
+ __u16 cause_code,
const void *payload,
size_t paylen);
void sctp_chunk_assign_tsn(sctp_chunk_t *);
sctp_cookie_param_t *
sctp_pack_cookie(const sctp_endpoint_t *, const sctp_association_t *,
const sctp_chunk_t *, int *cookie_len,
- const uint8_t *, int addrs_len);
-sctp_association_t *sctp_unpack_cookie(const sctp_endpoint_t *,
+ const __u8 *, int addrs_len);
+sctp_association_t *sctp_unpack_cookie(const sctp_endpoint_t *,
const sctp_association_t *,
sctp_chunk_t *, int priority, int *err);
int sctp_addip_addr_config(sctp_association_t *, sctp_param_t,
struct sockaddr_storage*, int);
/* 3rd level prototypes */
-uint32_t sctp_generate_tag(const sctp_endpoint_t *);
-uint32_t sctp_generate_tsn(const sctp_endpoint_t *);
+__u32 sctp_generate_tag(const sctp_endpoint_t *);
+__u32 sctp_generate_tsn(const sctp_endpoint_t *);
/* 4th level prototypes */
void sctp_param2sockaddr(sockaddr_storage_t *addr, const sctpParam_t param,
- uint16_t port);
+ __u16 port);
int sctp_addr2sockaddr(const sctpParam_t, sockaddr_storage_t *);
int sockaddr2sctp_addr(const sockaddr_storage_t *, sctpParam_t);
/* Get the size of a DATA chunk payload. */
-static inline uint16_t
-sctp_data_size(sctp_chunk_t *chunk)
+static inline __u16 sctp_data_size(sctp_chunk_t *chunk)
{
- uint16_t size;
+ __u16 size;
size = ntohs(chunk->chunk_hdr->length);
size -= sizeof(sctp_data_chunk_t);
- return(size);
-
-} /* sctp_data_size( ) */
+ return size;
+}
/* Compare two TSNs */
* s2, and
*
* (i1 < i2 and i2 - i1 > 2^(SERIAL_BITS - 1)) or
- * (i1 > i2 and i1 - i2 < 2^(SERIAL_BITS - 1))
+ * (i1 > i2 and i1 - i2 < 2^(SERIAL_BITS - 1))
*/
/*
* RFC 2960
- * 1.6 Serial Number Arithmetic
+ * 1.6 Serial Number Arithmetic
*
* Comparisons and arithmetic on TSNs in this document SHOULD use Serial
* Number Arithmetic as defined in [RFC1982] where SERIAL_BITS = 32.
TSN_SIGN_BIT = (1<<31)
};
-static inline int
-TSN_lt(__u32 s, __u32 t)
+static inline int TSN_lt(__u32 s, __u32 t)
{
return (((s) - (t)) & TSN_SIGN_BIT);
}
-static inline int
-TSN_lte(__u32 s, __u32 t)
+static inline int TSN_lte(__u32 s, __u32 t)
{
return (((s) == (t)) || (((s) - (t)) & TSN_SIGN_BIT));
}
/*
* RFC 2960
- * 1.6 Serial Number Arithmetic
+ * 1.6 Serial Number Arithmetic
*
- * Comparisons and arithmetic on Stream Sequence Numbers in this document
- * SHOULD use Serial Number Arithmetic as defined in [RFC1982] where
+ * Comparisons and arithmetic on Stream Sequence Numbers in this document
+ * SHOULD use Serial Number Arithmetic as defined in [RFC1982] where
* SERIAL_BITS = 16.
*/
enum {
SSN_SIGN_BIT = (1<<15)
};
-static inline int
-SSN_lt(__u16 s, __u16 t)
+static inline int SSN_lt(__u16 s, __u16 t)
{
return (((s) - (t)) & SSN_SIGN_BIT);
}
-static inline int
-SSN_lte(__u16 s, __u16 t)
+static inline int SSN_lte(__u16 s, __u16 t)
{
return (((s) == (t)) || (((s) - (t)) & SSN_SIGN_BIT));
}
/* Run sctp_add_cmd() generating a BUG() if there is a failure. */
-static inline void
-sctp_add_cmd_sf(sctp_cmd_seq_t *seq, sctp_verb_t verb, sctp_arg_t obj)
+static inline void sctp_add_cmd_sf(sctp_cmd_seq_t *seq, sctp_verb_t verb, sctp_arg_t obj)
{
- if (unlikely(!sctp_add_cmd(seq, verb, obj))) {
+ if (unlikely(!sctp_add_cmd(seq, verb, obj)))
BUG();
- }
-
-} /* sctp_add_cmd_sf() */
-
+}
#endif /* __sctp_sm_h__ */
#include <linux/sctp.h> /* We need sctp* header structs. */
/*
- * This is (almost) a direct quote from RFC 2553.
+ * This is (almost) a direct quote from RFC 2553.
*/
/*
* Desired design of maximum size and alignment
*/
#define _SS_MAXSIZE 128 /* Implementation specific max size */
-#define _SS_ALIGNSIZE (sizeof (int64_t))
+#define _SS_ALIGNSIZE (sizeof (__s64))
/* Implementation specific desired alignment */
/*
* Definitions used for sockaddr_storage structure paddings design.
_SS_PAD1SIZE + _SS_ALIGNSIZE))
struct sockaddr_storage {
- sa_family_t __ss_family; /* address family */
- /* Following fields are implementation specific */
- char __ss_pad1[_SS_PAD1SIZE];
- /* 6 byte pad, to make implementation */
- /* specific pad up to alignment field that */
- /* follows explicit in the data structure */
- int64_t __ss_align; /* field to force desired structure */
- /* storage alignment */
- char __ss_pad2[_SS_PAD2SIZE];
+ sa_family_t __ss_family; /* address family */
+ /* Following fields are implementation specific */
+ char __ss_pad1[_SS_PAD1SIZE];
+ /* 6 byte pad, to make implementation */
+ /* specific pad up to alignment field that */
+ /* follows explicit in the data structure */
+ __s64 __ss_align; /* field to force desired structure */
+ /* storage alignment */
+ char __ss_pad2[_SS_PAD2SIZE];
/* 112 byte pad to achieve desired size, */
/* _SS_MAXSIZE value minus size of ss_family */
/* __ss_pad1, __ss_align fields is 112 */
};
-/* A convenience structure for handling sockaddr structures.
+/* A convenience structure for handling sockaddr structures.
* We should wean ourselves off this.
- */
+ */
typedef union {
struct sockaddr_in v4;
struct sockaddr_in6 v6;
typedef struct SCTP_outqueue sctp_outqueue_t;
typedef struct SCTP_bind_addr sctp_bind_addr_t;
typedef struct sctp_opt sctp_opt_t;
-typedef struct sctp_endpoint_common sctp_endpoint_common_t;
+typedef struct sctp_endpoint_common sctp_endpoint_common_t;
-#include <net/sctp/sctp_tsnmap.h>
+#include <net/sctp/sctp_tsnmap.h>
#include <net/sctp/sctp_ulpevent.h>
#include <net/sctp/sctp_ulpqueue.h>
struct sctp_bind_bucket *chain;
} sctp_bind_hashbucket_t;
-/* Used for hashing all associations.
- */
+/* Used for hashing all associations. */
typedef struct sctp_hashbucket {
rwlock_t lock;
sctp_endpoint_common_t *chain;
/* The SCTP protocol structure. */
-struct SCTP_protocol {
+struct SCTP_protocol {
/* RFC2960 Section 14. Suggested SCTP Protocol Parameter Values
- *
- * The following protocol parameters are RECOMMENDED:
- *
- * RTO.Initial - 3 seconds
- * RTO.Min - 1 second
- * RTO.Max - 60 seconds
- * RTO.Alpha - 1/8 (3 when converted to right shifts.)
- * RTO.Beta - 1/4 (2 when converted to right shifts.)
- */
- uint32_t rto_initial;
- uint32_t rto_min;
- uint32_t rto_max;
+ *
+ * The following protocol parameters are RECOMMENDED:
+ *
+ * RTO.Initial - 3 seconds
+ * RTO.Min - 1 second
+ * RTO.Max - 60 seconds
+ * RTO.Alpha - 1/8 (3 when converted to right shifts.)
+ * RTO.Beta - 1/4 (2 when converted to right shifts.)
+ */
+ __u32 rto_initial;
+ __u32 rto_min;
+ __u32 rto_max;
- /* Note: rto_alpha and rto_beta are really defined as inverse
- * powers of two to facilitate integer operations.
+ /* Note: rto_alpha and rto_beta are really defined as inverse
+ * powers of two to facilitate integer operations.
*/
int rto_alpha;
int rto_beta;
/* Max.Burst - 4 */
int max_burst;
- /* Valid.Cookie.Life - 60 seconds
- */
- int valid_cookie_life;
+ /* Valid.Cookie.Life - 60 seconds */
+ int valid_cookie_life;
- /* Association.Max.Retrans - 10 attempts
- * Path.Max.Retrans - 5 attempts (per destination address)
- * Max.Init.Retransmits - 8 attempts
- */
+ /* Association.Max.Retrans - 10 attempts
+ * Path.Max.Retrans - 5 attempts (per destination address)
+ * Max.Init.Retransmits - 8 attempts
+ */
int max_retrans_association;
int max_retrans_path;
int max_retrans_init;
- /* HB.interval - 30 seconds
- */
- int hb_interval;
+ /* HB.interval - 30 seconds */
+ int hb_interval;
- /*
- * The following variables are implementation specific.
- */
+ /* The following variables are implementation specific. */
/* Default initialization values to be applied to new associations. */
- uint16_t max_instreams;
- uint16_t max_outstreams;
-
- /* This is a list of groups of functions for each address
- * family that we support.
- */
+ __u16 max_instreams;
+ __u16 max_outstreams;
+
+ /* This is a list of groups of functions for each address
+ * family that we support.
+ */
list_t address_families;
- /* This is the hash of all endpoints. */
+ /* This is the hash of all endpoints. */
int ep_hashsize;
sctp_hashbucket_t *ep_hashbucket;
*/
list_t local_addr_list;
spinlock_t local_addr_lock;
-
-}; /* struct SCTP_protocol */
+};
/*
*/
typedef struct sctp_func {
int (*queue_xmit) (struct sk_buff *skb);
- int (*setsockopt) (struct sock *sk,
- int level,
- int optname,
- char *optval,
+ int (*setsockopt) (struct sock *sk,
+ int level,
+ int optname,
+ char *optval,
int optlen);
- int (*getsockopt) (struct sock *sk,
- int level,
- int optname,
- char *optval,
+ int (*getsockopt) (struct sock *sk,
+ int level,
+ int optname,
+ char *optval,
int *optlen);
int (*get_dst_mtu) (const sockaddr_storage_t *address);
- uint16_t net_header_len;
+ __u16 net_header_len;
int sockaddr_len;
sa_family_t sa_family;
list_t list;
/* SCTP Socket type: UDP or TCP style. */
typedef enum {
- SCTP_SOCKET_UDP = 0,
+ SCTP_SOCKET_UDP = 0,
SCTP_SOCKET_UDP_HIGH_BANDWIDTH,
SCTP_SOCKET_TCP
} sctp_socket_type_t;
/* Per socket SCTP information. */
struct sctp_opt {
- /* What kind of a socket is this? */
- sctp_socket_type_t type;
+ /* What kind of a socket is this? */
+ sctp_socket_type_t type;
- /* What is our base endpointer? */
- sctp_endpoint_t *ep;
+ /* What is our base endpointer? */
+ sctp_endpoint_t *ep;
/* Various Socket Options. */
- uint16_t default_stream;
- uint32_t default_ppid;
+ __u16 default_stream;
+ __u32 default_ppid;
struct sctp_initmsg initmsg;
struct sctp_rtoinfo rtoinfo;
struct sctp_paddrparams paddrparam;
struct sctp_event_subscribe subscribe;
- uint32_t autoclose;
- uint8_t nodelay;
- uint8_t disable_fragments;
-
-}; /* struct sctp_opt */
+ __u32 autoclose;
+ __u8 nodelay;
+ __u8 disable_fragments;
+};
/* These are the parts of an association which we send in the cookie.
* Most of these are straight out of:
* RFC2960 12.2 Parameters necessary per association (i.e. the TCB)
- *
+ *
*/
typedef struct sctp_cookie {
/* My : Tag expected in every inbound packet and sent
* Verification: in the INIT or INIT ACK chunk.
- * Tag :
+ * Tag :
*/
- uint32_t my_vtag;
+ __u32 my_vtag;
/* Peer's : Tag expected in every outbound packet except
* Verification: in the INIT chunk.
- * Tag :
+ * Tag :
*/
- uint32_t peer_vtag;
-
+ __u32 peer_vtag;
+
/* The rest of these are not from the spec, but really need to
* be in the cookie.
*/
/* My Tie Tag : Assist in discovering a restarting association. */
- uint32_t my_ttag;
+ __u32 my_ttag;
/* Peer's Tie Tag: Assist in discovering a restarting association. */
- uint32_t peer_ttag;
-
+ __u32 peer_ttag;
+
/* When does this cookie expire? */
struct timeval expiration;
- /* Number of inbound/outbound streams which are set
+ /* Number of inbound/outbound streams which are set
* and negotiated during the INIT process. */
- uint16_t sinit_num_ostreams;
- uint16_t sinit_max_instreams;
+ __u16 sinit_num_ostreams;
+ __u16 sinit_max_instreams;
/* This is the first sequence number I used. */
- uint32_t initial_tsn;
+ __u32 initial_tsn;
/* This holds the originating address of the INIT packet. */
sockaddr_storage_t peer_addr;
-
- /* This is a shim for my peer's INIT packet, followed by
- * a copy of the raw address list of the association.
- * The length of the raw address list is saved in the
+
+ /* This is a shim for my peer's INIT packet, followed by
+ * a copy of the raw address list of the association.
+ * The length of the raw address list is saved in the
* raw_addr_list_len field, which will be used at the time when
- * the association TCB is re-constructed from the cookie.
+ * the association TCB is re-constructed from the cookie.
*/
- uint32_t raw_addr_list_len;
- sctp_init_chunk_t peer_init[0];
-} sctp_cookie_t;
+ __u32 raw_addr_list_len;
+ sctp_init_chunk_t peer_init[0];
+} sctp_cookie_t;
/* The format of our cookie that we send to our peer. */
typedef struct sctp_signed_cookie {
- uint8_t signature[SCTP_SECRET_SIZE];
- sctp_cookie_t c;
-} sctp_signed_cookie_t ;
+ __u8 signature[SCTP_SECRET_SIZE];
+ sctp_cookie_t c;
+} sctp_signed_cookie_t;
/* This convenience type allows us to avoid casting when walking
* through a parameter list.
*/
typedef union {
- uint8_t *v;
- sctp_paramhdr_t *p;
-
- sctp_cookie_preserve_param_t *bht;
- sctp_hostname_param_t *dns;
- sctp_cookie_param_t *cookie;
- sctp_supported_addrs_param_t *sat;
- sctp_ipv4addr_param_t *v4;
- sctp_ipv6addr_param_t *v6;
+ __u8 *v;
+ sctp_paramhdr_t *p;
+
+ sctp_cookie_preserve_param_t *bht;
+ sctp_hostname_param_t *dns;
+ sctp_cookie_param_t *cookie;
+ sctp_supported_addrs_param_t *sat;
+ sctp_ipv4addr_param_t *v4;
+ sctp_ipv6addr_param_t *v6;
} sctpParam_t;
/* This is another convenience type to allocate memory for address
- * params for the maximum size and pass such structures around
+ * params for the maximum size and pass such structures around
* internally.
*/
typedef union {
sockaddr_storage_t daddr;
unsigned long sent_at;
} sctp_sender_hb_info_t __attribute__((packed));
-
-
/* RFC2960 1.4 Key Terms
- *
+ *
* o Chunk: A unit of information within an SCTP packet, consisting of
* a chunk header and chunk-specific content.
*
* each chunk as well as a few other header pointers...
*/
struct SCTP_chunk {
- /* These first three elements MUST PRECISELY match the first
- * three elements of struct sk_buff. This allows us to reuse
- * all the skb_* queue management functions.
- */
- sctp_chunk_t *next;
- sctp_chunk_t *prev;
- struct sk_buff_head *list;
+ /* These first three elements MUST PRECISELY match the first
+ * three elements of struct sk_buff. This allows us to reuse
+ * all the skb_* queue management functions.
+ */
+ sctp_chunk_t *next;
+ sctp_chunk_t *prev;
+ struct sk_buff_head *list;
- /* This is our link to the per-transport transmitted list. */
- struct list_head transmitted_list;
+ /* This is our link to the per-transport transmitted list. */
+ struct list_head transmitted_list;
/* This field is used by chunks that hold fragmented data.
* For the first fragment this is the list that holds the rest of
struct list_head frag_list;
/* This points to the sk_buff containing the actual data. */
- struct sk_buff *skb;
+ struct sk_buff *skb;
- /* These are the SCTP headers by reverse order in a packet.
+ /* These are the SCTP headers by reverse order in a packet.
* Note that some of these may happen more than once. In that
* case, we point at the "current" one, whatever that means
* for that level of header.
*/
-
- /* We point this at the FIRST TLV parameter to chunk_hdr. */
- sctpParam_t param_hdr;
- union {
- uint8_t *v;
- sctp_datahdr_t *data_hdr;
- sctp_inithdr_t *init_hdr;
- sctp_sackhdr_t *sack_hdr;
+
+ /* We point this at the FIRST TLV parameter to chunk_hdr. */
+ sctpParam_t param_hdr;
+ union {
+ __u8 *v;
+ sctp_datahdr_t *data_hdr;
+ sctp_inithdr_t *init_hdr;
+ sctp_sackhdr_t *sack_hdr;
sctp_heartbeathdr_t *hb_hdr;
sctp_sender_hb_info_t *hbs_hdr;
- sctp_shutdownhdr_t *shutdown_hdr;
+ sctp_shutdownhdr_t *shutdown_hdr;
sctp_signed_cookie_t *cookie_hdr;
sctp_ecnehdr_t *ecne_hdr;
- sctp_cwrhdr_t *ecn_cwr_hdr;
+ sctp_cwrhdr_t *ecn_cwr_hdr;
sctp_errhdr_t *err_hdr;
- } subh;
-
- uint8_t *chunk_end;
-
+ } subh;
+
+ __u8 *chunk_end;
+
sctp_chunkhdr_t *chunk_hdr;
-
- sctp_sctphdr_t *sctp_hdr;
+
+ sctp_sctphdr_t *sctp_hdr;
/* This needs to be recoverable for SCTP_SEND_FAILED events. */
struct sctp_sndrcvinfo sinfo;
/* Which association does this belong to? */
- sctp_association_t *asoc;
+ sctp_association_t *asoc;
/* What endpoint received this chunk? */
sctp_endpoint_common_t *rcvr;
- /* We fill this in if we are calculating RTT. */
- unsigned long sent_at;
-
- uint8_t rtt_in_progress; /* Is this chunk used for RTT calculation? */
- uint8_t num_times_sent; /* How man times did we send this? */
- uint8_t has_tsn; /* Does this chunk have a TSN yet? */
- uint8_t singleton; /* Was this the only chunk in the packet? */
- uint8_t end_of_packet; /* Was this the last chunk in the packet? */
- uint8_t ecn_ce_done; /* Have we processed the ECN CE bit? */
- uint8_t pdiscard; /* Discard the whole packet now? */
- uint8_t tsn_gap_acked; /* Is this chunk acked by a GAP ACK? */
- uint8_t fast_retransmit; /* Is this chunk fast retransmitted? */
- uint8_t tsn_missing_report; /* Data chunk missing counter. */
+ /* We fill this in if we are calculating RTT. */
+ unsigned long sent_at;
+
+ __u8 rtt_in_progress; /* Is this chunk used for RTT calculation? */
+ __u8 num_times_sent; /* How man times did we send this? */
+ __u8 has_tsn; /* Does this chunk have a TSN yet? */
+ __u8 singleton; /* Was this the only chunk in the packet? */
+ __u8 end_of_packet; /* Was this the last chunk in the packet? */
+ __u8 ecn_ce_done; /* Have we processed the ECN CE bit? */
+ __u8 pdiscard; /* Discard the whole packet now? */
+ __u8 tsn_gap_acked; /* Is this chunk acked by a GAP ACK? */
+ __u8 fast_retransmit; /* Is this chunk fast retransmitted? */
+ __u8 tsn_missing_report; /* Data chunk missing counter. */
/* What is the origin IP address for this chunk? */
sockaddr_storage_t source;
* go. It is NULL if we have no preference.
*/
sctp_transport_t *transport;
-}; /* struct SCTP_chunk */
-
-sctp_chunk_t *sctp_make_chunk(const sctp_association_t *, uint8_t type,
- uint8_t flags, int size);
+};
+
+sctp_chunk_t *sctp_make_chunk(const sctp_association_t *, __u8 type,
+ __u8 flags, int size);
void sctp_free_chunk(sctp_chunk_t *);
sctp_chunk_t *sctp_copy_chunk(sctp_chunk_t *, int flags);
void *sctp_addto_chunk(sctp_chunk_t *chunk, int len, const void *data);
*/
struct sockaddr_storage_list {
list_t list;
- sockaddr_storage_t a;
-}; /* struct sockaddr_storage_list */
+ sockaddr_storage_t a;
+};
typedef sctp_chunk_t *(sctp_packet_phandler_t)(sctp_association_t *);
*/
struct SCTP_packet {
/* These are the SCTP header values (host order) for the packet. */
- uint16_t source_port;
- uint16_t destination_port;
- uint32_t vtag;
-
+ __u16 source_port;
+ __u16 destination_port;
+ __u32 vtag;
+
/* This contains the payload chunks. */
struct sk_buff_head chunks;
/* This is the total size of all chunks INCLUDING padding. */
size_t size;
- /* The packet is destined for this transport address.
- * The function we finally use to pass down to the next lower
- * layer lives in the transport structure.
- */
- sctp_transport_t *transport;
-
- /* Allow a callback for getting a high priority chunk
- * bundled early into the packet (This is used for ECNE).
- */
+ /* The packet is destined for this transport address.
+ * The function we finally use to pass down to the next lower
+ * layer lives in the transport structure.
+ */
+ sctp_transport_t *transport;
+
+ /* Allow a callback for getting a high priority chunk
+ * bundled early into the packet (This is used for ECNE).
+ */
sctp_packet_phandler_t *get_prepend_chunk;
/* This packet should advertise ECN capability to the network
* via the ECT bit.
*/
int ecn_capable;
-
+
/* This packet contains a COOKIE-ECHO chunk. */
int has_cookie_echo;
int malloced;
-
-}; /* struct SCTP_packet */
+};
typedef int (sctp_outqueue_thandler_t)(sctp_outqueue_t *, void *);
typedef int (sctp_outqueue_ehandler_t)(sctp_outqueue_t *);
typedef sctp_packet_t *(sctp_outqueue_ohandler_init_t)
(sctp_packet_t *,
sctp_transport_t *,
- uint16_t sport,
- uint16_t dport);
+ __u16 sport,
+ __u16 dport);
typedef sctp_packet_t *(sctp_outqueue_ohandler_config_t)
(sctp_packet_t *,
- uint32_t vtag,
+ __u32 vtag,
int ecn_capable,
sctp_packet_phandler_t *get_prepend_chunk);
typedef sctp_xmit_t (sctp_outqueue_ohandler_t)(sctp_packet_t *,
sctp_outqueue_ohandler_t sctp_packet_transmit_chunk;
sctp_outqueue_ohandler_force_t sctp_packet_transmit;
void sctp_packet_free(sctp_packet_t *);
-
+
/* This represents a remote transport address.
* For local transport addresses, we just use sockaddr_storage_t.
*
* RFC2960 Section 1.4 Key Terms
- *
+ *
* o Transport address: A Transport Address is traditionally defined
* by Network Layer address, Transport Layer protocol and Transport
* Layer port number. In the case of SCTP running over IP, a
/* This is the peer's IP address and port. */
sockaddr_storage_t ipaddr;
-
- /* These are the functions we call to handle LLP stuff. */
+
+ /* These are the functions we call to handle LLP stuff. */
sctp_func_t *af_specific;
- /* Which association do we belong to? */
- sctp_association_t *asoc;
+ /* Which association do we belong to? */
+ sctp_association_t *asoc;
/* RFC2960
*
- * 12.3 Per Transport Address Data
- *
- * For each destination transport address in the peer's
- * address list derived from the INIT or INIT ACK chunk, a
- * number of data elements needs to be maintained including:
- */
- uint32_t rtt; /* This is the most recent RTT. */
-
- /* RTO : The current retransmission timeout value. */
- uint32_t rto;
-
- /* RTTVAR : The current RTT variation. */
- uint32_t rttvar;
-
- /* SRTT : The current smoothed round trip time. */
- uint32_t srtt;
-
+ * 12.3 Per Transport Address Data
+ *
+ * For each destination transport address in the peer's
+ * address list derived from the INIT or INIT ACK chunk, a
+ * number of data elements needs to be maintained including:
+ */
+ __u32 rtt; /* This is the most recent RTT. */
+
+ /* RTO : The current retransmission timeout value. */
+ __u32 rto;
+
+ /* RTTVAR : The current RTT variation. */
+ __u32 rttvar;
+
+ /* SRTT : The current smoothed round trip time. */
+ __u32 srtt;
+
/* RTO-Pending : A flag used to track if one of the DATA
* chunks sent to this address is currently being
* used to compute a RTT. If this flag is 0,
* calculation completes (i.e. the DATA chunk
* is SACK'd) clear this flag.
*/
- int rto_pending;
+ int rto_pending;
/*
- * These are the congestion stats.
- */
- /* cwnd : The current congestion window. */
- uint32_t cwnd; /* This is the actual cwnd. */
-
- /* ssthresh : The current slow start threshold value. */
- uint32_t ssthresh;
-
+ * These are the congestion stats.
+ */
+ /* cwnd : The current congestion window. */
+ __u32 cwnd; /* This is the actual cwnd. */
+
+ /* ssthresh : The current slow start threshold value. */
+ __u32 ssthresh;
+
/* partial : The tracking method for increase of cwnd when in
- * bytes acked : congestion avoidance mode (see Section 6.2.2)
- */
- uint32_t partial_bytes_acked;
+ * bytes acked : congestion avoidance mode (see Section 6.2.2)
+ */
+ __u32 partial_bytes_acked;
/* Data that has been sent, but not acknowledged. */
- uint32_t flight_size;
-
+ __u32 flight_size;
+
/* PMTU : The current known path MTU. */
- uint32_t pmtu;
-
- /* When was the last time(in jiffies) that a data packet was sent on
+ __u32 pmtu;
+
+ /* When was the last time(in jiffies) that a data packet was sent on
* this transport? This is used to adjust the cwnd when the transport
* becomes inactive.
- */
+ */
unsigned long last_time_used;
-
+
/* Heartbeat interval: The endpoint sends out a Heartbeat chunk to
- * the destination address every heartbeat interval.
- */
- int hb_interval;
-
+ * the destination address every heartbeat interval.
+ */
+ int hb_interval;
+
/* When was the last time (in jiffies) that we heard from this
* transport? We use this to pick new active and retran paths.
*/
unsigned long last_time_heard;
- /* Last time(in jiffies) when cwnd is reduced due to the congestion
+ /* Last time(in jiffies) when cwnd is reduced due to the congestion
* indication based on ECNE chunk.
*/
- unsigned long last_time_ecne_reduced;
+ unsigned long last_time_ecne_reduced;
/* state : The current state of this destination,
* : i.e. DOWN, UP, ALLOW-HB, NO-HEARTBEAT, etc.
int active;
int hb_allowed;
} state;
-
+
/* These are the error stats for this destination. */
-
- /* Error count : The current error count for this destination. */
+
+ /* Error count : The current error count for this destination. */
unsigned short error_count;
-
+
/* Error : Current error threshold for this destination
* Threshold : i.e. what value marks the destination down if
* : errorCount reaches this value.
- */
+ */
unsigned short error_threshold;
- /* This is the max_retrans value for the transport and will
- * be initialized to proto.max_retrans.path. This can be changed
+ /* This is the max_retrans value for the transport and will
+ * be initialized to proto.max_retrans.path. This can be changed
* using SCTP_SET_PEER_ADDR_PARAMS socket option.
- */
+ */
int max_retrans;
-
- /* We use this name for debugging output... */
- char *debug_name;
-
- /* Per : A timer used by each destination.
- * Destination :
- * Timer :
- *
- * [Everywhere else in the text this is called T3-rtx. -ed]
- */
- struct timer_list T3_rtx_timer;
- /* Heartbeat timer is per destination. */
- struct timer_list hb_timer;
+ /* We use this name for debugging output... */
+ char *debug_name;
+
+ /* Per : A timer used by each destination.
+ * Destination :
+ * Timer :
+ *
+ * [Everywhere else in the text this is called T3-rtx. -ed]
+ */
+ struct timer_list T3_rtx_timer;
+
+ /* Heartbeat timer is per destination. */
+ struct timer_list hb_timer;
/* Since we're using per-destination retransmission timers
* (see above), we're also using per-destination "transmitted"
* queues. This probably ought to be a private struct
* accessible only within the outqueue, but it's not, yet.
*/
- struct list_head transmitted;
+ struct list_head transmitted;
/* We build bundle-able packets for this transport here. */
sctp_packet_t packet;
struct list_head send_ready;
int malloced; /* Is this structure kfree()able? */
-}; /* struct SCTP_transport */
+};
extern sctp_transport_t *sctp_transport_new(const sockaddr_storage_t *, int);
-extern sctp_transport_t *sctp_transport_init(sctp_transport_t *,
+extern sctp_transport_t *sctp_transport_init(sctp_transport_t *,
const sockaddr_storage_t *, int);
extern void sctp_transport_set_owner(sctp_transport_t *, sctp_association_t *);
extern void sctp_transport_free(sctp_transport_t *);
extern void sctp_transport_reset_timers(sctp_transport_t *);
extern void sctp_transport_hold(sctp_transport_t *);
extern void sctp_transport_put(sctp_transport_t *);
-extern void sctp_transport_update_rto(sctp_transport_t *, uint32_t);
-extern void sctp_transport_raise_cwnd(sctp_transport_t *, uint32_t, uint32_t);
+extern void sctp_transport_update_rto(sctp_transport_t *, __u32);
+extern void sctp_transport_raise_cwnd(sctp_transport_t *, __u32, __u32);
extern void sctp_transport_lower_cwnd(sctp_transport_t *, sctp_lower_cwnd_t);
/* This is the structure we use to queue packets as they come into
* fragment reassembly and chunk unbundling.
*/
struct SCTP_inqueue {
- /* This is actually a queue of sctp_chunk_t each
- * containing a partially decoded packet.
- */
- struct sk_buff_head in;
+ /* This is actually a queue of sctp_chunk_t each
+ * containing a partially decoded packet.
+ */
+ struct sk_buff_head in;
/* This is the packet which is currently off the in queue and is
* being worked on through the inbound chunk processing.
*/
sctp_chunk_t *in_progress;
- /* This is the delayed task to finish delivering inbound
- * messages.
- */
- struct tq_struct immediate;
+ /* This is the delayed task to finish delivering inbound
+ * messages.
+ */
+ struct tq_struct immediate;
- int malloced; /* Is this structure kfree()able? */
-}; /* struct SCTP_inqueue */
+ int malloced; /* Is this structure kfree()able? */
+};
sctp_inqueue_t *sctp_inqueue_new(void);
void sctp_inqueue_init(sctp_inqueue_t *);
sctp_chunk_t *sctp_pop_inqueue(sctp_inqueue_t *);
void sctp_inqueue_set_th_handler(sctp_inqueue_t *,
void (*)(void *), void *);
-
/* This is the structure we use to hold outbound chunks. You push
* chunks in and they automatically pop out the other end as bundled
*
* This structure covers sections 6.3, 6.4, 6.7, 6.8, 6.10, 7., 8.1,
* and 8.2 of the v13 draft.
- *
+ *
* It handles retransmissions. The connection to the timeout portion
* of the state machine is through sctp_..._timeout() and timeout_handler.
*
* When free()'d, it empties itself out via output_handler().
*/
struct SCTP_outqueue {
- sctp_association_t *asoc;
+ sctp_association_t *asoc;
- /* BUG: This really should be an array of streams.
- * This really holds a list of chunks (one stream).
+ /* BUG: This really should be an array of streams.
+ * This really holds a list of chunks (one stream).
* FIXME: If true, why so?
*/
- struct sk_buff_head out;
+ struct sk_buff_head out;
/* These are control chunks we want to send. */
- struct sk_buff_head control;
+ struct sk_buff_head control;
/* These are chunks that have been sacked but are above the
- * CTSN, or cumulative tsn ack point.
+ * CTSN, or cumulative tsn ack point.
*/
- struct list_head sacked;
+ struct list_head sacked;
/* Put chunks on this list to schedule them for
* retransmission.
*/
- struct list_head retransmit;
+ struct list_head retransmit;
- /* Call these functions to send chunks down to the next lower
- * layer. This is always SCTP_packet, but we separate the two
- * structures to make testing simpler.
- */
- sctp_outqueue_ohandler_init_t *init_output;
- sctp_outqueue_ohandler_config_t *config_output;
- sctp_outqueue_ohandler_t *append_output;
- sctp_outqueue_ohandler_t *build_output;
- sctp_outqueue_ohandler_force_t *force_output;
+ /* Call these functions to send chunks down to the next lower
+ * layer. This is always SCTP_packet, but we separate the two
+ * structures to make testing simpler.
+ */
+ sctp_outqueue_ohandler_init_t *init_output;
+ sctp_outqueue_ohandler_config_t *config_output;
+ sctp_outqueue_ohandler_t *append_output;
+ sctp_outqueue_ohandler_t *build_output;
+ sctp_outqueue_ohandler_force_t *force_output;
- /* How many unackd bytes do we have in-flight? */
- uint32_t outstanding_bytes;
+ /* How many unackd bytes do we have in-flight? */
+ __u32 outstanding_bytes;
- /* Is this structure empty? */
- int empty;
+ /* Is this structure empty? */
+ int empty;
/* Are we kfree()able? */
- int malloced;
-}; /* struct SCTP_outqueue */
+ int malloced;
+};
sctp_outqueue_t *sctp_outqueue_new(sctp_association_t *);
void sctp_outqueue_init(sctp_association_t *, sctp_outqueue_t *);
sctp_outqueue_ohandler_t build,
sctp_outqueue_ohandler_force_t force);
void sctp_outqueue_restart(sctp_outqueue_t *);
-void sctp_retransmit(sctp_outqueue_t *, sctp_transport_t *, uint8_t);
+void sctp_retransmit(sctp_outqueue_t *, sctp_transport_t *, __u8);
/* These bind address data fields common between endpoints and associations */
struct SCTP_bind_addr {
-
+
/* RFC 2960 12.1 Parameters necessary for the SCTP instance
*
* SCTP Port: The local SCTP port number the endpoint is
- * bound to.
- */
- uint16_t port;
+ * bound to.
+ */
+ __u16 port;
/* RFC 2960 12.1 Parameters necessary for the SCTP instance
*
- * Address List: The list of IP addresses that this instance
- * has bound. This information is passed to one's
- * peer(s) in INIT and INIT ACK chunks.
- */
+ * Address List: The list of IP addresses that this instance
+ * has bound. This information is passed to one's
+ * peer(s) in INIT and INIT ACK chunks.
+ */
list_t address_list;
- int malloced; /* Are we kfree()able? */
-}; /* struct SCTP_bind_addr */
+ int malloced; /* Are we kfree()able? */
+};
sctp_bind_addr_t *sctp_bind_addr_new(int gfp_mask);
-void sctp_bind_addr_init(sctp_bind_addr_t *, uint16_t port);
+void sctp_bind_addr_init(sctp_bind_addr_t *, __u16 port);
void sctp_bind_addr_free(sctp_bind_addr_t *);
-int sctp_bind_addr_copy(sctp_bind_addr_t *dest, const sctp_bind_addr_t *src,
+int sctp_bind_addr_copy(sctp_bind_addr_t *dest, const sctp_bind_addr_t *src,
sctp_scope_t scope, int priority,int flags);
int sctp_add_bind_addr(sctp_bind_addr_t *, sockaddr_storage_t *,
int priority);
int sctp_del_bind_addr(sctp_bind_addr_t *, sockaddr_storage_t *);
int sctp_bind_addr_has_addr(sctp_bind_addr_t *, const sockaddr_storage_t *);
-sctpParam_t sctp_bind_addrs_to_raw(const sctp_bind_addr_t *bp,
+sctpParam_t sctp_bind_addrs_to_raw(const sctp_bind_addr_t *bp,
int *addrs_len,
int priority);
-int sctp_raw_to_bind_addrs(sctp_bind_addr_t *bp,
- uint8_t *raw_addr_list,
+int sctp_raw_to_bind_addrs(sctp_bind_addr_t *bp,
+ __u8 *raw_addr_list,
int addrs_len,
unsigned short port,
int priority);
SCTP_EP_TYPE_ASSOCIATION,
} sctp_endpoint_type_t;
-/*
- * A common base class to bridge the implmentation view of a
- * socket (usually listening) endpoint versus an association's
- * local endpoint.
+/*
+ * A common base class to bridge the implmentation view of a
+ * socket (usually listening) endpoint versus an association's
+ * local endpoint.
* This common structure is useful for several purposes:
- * 1) Common interface for lookup routines.
+ * 1) Common interface for lookup routines.
* a) Subfunctions work for either endpoint or association
* b) Single interface to lookup allows hiding the lookup lock rather
- * than acquiring it externally.
- * 2) Common interface for the inbound chunk handling/state machine.
- * 3) Common object handling routines for reference counting, etc.
+ * than acquiring it externally.
+ * 2) Common interface for the inbound chunk handling/state machine.
+ * 3) Common object handling routines for reference counting, etc.
* 4) Disentangle association lookup from endpoint lookup, where we
- * do not have to find our endpoint to find our association.
+ * do not have to find our endpoint to find our association.
*
*/
-struct sctp_endpoint_common {
+struct sctp_endpoint_common {
/* Fields to help us manage our entries in the hash tables. */
sctp_endpoint_common_t *next;
- sctp_endpoint_common_t **pprev;
+ sctp_endpoint_common_t **pprev;
int hashent;
-
+
/* Runtime type information. What kind of endpoint is this? */
sctp_endpoint_type_t type;
-
- /* Some fields to help us manage this object.
+
+ /* Some fields to help us manage this object.
* refcnt - Reference count access to this object.
- * dead - Do not attempt to use this object.
- * malloced - Do we need to kfree this object?
+ * dead - Do not attempt to use this object.
+ * malloced - Do we need to kfree this object?
*/
atomic_t refcnt;
char dead;
char malloced;
/* What socket does this endpoint belong to? */
- struct sock *sk;
+ struct sock *sk;
/* This is where we receive inbound chunks. */
sctp_inqueue_t inqueue;
- /* This substructure includes the defining parameters of the
+ /* This substructure includes the defining parameters of the
* endpoint:
* bind_addr.port is our shared port number.
* bind_addr.address_list is our set of local IP addresses.
- *
*/
sctp_bind_addr_t bind_addr;
/* Protection during address list comparisons. */
rwlock_t addr_lock;
-}; /* sctp_endpoint_common_t */
+};
/* RFC Section 1.4 Key Terms
- *
+ *
* o SCTP endpoint: The logical sender/receiver of SCTP packets. On a
* multi-homed host, an SCTP endpoint is represented to its peers as a
* combination of a set of eligible destination transport addresses to
* which SCTP packets can be sent and a set of eligible source
* transport addresses from which SCTP packets can be received.
- * All transport addresses used by an SCTP endpoint must use the
+ * All transport addresses used by an SCTP endpoint must use the
* same port number, but can use multiple IP addresses. A transport
* address used by an SCTP endpoint must not be used by another
* SCTP endpoint. In other words, a transport address is unique
struct SCTP_endpoint {
/* Common substructure for endpoint and association. */
- sctp_endpoint_common_t base;
-
- /* These are the system-wide defaults and other stuff which is
- * endpoint-independent.
- */
- sctp_protocol_t *proto;
+ sctp_endpoint_common_t base;
+
+ /* These are the system-wide defaults and other stuff which is
+ * endpoint-independent.
+ */
+ sctp_protocol_t *proto;
/* Associations: A list of current associations and mappings
- * to the data consumers for each association. This
- * may be in the form of a hash table or other
- * implementation dependent structure. The data
- * consumers may be process identification
- * information such as file descriptors, named pipe
- * pointer, or table pointers dependent on how SCTP
- * is implemented.
- */
+ * to the data consumers for each association. This
+ * may be in the form of a hash table or other
+ * implementation dependent structure. The data
+ * consumers may be process identification
+ * information such as file descriptors, named pipe
+ * pointer, or table pointers dependent on how SCTP
+ * is implemented.
+ */
/* This is really a list of sctp_association_t entries. */
list_t asocs;
/* Secret Key: A secret key used by this endpoint to compute
- * the MAC. This SHOULD be a cryptographic quality
- * random number with a sufficient length.
- * Discussion in [RFC1750] can be helpful in
- * selection of the key.
- */
- uint8_t secret_key[SCTP_HOW_MANY_SECRETS][SCTP_SECRET_SIZE];
- int current_key;
- int last_key;
- int key_changed_at;
+ * the MAC. This SHOULD be a cryptographic quality
+ * random number with a sufficient length.
+ * Discussion in [RFC1750] can be helpful in
+ * selection of the key.
+ */
+ __u8 secret_key[SCTP_HOW_MANY_SECRETS][SCTP_SECRET_SIZE];
+ int current_key;
+ int last_key;
+ int key_changed_at;
- /* Default timeouts. */
- int timeouts[SCTP_NUM_TIMEOUT_TYPES];
+ /* Default timeouts. */
+ int timeouts[SCTP_NUM_TIMEOUT_TYPES];
- /* Various thresholds. */
+ /* Various thresholds. */
- /* Name for debugging output... */
- char *debug_name;
-}; /* sctp_endpoint_t */
+ /* Name for debugging output... */
+ char *debug_name;
+};
/* Recover the outter endpoint structure. */
-static inline sctp_endpoint_t *sctp_ep(sctp_endpoint_common_t* base)
-{
+static inline sctp_endpoint_t *sctp_ep(sctp_endpoint_common_t *base)
+{
sctp_endpoint_t *ep;
-
+
/* We are not really a list, but the list_entry() macro is
- * really quite generic to find the address of an outter struct.
+ * really quite generic to find the address of an outter struct.
*/
ep = list_entry(base, sctp_endpoint_t, base);
return ep;
-
-} /* sctp_ep() */
+}
/* These are function signatures for manipulating endpoints. */
sctp_endpoint_t *sctp_endpoint_new(sctp_protocol_t *, struct sock *, int);
sctp_association_t *sctp_endpoint_lookup_assoc(const sctp_endpoint_t *ep,
const sockaddr_storage_t *paddr,
sctp_transport_t **);
-sctp_endpoint_t *sctp_endpoint_is_match(sctp_endpoint_t *,
- const sockaddr_storage_t *);
+sctp_endpoint_t *sctp_endpoint_is_match(sctp_endpoint_t *,
+ const sockaddr_storage_t *);
void sctp_process_init(sctp_association_t *asoc, sctp_cid_t cid,
const sockaddr_storage_t *peer_addr,
sctp_init_chunk_t *peer_init, int priority);
int sctp_process_param(sctp_association_t *asoc,
- sctpParam_t param,
+ sctpParam_t param,
const sockaddr_storage_t *peer_addr,
sctp_cid_t cid, int priority);
-uint32_t sctp_generate_tag(const sctp_endpoint_t *ep);
-uint32_t sctp_generate_tsn(const sctp_endpoint_t *ep);
+__u32 sctp_generate_tag(const sctp_endpoint_t *ep);
+__u32 sctp_generate_tsn(const sctp_endpoint_t *ep);
/* RFC2960
- *
+ *
* 12. Recommended Transmission Control Block (TCB) Parameters
- *
+ *
* This section details a recommended set of parameters that should
* be contained within the TCB for an implementation. This section is
* for illustrative purposes and should not be deemed as requirements
* on an implementation or as an exhaustive list of all parameters
* inside an SCTP TCB. Each implementation may need its own additional
- * parameters for optimization.
+ * parameters for optimization.
*/
/* Here we have information about each individual association. */
struct SCTP_association {
-
- /* A base structure common to endpoint and association.
+
+ /* A base structure common to endpoint and association.
* In this context, it represents the associations's view
- * of the local endpoint of the association.
+ * of the local endpoint of the association.
*/
sctp_endpoint_common_t base;
* sctp_association_t data structure. Used for mapping an
* association id to an association.
*/
- uint32_t eyecatcher;
+ __u32 eyecatcher;
- /* This is our parent endpoint. */
- sctp_endpoint_t *ep;
+ /* This is our parent endpoint. */
+ sctp_endpoint_t *ep;
- /* These are those association elements needed in the cookie. */
- sctp_cookie_t c;
+ /* These are those association elements needed in the cookie. */
+ sctp_cookie_t c;
/* This is all information about our peer. */
struct {
- /* rwnd
+ /* rwnd
*
* Peer Rwnd : Current calculated value of the peer's rwnd.
*/
- uint32_t rwnd;
+ __u32 rwnd;
/* transport_addr_list
*
* List : associate an inbound packet with a given
* : association. Normally this information is
* : hashed or keyed for quick lookup and access
- * : of the TCB.
+ * : of the TCB.
*
* It is a list of SCTP_transport's.
*/
/* port
* The transport layer port number.
*/
- uint16_t port;
-
+ __u16 port;
+
/* primary_path
*
* Primary : This is the current primary destination
* Path : transport address of the peer endpoint. It
- * : may also specify a source transport address
- * : on this endpoint.
+ * : may also specify a source transport address
+ * : on this endpoint.
*
* All of these paths live on transport_addr_list.
*
* transmit new data and most control chunks.
*/
sctp_transport_t *primary_path;
-
+
/* active_path
- * The path that we are currently using to
+ * The path that we are currently using to
* transmit new data and most control chunks.
*/
sctp_transport_t *active_path;
/* Pointer to last transport I have sent on. */
sctp_transport_t *last_sent_to;
-
+
/* This is the last transport I have recieved DATA on. */
sctp_transport_t *last_data_from;
* TSN : sequence. This value is set initially by
* : taking the peer's Initial TSN, received in
* : the INIT or INIT ACK chunk, and subtracting
- * : one from it.
+ * : one from it.
*
* Throughout most of the specification this is called the
* "Cumulative TSN ACK Point". In this case, we
* ignore the advice in 12.2 in favour of the term
* used in the bulk of the text. This value is hidden
- * in tsn_map--we get it by calling sctp_tsnmap_get_ctsn().
+ * in tsn_map--we get it by calling sctp_tsnmap_get_ctsn().
*/
sctp_tsnmap_t tsn_map;
- uint8_t _map[sctp_tsnmap_storage_size(SCTP_TSN_MAP_SIZE)];
+ __u8 _map[sctp_tsnmap_storage_size(SCTP_TSN_MAP_SIZE)];
/* We record duplicate TSNs here. We clear this after
- * every SACK.
+ * every SACK.
* FIXME: We should move this into the tsnmap? --jgrimm
*/
sctp_dup_tsn_t dup_tsns[SCTP_MAX_DUP_TSNS];
/* Do we need to sack the peer? */
int sack_needed;
-
- /* These are capabilities which our peer advertised. */
- uint8_t ecn_capable; /* Can peer do ECN? */
- uint8_t ipv4_address; /* Peer understands IPv4 addresses? */
- uint8_t ipv6_address; /* Peer understands IPv6 addresses? */
- uint8_t hostname_address;/* Peer understands DNS addresses? */
+
+ /* These are capabilities which our peer advertised. */
+ __u8 ecn_capable; /* Can peer do ECN? */
+ __u8 ipv4_address; /* Peer understands IPv4 addresses? */
+ __u8 ipv6_address; /* Peer understands IPv6 addresses? */
+ __u8 hostname_address;/* Peer understands DNS addresses? */
sctp_inithdr_t i;
int cookie_len;
void *cookie;
/* ADDIP Extention (ADDIP) --xguo */
/* <expected peer-serial-number> minus 1 (ADDIP sec. 4.2 C1) */
- uint32_t addip_serial;
+ __u32 addip_serial;
} peer;
- /* State : A state variable indicating what state the
- * : association is in, i.e. COOKIE-WAIT,
- * : COOKIE-ECHOED, ESTABLISHED, SHUTDOWN-PENDING,
- * : SHUTDOWN-SENT, SHUTDOWN-RECEIVED, SHUTDOWN-ACK-SENT.
- *
- * Note: No "CLOSED" state is illustrated since if a
- * association is "CLOSED" its TCB SHOULD be removed.
- *
- * In this implementation we DO have a CLOSED
- * state which is used during initiation and shutdown.
- *
- * State takes values from SCTP_STATE_*.
- */
- sctp_state_t state;
+ /* State : A state variable indicating what state the
+ * : association is in, i.e. COOKIE-WAIT,
+ * : COOKIE-ECHOED, ESTABLISHED, SHUTDOWN-PENDING,
+ * : SHUTDOWN-SENT, SHUTDOWN-RECEIVED, SHUTDOWN-ACK-SENT.
+ *
+ * Note: No "CLOSED" state is illustrated since if a
+ * association is "CLOSED" its TCB SHOULD be removed.
+ *
+ * In this implementation we DO have a CLOSED
+ * state which is used during initiation and shutdown.
+ *
+ * State takes values from SCTP_STATE_*.
+ */
+ sctp_state_t state;
- /* When did we enter this state? */
- int state_timestamp;
-
- /* The cookie life I award for any cookie. */
- struct timeval cookie_life;
- uint32_t cookie_preserve;
+ /* When did we enter this state? */
+ int state_timestamp;
- /* Overall : The overall association error count.
- * Error Count : [Clear this any time I get something.]
- */
- int overall_error_count;
-
- /* Overall : The threshold for this association that if
- * Error : the Overall Error Count reaches will cause
- * Threshold : this association to be torn down.
- */
- int overall_error_threshold;
+ /* The cookie life I award for any cookie. */
+ struct timeval cookie_life;
+ __u32 cookie_preserve;
+
+ /* Overall : The overall association error count.
+ * Error Count : [Clear this any time I get something.]
+ */
+ int overall_error_count;
+
+ /* Overall : The threshold for this association that if
+ * Error : the Overall Error Count reaches will cause
+ * Threshold : this association to be torn down.
+ */
+ int overall_error_threshold;
/* These are the association's initial, max, and min RTO values.
* These values will be initialized by system defaults, but can
- * be modified via the SCTP_RTOINFO socket option.
- */
- uint32_t rto_initial;
- uint32_t rto_max;
- uint32_t rto_min;
+ * be modified via the SCTP_RTOINFO socket option.
+ */
+ __u32 rto_initial;
+ __u32 rto_max;
+ __u32 rto_min;
/* Maximum number of new data packets that can be sent in a burst. */
int max_burst;
/* This is the max_retrans value for the association. This value will
* be initialized initialized from system defaults, but can be
* modified by the SCTP_ASSOCINFO socket option.
- */
+ */
int max_retrans;
/* Maximum number of times the endpoint will retransmit INIT */
- uint16_t max_init_attempts;
+ __u16 max_init_attempts;
/* How many times have we resent an INIT? */
- uint16_t init_retries;
+ __u16 init_retries;
/* The largest timeout or RTO value to use in attempting an INIT */
- uint16_t max_init_timeo;
+ __u16 max_init_timeo;
- int timeouts[SCTP_NUM_TIMEOUT_TYPES];
- struct timer_list timers[SCTP_NUM_TIMEOUT_TYPES];
+ int timeouts[SCTP_NUM_TIMEOUT_TYPES];
+ struct timer_list timers[SCTP_NUM_TIMEOUT_TYPES];
/* Transport to which SHUTDOWN chunk was last sent. */
sctp_transport_t *shutdown_last_sent_to;
- /* Next TSN : The next TSN number to be assigned to a new
- * : DATA chunk. This is sent in the INIT or INIT
- * : ACK chunk to the peer and incremented each
- * : time a DATA chunk is assigned a TSN
- * : (normally just prior to transmit or during
- * : fragmentation).
- */
- uint32_t next_tsn;
+ /* Next TSN : The next TSN number to be assigned to a new
+ * : DATA chunk. This is sent in the INIT or INIT
+ * : ACK chunk to the peer and incremented each
+ * : time a DATA chunk is assigned a TSN
+ * : (normally just prior to transmit or during
+ * : fragmentation).
+ */
+ __u32 next_tsn;
- /*
+ /*
* Last Rcvd : This is the last TSN received in sequence. This value
* TSN : is set initially by taking the peer's Initial TSN,
* : received in the INIT or INIT ACK chunk, and
* Most of RFC 2960 refers to this as the Cumulative TSN Ack Point.
*/
- uint32_t ctsn_ack_point;
-
- /* The number of unacknowledged data chunks. Reported through
- * the SCTP_STATUS sockopt.
+ __u32 ctsn_ack_point;
+
+ /* The number of unacknowledged data chunks. Reported through
+ * the SCTP_STATUS sockopt.
*/
- uint16_t unack_data;
+ __u16 unack_data;
/* This is the association's receive buffer space. This value is used
- * to set a_rwnd field in an INIT or a SACK chunk.
+ * to set a_rwnd field in an INIT or a SACK chunk.
*/
- uint32_t rwnd;
+ __u32 rwnd;
/* Number of bytes by which the rwnd has slopped. The rwnd is allowed
* to slop over a maximum of the association's frag_point.
*/
- uint32_t rwnd_over;
+ __u32 rwnd_over;
- /* This is the sndbuf size in use for the association.
- * This corresponds to the sndbuf size for the association,
+ /* This is the sndbuf size in use for the association.
+ * This corresponds to the sndbuf size for the association,
* as specified in the sk->sndbuf.
*/
int sndbuf_used;
*/
wait_queue_head_t wait;
- /* Association : The smallest PMTU discovered for all of the
- * PMTU : peer's transport addresses.
- */
- uint32_t pmtu;
+ /* Association : The smallest PMTU discovered for all of the
+ * PMTU : peer's transport addresses.
+ */
+ __u32 pmtu;
/* The message size at which SCTP fragmentation will occur. */
- uint32_t frag_point;
-
- /* Ack State : This flag indicates if the next received
- * : packet is to be responded to with a
- * : SACK. This is initializedto 0. When a packet
- * : is received it is incremented. If this value
- * : reaches 2 or more, a SACK is sent and the
- * : value is reset to 0. Note: This is used only
- * : when no DATA chunks are received out of
- * : order. When DATA chunks are out of order,
- * : SACK's are not delayed (see Section 6).
- */
- /* Do we need to send an ack?
- * When counters[SctpCounterAckState] is above 1 we do!
- */
- int counters[SCTP_NUMBER_COUNTERS];
-
- struct {
- uint16_t stream;
- uint32_t ppid;
- } defaults;
-
- /* This tracks outbound ssn for a given stream. */
- uint16_t ssn[SCTP_MAX_STREAM];
-
- /* All outbound chunks go through this structure. */
- sctp_outqueue_t outqueue;
-
- /* A smart pipe that will handle reordering and fragmentation,
+ __u32 frag_point;
+
+ /* Ack State : This flag indicates if the next received
+ * : packet is to be responded to with a
+ * : SACK. This is initializedto 0. When a packet
+ * : is received it is incremented. If this value
+ * : reaches 2 or more, a SACK is sent and the
+ * : value is reset to 0. Note: This is used only
+ * : when no DATA chunks are received out of
+ * : order. When DATA chunks are out of order,
+ * : SACK's are not delayed (see Section 6).
+ */
+ /* Do we need to send an ack?
+ * When counters[SctpCounterAckState] is above 1 we do!
+ */
+ int counters[SCTP_NUMBER_COUNTERS];
+
+ struct {
+ __u16 stream;
+ __u32 ppid;
+ } defaults;
+
+ /* This tracks outbound ssn for a given stream. */
+ __u16 ssn[SCTP_MAX_STREAM];
+
+ /* All outbound chunks go through this structure. */
+ sctp_outqueue_t outqueue;
+
+ /* A smart pipe that will handle reordering and fragmentation,
* as well as handle passing events up to the ULP.
* In the future, we should make this at least dynamic, if
- * not also some sparse structure.
+ * not also some sparse structure.
*/
- sctp_ulpqueue_t ulpq;
- uint8_t _ssnmap[sctp_ulpqueue_storage_size(SCTP_MAX_STREAM)];
+ sctp_ulpqueue_t ulpq;
+ __u8 _ssnmap[sctp_ulpqueue_storage_size(SCTP_MAX_STREAM)];
/* Need to send an ECNE Chunk? */
- int need_ecne;
-
+ int need_ecne;
+
/* Last TSN that caused an ECNE Chunk to be sent. */
- uint32_t last_ecne_tsn;
+ __u32 last_ecne_tsn;
/* Last TSN that caused a CWR Chunk to be sent. */
- uint32_t last_cwr_tsn;
-
+ __u32 last_cwr_tsn;
+
/* How many duplicated TSNs have we seen? */
- int numduptsns;
+ int numduptsns;
+
+ /* Number of seconds of idle time before an association is closed. */
+ __u32 autoclose;
- /* Number of seconds of idle time before an association is closed. */
- uint32_t autoclose;
-
- /* Name for debugging output... */
- char *debug_name;
+ /* Name for debugging output... */
+ char *debug_name;
/* These are to support
* "SCTP Extensions for Dynamic Reconfiguration of IP Addresses
int addip_enable;
/* ADDIP Section 4.1.1 Congestion Control of ASCONF Chunks
- *
+ *
* R1) One and only one ASCONF Chunk MAY be in transit and
* unacknowledged at any one time. If a sender, after sending
* an ASCONF chunk, decides it needs to transfer another
* subsequent ASCONF. Note this restriction binds each side,
* so at any time two ASCONF may be in-transit on any given
* association (one sent from each endpoint).
- *
+ *
* [This is our one-and-only-one ASCONF in flight. If we do
* not have an ASCONF in flight, this is NULL.]
*/
sctp_chunk_t *addip_last_asconf;
/* ADDIP Section 4.2 Upon reception of an ASCONF Chunk.
- *
+ *
* IMPLEMENTATION NOTE: As an optimization a receiver may wish
* to save the last ASCONF-ACK for some predetermined period
* of time and instead of re-processing the ASCONF (with the
* essential that these transfers MUST NOT cause congestion
* within the network. To achieve this, we place these
* restrictions on the transfer of ASCONF Chunks:
- *
+ *
* R1) One and only one ASCONF Chunk MAY be in transit and
* unacknowledged at any one time. If a sender, after sending
* an ASCONF chunk, decides it needs to transfer another
* subsequent ASCONF. Note this restriction binds each side,
* so at any time two ASCONF may be in-transit on any given
* association (one sent from each endpoint).
- *
- *
+ *
+ *
* [I really think this is EXACTLY the sort of intelligence
* which already resides in SCTP_outqueue. Please move this
- * queue and its supporting logic down there. --piggy] */
+ * queue and its supporting logic down there. --piggy]
+ */
struct sk_buff_head addip_chunks;
/* ADDIP Section 4.1 ASCONF Chunk Procedures
- *
+ *
* A2) A serial number should be assigned to the Chunk. The
* serial number should be a monotonically increasing
* number. All serial numbers are defined to be initialized at
* the start of the association to the same value as the
* Initial TSN.
- *
+ *
* [and]
*
* ADDIP
* 3.1.1 Address/Stream Configuration Change Chunk (ASCONF)
- *
+ *
* Serial Number : 32 bits (unsigned integer)
- *
+ *
* This value represents a Serial Number for the ASCONF
* Chunk. The valid range of Serial Number is from 0 to
* 4294967295 (2**32 - 1). Serial Numbers wrap back to 0
* after reaching 4294967295.
*/
- uint32_t addip_serial;
-}; /* sctp_association_t */
+ __u32 addip_serial;
+};
/* An eyecatcher for determining if we are really looking at an
static inline sctp_association_t *sctp_assoc(sctp_endpoint_common_t *base)
{
sctp_association_t *asoc;
-
- /* We are not really a list, but the list_entry() macro is
- * really quite generic find the address of an outter struct.
+
+ /* We are not really a list, but the list_entry() macro is
+ * really quite generic find the address of an outter struct.
*/
asoc = list_entry(base, sctp_association_t, base);
- return asoc;
-
-} /* sctp_asoc() */
+ return asoc;
+}
/* These are function signatures for manipulating associations. */
sctp_association_new(const sctp_endpoint_t *, const struct sock *,
sctp_scope_t scope, int priority);
sctp_association_t *
-sctp_association_init(sctp_association_t *, const sctp_endpoint_t *,
- const struct sock *, sctp_scope_t scope,
+sctp_association_init(sctp_association_t *, const sctp_endpoint_t *,
+ const struct sock *, sctp_scope_t scope,
int priority);
void sctp_association_free(sctp_association_t *);
void sctp_association_put(sctp_association_t *);
void sctp_association_hold(sctp_association_t *);
sctp_transport_t *sctp_assoc_choose_shutdown_transport(sctp_association_t *);
-sctp_transport_t *sctp_assoc_lookup_paddr(const sctp_association_t *,
+sctp_transport_t *sctp_assoc_lookup_paddr(const sctp_association_t *,
const sockaddr_storage_t *);
-sctp_transport_t *sctp_assoc_add_peer(sctp_association_t *,
+sctp_transport_t *sctp_assoc_add_peer(sctp_association_t *,
const sockaddr_storage_t *address,
const int priority);
void sctp_assoc_control_transport(sctp_association_t *, sctp_transport_t *,
sctp_transport_cmd_t, sctp_sn_error_t);
-sctp_transport_t *sctp_assoc_lookup_tsn(sctp_association_t *, uint32_t);
-sctp_transport_t *sctp_assoc_is_match(sctp_association_t *,
- const sockaddr_storage_t *,
+sctp_transport_t *sctp_assoc_lookup_tsn(sctp_association_t *, __u32);
+sctp_transport_t *sctp_assoc_is_match(sctp_association_t *,
+ const sockaddr_storage_t *,
const sockaddr_storage_t *);
void sctp_assoc_migrate(sctp_association_t *, struct sock *);
void sctp_assoc_update(sctp_association_t *dst, sctp_association_t *src);
-uint32_t __sctp_association_get_next_tsn(sctp_association_t *);
-uint32_t __sctp_association_get_tsn_block(sctp_association_t *, int);
-uint16_t __sctp_association_get_next_ssn(sctp_association_t *, uint16_t sid);
+__u32 __sctp_association_get_next_tsn(sctp_association_t *);
+__u32 __sctp_association_get_tsn_block(sctp_association_t *, int);
+__u16 __sctp_association_get_next_ssn(sctp_association_t *, __u16 sid);
int sctp_cmp_addr(const sockaddr_storage_t *ss1,
const sockaddr_storage_t *ss2);
char *label;
atomic_t *counter;
} sctp_dbg_objcnt_entry_t;
-
-
#endif /* __sctp_structs_h__ */
*/
typedef struct sctp_tsnmap {
-
+
/* This array counts the number of chunks with each TSN.
* It points at one of the two buffers with which we will
- * ping-pong between.
+ * ping-pong between.
*/
- uint8_t *tsn_map;
+ __u8 *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).
*/
- uint32_t overflow_tsn;
+ __u32 overflow_tsn;
/* This is the overflow array for tsn_map.
* It points at one of the other ping-pong buffers.
*/
- uint8_t *overflow_map;
+ __u8 *overflow_map;
/* This is the TSN at tsn_map[0]. */
- uint32_t base_tsn;
+ __u32 base_tsn;
- /* Last Rcvd : This is the last TSN received in
+ /* Last Rcvd : This is the last TSN received in
* TSN : sequence. This value is set initially by
* : taking the peer's Initial TSN, received in
* : the INIT or INIT ACK chunk, and subtracting
- * : one from it.
+ * : one from it.
*
* Throughout most of the specification this is called the
* "Cumulative TSN ACK Point". In this case, we
* ignore the advice in 12.2 in favour of the term
* used in the bulk of the text.
- */
- uint32_t cumulative_tsn_ack_point;
+ */
+ __u32 cumulative_tsn_ack_point;
/* 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
- * potentially track more than this quantity.
+ * to the size of tsn_map. Note: the overflow_map allows us to
+ * potentially track more than this quantity.
*/
- uint16_t len;
+ __u16 len;
/* This is the highest TSN we've marked. */
- uint32_t max_tsn_seen;
+ __u32 max_tsn_seen;
/* No. of data chunks pending receipt. used by SCTP_STATUS sockopt */
- uint16_t pending_data;
+ __u16 pending_data;
int malloced;
- uint8_t raw_map[0];
+ __u8 raw_map[0];
} sctp_tsnmap_t;
typedef struct sctp_tsnmap_iter {
- uint32_t start;
+ __u32 start;
} sctp_tsnmap_iter_t;
/* Create a new tsnmap. */
-sctp_tsnmap_t *sctp_tsnmap_new(uint16_t len, uint32_t initial_tsn,
+sctp_tsnmap_t *sctp_tsnmap_new(__u16 len, __u32 initial_tsn,
int priority);
/* Dispose of a tsnmap. */
void sctp_tsnmap_free(sctp_tsnmap_t *map);
-/* This macro assists in creation of external storage for variable length
+/* 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(uint8_t) * (count) * 2)
+#define sctp_tsnmap_storage_size(count) (sizeof(__u8) * (count) * 2)
/* Initialize a block of memory as a tsnmap. */
-sctp_tsnmap_t *sctp_tsnmap_init(sctp_tsnmap_t *map, uint16_t len,
- uint32_t initial_tsn);
+sctp_tsnmap_t *sctp_tsnmap_init(sctp_tsnmap_t *map, __u16 len, __u32 initial_tsn);
* >0 if the TSN has been seen (duplicate)
* <0 if the TSN is invalid (too large to track)
*/
-int sctp_tsnmap_check(const sctp_tsnmap_t *map, uint32_t tsn);
+int sctp_tsnmap_check(const sctp_tsnmap_t *map, __u32 tsn);
/* Mark this TSN as seen. */
-void sctp_tsnmap_mark(sctp_tsnmap_t *map, uint32_t tsn);
+void sctp_tsnmap_mark(sctp_tsnmap_t *map, __u32 tsn);
/* Retrieve the Cumulative TSN ACK Point. */
-uint32_t sctp_tsnmap_get_ctsn(const sctp_tsnmap_t *map);
+__u32 sctp_tsnmap_get_ctsn(const sctp_tsnmap_t *map);
/* Retrieve the highest TSN we've seen. */
-uint32_t sctp_tsnmap_get_max_tsn_seen(const sctp_tsnmap_t *map);
+__u32 sctp_tsnmap_get_max_tsn_seen(const sctp_tsnmap_t *map);
/* Is there a gap in the TSN map? */
int sctp_tsnmap_has_gap(const sctp_tsnmap_t *map);
/* Get the next gap ack blocks. We return 0 if there are no more
* gap ack blocks.
*/
-int
-sctp_tsnmap_next_gap_ack(const sctp_tsnmap_t *map, sctp_tsnmap_iter_t *iter,
- uint16_t *start, uint16_t *end);
+int sctp_tsnmap_next_gap_ack(const sctp_tsnmap_t *map, sctp_tsnmap_iter_t *iter,
+ __u16 *start, __u16 *end);
#endif /* __sctp_tsnmap_h__ */
#define __sctp_ulpevent_h__
/* A structure to carry information to the ULP (e.g. Sockets API) */
-/* Warning: This sits inside an skb.cb[] area. Be very careful of
- * growing this structure as it is at the maximum limit now.
+/* Warning: This sits inside an skb.cb[] area. Be very careful of
+ * growing this structure as it is at the maximum limit now.
*/
typedef struct sctp_ulpevent {
int malloced;
} sctp_ulpevent_t;
-sctp_ulpevent_t *
-sctp_ulpevent_new(int size, int msg_flags, int priority);
+sctp_ulpevent_t *sctp_ulpevent_new(int size, int msg_flags, int priority);
-sctp_ulpevent_t *
-sctp_ulpevent_init(sctp_ulpevent_t *event, struct sk_buff *skb, int msg_flags);
+sctp_ulpevent_t *sctp_ulpevent_init(sctp_ulpevent_t *event, struct sk_buff *skb, int msg_flags);
-void
-sctp_ulpevent_free(sctp_ulpevent_t *event);
+void sctp_ulpevent_free(sctp_ulpevent_t *event);
+int sctp_ulpevent_is_notification(const sctp_ulpevent_t *event);
-int
-sctp_ulpevent_is_notification(const sctp_ulpevent_t *event);
-
-sctp_ulpevent_t *
-sctp_ulpevent_make_assoc_change(const struct SCTP_association *asoc,
- uint16_t flags,
- uint16_t state,
- uint16_t error,
- uint16_t outbound,
- uint16_t inbound,
+sctp_ulpevent_t *sctp_ulpevent_make_assoc_change(
+ const struct SCTP_association *asoc,
+ __u16 flags,
+ __u16 state,
+ __u16 error,
+ __u16 outbound,
+ __u16 inbound,
int priority);
-sctp_ulpevent_t *
-sctp_ulpevent_make_peer_addr_change(const struct SCTP_association *asoc,
- const struct sockaddr_storage *aaddr,
- int flags,
- int state,
- int error,
- int priority);
+sctp_ulpevent_t *sctp_ulpevent_make_peer_addr_change(
+ const struct SCTP_association *asoc,
+ const struct sockaddr_storage *aaddr,
+ int flags,
+ int state,
+ int error,
+ int priority);
-sctp_ulpevent_t *
-sctp_ulpevent_make_remote_error(const struct SCTP_association *asoc,
+sctp_ulpevent_t *sctp_ulpevent_make_remote_error(
+ const struct SCTP_association *asoc,
+ struct SCTP_chunk *chunk,
+ __u16 flags,
+ int priority);
+sctp_ulpevent_t *sctp_ulpevent_make_send_failed(
+ const struct SCTP_association *asoc,
struct SCTP_chunk *chunk,
- uint16_t flags,
+ __u16 flags,
+ __u32 error,
int priority);
-sctp_ulpevent_t *
-sctp_ulpevent_make_send_failed(const struct SCTP_association *asoc,
- struct SCTP_chunk *chunk,
- uint16_t flags,
- uint32_t error,
- int priority);
-sctp_ulpevent_t *
-sctp_ulpevent_make_shutdown_event(const struct SCTP_association *asoc,
- uint16_t flags,
- int priority);
+sctp_ulpevent_t *sctp_ulpevent_make_shutdown_event(
+ const struct SCTP_association *asoc,
+ __u16 flags,
+ int priority);
-sctp_ulpevent_t *
-sctp_ulpevent_make_rcvmsg(struct SCTP_association *asoc,
- struct SCTP_chunk *chunk,
- int priority);
+sctp_ulpevent_t *sctp_ulpevent_make_rcvmsg(struct SCTP_association *asoc,
+ struct SCTP_chunk *chunk,
+ int priority);
-void
-sctp_ulpevent_read_sndrcvinfo(const sctp_ulpevent_t *event,
- struct msghdr *msghdr);
+void sctp_ulpevent_read_sndrcvinfo(const sctp_ulpevent_t *event,
+ struct msghdr *msghdr);
-uint16_t
-sctp_ulpevent_get_notification_type(const sctp_ulpevent_t *event);
+__u16 sctp_ulpevent_get_notification_type(const sctp_ulpevent_t *event);
/* Given an event subscription, is this event enabled? */
-static inline int
-sctp_ulpevent_is_enabled(const sctp_ulpevent_t *event,
- const struct sctp_event_subscribe *mask)
+static inline int sctp_ulpevent_is_enabled(const sctp_ulpevent_t *event,
+ const struct sctp_event_subscribe *mask)
{
- const char *amask = (const char *)mask;
- uint16_t sn_type;
-
+ const char *amask = (const char *) mask;
+ __u16 sn_type;
int enabled = 1;
+
if (sctp_ulpevent_is_notification(event)) {
sn_type = sctp_ulpevent_get_notification_type(event);
enabled = amask[sn_type - SCTP_SN_TYPE_BASE];
}
- return(enabled);
-
-} /* sctp_ulpevent_is_enabled() */
+ return enabled;
+}
#endif /* __sctp_ulpevent_h__ */
sctp_association_t *asoc;
struct sk_buff_head reasm;
struct sk_buff_head lobby;
- uint16_t ssn[0];
+ __u16 ssn[0];
} sctp_ulpqueue_t;
-/* This macro assists in creation of external storage for variable length
- * internal buffers.
+/* This macro assists in creation of external storage for variable length
+ * internal buffers.
*/
-#define sctp_ulpqueue_storage_size(inbound) (sizeof(uint16_t) * (inbound))
+#define sctp_ulpqueue_storage_size(inbound) (sizeof(__u16) * (inbound))
-sctp_ulpqueue_t *
-sctp_ulpqueue_new(sctp_association_t *asoc,
- uint16_t inbound,
- int priority);
+sctp_ulpqueue_t *sctp_ulpqueue_new(sctp_association_t *asoc,
+ __u16 inbound,
+ int priority);
-sctp_ulpqueue_t *
-sctp_ulpqueue_init(sctp_ulpqueue_t *ulpq,
- sctp_association_t *asoc,
- uint16_t inbound);
+sctp_ulpqueue_t *sctp_ulpqueue_init(sctp_ulpqueue_t *ulpq,
+ sctp_association_t *asoc,
+ __u16 inbound);
-void
-sctp_ulpqueue_free(sctp_ulpqueue_t *);
+void sctp_ulpqueue_free(sctp_ulpqueue_t *);
/* Add a new DATA chunk for processing. */
-int
-sctp_ulpqueue_tail_data(sctp_ulpqueue_t *,
- sctp_chunk_t *chunk,
- int priority);
+int sctp_ulpqueue_tail_data(sctp_ulpqueue_t *,
+ sctp_chunk_t *chunk,
+ int priority);
/* Add a new event for propogation to the ULP. */
-int
-sctp_ulpqueue_tail_event(sctp_ulpqueue_t *,
- sctp_ulpevent_t *event);
+int sctp_ulpqueue_tail_event(sctp_ulpqueue_t *,
+ sctp_ulpevent_t *event);
/* Is the ulpqueue empty. */
-int
-sctp_ulpqueue_is_empty(sctp_ulpqueue_t *);
+int sctp_ulpqueue_is_empty(sctp_ulpqueue_t *);
-int
-sctp_ulpqueue_is_data_empty(sctp_ulpqueue_t *);
+int sctp_ulpqueue_is_data_empty(sctp_ulpqueue_t *);
#endif /* __sctp_ulpqueue_h__ */
typedef void * sctp_assoc_t;
-/* The following symbols come from the Sockets API Extensions for
+/* The following symbols come from the Sockets API Extensions for
* SCTP <draft-ietf-tsvwg-sctpsocket-04.txt>.
*/
enum sctp_optname {
#define SCTP_SOCKOPT_BINDX_REM SCTP_SOCKOPT_BINDX_REM
SCTP_SOCKOPT_PEELOFF, /* peel off association. */
#define SCTP_SOCKOPT_PEELOFF SCTP_SOCKOPT_PEELOFF
-}; /* enum sctp_optname */
+};
-/*
+/*
* 5.2 SCTP msg_control Structures
*
* A key element of all SCTP-specific socket extensions is the use of
* contained in in the cmsg_data[] member.
*/
-/*
+/*
* 5.2.1 SCTP Initiation Structure (SCTP_INIT)
- *
+ *
* This cmsghdr structure provides information for initializing new
* SCTP associations with sendmsg(). The SCTP_INITMSG socket option
* uses this same data structure. This structure is not used for
*
*/
struct sctp_initmsg {
- uint16_t sinit_num_ostreams;
- uint16_t sinit_max_instreams;
- uint16_t sinit_max_attempts;
- uint16_t sinit_max_init_timeo;
-}; /* struct sctp_initmsg */
+ __u16 sinit_num_ostreams;
+ __u16 sinit_max_instreams;
+ __u16 sinit_max_attempts;
+ __u16 sinit_max_init_timeo;
+};
-/*
+/*
* 5.2.2 SCTP Header Information Structure (SCTP_SNDRCV)
- *
+ *
* This cmsghdr structure specifies SCTP options for sendmsg() and
* describes SCTP header information about a received message through
* recvmsg().
*
*/
struct sctp_sndrcvinfo {
- uint16_t sinfo_stream;
- uint16_t sinfo_ssn;
- uint16_t sinfo_flags;
- uint32_t sinfo_ppid;
- uint32_t sinfo_context;
- uint32_t sinfo_timetolive;
- uint32_t sinfo_tsn;
- sctp_assoc_t sinfo_assoc_id;
-}; /* struct sctp_sndrcvinfo */
+ __u16 sinfo_stream;
+ __u16 sinfo_ssn;
+ __u16 sinfo_flags;
+ __u32 sinfo_ppid;
+ __u32 sinfo_context;
+ __u32 sinfo_timetolive;
+ __u32 sinfo_tsn;
+ sctp_assoc_t sinfo_assoc_id;
+};
/*
* sinfo_flags: 16 bits (unsigned integer)
MSG_ADDR_OVER = 2, /* Override the primary destination. */
MSG_ABORT=4, /* Send an ABORT message to the peer. */
/* MSG_EOF is already defined per socket.h */
-}; /* enum sctp_sinfo_flags */
+};
typedef union {
- u_int8_t raw;
- struct sctp_initmsg init;
- struct sctp_sndrcvinfo sndrcv;
+ __u8 raw;
+ struct sctp_initmsg init;
+ struct sctp_sndrcvinfo sndrcv;
} sctp_cmsg_data_t;
/* These are cmsg_types. */
typedef enum sctp_cmsg_type {
- SCTP_INIT, /* 5.2.1 SCTP Initiation Structure */
- SCTP_SNDRCV, /* 5.2.2 SCTP Header Information Structure */
+ SCTP_INIT, /* 5.2.1 SCTP Initiation Structure */
+ SCTP_SNDRCV, /* 5.2.2 SCTP Header Information Structure */
} sctp_cmsg_t;
-/*
+/*
* 5.3.1.1 SCTP_ASSOC_CHANGE
*
* Communication notifications inform the ULP that an SCTP association
*/
struct sctp_assoc_change {
- uint16_t sac_type;
- uint16_t sac_flags;
- uint32_t sac_length;
- uint16_t sac_state;
- uint16_t sac_error;
- uint16_t sac_outbound_streams;
- uint16_t sac_inbound_streams;
+ __u16 sac_type;
+ __u16 sac_flags;
+ __u32 sac_length;
+ __u16 sac_state;
+ __u16 sac_error;
+ __u16 sac_outbound_streams;
+ __u16 sac_inbound_streams;
sctp_assoc_t sac_assoc_id;
-}; /* sctp_assoc_change */
+};
/*
* sac_state: 32 bits (signed integer)
*
* This field holds one of a number of values that communicate the
* event that happened to the association. They include:
- *
+ *
* Note: The following state names deviate from the API draft as
* the names clash too easily with other kernel symbols.
*/
SCTP_RESTART,
SCTP_SHUTDOWN_COMP,
SCTP_CANT_STR_ASSOC,
-}; /* sctp_sac_state */
+};
-/*
+/*
* 5.3.1.2 SCTP_PEER_ADDR_CHANGE
*
* When a destination address on a multi-homed peer encounters a change
* an interface details event is sent. The information has the
* following structure:
*/
-struct sctp_paddr_change{
- uint16_t spc_type;
- uint16_t spc_flags;
- uint32_t spc_length;
+struct sctp_paddr_change {
+ __u16 spc_type;
+ __u16 spc_flags;
+ __u32 spc_length;
struct sockaddr_storage spc_aaddr;
int spc_state;
int spc_error;
sctp_assoc_t spc_assoc_id;
-}; /* sctp_paddr_change */
+};
/*
* spc_state: 32 bits (signed integer)
ADDRESS_REMOVED,
ADDRESS_ADDED,
ADDRESS_MADE_PRIM,
-}; /* sctp_spc_state */
+};
-/*
+/*
* 5.3.1.3 SCTP_REMOTE_ERROR
*
* A remote peer may send an Operational Error message to its peer.
* error formats. SCTP error TLVs have the format:
*/
struct sctp_remote_error {
- uint16_t sre_type;
- uint16_t sre_flags;
- uint32_t sre_length;
- uint16_t sre_error;
- uint16_t sre_len;
- sctp_assoc_t sre_assoc_id;
- uint8_t sre_data[0];
+ __u16 sre_type;
+ __u16 sre_flags;
+ __u32 sre_length;
+ __u16 sre_error;
+ __u16 sre_len;
+ sctp_assoc_t sre_assoc_id;
+ __u8 sre_data[0];
};
-/*
+/*
* 5.3.1.4 SCTP_SEND_FAILED
*
* If SCTP cannot deliver a message it may return the message as a
* notification.
*/
struct sctp_send_failed {
- uint16_t ssf_type;
- uint16_t ssf_flags;
- uint32_t ssf_length;
- uint32_t ssf_error;
- struct sctp_sndrcvinfo ssf_info;
- sctp_assoc_t ssf_assoc_id;
- uint8_t ssf_data[0];
+ __u16 ssf_type;
+ __u16 ssf_flags;
+ __u32 ssf_length;
+ __u32 ssf_error;
+ struct sctp_sndrcvinfo ssf_info;
+ sctp_assoc_t ssf_assoc_id;
+ __u8 ssf_data[0];
};
/*
*/
struct sctp_shutdown_event {
- uint16_t sse_type;
- uint16_t sse_flags;
- uint32_t sse_length;
- sctp_assoc_t sse_assoc_id;
+ __u16 sse_type;
+ __u16 sse_flags;
+ __u32 sse_length;
+ sctp_assoc_t sse_assoc_id;
};
-/*
+/*
* 5.3.1.6 SCTP_ADAPTION_INDICATION
*
* When a peer sends a Adaption Layer Indication parameter , SCTP
- * delivers this notification to inform the application
+ * delivers this notification to inform the application
* that of the peers requested adaption layer.
*/
struct sctp_adaption_event {
- uint16_t sai_type;
- uint16_t sai_flags;
- uint32_t sai_length;
- uint32_t sai_adaptation_bits;
- sctp_assoc_t sse_assoc_id;
+ __u16 sai_type;
+ __u16 sai_flags;
+ __u32 sai_length;
+ __u32 sai_adaptation_bits;
+ sctp_assoc_t sse_assoc_id;
};
-/*
+/*
* 5.3.1.7 SCTP_PARTIAL_DELIVERY_EVENT
*
* When a reciever is engaged in a partial delivery of a
* message this notification will be used to inidicate
- * various events.
+ * various events.
*/
struct sctp_rcv_pdapi_event {
- uint16_t pdapi_type;
- uint16_t pdapi_flags;
- uint32_t pdapi_length;
- uint32_t pdapi_indication;
- sctp_assoc_t pdapi_assoc_id;
+ __u16 pdapi_type;
+ __u16 pdapi_flags;
+ __u32 pdapi_length;
+ __u32 pdapi_indication;
+ sctp_assoc_t pdapi_assoc_id;
};
-/*
- * Described in Section 7.3
+/*
+ * Described in Section 7.3
* Ancillary Data and Notification Interest Options
*/
struct sctp_event_subscribe {
- uint8_t sctp_data_io_event;
- uint8_t sctp_association_event;
- uint8_t sctp_address_event;
- uint8_t sctp_send_failure_event;
- uint8_t sctp_peer_error_event;
- uint8_t sctp_shutdown_event;
- uint8_t sctp_partial_delivery_event;
- uint8_t sctp_adaption_layer_event;
-}; /* struct sctp_event_subscribe */
-
-/*
+ __u8 sctp_data_io_event;
+ __u8 sctp_association_event;
+ __u8 sctp_address_event;
+ __u8 sctp_send_failure_event;
+ __u8 sctp_peer_error_event;
+ __u8 sctp_shutdown_event;
+ __u8 sctp_partial_delivery_event;
+ __u8 sctp_adaption_layer_event;
+};
+
+/*
* 5.3.1 SCTP Notification Structure
*
* The notification structure is defined as the union of all
*/
union sctp_notification {
struct {
- uint16_t sn_type; /* Notification type. */
- uint16_t sn_flags;
- uint32_t sn_length;
+ __u16 sn_type; /* Notification type. */
+ __u16 sn_flags;
+ __u32 sn_length;
} h;
- struct sctp_assoc_change sn_assoc_change;
- struct sctp_paddr_change sn_padr_change;
- struct sctp_remote_error sn_remote_error;
+ struct sctp_assoc_change sn_assoc_change;
+ struct sctp_paddr_change sn_padr_change;
+ struct sctp_remote_error sn_remote_error;
struct sctp_send_failed sn_send_failed;
- struct sctp_shutdown_event sn_shutdown_event;
+ struct sctp_shutdown_event sn_shutdown_event;
struct sctp_adaption_event sn_adaption_event;
- struct sctp_rcv_pdapi_event sn_rcv_pdapi_event;
-};
+ struct sctp_rcv_pdapi_event sn_rcv_pdapi_event;
+};
/* Section 5.3.1
* All standard values for sn_type flags are greater than 2^15.
SCTP_ADAPTION_INDICATION,
};
-/* Notification error codes used to fill up the error fields in some
+/* Notification error codes used to fill up the error fields in some
* notifications.
* SCTP_PEER_ADDRESS_CHAGE : spc_error
* SCTP_ASSOC_CHANGE : sac_error
- * These names should be potentially included in the draft 04 of the SCTP
+ * These names should be potentially included in the draft 04 of the SCTP
* sockets API specification.
- */
+ */
typedef enum sctp_sn_error {
SCTP_FAILED_THRESHOLD,
SCTP_RECEIVED_SACK,
SCTP_PEER_FAULTY,
} sctp_sn_error_t;
-/*
+/*
*
* 7.1.14 Peer Address Parameters
*
*/
struct sctp_paddrparams {
- struct sockaddr_storage spp_address;
- uint32_t spp_hbinterval;
- uint16_t spp_pathmaxrxt;
- sctp_assoc_t spp_assoc_id;
+ struct sockaddr_storage spp_address;
+ __u32 spp_hbinterval;
+ __u16 spp_pathmaxrxt;
+ sctp_assoc_t spp_assoc_id;
};
-/*
+/*
* 7.2.2 Peer Address Information
*
* Applications can retrieve information about a specific peer address
*/
struct sctp_paddrinfo {
- sctp_assoc_t spinfo_assoc_id;
- struct sockaddr_storage spinfo_address;
- int32_t spinfo_state;
- uint32_t spinfo_cwnd;
- uint32_t spinfo_srtt;
- uint32_t spinfo_rto;
- uint32_t spinfo_mtu;
+ sctp_assoc_t spinfo_assoc_id;
+ struct sockaddr_storage spinfo_address;
+ __s32 spinfo_state;
+ __u32 spinfo_cwnd;
+ __u32 spinfo_srtt;
+ __u32 spinfo_rto;
+ __u32 spinfo_mtu;
};
-/*
+/*
* 7.1.1 Retransmission Timeout Parameters (SCTP_RTOINFO)
*
* The protocol parameters used to initialize and bound retransmission
*/
struct sctp_rtoinfo {
- uint32_t srto_initial;
- uint32_t srto_max;
- uint32_t srto_min;
- sctp_assoc_t srto_assoc_id;
+ __u32 srto_initial;
+ __u32 srto_max;
+ __u32 srto_min;
+ sctp_assoc_t srto_assoc_id;
};
-/*
+/*
* 7.1.2 Association Retransmission Parameter (SCTP_ASSOCRTXINFO)
*
* The protocol parameter used to set the number of retransmissions
*/
struct sctp_assocparams {
- uint16_t sasoc_asocmaxrxt;
- sctp_assoc_t sasoc_assoc_id;
+ __u16 sasoc_asocmaxrxt;
+ sctp_assoc_t sasoc_assoc_id;
};
-/*
+/*
* 7.1.9 Set Primary Address (SCTP_SET_PRIMARY_ADDR)
*
* Requests that the peer mark the enclosed address as the association
*/
struct sctp_setprim {
- struct sockaddr_storage ssp_addr;
- sctp_assoc_t ssp_assoc_id;
+ struct sockaddr_storage ssp_addr;
+ sctp_assoc_t ssp_assoc_id;
};
-/*
+/*
* 7.1.10 Set Peer Primary Address (SCTP_SET_PEER_PRIMARY_ADDR)
*
* Requests that the local SCTP stack use the enclosed peer address as
*/
struct sctp_setpeerprim {
- struct sockaddr_storage sspp_addr;
- sctp_assoc_t sspp_assoc_id;
+ struct sockaddr_storage sspp_addr;
+ sctp_assoc_t sspp_assoc_id;
};
-/*
+/*
* 7.2.1 Association Status (SCTP_STATUS)
*
* Applications can retrieve current status information about an
* used to access this information:
*/
struct sctp_status {
- sctp_assoc_t sstat_assoc_id;
- int32_t sstat_state;
- uint32_t sstat_rwnd;
- uint16_t sstat_unackdata;
- uint16_t sstat_penddata;
- uint16_t sstat_instrms;
- uint16_t sstat_outstrms;
- uint32_t sstat_fragmentation_point;
- struct sctp_paddrinfo sstat_primary;
+ sctp_assoc_t sstat_assoc_id;
+ __s32 sstat_state;
+ __u32 sstat_rwnd;
+ __u16 sstat_unackdata;
+ __u16 sstat_penddata;
+ __u16 sstat_instrms;
+ __u16 sstat_outstrms;
+ __u32 sstat_fragmentation_point;
+ struct sctp_paddrinfo sstat_primary;
};
-/*
- * 7.1.12 Set Adaption Layer Indicator
- *
- * Requests that the local endpoint set the specified Adaption Layer
+/*
+ * 7.1.12 Set Adaption Layer Indicator
+ *
+ * Requests that the local endpoint set the specified Adaption Layer
* Indication parameter for all future
- * INIT and INIT-ACK exchanges.
+ * INIT and INIT-ACK exchanges.
*/
struct sctp_setadaption {
- u_int32_t ssb_adaption_ind;
+ __u32 ssb_adaption_ind;
};
-/*
+/*
* 7.1.12 Set default message time outs (SCTP_SET_STREAM_TIMEOUTS)
- *
+ *
* This option requests that the requested stream apply a
* default time-out for messages in queue.
*/
struct sctp_setstrm_timeout {
- sctp_assoc_t ssto_assoc_id;
- u_int32_t ssto_timeout;
- u_int16_t ssto_streamid_start;
- u_int16_t ssto_streamid_end;
+ sctp_assoc_t ssto_assoc_id;
+ __u32 ssto_timeout;
+ __u16 ssto_streamid_start;
+ __u16 ssto_streamid_end;
};
enum sctp_msg_flags {
MSG_NOTIFICATION = 0x8000,
#define MSG_NOTIFICATION MSG_NOTIFICATION
-}; /* enum sctp_msg_flags */
+};
-/*
+/*
* 8.1 sctp_bindx()
*
- * The flags parameter is formed from the bitwise OR of zero or more of the
+ * The flags parameter is formed from the bitwise OR of zero or more of the
* following currently defined flags:
*/
#define BINDX_ADD_ADDR 0x01
#define BINDX_REM_ADDR 0x02
-/* This is the structure that is passed as an argument(optval) to
+/* This is the structure that is passed as an argument(optval) to
* getsockopt(SCTP_SOCKOPT_PEELOFF).
*/
typedef struct {
-- network-level fault tolerance through supporting of multi-
homing at either or both ends of an association."
+ This protocol support is also available as a module ( = code which
+ can be inserted in and removed from the running kernel whenever you
+ want). The module will be called sctp.o. If you want to compile it
+ as a module, say M here and read <file:Documentation/modules.txt>.
+
If in doubt, say N.
CONFIG_SCTP_ADLER32
obj-$(CONFIG_IP_SCTP) += sctp.o
-obj-y := sctp_sm_statetable.o sctp_sm_statefuns.o sctp_sm_sideeffect.o \
- sctp_protocol.o sctp_endpointola.o sctp_associola.o \
- sctp_transport.o sctp_sm_make_chunk.o sctp_ulpevent.o \
- sctp_inqueue.o sctp_outqueue.o sctp_ulpqueue.o sctp_command.o \
- sctp_tsnmap.o sctp_bind_addr.o sctp_socket.o sctp_primitive.o \
- sctp_output.o sctp_input.o sctp_hashdriver.o sctp_sla1.o \
- sctp_debug.o
+sctp-y := sctp_sm_statetable.o sctp_sm_statefuns.o sctp_sm_sideeffect.o \
+ sctp_protocol.o sctp_endpointola.o sctp_associola.o \
+ sctp_transport.o sctp_sm_make_chunk.o sctp_ulpevent.o \
+ sctp_inqueue.o sctp_outqueue.o sctp_ulpqueue.o sctp_command.o \
+ sctp_tsnmap.o sctp_bind_addr.o sctp_socket.o sctp_primitive.o \
+ sctp_output.o sctp_input.o sctp_hashdriver.o sctp_sla1.o \
+ sctp_debug.o
ifeq ($(CONFIG_SCTP_ADLER32), y)
-obj-y += sctp_adler32.o
+sctp-y += sctp_adler32.o
else
-obj-y += sctp_crc32c.o
+sctp-y += sctp_crc32c.o
endif
-obj-$(CONFIG_SCTP_DBG_OBJCNT) += sctp_objcnt.o
-obj-$(CONFIG_SYSCTL) += sctp_sysctl.o
+sctp-$(CONFIG_SCTP_DBG_OBJCNT) += sctp_objcnt.o
+sctp-$(CONFIG_SYSCTL) += sctp_sysctl.o
-obj-$(subst m,y,$(CONFIG_IPV6)) += sctp_ipv6.o
+sctp-$(subst m,y,$(CONFIG_IPV6)) += sctp_ipv6.o
-sctp-objs := $(obj-y)
+sctp-objs := $(sctp-y)
include $(TOPDIR)/Rules.make
/* This is an entry point for external calls
* Define this function in the header file. This is
* direct from rfc1950, ...
-
+ *
* The following C code computes the Adler-32 checksum of a data buffer.
- It is written for clarity, not for speed. The sample code is in the
- ANSI C programming language. Non C users may find it easier to read
- with these hints:
-
- & Bitwise AND operator.
- >> Bitwise right shift operator. When applied to an
- unsigned quantity, as here, right shift inserts zero bit(s)
- at the left.
- << Bitwise left shift operator. Left shift inserts zero
- bit(s) at the right.
- ++ "n++" increments the variable n.
- % modulo operator: a % b is the remainder of a divided by b.
-
+ * It is written for clarity, not for speed. The sample code is in the
+ * ANSI C programming language. Non C users may find it easier to read
+ * with these hints:
+ *
+ * & Bitwise AND operator.
+ * >> Bitwise right shift operator. When applied to an
+ * unsigned quantity, as here, right shift inserts zero bit(s)
+ * at the left.
+ * << Bitwise left shift operator. Left shift inserts zero
+ * bit(s) at the right.
+ * ++ "n++" increments the variable n.
+ * % modulo operator: a % b is the remainder of a divided by b.
+ *
* Well, the above is a bit of a lie, I have optimized this a small
* tad, but I have commented the original lines below
*/
* it back and we will incorporate it :-)
*/
-
unsigned long update_adler32(unsigned long adler,
unsigned char *buf, int len)
{
- uint32_t s1 = adler & 0xffff;
- uint32_t s2 = (adler >> 16) & 0xffff;
+ __u32 s1 = adler & 0xffff;
+ __u32 s2 = (adler >> 16) & 0xffff;
int n;
-
- for (n = 0; n < len; n++,buf++) {
-
- /* s1 = (s1 + buf[n]) % BASE */
- /* first we add */
- s1 = (s1 + *buf);
- /* Now if we need to, we do a mod by
- * subtracting. It seems a bit faster
- * since I really will only ever do
- * one subtract at the MOST, since buf[n]
- * is a max of 255.
- */
- if(s1 >= BASE){
- s1 -= BASE;
- }
- /* s2 = (s2 + s1) % BASE */
- /* first we add */
- s2 = (s2 + s1);
- /* again, it is more efficent (it seems) to
- * subtract since the most s2 will ever be
- * is (BASE-1 + BASE-1) in the worse case.
- * This would then be (2 * BASE) - 2, which
- * will still only do one subtract. On Intel
- * this is much better to do this way and
- * avoid the divide. Have not -pg'd on
- * sparc.
- */
- if(s2 >= BASE){
- /* s2 %= BASE;*/
- s2 -= BASE;
- }
- }
- /* Return the adler32 of the bytes buf[0..len-1] */
- return (s2 << 16) + s1;
+
+ for (n = 0; n < len; n++,buf++) {
+ /* s1 = (s1 + buf[n]) % BASE */
+ /* first we add */
+ s1 = (s1 + *buf);
+
+ /* Now if we need to, we do a mod by
+ * subtracting. It seems a bit faster
+ * since I really will only ever do
+ * one subtract at the MOST, since buf[n]
+ * is a max of 255.
+ */
+ if(s1 >= BASE)
+ s1 -= BASE;
+
+ /* s2 = (s2 + s1) % BASE */
+ /* first we add */
+ s2 = (s2 + s1);
+
+ /* again, it is more efficent (it seems) to
+ * subtract since the most s2 will ever be
+ * is (BASE-1 + BASE-1) in the worse case.
+ * This would then be (2 * BASE) - 2, which
+ * will still only do one subtract. On Intel
+ * this is much better to do this way and
+ * avoid the divide. Have not -pg'd on
+ * sparc.
+ */
+ if (s2 >= BASE) {
+ /* s2 %= BASE;*/
+ s2 -= BASE;
+ }
+ }
+
+ /* Return the adler32 of the bytes buf[0..len-1] */
+ return (s2 << 16) + s1;
}
-uint32_t
-count_crc(uint8_t *ptr,
- uint16_t count)
+__u32 count_crc(__u8 *ptr, __u16 count)
{
- /*
- * Update a running Adler-32 checksum with the bytes
- * buf[0..len-1] and return the updated checksum. The Adler-32
- * checksum should be initialized to 1.
- */
- uint32_t adler = 1L;
- uint32_t zero = 0L;
+ /*
+ * Update a running Adler-32 checksum with the bytes
+ * buf[0..len-1] and return the updated checksum. The Adler-32
+ * checksum should be initialized to 1.
+ */
+ __u32 adler = 1L;
+ __u32 zero = 0L;
/* Calculate the CRC up to the checksum field. */
- adler = update_adler32(adler, ptr,
- sizeof(struct sctphdr) - sizeof(uint32_t));
+ adler = update_adler32(adler, ptr,
+ sizeof(struct sctphdr) - sizeof(__u32));
/* Skip over the checksum field. */
- adler = update_adler32(adler, &zero, sizeof(uint32_t));
+ adler = update_adler32(adler, &zero, sizeof(__u32));
ptr += sizeof(struct sctphdr);
count -= sizeof(struct sctphdr);
+
/* Calculate the rest of the Adler-32. */
adler = update_adler32(adler, ptr, count);
-
- return(adler);
-}
+ return adler;
+}
*/
static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_associola.c,v 1.48 2002/08/16 19:30:49 jgrimm Exp $";
-#include <linux/config.h>
#include <linux/types.h>
#include <linux/fcntl.h>
#include <linux/poll.h>
#include <net/ipv6.h>
#include <net/sctp/sctp.h>
-
/* Forward declarations for internal functions. */
static void sctp_assoc_bh_rcv(sctp_association_t *asoc);
/* 1st Level Abstractions. */
/* Allocate and initialize a new association */
-sctp_association_t *
-sctp_association_new(const sctp_endpoint_t *ep, const struct sock *sk,
- sctp_scope_t scope, int priority)
+sctp_association_t *sctp_association_new(const sctp_endpoint_t *ep,
+ const struct sock *sk,
+ sctp_scope_t scope, int priority)
{
- sctp_association_t *asoc;
+ sctp_association_t *asoc;
- asoc = t_new(sctp_association_t, priority);
- if (!asoc) {
+ asoc = t_new(sctp_association_t, priority);
+ if (!asoc)
goto fail;
- }
- if (!sctp_association_init(asoc, ep, sk, scope, priority)) {
+ if (!sctp_association_init(asoc, ep, sk, scope, priority))
goto fail_init;
- }
asoc->base.malloced = 1;
SCTP_DBG_OBJCNT_INC(assoc);
return asoc;
-
+
fail_init:
kfree(asoc);
fail:
return NULL;
-
-} /* sctp_association_new() */
+}
/* Intialize a new association from provided memory. */
-sctp_association_t *
-sctp_association_init(sctp_association_t *asoc, const sctp_endpoint_t *ep,
- const struct sock *sk, sctp_scope_t scope,
- int priority)
+sctp_association_t *sctp_association_init(sctp_association_t *asoc,
+ const sctp_endpoint_t *ep,
+ const struct sock *sk,
+ sctp_scope_t scope,
+ int priority)
{
- int i;
sctp_opt_t *sp;
+ int i;
/* Retrieve the SCTP per socket area. */
- sp = sctp_sk((struct sock *)sk);
+ sp = sctp_sk((struct sock *)sk);
/* Init all variables to a known value. */
- memset(asoc, 0, sizeof(sctp_association_t));
+ memset(asoc, 0, sizeof(sctp_association_t));
/* Discarding const is appropriate here. */
- asoc->ep = (sctp_endpoint_t *)ep;
+ asoc->ep = (sctp_endpoint_t *)ep;
sctp_endpoint_hold(asoc->ep);
/* Hold the sock. */
asoc->base.sk = (struct sock *)sk;
sock_hold(asoc->base.sk);
- /* Initialize the common base substructure. */
+ /* Initialize the common base substructure. */
asoc->base.type = SCTP_EP_TYPE_ASSOCIATION;
/* Initialize the object handling fields. */
/* Initialize the bind addr area. */
sctp_bind_addr_init(&asoc->base.bind_addr, ep->base.bind_addr.port);
asoc->base.addr_lock = RW_LOCK_UNLOCKED;
-
- asoc->state = SCTP_STATE_CLOSED;
- asoc->state_timestamp = jiffies;
- /* Set things that have constant value. */
- asoc->cookie_life.tv_sec = SCTP_DEFAULT_COOKIE_LIFE_SEC;
- asoc->cookie_life.tv_usec = SCTP_DEFAULT_COOKIE_LIFE_USEC;
+ asoc->state = SCTP_STATE_CLOSED;
+ asoc->state_timestamp = jiffies;
+
+ /* Set things that have constant value. */
+ asoc->cookie_life.tv_sec = SCTP_DEFAULT_COOKIE_LIFE_SEC;
+ asoc->cookie_life.tv_usec = SCTP_DEFAULT_COOKIE_LIFE_USEC;
- asoc->pmtu = 0;
+ asoc->pmtu = 0;
asoc->frag_point = 0;
/* Initialize the default association max_retrans and RTO values. */
asoc->overall_error_threshold = 0;
asoc->overall_error_count = 0;
- /* Initialize the maximum mumber of new data packets that can be sent
- * in a burst.
+ /* Initialize the maximum mumber of new data packets that can be sent
+ * in a burst.
*/
asoc->max_burst = ep->proto->max_burst;
- /* Copy things from the endpoint. */
- for (i = SCTP_EVENT_TIMEOUT_NONE; i < SCTP_NUM_TIMEOUT_TYPES; ++i) {
- asoc->timeouts[i] = ep->timeouts[i];
- init_timer(&asoc->timers[i]);
- asoc->timers[i].function = sctp_timer_events[i];
- asoc->timers[i].data = (unsigned long)asoc;
- }
+ /* Copy things from the endpoint. */
+ for (i = SCTP_EVENT_TIMEOUT_NONE; i < SCTP_NUM_TIMEOUT_TYPES; ++i) {
+ asoc->timeouts[i] = ep->timeouts[i];
+ init_timer(&asoc->timers[i]);
+ asoc->timers[i].function = sctp_timer_events[i];
+ asoc->timers[i].data = (unsigned long) asoc;
+ }
- /* Pull default initialization values from the sock options.
- * Note: This assumes that the values have already been
- * validated in the sock.
+ /* Pull default initialization values from the sock options.
+ * Note: This assumes that the values have already been
+ * validated in the sock.
*/
- asoc->c.sinit_max_instreams = sp->initmsg.sinit_max_instreams;
- asoc->c.sinit_num_ostreams = sp->initmsg.sinit_num_ostreams;
- asoc->max_init_attempts = sp->initmsg.sinit_max_attempts;
+ asoc->c.sinit_max_instreams = sp->initmsg.sinit_max_instreams;
+ asoc->c.sinit_num_ostreams = sp->initmsg.sinit_num_ostreams;
+ asoc->max_init_attempts = sp->initmsg.sinit_max_attempts;
asoc->max_init_timeo = sp->initmsg.sinit_max_init_timeo * HZ;
/* RFC 2960 6.5 Stream Identifier and Stream Sequence Number
- *
+ *
* The stream sequence number in all the streams shall start
* from 0 when the association is established. Also, when the
* stream sequence number reaches the value 65535 the next
* stream sequence number shall be set to 0.
*/
- for(i=0; i < SCTP_MAX_STREAM; i++) {
- asoc->ssn[i] = 0;
- }
+ for (i = 0; i < SCTP_MAX_STREAM; i++)
+ asoc->ssn[i] = 0;
- /* Set the local window size for receive.
- * This is also the rcvbuf space per association.
- * RFC 6 - A SCTP receiver MUST be able to receive a minimum of
- * 1500 bytes in one SCTP packet.
+ /* Set the local window size for receive.
+ * This is also the rcvbuf space per association.
+ * RFC 6 - A SCTP receiver MUST be able to receive a minimum of
+ * 1500 bytes in one SCTP packet.
*/
- if (sk->rcvbuf < SCTP_DEFAULT_MINWINDOW) {
+ if (sk->rcvbuf < SCTP_DEFAULT_MINWINDOW)
asoc->rwnd = SCTP_DEFAULT_MINWINDOW;
- } else {
+ else
asoc->rwnd = sk->rcvbuf;
- }
asoc->rwnd_over = 0;
- /* Use my own max window until I learn something better. */
+ /* Use my own max window until I learn something better. */
asoc->peer.rwnd = SCTP_DEFAULT_MAXWINDOW;
/* Set the sndbuf size for transmit. */
asoc->sndbuf_used = 0;
-
+
init_waitqueue_head(&asoc->wait);
- asoc->c.my_vtag = sctp_generate_tag(ep);
- asoc->peer.i.init_tag = 0; /* INIT needs a vtag of 0. */
+ asoc->c.my_vtag = sctp_generate_tag(ep);
+ asoc->peer.i.init_tag = 0; /* INIT needs a vtag of 0. */
asoc->c.peer_vtag = 0;
asoc->c.my_ttag = 0;
- asoc->c.peer_ttag = 0;
+ asoc->c.peer_ttag = 0;
- asoc->c.initial_tsn = sctp_generate_tsn(ep);
+ asoc->c.initial_tsn = sctp_generate_tsn(ep);
asoc->next_tsn = asoc->c.initial_tsn;
-
- asoc->ctsn_ack_point = asoc->next_tsn - 1;
+
+ asoc->ctsn_ack_point = asoc->next_tsn - 1;
asoc->unack_data = 0;
- SCTP_DEBUG_PRINTK("myctsnap for %s INIT as 0x%x.\n",
+ SCTP_DEBUG_PRINTK("myctsnap for %s INIT as 0x%x.\n",
asoc->ep->debug_name,
asoc->ctsn_ack_point);
-
+
/* ADDIP Section 4.1 Asconf Chunk Procedures
- *
+ *
* When an endpoint has an ASCONF signaled change to be sent to the
* remote endpoint it should do the following:
* ...
*/
asoc->addip_serial = asoc->c.initial_tsn;
- /* Make an empty list of remote transport addresses. */
+ /* Make an empty list of remote transport addresses. */
INIT_LIST_HEAD(&asoc->peer.transport_addr_list);
-
- /* RFC 2960 5.1 Normal Establishment of an Association
+
+ /* RFC 2960 5.1 Normal Establishment of an Association
+ *
+ * After the reception of the first data chunk in an
+ * association the endpoint must immediately respond with a
+ * sack to acknowledge the data chunk. Subsequent
+ * acknowledgements should be done as described in Section
+ * 6.2.
*
- * After the reception of the first data chunk in an
- * association the endpoint must immediately respond with a
- * sack to acknowledge the data chunk. Subsequent
- * acknowledgements should be done as described in Section
- * 6.2.
- *
- * [We implement this by telling a new association that it
- * already received one packet.]
+ * [We implement this by telling a new association that it
+ * already received one packet.]
*/
- asoc->peer.sack_needed = 1;
+ asoc->peer.sack_needed = 1;
- /* Create an input queue. */
- sctp_inqueue_init(&asoc->base.inqueue);
- sctp_inqueue_set_th_handler(&asoc->base.inqueue,
- (void (*)(void *))sctp_assoc_bh_rcv,
+ /* Create an input queue. */
+ sctp_inqueue_init(&asoc->base.inqueue);
+ sctp_inqueue_set_th_handler(&asoc->base.inqueue,
+ (void (*)(void *))sctp_assoc_bh_rcv,
asoc);
-
- /* Create an output queue. */
- sctp_outqueue_init(asoc, &asoc->outqueue);
- sctp_outqueue_set_output_handlers(&asoc->outqueue,
- sctp_packet_init,
- sctp_packet_config,
- sctp_packet_append_chunk,
- sctp_packet_transmit_chunk,
- sctp_packet_transmit);
-
- if (NULL == sctp_ulpqueue_init(&asoc->ulpq, asoc, SCTP_MAX_STREAM)) {
+
+ /* Create an output queue. */
+ sctp_outqueue_init(asoc, &asoc->outqueue);
+ sctp_outqueue_set_output_handlers(&asoc->outqueue,
+ sctp_packet_init,
+ sctp_packet_config,
+ sctp_packet_append_chunk,
+ sctp_packet_transmit_chunk,
+ sctp_packet_transmit);
+
+ if (NULL == sctp_ulpqueue_init(&asoc->ulpq, asoc, SCTP_MAX_STREAM))
goto fail_init;
- }
- /* Set up the tsn tracking. */
+ /* Set up the tsn tracking. */
sctp_tsnmap_init(&asoc->peer.tsn_map, SCTP_TSN_MAP_SIZE, 0);
asoc->peer.next_dup_tsn = 0;
- skb_queue_head_init(&asoc->addip_chunks);
+ skb_queue_head_init(&asoc->addip_chunks);
asoc->need_ecne = 0;
-
- asoc->debug_name = "unnamedasoc";
- asoc->eyecatcher = SCTP_ASSOC_EYECATCHER;
+
+ asoc->debug_name = "unnamedasoc";
+ asoc->eyecatcher = SCTP_ASSOC_EYECATCHER;
/* Assume that peer would support both address types unless we are
- * told otherwise.
+ * told otherwise.
*/
asoc->peer.ipv4_address = 1;
asoc->peer.ipv6_address = 1;
fail_init:
sctp_endpoint_put(asoc->ep);
sock_put(asoc->base.sk);
- return NULL;
-
-} /* sctp_association_init() */
+ return NULL;
+}
-/* Free this association if possible. There may still be users, so
- * the actual deallocation may be delayed.
-*/
-void
-sctp_association_free(sctp_association_t *asoc)
+/* Free this association if possible. There may still be users, so
+ * the actual deallocation may be delayed.
+ */
+void sctp_association_free(sctp_association_t *asoc)
{
- sctp_transport_t *transport;
+ sctp_transport_t *transport;
sctp_endpoint_t *ep;
- list_t *pos, *temp;
+ list_t *pos, *temp;
int i;
ep = asoc->ep;
list_del(&asoc->asocs);
- /* Mark as dead, so other users can know this structure is
- * going away.
+ /* Mark as dead, so other users can know this structure is
+ * going away.
*/
asoc->base.dead = 1;
- /* Dispose of any data lying around in the outqueue. */
- sctp_outqueue_free(&asoc->outqueue);
+ /* Dispose of any data lying around in the outqueue. */
+ sctp_outqueue_free(&asoc->outqueue);
/* Dispose of any pending messages for the upper layer. */
sctp_ulpqueue_free(&asoc->ulpq);
/* Clean up the bound address list. */
sctp_bind_addr_free(&asoc->base.bind_addr);
-
- /* Do we need to go through all of our timers and
- * delete them? To be safe we will try to delete all, but we
+
+ /* Do we need to go through all of our timers and
+ * delete them? To be safe we will try to delete all, but we
* should be able to go through and make a guess based
- * on our state.
+ * on our state.
*/
-
- for (i = SCTP_EVENT_TIMEOUT_NONE; i < SCTP_NUM_TIMEOUT_TYPES; ++i) {
- if (timer_pending(&asoc->timers[i])
- && del_timer(&asoc->timers[i])) {
+ for (i = SCTP_EVENT_TIMEOUT_NONE; i < SCTP_NUM_TIMEOUT_TYPES; ++i) {
+ if (timer_pending(&asoc->timers[i]) &&
+ del_timer(&asoc->timers[i]))
sctp_association_put(asoc);
- }
- }
+ }
/* Release the transport structures. */
- list_for_each_safe(pos, temp, &asoc->peer.transport_addr_list) {
+ list_for_each_safe(pos, temp, &asoc->peer.transport_addr_list) {
transport = list_entry(pos, sctp_transport_t, transports);
list_del(pos);
sctp_transport_free(transport);
- } /* for (all transports). */
-
- asoc->eyecatcher = 0;
+ }
- sctp_association_put(asoc);
+ asoc->eyecatcher = 0;
-} /* sctp_association_free() */
+ sctp_association_put(asoc);
+}
/* Cleanup and free up an association. */
-static void
-sctp_association_destroy(sctp_association_t *asoc)
+static void sctp_association_destroy(sctp_association_t *asoc)
{
SCTP_ASSERT(asoc->base.dead, "Assoc is not dead", return);
- sctp_endpoint_put(asoc->ep);
+ sctp_endpoint_put(asoc->ep);
sock_put(asoc->base.sk);
if (asoc->base.malloced) {
kfree(asoc);
SCTP_DBG_OBJCNT_DEC(assoc);
}
-
-} /* sctp_association_destroy() */
+}
/* Add a transport address to an association. */
-sctp_transport_t *
-sctp_assoc_add_peer(sctp_association_t *asoc, const sockaddr_storage_t *addr,
- int priority)
+sctp_transport_t *sctp_assoc_add_peer(sctp_association_t *asoc,
+ const sockaddr_storage_t *addr,
+ int priority)
{
- sctp_transport_t *peer;
- sctp_opt_t *sp;
- const uint16_t *port;
+ sctp_transport_t *peer;
+ sctp_opt_t *sp;
+ const __u16 *port;
- switch (addr->sa.sa_family){
+ switch (addr->sa.sa_family) {
case AF_INET:
port = &addr->v4.sin_port;
break;
+
case AF_INET6:
SCTP_V6(
port = &addr->v6.sin6_port;
break;
);
+
default:
return NULL;
- } /* switch addr type */
+ };
- /* Set the port if it has not been set yet. */
+ /* Set the port if it has not been set yet. */
if (0 == asoc->peer.port) {
asoc->peer.port = *port;
}
/* Check to see if this is a duplicate. */
peer = sctp_assoc_lookup_paddr(asoc, addr);
- if (peer) { return peer; }
+ if (peer)
+ return peer;
- peer = sctp_transport_new(addr, priority);
- if (NULL == peer) { return NULL; }
+ peer = sctp_transport_new(addr, priority);
+ if (NULL == peer)
+ return NULL;
sctp_transport_set_owner(peer, asoc);
- /* If this is the first transport addr on this association,
- * initialize the association PMTU to the peer's PMTU.
- * If not and the current association PMTU is higher than the new
- * peer's PMTU, reset the association PMTU to the new peer's PMTU.
- */
- if (asoc->pmtu) {
- asoc->pmtu = min_t(int, peer->pmtu, asoc->pmtu);
- } else {
- asoc->pmtu = peer->pmtu;
- }
+ /* If this is the first transport addr on this association,
+ * initialize the association PMTU to the peer's PMTU.
+ * If not and the current association PMTU is higher than the new
+ * peer's PMTU, reset the association PMTU to the new peer's PMTU.
+ */
+ if (asoc->pmtu) {
+ asoc->pmtu = min_t(int, peer->pmtu, asoc->pmtu);
+ } else {
+ asoc->pmtu = peer->pmtu;
+ }
- SCTP_DEBUG_PRINTK("sctp_assoc_add_peer:association %p PMTU set to "
+ SCTP_DEBUG_PRINTK("sctp_assoc_add_peer:association %p PMTU set to "
"%d\n", asoc, asoc->pmtu);
- asoc->frag_point = asoc->pmtu -
+ asoc->frag_point = asoc->pmtu -
(SCTP_IP_OVERHEAD + sizeof(sctp_data_chunk_t));
/* The asoc->peer.port might not be meaningful as of now, but
asoc->base.bind_addr.port,
asoc->peer.port);
- /* 7.2.1 Slow-Start
- *
- * o The initial cwnd before data transmission or after a
- * sufficiently long idle period MUST be <= 2*MTU.
- *
- * o The initial value of ssthresh MAY be arbitrarily high
- * (for example, implementations MAY use the size of the
- * receiver advertised window).
- */
+ /* 7.2.1 Slow-Start
+ *
+ * o The initial cwnd before data transmission or after a
+ * sufficiently long idle period MUST be <= 2*MTU.
+ *
+ * o The initial value of ssthresh MAY be arbitrarily high
+ * (for example, implementations MAY use the size of the
+ * receiver advertised window).
+ */
peer->cwnd = asoc->pmtu * 2;
- /* At this point, we may not have the receiver's advertised window,
- * so initialize ssthresh to the default value and it will be set
+
+ /* At this point, we may not have the receiver's advertised window,
+ * so initialize ssthresh to the default value and it will be set
* later when we process the INIT.
*/
- peer->ssthresh = SCTP_DEFAULT_MAXWINDOW;
+ peer->ssthresh = SCTP_DEFAULT_MAXWINDOW;
peer->partial_bytes_acked = 0;
peer->flight_size = 0;
- peer->error_threshold = peer->max_retrans;
+ peer->error_threshold = peer->max_retrans;
- /* Update the overall error threshold value of the association
+ /* Update the overall error threshold value of the association
* taking the new peer's error threshold into account.
- */
- asoc->overall_error_threshold =
+ */
+ asoc->overall_error_threshold =
min(asoc->overall_error_threshold + peer->error_threshold,
asoc->max_retrans);
-
+
/* Initialize the peer's heartbeat interval based on the
- * sock configured value.
- */
+ * sock configured value.
+ */
sp = sctp_sk(asoc->base.sk);
peer->hb_interval = sp->paddrparam.spp_hbinterval * HZ;
/* Attach the remote transport to our asoc. */
list_add_tail(&peer->transports, &asoc->peer.transport_addr_list);
-
- /* If we do not yet have a primary path, set one. */
- if (NULL == asoc->peer.primary_path) {
- asoc->peer.primary_path = peer;
- asoc->peer.active_path = peer;
- asoc->peer.retran_path = peer;
- }
- if (asoc->peer.active_path == asoc->peer.retran_path) {
- asoc->peer.retran_path = peer;
- }
+ /* If we do not yet have a primary path, set one. */
+ if (NULL == asoc->peer.primary_path) {
+ asoc->peer.primary_path = peer;
+ asoc->peer.active_path = peer;
+ asoc->peer.retran_path = peer;
+ }
- return peer;
+ if (asoc->peer.active_path == asoc->peer.retran_path)
+ asoc->peer.retran_path = peer;
-} /* sctp_assoc_add_peer() */
+ return peer;
+}
/* Lookup a transport by address. */
-sctp_transport_t *
-sctp_assoc_lookup_paddr(const sctp_association_t *asoc,
- const sockaddr_storage_t *address)
+sctp_transport_t *sctp_assoc_lookup_paddr(const sctp_association_t *asoc,
+ const sockaddr_storage_t *address)
{
sctp_transport_t *t;
list_t *pos;
/* Cycle through all transports searching for a peer address. */
- list_for_each(pos, &asoc->peer.transport_addr_list) {
+ list_for_each(pos, &asoc->peer.transport_addr_list) {
t = list_entry(pos, sctp_transport_t, transports);
- if ( sctp_cmp_addr_exact(address, &t->ipaddr) ) { return t; }
- } /* for (all transports) */
-
- return NULL;
+ if (sctp_cmp_addr_exact(address, &t->ipaddr))
+ return t;
+ }
-} /* sctp_assoc_lookup_paddr() */
+ return NULL;
+}
/* Engage in transport control operations.
* Mark the transport up or down and send a notification to the user.
* Select and update the new active and retran paths.
*/
-void
-sctp_assoc_control_transport(sctp_association_t *asoc,
- sctp_transport_t *transport,
- sctp_transport_cmd_t command,
- sctp_sn_error_t error)
+void sctp_assoc_control_transport(sctp_association_t *asoc,
+ sctp_transport_t *transport,
+ sctp_transport_cmd_t command,
+ sctp_sn_error_t error)
{
sctp_transport_t *t = NULL;
sctp_transport_t *first;
list_t *pos;
int spc_state = 0;
-
/* Record the transition on the transport. */
switch (command) {
case SCTP_TRANSPORT_UP:
transport->state.active = 1;
spc_state = ADDRESS_AVAILABLE;
break;
+
case SCTP_TRANSPORT_DOWN:
transport->state.active = 0;
spc_state = ADDRESS_UNREACHABLE;
break;
+
default:
BUG();
- } /* switch (command) */
-
+ };
/* Generate and send a SCTP_PEER_ADDR_CHANGE notification to the
- * user.
+ * user.
*/
event = sctp_ulpevent_make_peer_addr_change(asoc,
- (struct sockaddr_storage *)&transport->ipaddr,
+ (struct sockaddr_storage *) &transport->ipaddr,
0, spc_state, error, GFP_ATOMIC);
- if (event) {
+ if (event)
sctp_ulpqueue_tail_event(&asoc->ulpq, event);
- }
-
/* Select new active and retran paths. */
list_for_each(pos, &asoc->peer.transport_addr_list) {
t = list_entry(pos, sctp_transport_t, transports);
- if (!t->state.active) { continue; }
-
+ if (!t->state.active)
+ continue;
if (!first || t->last_time_heard > first->last_time_heard) {
second = first;
first = t;
}
- if (!second || t->last_time_heard > second->last_time_heard) {
+ if (!second || t->last_time_heard > second->last_time_heard)
second = t;
- }
-
- } /* for (all transports) */
+ }
/* RFC 2960 6.4 Multi-Homed SCTP Endpoints
*
* [If the primary is active but not most recent, bump the most
* recently used transport.]
*/
- if (asoc->peer.primary_path->state.active
- && first != asoc->peer.primary_path) {
+ if (asoc->peer.primary_path->state.active &&
+ first != asoc->peer.primary_path) {
second = first;
first = asoc->peer.primary_path;
}
-
/* If we failed to find a usable transport, just camp on the
* primary, even if it is inactive.
*/
- if ( NULL == first ) {
+ if (NULL == first) {
first = asoc->peer.primary_path;
second = asoc->peer.primary_path;
}
/* Set the active and retran transports. */
asoc->peer.active_path = first;
asoc->peer.retran_path = second;
-
- return;
-
-} /* sctp_assoc_control_transport() */
-
+}
/* Hold a reference to an association. */
void sctp_association_hold(sctp_association_t *asoc)
{
atomic_inc(&asoc->base.refcnt);
-
-} /* sctp_association_hold() */
+}
/* Release a reference to an association and cleanup
- * if there are no more references.
+ * if there are no more references.
*/
void sctp_association_put(sctp_association_t *asoc)
{
- if (atomic_dec_and_test(&asoc->base.refcnt)) {
+ if (atomic_dec_and_test(&asoc->base.refcnt))
sctp_association_destroy(asoc);
- }
-
-} /* sctp_association_put() */
-
+}
/* Allocate the next TSN, Transmission Sequence Number, for the given
* association.
*/
-uint32_t
-__sctp_association_get_next_tsn(sctp_association_t *asoc)
+__u32 __sctp_association_get_next_tsn(sctp_association_t *asoc)
{
- /* From Section 1.6 Serial Number Arithmetic:
- * Transmission Sequence Numbers wrap around when they reach
- * 2**32 - 1. That is, the next TSN a DATA chunk MUST use
- * after transmitting TSN = 2*32 - 1 is TSN = 0.
- */
- uint32_t retval = asoc->next_tsn;
+ /* From Section 1.6 Serial Number Arithmetic:
+ * Transmission Sequence Numbers wrap around when they reach
+ * 2**32 - 1. That is, the next TSN a DATA chunk MUST use
+ * after transmitting TSN = 2*32 - 1 is TSN = 0.
+ */
+ __u32 retval = asoc->next_tsn;
asoc->next_tsn++;
asoc->unack_data++;
return retval;
-
-} /* __sctp_association_get_next_tsn() */
-
+}
/* Allocate 'num' TSNs by incrementing the association's TSN by num. */
-uint32_t
-__sctp_association_get_tsn_block(sctp_association_t *asoc, int num)
+__u32 __sctp_association_get_tsn_block(sctp_association_t *asoc, int num)
{
- uint32_t retval = asoc->next_tsn;
- asoc->next_tsn += num;
- asoc->unack_data += num;
+ __u32 retval = asoc->next_tsn;
- return retval;
-
-} /* __sctp_association_get_tsn_block() */
+ asoc->next_tsn += num;
+ asoc->unack_data += num;
+ return retval;
+}
/* Fetch the next Stream Sequence Number for stream number 'sid'. */
-uint16_t
-__sctp_association_get_next_ssn(sctp_association_t *asoc, uint16_t sid)
+__u16 __sctp_association_get_next_ssn(sctp_association_t *asoc, __u16 sid)
{
- return(asoc->ssn[sid]++);
-
-} /* __sctp_association_get_next_ssn() */
-
+ return asoc->ssn[sid]++;
+}
/* Compare two addresses to see if they match. Wildcard addresses
* always match within their address family.
- *
+ *
* FIXME: We do not match address scopes correctly.
*/
-int
-sctp_cmp_addr(const sockaddr_storage_t *ss1, const sockaddr_storage_t *ss2)
+int sctp_cmp_addr(const sockaddr_storage_t *ss1, const sockaddr_storage_t *ss2)
{
int len;
const void *base1;
const void *base2;
- if (ss1->sa.sa_family != ss2->sa.sa_family) {
+ if (ss1->sa.sa_family != ss2->sa.sa_family)
return 0;
- }
- if (ss1->v4.sin_port != ss2->v4.sin_port) {
+ if (ss1->v4.sin_port != ss2->v4.sin_port)
return 0;
- }
switch (ss1->sa.sa_family) {
case AF_INET:
if (INADDR_ANY == ss1->v4.sin_addr.s_addr ||
- INADDR_ANY == ss2->v4.sin_addr.s_addr) {
+ INADDR_ANY == ss2->v4.sin_addr.s_addr)
goto match;
- }
len = sizeof(struct in_addr);
base1 = &ss1->v4.sin_addr;
base2 = &ss2->v4.sin_addr;
break;
+
case AF_INET6:
SCTP_V6(
- if (IPV6_ADDR_ANY ==
- sctp_ipv6_addr_type(&ss1->v6.sin6_addr)) {
+ if (IPV6_ADDR_ANY ==
+ sctp_ipv6_addr_type(&ss1->v6.sin6_addr))
goto match;
- }
- if (IPV6_ADDR_ANY ==
- sctp_ipv6_addr_type(&ss2->v6.sin6_addr)) {
+ if (IPV6_ADDR_ANY ==
+ sctp_ipv6_addr_type(&ss2->v6.sin6_addr))
goto match;
- }
+
len = sizeof(struct in6_addr);
base1 = &ss1->v6.sin6_addr;
base2 = &ss2->v6.sin6_addr;
break;
)
-
+
default:
printk(KERN_WARNING
"WARNING, bogus socket address family %d\n",
ss1->sa.sa_family);
return 0;
- }
+ };
return (0 == memcmp(base1, base2, len));
- match:
- return 1;
-} /* sctp_cmp_addr() */
+match:
+ return 1;
+}
/* Compare two addresses to see if they match. Wildcard addresses
* only match themselves.
*
* FIXME: We do not match address scopes correctly.
*/
-int
-sctp_cmp_addr_exact(const sockaddr_storage_t *ss1,
- const sockaddr_storage_t *ss2)
+int sctp_cmp_addr_exact(const sockaddr_storage_t *ss1,
+ const sockaddr_storage_t *ss2)
{
int len;
const void *base1;
const void *base2;
-
- if (ss1->sa.sa_family != ss2->sa.sa_family) {
+ if (ss1->sa.sa_family != ss2->sa.sa_family)
return 0;
- }
-
- if (ss1->v4.sin_port != ss2->v4.sin_port) {
+ if (ss1->v4.sin_port != ss2->v4.sin_port)
return 0;
- }
-
- switch(ss1->sa.sa_family) {
+
+ switch (ss1->sa.sa_family) {
case AF_INET:
len = sizeof(struct in_addr);
base1 = &ss1->v4.sin_addr;
base2 = &ss2->v4.sin_addr;
break;
+
case AF_INET6:
SCTP_V6(
len = sizeof(struct in6_addr);
base2 = &ss2->v6.sin6_addr;
break;
)
+
default:
printk(KERN_WARNING
"WARNING, bogus socket address family %d\n",
ss1->sa.sa_family);
return 0;
- }
+ };
return (0 == memcmp(base1, base2, len));
-
-} /* sctp_cmp_addr_exact() */
-
+}
/* Return an ecne chunk to get prepended to a packet.
* Note: We are sly and return a shared, prealloced chunk.
*/
-sctp_chunk_t *
-sctp_get_ecne_prepend(sctp_association_t *asoc)
+sctp_chunk_t *sctp_get_ecne_prepend(sctp_association_t *asoc)
{
sctp_chunk_t *chunk;
int need_ecne;
- uint32_t lowest_tsn;
+ __u32 lowest_tsn;
- /* Can be called from task or bh. Both need_ecne and
- * last_ecne_tsn are written during bh.
+ /* Can be called from task or bh. Both need_ecne and
+ * last_ecne_tsn are written during bh.
*/
-
need_ecne = asoc->need_ecne;
lowest_tsn = asoc->last_ecne_tsn;
* out the network. If we run out of memory, just return
* NULL.
*/
-
} else {
chunk = NULL;
}
return chunk;
-
-} /* sctp_get_ecne_prepend(asoc) */
-
+}
/* Use this function for the packet prepend callback when no ECNE
* packet is desired (e.g. some packets don't like to be bundled).
*/
-sctp_chunk_t *
-sctp_get_no_prepend(sctp_association_t *asoc)
+sctp_chunk_t *sctp_get_no_prepend(sctp_association_t *asoc)
{
return NULL;
-
-} /* sctp_get_no_prepend(asoc) */
-
+}
/*
- * Find which transport this TSN was sent on.
+ * Find which transport this TSN was sent on.
*/
-sctp_transport_t *
-sctp_assoc_lookup_tsn(sctp_association_t *asoc, uint32_t tsn)
+sctp_transport_t *sctp_assoc_lookup_tsn(sctp_association_t *asoc, __u32 tsn)
{
sctp_transport_t *active;
sctp_transport_t *match;
list_t *entry, *pos;
sctp_transport_t *transport;
-
sctp_chunk_t *chunk;
-
- uint32_t key = htonl(tsn);
-
+ __u32 key = htonl(tsn);
+
match = NULL;
- /*
+ /*
* FIXME: In general, find a more efficient data structure for
- * searching.
+ * searching.
*/
- /*
+ /*
* The general strategy is to search each transport's transmitted
* list. Return which transport this TSN lives on.
- *
+ *
* Let's be hopeful and check the active_path first.
* Another optimization would be to know if there is only one
* outbound path and not have to look for the TSN at all.
- *
+ *
*/
active = asoc->peer.active_path;
-
- list_for_each(entry, &active->transmitted) {
+ list_for_each(entry, &active->transmitted) {
chunk = list_entry(entry, sctp_chunk_t, transmitted_list);
-
- if (key == chunk->subh.data_hdr->tsn){
+
+ if (key == chunk->subh.data_hdr->tsn) {
match = active;
goto out;
}
- } /* for (each unacknowledged TSN) */
-
- /* If not found, go search all the other transports. */
+ }
+ /* If not found, go search all the other transports. */
list_for_each(pos, &asoc->peer.transport_addr_list) {
transport = list_entry(pos, sctp_transport_t, transports);
- if (transport == active) { break; }
+ if (transport == active)
+ break;
list_for_each(entry, &transport->transmitted) {
-
- chunk = list_entry(entry, sctp_chunk_t,
+ chunk = list_entry(entry, sctp_chunk_t,
transmitted_list);
if (key == chunk->subh.data_hdr->tsn) {
match = transport;
goto out;
}
- } /* for (all unacknowledged TSNs) */
-
- } /* for (all transports) */
-
+ }
+ }
out:
return match;
-
-} /* sctp_assoc_lookup_tsn() */
+}
/* Is this the association we are looking for? */
-sctp_transport_t *
-sctp_assoc_is_match(sctp_association_t *asoc, const sockaddr_storage_t *laddr,
- const sockaddr_storage_t *paddr)
+sctp_transport_t *sctp_assoc_is_match(sctp_association_t *asoc,
+ const sockaddr_storage_t *laddr,
+ const sockaddr_storage_t *paddr)
{
sctp_transport_t *transport;
if ((asoc->base.bind_addr.port == laddr->v4.sin_port) &&
(asoc->peer.port == paddr->v4.sin_port)) {
-
transport = sctp_assoc_lookup_paddr(asoc, paddr);
- if (!transport) { goto out; }
-
- if (sctp_bind_addr_has_addr(&asoc->base.bind_addr, laddr)) {
+ if (!transport)
+ goto out;
+
+ if (sctp_bind_addr_has_addr(&asoc->base.bind_addr, laddr))
goto out;
- }
-
}
transport = NULL;
out:
sctp_read_unlock(&asoc->base.addr_lock);
return transport;
-
-} /* sctp_assoc_is_match() */
+}
/* Do delayed input processing. This is scheduled by sctp_rcv(). */
-static void
-sctp_assoc_bh_rcv(sctp_association_t *asoc)
+static void sctp_assoc_bh_rcv(sctp_association_t *asoc)
{
- sctp_endpoint_t *ep;
- sctp_chunk_t *chunk;
+ sctp_endpoint_t *ep;
+ sctp_chunk_t *chunk;
struct sock *sk;
- sctp_inqueue_t *inqueue;
- int state, subtype;
+ sctp_inqueue_t *inqueue;
+ int state, subtype;
sctp_assoc_t associd = sctp_assoc2id(asoc);
- int error = 0;
+ int error = 0;
/* The association should be held so we should be safe. */
- ep = asoc->ep;
+ ep = asoc->ep;
sk = asoc->base.sk;
- inqueue = &asoc->base.inqueue;
- while (NULL != (chunk = sctp_pop_inqueue(inqueue))) {
- state = asoc->state;
- subtype = chunk->chunk_hdr->type;
+ inqueue = &asoc->base.inqueue;
+ while (NULL != (chunk = sctp_pop_inqueue(inqueue))) {
+ state = asoc->state;
+ subtype = chunk->chunk_hdr->type;
/* Remember where the last DATA chunk came from so we
* know where to send the SACK.
*/
- if (sctp_chunk_is_data(chunk)) {
+ if (sctp_chunk_is_data(chunk))
asoc->peer.last_data_from = chunk->transport;
- }
- if (chunk->transport) {
+ if (chunk->transport)
chunk->transport->last_time_heard = jiffies;
- }
/* Run through the state machine. */
- error = sctp_do_sm(SCTP_EVENT_T_CHUNK, SCTP_ST_CHUNK(subtype),
+ error = sctp_do_sm(SCTP_EVENT_T_CHUNK, SCTP_ST_CHUNK(subtype),
state, ep, asoc, chunk, GFP_ATOMIC);
- /* Check to see if the association is freed in response to
+ /* Check to see if the association is freed in response to
* the incoming chunk. If so, get out of the while loop.
*/
- if (!sctp_id2assoc(sk, associd)) { goto out; }
-
- if (error != 0) { goto err_out; }
+ if (!sctp_id2assoc(sk, associd))
+ goto out;
- } /* while (we have more chunks to receive) */
+ if (error != 0)
+ goto err_out;
+ }
err_out:
- /* Is this the right way to pass errors up to the ULP? */
- if (error) {
- sk->err = -error;
- }
-
+ /* Is this the right way to pass errors up to the ULP? */
+ if (error)
+ sk->err = -error;
out:
- return;
-
-} /* sctp_bh_rcv_asoc() */
+}
/* This routine moves an association from its old sk to a new sk. */
-void
-sctp_assoc_migrate(sctp_association_t *assoc, struct sock *newsk)
+void sctp_assoc_migrate(sctp_association_t *assoc, struct sock *newsk)
{
sctp_opt_t *newsp = sctp_sk(newsk);
/* Delete the association from the old endpoint's list of
- * associations.
+ * associations.
*/
list_del(&assoc->asocs);
assoc->base.sk = newsk;
sock_hold(assoc->base.sk);
- /* Add the association to the new endpoint's list of associations. */
+ /* Add the association to the new endpoint's list of associations. */
sctp_endpoint_add_asoc(newsp->ep, assoc);
-
-} /* sctp_assoc_migrate() */
+}
/* Update an association (possibly from unexpected COOKIE-ECHO processing). */
-void
-sctp_assoc_update(sctp_association_t *asoc, sctp_association_t *new)
+void sctp_assoc_update(sctp_association_t *asoc, sctp_association_t *new)
{
int i;
-
+
/* Copy in new parameters of peer. */
asoc->c = new->c;
- asoc->peer.rwnd = new->peer.rwnd;
+ asoc->peer.rwnd = new->peer.rwnd;
asoc->peer.next_dup_tsn = new->peer.next_dup_tsn;
asoc->peer.sack_needed = new->peer.sack_needed;
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_SIZE,
asoc->peer.i.initial_tsn);
-
+
/* FIXME:
* Do we need to copy primary_path etc?
- *
+ *
* More explicitly, addresses may have been removed and
* this needs accounting for.
*/
-
-
+
/* If the case is A (association restart), use
* initial_tsn as next_tsn. If the case is B, use
* current next_tsn in case there is data sent to peer
* has been discarded and needs retransmission.
*/
- if (SCTP_STATE_ESTABLISHED == asoc->state){
+ if (SCTP_STATE_ESTABLISHED == asoc->state) {
asoc->next_tsn = new->next_tsn;
- asoc->ctsn_ack_point = new->ctsn_ack_point;
-
+ asoc->ctsn_ack_point = new->ctsn_ack_point;
+
/* Reinitialize SSN for both local streams
* and peer's streams.
*/
- for(i=0; i < SCTP_MAX_STREAM; i++) {
+ for (i = 0; i < SCTP_MAX_STREAM; i++) {
asoc->ssn[i] = 0;
asoc->ulpq.ssn[i] = 0;
- }
-
+ }
} else {
- asoc->ctsn_ack_point = asoc->next_tsn-1;
+ asoc->ctsn_ack_point = asoc->next_tsn - 1;
}
-
- return;
+}
-} /* sctp_assoc_update() */
-
-/* Choose the transport for sending a shutdown packet.
+/* Choose the transport for sending a shutdown packet.
* Round-robin through the active transports, else round-robin
* through the inactive transports as this is the next best thing
- * we can try.
+ * we can try.
*/
-sctp_transport_t *
-sctp_assoc_choose_shutdown_transport(sctp_association_t *asoc) {
+sctp_transport_t *sctp_assoc_choose_shutdown_transport(sctp_association_t *asoc)
+{
sctp_transport_t *t, *next;
list_t *head = &asoc->peer.transport_addr_list;
list_t *pos;
-
/* If this is the first time SHUTDOWN is sent, use the active
* path.
*/
- if (!asoc->shutdown_last_sent_to) {
+ if (!asoc->shutdown_last_sent_to)
return asoc->peer.active_path;
- }
/* Otherwise, find the next transport in a round-robin fashion. */
next = NULL;
while (1) {
-
/* Skip the head. */
- if (pos->next == head) {
+ if (pos->next == head)
pos = head->next;
- } else {
+ else
pos = pos->next;
- }
-
+
t = list_entry(pos, sctp_transport_t, transports);
-
+
/* Try to find an active transport. */
if (t->state.active) {
- break;
+ break;
} else {
/* Keep track of the next transport in case
* we don't find any active transport.
*/
- if (!next) {
+ if (!next)
next = t;
- }
}
/* We have exhausted the list, but didn't find any
* other active transports. If so, use the next
- * transport.
- */
+ * transport.
+ */
if (t == asoc->shutdown_last_sent_to) {
t = next;
break;
}
return t;
-
-} /* sctp_assoc_choose_shutdown_transport() */
+}
static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_bind_addr.c,v 1.16 2002/07/12 15:15:45 jgrimm Exp $";
-#include <linux/config.h>
#include <linux/types.h>
#include <linux/sched.h>
#include <linux/in.h>
#include <net/sctp/sctp_sm.h>
/* Forward declarations for internal helpers. */
-static int sctp_copy_one_addr(sctp_bind_addr_t *, sockaddr_storage_t *,
+static int sctp_copy_one_addr(sctp_bind_addr_t *, sockaddr_storage_t *,
sctp_scope_t scope, int priority, int flags);
static void sctp_bind_addr_clean(sctp_bind_addr_t *);
/* Copy 'src' to 'dest' taking 'scope' into account. Omit addresses
* in 'src' which have a broader scope than 'scope'.
*/
-int
-sctp_bind_addr_copy(sctp_bind_addr_t *dest, const sctp_bind_addr_t *src,
- sctp_scope_t scope, int priority, int flags)
+int sctp_bind_addr_copy(sctp_bind_addr_t *dest, const sctp_bind_addr_t *src,
+ sctp_scope_t scope, int priority, int flags)
{
- int error = 0;
struct sockaddr_storage_list *addr;
list_t *pos;
+ int error = 0;
/* All addresses share the same port. */
dest->port = src->port;
/* Extract the addresses which are relevant for this scope. */
list_for_each(pos, &src->address_list) {
- addr = list_entry(pos, struct sockaddr_storage_list, list);
- error = sctp_copy_one_addr(dest, &addr->a, scope, priority,
- flags);
- if (error < 0) { goto out; }
-
- } /* for (each address in the source bind_addr) */
-
- out:
- if (0 != error) {
- sctp_bind_addr_clean(dest);
+ addr = list_entry(pos, struct sockaddr_storage_list, list);
+ error = sctp_copy_one_addr(dest, &addr->a, scope,
+ priority, flags);
+ if (error < 0)
+ goto out;
}
+
+out:
+ if (error)
+ sctp_bind_addr_clean(dest);
return error;
-
-} /* sctp_bind_addr_copy() */
+}
/* Create a new SCTP_bind_addr from nothing. */
-sctp_bind_addr_t *
-sctp_bind_addr_new(int priority)
+sctp_bind_addr_t *sctp_bind_addr_new(int priority)
{
sctp_bind_addr_t *retval;
retval = t_new(sctp_bind_addr_t, priority);
- if (NULL == retval) { goto nomem; }
+ if (!retval)
+ goto nomem;
sctp_bind_addr_init(retval, 0);
retval->malloced = 1;
SCTP_DBG_OBJCNT_INC(bind_addr);
- nomem:
- return retval;
-
-} /* sctp_bind_addr_new() */
+nomem:
+ return retval;
+}
/* Initialize the SCTP_bind_addr structure for either an endpoint or
* an association.
*/
-void
-sctp_bind_addr_init(sctp_bind_addr_t *bp, uint16_t port)
+void sctp_bind_addr_init(sctp_bind_addr_t *bp, __u16 port)
{
bp->malloced = 0;
INIT_LIST_HEAD(&bp->address_list);
bp->port = port;
-
-} /* sctp_bind_addr_init() */
+}
/* Dispose of the address list. */
-static void
-sctp_bind_addr_clean(sctp_bind_addr_t *bp)
+static void sctp_bind_addr_clean(sctp_bind_addr_t *bp)
{
struct sockaddr_storage_list *addr;
list_t *pos, *temp;
list_del(pos);
kfree(addr);
SCTP_DBG_OBJCNT_DEC(addr);
- } /* for (each bound address) */
-
-} /* sctp_bind_addr_clean() */
+ }
+}
-/* Dispose of an SCTP_bind_addr structure
- */
-void
-sctp_bind_addr_free(sctp_bind_addr_t *bp)
+/* Dispose of an SCTP_bind_addr structure */
+void sctp_bind_addr_free(sctp_bind_addr_t *bp)
{
-
/* Empty the bind address list. */
sctp_bind_addr_clean(bp);
kfree(bp);
SCTP_DBG_OBJCNT_DEC(bind_addr);
}
-
-} /* sctp_bind_addr_free() */
+}
/* Add an address to the bind address list in the SCTP_bind_addr structure. */
-int
-sctp_add_bind_addr(sctp_bind_addr_t *bp, sockaddr_storage_t *new,
- int priority)
+int sctp_add_bind_addr(sctp_bind_addr_t *bp, sockaddr_storage_t *new,
+ int priority)
{
-
struct sockaddr_storage_list *addr;
-
- /* Add the address to the bind address list. */
- addr = t_new(struct sockaddr_storage_list, priority);
- if (NULL == addr) {
- return -ENOMEM;
- }
-
- addr->a = *new;
-
- /* Fix up the port if it has not yet been set.
+
+ /* Add the address to the bind address list. */
+ addr = t_new(struct sockaddr_storage_list, priority);
+ if (!addr)
+ return -ENOMEM;
+
+ addr->a = *new;
+
+ /* Fix up the port if it has not yet been set.
* Both v4 and v6 have the port at the same offset.
*/
- if (0 == addr->a.v4.sin_port) {
+ if (!addr->a.v4.sin_port)
addr->a.v4.sin_port = bp->port;
- }
INIT_LIST_HEAD(&addr->list);
list_add_tail(&addr->list, &bp->address_list);
SCTP_DBG_OBJCNT_INC(addr);
return 0;
-
-} /* sctp_add_bind_addr() */
+}
/* Delete an address from the bind address list in the SCTP_bind_addr
* structure.
*/
-int
-sctp_del_bind_addr(sctp_bind_addr_t *bp, sockaddr_storage_t *del_addr)
+int sctp_del_bind_addr(sctp_bind_addr_t *bp, sockaddr_storage_t *del_addr)
{
list_t *pos, *temp;
- struct sockaddr_storage_list *addr;
+ struct sockaddr_storage_list *addr;
list_for_each_safe(pos, temp, &bp->address_list) {
addr = list_entry(pos, struct sockaddr_storage_list, list);
list_del(pos);
kfree(addr);
SCTP_DBG_OBJCNT_DEC(addr);
+
return 0;
}
- } /* for (each bound address) */
-
- return -EINVAL;
+ }
-} /* sctp_del_bind_addr() */
+ return -EINVAL;
+}
/* Create a network byte-order representation of all the addresses
* formated as SCTP parameters.
*
* The second argument is the return value for the length.
*/
-sctpParam_t
-sctp_bind_addrs_to_raw(const sctp_bind_addr_t *bp, int *addrs_len,
- int priority)
+sctpParam_t sctp_bind_addrs_to_raw(const sctp_bind_addr_t *bp, int *addrs_len,
+ int priority)
{
- sctpParam_t rawaddr;
+ sctpParam_t rawaddr;
sctpParam_t addrparms;
sctpParam_t retval;
int addrparms_len;
- sctpIpAddress_t rawaddr_space;
- int len;
+ sctpIpAddress_t rawaddr_space;
+ int len;
struct sockaddr_storage_list *addr;
list_t *pos;
/* Allocate enough memory at once. */
list_for_each(pos, &bp->address_list) {
len += sizeof(sctp_ipv6addr_param_t);
- } /* for (each bound address) */
+ }
- addrparms.v = kmalloc(len, priority);
- if (NULL == addrparms.v) { goto end_raw; }
-
- retval = addrparms;
+ addrparms.v = kmalloc(len, priority);
+ if (!addrparms.v)
+ goto end_raw;
+
+ retval = addrparms;
rawaddr.v4 = &rawaddr_space.v4;
list_for_each(pos, &bp->address_list) {
memcpy(addrparms.v, rawaddr.v, len);
addrparms.v += len;
addrparms_len += len;
- } /* for (all addresses in the bind address list) */
-
+ }
+
end_raw:
-
- *addrs_len = addrparms_len;
- return retval;
-
-} /* sctp_bind_addrs_to_raw() */
+ *addrs_len = addrparms_len;
+ return retval;
+}
/*
- * Create an address list out of the raw address list format (IPv4 and IPv6
- * address parameters).
- *
+ * Create an address list out of the raw address list format (IPv4 and IPv6
+ * address parameters).
*/
-int
-sctp_raw_to_bind_addrs(sctp_bind_addr_t *bp, uint8_t *raw_addr_list,
- int addrs_len, uint16_t port, int priority)
+int sctp_raw_to_bind_addrs(sctp_bind_addr_t *bp, __u8 *raw_addr_list,
+ int addrs_len, __u16 port, int priority)
{
- sctpParam_t rawaddr;
+ sctpParam_t rawaddr;
sockaddr_storage_t addr;
int retval = 0;
- int len;
+ int len;
/* Convert the raw address to standard address format */
while (addrs_len) {
rawaddr.v = raw_addr_list;
- if (SCTP_PARAM_IPV4_ADDRESS==rawaddr.p->type
+ if (SCTP_PARAM_IPV4_ADDRESS==rawaddr.p->type
|| SCTP_PARAM_IPV6_ADDRESS==rawaddr.p->type) {
-
sctp_param2sockaddr(&addr, rawaddr, port);
retval = sctp_add_bind_addr(bp, &addr, priority);
if (retval) {
break;
}
}
-
+
return retval;
-
-} /* sctp_raw_to_bind_addrs() */
+}
/********************************************************************
* 2nd Level Abstractions
********************************************************************/
/* Does this contain a specified address? */
-int
-sctp_bind_addr_has_addr(sctp_bind_addr_t *bp, const sockaddr_storage_t *addr)
+int sctp_bind_addr_has_addr(sctp_bind_addr_t *bp, const sockaddr_storage_t *addr)
{
- list_t *pos;
- struct sockaddr_storage_list *laddr;
-
- list_for_each(pos, &bp->address_list) {
- laddr = list_entry(pos, struct sockaddr_storage_list, list);
- if (sctp_cmp_addr(&laddr->a, addr)) {
+ struct sockaddr_storage_list *laddr;
+ list_t *pos;
+
+ list_for_each(pos, &bp->address_list) {
+ laddr = list_entry(pos, struct sockaddr_storage_list, list);
+ if (sctp_cmp_addr(&laddr->a, addr))
return 1;
- }
- } /* for (all bound addresses). */
-
- return 0;
-
-} /* sctp_bind_addr_has_addr() */
+ }
+
+ return 0;
+}
/* Copy out addresses from the global local address list. */
-static int
-sctp_copy_one_addr(sctp_bind_addr_t *dest, sockaddr_storage_t *addr,
- sctp_scope_t scope, int priority, int flags)
+static int sctp_copy_one_addr(sctp_bind_addr_t *dest, sockaddr_storage_t *addr,
+ sctp_scope_t scope, int priority, int flags)
{
sctp_protocol_t *proto = sctp_get_protocol();
int error = 0;
if (sctp_is_any(addr)) {
- error = sctp_copy_local_addr_list(proto, dest, scope,
+ error = sctp_copy_local_addr_list(proto, dest, scope,
priority, flags);
} else if (sctp_in_scope(addr, scope)) {
/* Now that the address is in scope, check to see if
* the address type is supported by local sock as
* well as the remote peer.
*/
- if ((((AF_INET == addr->sa.sa_family)
- && (flags & SCTP_ADDR4_PEERSUPP)))
- || (((AF_INET6 == addr->sa.sa_family)
- && (flags & SCTP_ADDR6_ALLOWED)
- && (flags & SCTP_ADDR6_PEERSUPP)))) {
-
+ if ((((AF_INET == addr->sa.sa_family) &&
+ (flags & SCTP_ADDR4_PEERSUPP))) ||
+ (((AF_INET6 == addr->sa.sa_family) &&
+ (flags & SCTP_ADDR6_ALLOWED) &&
+ (flags & SCTP_ADDR6_PEERSUPP))))
error = sctp_add_bind_addr(dest, addr, priority);
- }
}
return error;
-
-} /* sctp_copy_one_addr() */
-
+}
/* Is addr one of the wildcards? */
-int
-sctp_is_any(const sockaddr_storage_t *addr)
+int sctp_is_any(const sockaddr_storage_t *addr)
{
int retval = 0;
switch (addr->sa.sa_family) {
case AF_INET:
- if (INADDR_ANY == addr->v4.sin_addr.s_addr) {
+ if (INADDR_ANY == addr->v4.sin_addr.s_addr)
retval = 1;
- }
break;
+
case AF_INET6:
SCTP_V6(
if (IPV6_ADDR_ANY ==
- sctp_ipv6_addr_type(&addr->v6.sin6_addr)) {
+ sctp_ipv6_addr_type(&addr->v6.sin6_addr))
retval = 1;
- }
);
break;
+
default:
break;
- }
+ };
return retval;
-
-} /* sctp_is_any() */
+}
/* Is 'addr' valid for 'scope'? */
-int
-sctp_in_scope(const sockaddr_storage_t *addr, sctp_scope_t scope)
+int sctp_in_scope(const sockaddr_storage_t *addr, sctp_scope_t scope)
{
-
sctp_scope_t addr_scope = sctp_scope(addr);
switch (addr->sa.sa_family) {
case AF_INET:
/* According to the SCTP IPv4 address scoping document -
- * <draft-stewart-tsvwg-sctp-ipv4-00.txt>, the scope has
- * a heirarchy of 5 levels:
+ * <draft-stewart-tsvwg-sctp-ipv4-00.txt>, the scope has
+ * a heirarchy of 5 levels:
* Level 0 - unusable SCTP addresses
- * Level 1 - loopback address
- * Level 2 - link-local addresses
- * Level 3 - private addresses.
- * Level 4 - global addresses
- * For INIT and INIT-ACK address list, let L be the level of
- * of requested destination address, sender and receiver
+ * Level 1 - loopback address
+ * Level 2 - link-local addresses
+ * Level 3 - private addresses.
+ * Level 4 - global addresses
+ * For INIT and INIT-ACK address list, let L be the level of
+ * of requested destination address, sender and receiver
* SHOULD include all of its addresses with level greater
- * than or equal to L.
- */
+ * than or equal to L.
+ */
/* The unusable SCTP addresses will not be considered with
* any defined scopes.
*/
- if (SCTP_SCOPE_UNUSABLE == addr_scope) {
- return 0 ;
- }
+ if (SCTP_SCOPE_UNUSABLE == addr_scope)
+ return 0;
/* Note that we are assuming that the scoping are the same
* for both IPv4 addresses and IPv6 addresses, i.e., if the
- * scope is link local, both IPv4 link local addresses and
- * IPv6 link local addresses would be treated as in the
+ * scope is link local, both IPv4 link local addresses and
+ * IPv6 link local addresses would be treated as in the
* scope. There is no filtering for IPv4 vs. IPv6 addresses
* based on scoping alone.
- */
- if (addr_scope <= scope) {
+ */
+ if (addr_scope <= scope)
return 1;
- }
break;
case AF_INET6:
-
/* FIXME:
* This is almost certainly wrong since scopes have an
* heirarchy. I don't know what RFC to look at.
* There may be some guidance in the SCTP implementors
* guide (an Internet Draft as of October 2001).
*
- * Further verification on the correctness of the IPv6
+ * Further verification on the correctness of the IPv6
* scoping is needed. According to the IPv6 scoping draft,
- * the link local and site local address may require
- * further scoping.
+ * the link local and site local address may require
+ * further scoping.
*
* Is the heirachy of the IPv6 scoping the same as what's
- * defined for IPv4?
- * If the same heirarchy indeed applies to both famiies,
- * this function can be simplified with one set of code.
+ * defined for IPv4?
+ * If the same heirarchy indeed applies to both famiies,
+ * this function can be simplified with one set of code.
* (see the comments for IPv4 above)
- *
*/
- if (addr_scope <= scope) {
+ if (addr_scope <= scope)
return 1;
- }
-
break;
-
+
default:
return 0;
- }
+ };
return 0;
-
-} /* sctp_in_scope() */
+}
/********************************************************************
* 3rd Level Abstractions
********************************************************************/
/* What is the scope of 'addr'? */
-sctp_scope_t
-sctp_scope(const sockaddr_storage_t *addr)
+sctp_scope_t sctp_scope(const sockaddr_storage_t *addr)
{
-
sctp_scope_t retval = SCTP_SCOPE_GLOBAL;
switch (addr->sa.sa_family) {
case AF_INET:
/* We are checking the loopback, private and other address
- * scopes as defined in RFC 1918.
+ * scopes as defined in RFC 1918.
* The IPv4 scoping is based on the draft for SCTP IPv4
* scoping <draft-stewart-tsvwg-sctp-ipv4-00.txt>.
* The set of SCTP address scope hopefully can cover both
* types of addresses.
*/
- /* Should IPv4 scoping be a sysctl configurable option
- * so users can turn it off (default on) for certain
- * unconventional networking environments?
+
+ /* Should IPv4 scoping be a sysctl configurable option
+ * so users can turn it off (default on) for certain
+ * unconventional networking environments?
*/
+
/* Check for unusable SCTP addresses. */
if (IS_IPV4_UNUSABLE_ADDRESS(&addr->v4.sin_addr.s_addr)) {
- retval = SCTP_SCOPE_UNUSABLE ;
-
+ retval = SCTP_SCOPE_UNUSABLE;
} else if (LOOPBACK(addr->v4.sin_addr.s_addr)) {
retval = SCTP_SCOPE_LOOPBACK;
-
} else if (IS_IPV4_LINK_ADDRESS(&addr->v4.sin_addr.s_addr)) {
retval = SCTP_SCOPE_LINK;
-
- } else if (IS_IPV4_PRIVATE_ADDRESS(&addr->v4.sin_addr.s_addr)){
+ } else if (IS_IPV4_PRIVATE_ADDRESS(&addr->v4.sin_addr.s_addr)) {
retval = SCTP_SCOPE_PRIVATE;
-
} else {
retval = SCTP_SCOPE_GLOBAL;
-
}
-
-
break;
case AF_INET6:
- {
+ {
SCTP_V6(
int v6scope;
v6scope = ipv6_addr_scope((struct in6_addr *)
&addr->v6.sin6_addr);
- /* The IPv6 scope is really a set of bit
+ /* The IPv6 scope is really a set of bit
* fields. See IFA_* in <net/if_inet6.h>.
- * Mapping them to the generic SCTP scope
- * set is an attempt to have code
+ * Mapping them to the generic SCTP scope
+ * set is an attempt to have code
* consistencies with the IPv4 scoping.
*/
switch (v6scope) {
case IFA_HOST:
retval = SCTP_SCOPE_LOOPBACK;
break;
+
case IFA_LINK:
retval = SCTP_SCOPE_LINK;
break;
+
case IFA_SITE:
retval = SCTP_SCOPE_PRIVATE;
break;
+
default:
retval = SCTP_SCOPE_GLOBAL;
- }
+ break;
+ };
);
-
break;
}
default:
retval = SCTP_SCOPE_GLOBAL;
- }
+ break;
+ };
return retval;
-
-} /* sctp_scope() */
+}
/* This function checks if the address is a valid address to be used for
- * SCTP.
+ * SCTP.
*
* Output:
- * Return 0 - If the address is a non-unicast or an illegal address.
+ * Return 0 - If the address is a non-unicast or an illegal address.
* Return 1 - If the address is a unicast.
*/
-int
-sctp_addr_is_valid(const sockaddr_storage_t *addr)
+int sctp_addr_is_valid(const sockaddr_storage_t *addr)
{
unsigned short sa_family = addr->sa.sa_family;
switch (sa_family) {
case AF_INET:
/* Is this a non-unicast address or a unusable SCTP address? */
- if (IS_IPV4_UNUSABLE_ADDRESS(&addr->v4.sin_addr.s_addr)) {
+ if (IS_IPV4_UNUSABLE_ADDRESS(&addr->v4.sin_addr.s_addr))
return 0;
- }
break;
case AF_INET6:
SCTP_V6(
{
- int ret;
-
- ret = sctp_ipv6_addr_type(&addr->v6.sin6_addr);
+ int ret = sctp_ipv6_addr_type(&addr->v6.sin6_addr);
+
/* Is this a non-unicast address */
- if (!(ret & IPV6_ADDR_UNICAST)) {
+ if (!(ret & IPV6_ADDR_UNICAST))
return 0;
- }
break;
});
+
default:
return 0;
- }
+ };
return 1;
-
-
-} /* sctp_addr_is_valid() */
-
+}
static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_command.c,v 1.4 2002/04/24 16:33:39 jgrimm Exp $";
-#include <linux/config.h>
#include <linux/types.h>
#include <net/sctp/sctp.h>
#include <net/sctp/sctp_sm.h>
/* Create a new sctp_command_sequence. */
-sctp_cmd_seq_t *
-sctp_new_cmd_seq(int priority)
+sctp_cmd_seq_t *sctp_new_cmd_seq(int priority)
{
- sctp_cmd_seq_t *retval;
-
- retval = t_new(sctp_cmd_seq_t, priority);
+ sctp_cmd_seq_t *retval = t_new(sctp_cmd_seq_t, priority);
+ /* XXX Check for NULL? -DaveM */
sctp_init_cmd_seq(retval);
return retval;
-
-} /* sctp_new_cmd_seq() */
+}
/* Initialize a block of memory as a command sequence. */
-int
-sctp_init_cmd_seq(sctp_cmd_seq_t *seq)
+int sctp_init_cmd_seq(sctp_cmd_seq_t *seq)
{
memset(seq, 0, sizeof(sctp_cmd_seq_t));
return 1; /* We always succeed. */
-
-} /* sctp_init_cmd_seq() */
+}
/* Add a command to a sctp_cmd_seq_t.
* Return 0 if the command sequence is full.
*/
-int
-sctp_add_cmd(sctp_cmd_seq_t *seq, sctp_verb_t verb, sctp_arg_t obj)
+int sctp_add_cmd(sctp_cmd_seq_t *seq, sctp_verb_t verb, sctp_arg_t obj)
{
- if (seq->next_free_slot >= SCTP_MAX_NUM_COMMANDS) {
+ if (seq->next_free_slot >= SCTP_MAX_NUM_COMMANDS)
goto fail;
- }
seq->cmds[seq->next_free_slot].verb = verb;
seq->cmds[seq->next_free_slot++].obj = obj;
return 1;
- fail:
- return 0;
-} /* sctp_add_cmd() */
+fail:
+ return 0;
+}
/* Rewind an sctp_cmd_seq_t to iterate from the start. */
-int
-sctp_rewind_sequence(sctp_cmd_seq_t *seq)
+int sctp_rewind_sequence(sctp_cmd_seq_t *seq)
{
seq->next_cmd = 0;
return 1; /* We always succeed. */
-
-} /* sctp_rewind_sequence() */
+}
/* Return the next command structure in a sctp_cmd_seq.
* Returns NULL at the end of the sequence.
*/
-sctp_cmd_t *
-sctp_next_cmd(sctp_cmd_seq_t *seq)
+sctp_cmd_t *sctp_next_cmd(sctp_cmd_seq_t *seq)
{
sctp_cmd_t *retval = NULL;
- if (seq->next_cmd < seq->next_free_slot) {
+ if (seq->next_cmd < seq->next_free_slot)
retval = &seq->cmds[seq->next_cmd++];
- }
return retval;
-
-} /* sctp_next_cmd() */
+}
/* Dispose of a command sequence. */
-void
-sctp_free_cmd_seq(sctp_cmd_seq_t *seq)
+void sctp_free_cmd_seq(sctp_cmd_seq_t *seq)
{
kfree(seq);
-
-} /* sctp_free_cmd_seq() */
+}
*
* The code has now been modified specifically for SCTP knowledge.
*/
-
#include <linux/types.h>
#include <net/sctp/sctp.h>
+
+#define CRC32C_POLY 0x1EDC6F41
+#define CRC32C(c,d) (c=(c>>8)^crc_c[(c^(d))&0xFF])
+/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
+/* Copyright 2001, D. Otis. Use this program, code or tables */
+/* extracted from it, as desired without restriction. */
+/* */
+/* 32 Bit Reflected CRC table generation for SCTP. */
+/* To accommodate serial byte data being shifted out least */
+/* significant bit first, the table's 32 bit words are reflected */
+/* which flips both byte and bit MS and LS positions. The CRC */
+/* is calculated MS bits first from the perspective of the serial*/
+/* stream. The x^32 term is implied and the x^0 term may also */
+/* be shown as +1. The polynomial code used is 0x1EDC6F41. */
+/* Castagnoli93 */
+/* x^32+x^28+x^27+x^26+x^25+x^23+x^22+x^20+x^19+x^18+x^14+x^13+ */
+/* x^11+x^10+x^9+x^8+x^6+x^0 */
+/* Guy Castagnoli Stefan Braeuer and Martin Herrman */
+/* "Optimization of Cyclic Redundancy-Check Codes */
+/* with 24 and 32 Parity Bits", */
+/* IEEE Transactions on Communications, Vol.41, No.6, June 1993 */
+/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
+__u32 crc_c[256] = {
+ 0x00000000, 0xF26B8303, 0xE13B70F7, 0x1350F3F4,
+ 0xC79A971F, 0x35F1141C, 0x26A1E7E8, 0xD4CA64EB,
+ 0x8AD958CF, 0x78B2DBCC, 0x6BE22838, 0x9989AB3B,
+ 0x4D43CFD0, 0xBF284CD3, 0xAC78BF27, 0x5E133C24,
+ 0x105EC76F, 0xE235446C, 0xF165B798, 0x030E349B,
+ 0xD7C45070, 0x25AFD373, 0x36FF2087, 0xC494A384,
+ 0x9A879FA0, 0x68EC1CA3, 0x7BBCEF57, 0x89D76C54,
+ 0x5D1D08BF, 0xAF768BBC, 0xBC267848, 0x4E4DFB4B,
+ 0x20BD8EDE, 0xD2D60DDD, 0xC186FE29, 0x33ED7D2A,
+ 0xE72719C1, 0x154C9AC2, 0x061C6936, 0xF477EA35,
+ 0xAA64D611, 0x580F5512, 0x4B5FA6E6, 0xB93425E5,
+ 0x6DFE410E, 0x9F95C20D, 0x8CC531F9, 0x7EAEB2FA,
+ 0x30E349B1, 0xC288CAB2, 0xD1D83946, 0x23B3BA45,
+ 0xF779DEAE, 0x05125DAD, 0x1642AE59, 0xE4292D5A,
+ 0xBA3A117E, 0x4851927D, 0x5B016189, 0xA96AE28A,
+ 0x7DA08661, 0x8FCB0562, 0x9C9BF696, 0x6EF07595,
+ 0x417B1DBC, 0xB3109EBF, 0xA0406D4B, 0x522BEE48,
+ 0x86E18AA3, 0x748A09A0, 0x67DAFA54, 0x95B17957,
+ 0xCBA24573, 0x39C9C670, 0x2A993584, 0xD8F2B687,
+ 0x0C38D26C, 0xFE53516F, 0xED03A29B, 0x1F682198,
+ 0x5125DAD3, 0xA34E59D0, 0xB01EAA24, 0x42752927,
+ 0x96BF4DCC, 0x64D4CECF, 0x77843D3B, 0x85EFBE38,
+ 0xDBFC821C, 0x2997011F, 0x3AC7F2EB, 0xC8AC71E8,
+ 0x1C661503, 0xEE0D9600, 0xFD5D65F4, 0x0F36E6F7,
+ 0x61C69362, 0x93AD1061, 0x80FDE395, 0x72966096,
+ 0xA65C047D, 0x5437877E, 0x4767748A, 0xB50CF789,
+ 0xEB1FCBAD, 0x197448AE, 0x0A24BB5A, 0xF84F3859,
+ 0x2C855CB2, 0xDEEEDFB1, 0xCDBE2C45, 0x3FD5AF46,
+ 0x7198540D, 0x83F3D70E, 0x90A324FA, 0x62C8A7F9,
+ 0xB602C312, 0x44694011, 0x5739B3E5, 0xA55230E6,
+ 0xFB410CC2, 0x092A8FC1, 0x1A7A7C35, 0xE811FF36,
+ 0x3CDB9BDD, 0xCEB018DE, 0xDDE0EB2A, 0x2F8B6829,
+ 0x82F63B78, 0x709DB87B, 0x63CD4B8F, 0x91A6C88C,
+ 0x456CAC67, 0xB7072F64, 0xA457DC90, 0x563C5F93,
+ 0x082F63B7, 0xFA44E0B4, 0xE9141340, 0x1B7F9043,
+ 0xCFB5F4A8, 0x3DDE77AB, 0x2E8E845F, 0xDCE5075C,
+ 0x92A8FC17, 0x60C37F14, 0x73938CE0, 0x81F80FE3,
+ 0x55326B08, 0xA759E80B, 0xB4091BFF, 0x466298FC,
+ 0x1871A4D8, 0xEA1A27DB, 0xF94AD42F, 0x0B21572C,
+ 0xDFEB33C7, 0x2D80B0C4, 0x3ED04330, 0xCCBBC033,
+ 0xA24BB5A6, 0x502036A5, 0x4370C551, 0xB11B4652,
+ 0x65D122B9, 0x97BAA1BA, 0x84EA524E, 0x7681D14D,
+ 0x2892ED69, 0xDAF96E6A, 0xC9A99D9E, 0x3BC21E9D,
+ 0xEF087A76, 0x1D63F975, 0x0E330A81, 0xFC588982,
+ 0xB21572C9, 0x407EF1CA, 0x532E023E, 0xA145813D,
+ 0x758FE5D6, 0x87E466D5, 0x94B49521, 0x66DF1622,
+ 0x38CC2A06, 0xCAA7A905, 0xD9F75AF1, 0x2B9CD9F2,
+ 0xFF56BD19, 0x0D3D3E1A, 0x1E6DCDEE, 0xEC064EED,
+ 0xC38D26C4, 0x31E6A5C7, 0x22B65633, 0xD0DDD530,
+ 0x0417B1DB, 0xF67C32D8, 0xE52CC12C, 0x1747422F,
+ 0x49547E0B, 0xBB3FFD08, 0xA86F0EFC, 0x5A048DFF,
+ 0x8ECEE914, 0x7CA56A17, 0x6FF599E3, 0x9D9E1AE0,
+ 0xD3D3E1AB, 0x21B862A8, 0x32E8915C, 0xC083125F,
+ 0x144976B4, 0xE622F5B7, 0xF5720643, 0x07198540,
+ 0x590AB964, 0xAB613A67, 0xB831C993, 0x4A5A4A90,
+ 0x9E902E7B, 0x6CFBAD78, 0x7FAB5E8C, 0x8DC0DD8F,
+ 0xE330A81A, 0x115B2B19, 0x020BD8ED, 0xF0605BEE,
+ 0x24AA3F05, 0xD6C1BC06, 0xC5914FF2, 0x37FACCF1,
+ 0x69E9F0D5, 0x9B8273D6, 0x88D28022, 0x7AB90321,
+ 0xAE7367CA, 0x5C18E4C9, 0x4F48173D, 0xBD23943E,
+ 0xF36E6F75, 0x0105EC76, 0x12551F82, 0xE03E9C81,
+ 0x34F4F86A, 0xC69F7B69, 0xD5CF889D, 0x27A40B9E,
+ 0x79B737BA, 0x8BDCB4B9, 0x988C474D, 0x6AE7C44E,
+ 0xBE2DA0A5, 0x4C4623A6, 0x5F16D052, 0xAD7D5351,
+};
-#define CRC32C_POLY 0x1EDC6F41
-#define CRC32C(c,d) (c=(c>>8)^crc_c[(c^(d))&0xFF])
-/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
-/* Copyright 2001, D. Otis. Use this program, code or tables */
-/* extracted from it, as desired without restriction. */
-/* */
-/* 32 Bit Reflected CRC table generation for SCTP. */
-/* To accommodate serial byte data being shifted out least */
-/* significant bit first, the table's 32 bit words are reflected */
-/* which flips both byte and bit MS and LS positions. The CRC */
-/* is calculated MS bits first from the perspective of the serial*/
-/* stream. The x^32 term is implied and the x^0 term may also */
-/* be shown as +1. The polynomial code used is 0x1EDC6F41. */
-/* Castagnoli93 */
-/* x^32+x^28+x^27+x^26+x^25+x^23+x^22+x^20+x^19+x^18+x^14+x^13+ */
-/* x^11+x^10+x^9+x^8+x^6+x^0 */
-/* Guy Castagnoli Stefan Braeuer and Martin Herrman */
-/* "Optimization of Cyclic Redundancy-Check Codes */
-/* with 24 and 32 Parity Bits", */
-/* IEEE Transactions on Communications, Vol.41, No.6, June 1993 */
-/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
-unsigned long crc_c[256] =
-{
- 0x00000000L, 0xF26B8303L, 0xE13B70F7L, 0x1350F3F4L,
- 0xC79A971FL, 0x35F1141CL, 0x26A1E7E8L, 0xD4CA64EBL,
- 0x8AD958CFL, 0x78B2DBCCL, 0x6BE22838L, 0x9989AB3BL,
- 0x4D43CFD0L, 0xBF284CD3L, 0xAC78BF27L, 0x5E133C24L,
- 0x105EC76FL, 0xE235446CL, 0xF165B798L, 0x030E349BL,
- 0xD7C45070L, 0x25AFD373L, 0x36FF2087L, 0xC494A384L,
- 0x9A879FA0L, 0x68EC1CA3L, 0x7BBCEF57L, 0x89D76C54L,
- 0x5D1D08BFL, 0xAF768BBCL, 0xBC267848L, 0x4E4DFB4BL,
- 0x20BD8EDEL, 0xD2D60DDDL, 0xC186FE29L, 0x33ED7D2AL,
- 0xE72719C1L, 0x154C9AC2L, 0x061C6936L, 0xF477EA35L,
- 0xAA64D611L, 0x580F5512L, 0x4B5FA6E6L, 0xB93425E5L,
- 0x6DFE410EL, 0x9F95C20DL, 0x8CC531F9L, 0x7EAEB2FAL,
- 0x30E349B1L, 0xC288CAB2L, 0xD1D83946L, 0x23B3BA45L,
- 0xF779DEAEL, 0x05125DADL, 0x1642AE59L, 0xE4292D5AL,
- 0xBA3A117EL, 0x4851927DL, 0x5B016189L, 0xA96AE28AL,
- 0x7DA08661L, 0x8FCB0562L, 0x9C9BF696L, 0x6EF07595L,
- 0x417B1DBCL, 0xB3109EBFL, 0xA0406D4BL, 0x522BEE48L,
- 0x86E18AA3L, 0x748A09A0L, 0x67DAFA54L, 0x95B17957L,
- 0xCBA24573L, 0x39C9C670L, 0x2A993584L, 0xD8F2B687L,
- 0x0C38D26CL, 0xFE53516FL, 0xED03A29BL, 0x1F682198L,
- 0x5125DAD3L, 0xA34E59D0L, 0xB01EAA24L, 0x42752927L,
- 0x96BF4DCCL, 0x64D4CECFL, 0x77843D3BL, 0x85EFBE38L,
- 0xDBFC821CL, 0x2997011FL, 0x3AC7F2EBL, 0xC8AC71E8L,
- 0x1C661503L, 0xEE0D9600L, 0xFD5D65F4L, 0x0F36E6F7L,
- 0x61C69362L, 0x93AD1061L, 0x80FDE395L, 0x72966096L,
- 0xA65C047DL, 0x5437877EL, 0x4767748AL, 0xB50CF789L,
- 0xEB1FCBADL, 0x197448AEL, 0x0A24BB5AL, 0xF84F3859L,
- 0x2C855CB2L, 0xDEEEDFB1L, 0xCDBE2C45L, 0x3FD5AF46L,
- 0x7198540DL, 0x83F3D70EL, 0x90A324FAL, 0x62C8A7F9L,
- 0xB602C312L, 0x44694011L, 0x5739B3E5L, 0xA55230E6L,
- 0xFB410CC2L, 0x092A8FC1L, 0x1A7A7C35L, 0xE811FF36L,
- 0x3CDB9BDDL, 0xCEB018DEL, 0xDDE0EB2AL, 0x2F8B6829L,
- 0x82F63B78L, 0x709DB87BL, 0x63CD4B8FL, 0x91A6C88CL,
- 0x456CAC67L, 0xB7072F64L, 0xA457DC90L, 0x563C5F93L,
- 0x082F63B7L, 0xFA44E0B4L, 0xE9141340L, 0x1B7F9043L,
- 0xCFB5F4A8L, 0x3DDE77ABL, 0x2E8E845FL, 0xDCE5075CL,
- 0x92A8FC17L, 0x60C37F14L, 0x73938CE0L, 0x81F80FE3L,
- 0x55326B08L, 0xA759E80BL, 0xB4091BFFL, 0x466298FCL,
- 0x1871A4D8L, 0xEA1A27DBL, 0xF94AD42FL, 0x0B21572CL,
- 0xDFEB33C7L, 0x2D80B0C4L, 0x3ED04330L, 0xCCBBC033L,
- 0xA24BB5A6L, 0x502036A5L, 0x4370C551L, 0xB11B4652L,
- 0x65D122B9L, 0x97BAA1BAL, 0x84EA524EL, 0x7681D14DL,
- 0x2892ED69L, 0xDAF96E6AL, 0xC9A99D9EL, 0x3BC21E9DL,
- 0xEF087A76L, 0x1D63F975L, 0x0E330A81L, 0xFC588982L,
- 0xB21572C9L, 0x407EF1CAL, 0x532E023EL, 0xA145813DL,
- 0x758FE5D6L, 0x87E466D5L, 0x94B49521L, 0x66DF1622L,
- 0x38CC2A06L, 0xCAA7A905L, 0xD9F75AF1L, 0x2B9CD9F2L,
- 0xFF56BD19L, 0x0D3D3E1AL, 0x1E6DCDEEL, 0xEC064EEDL,
- 0xC38D26C4L, 0x31E6A5C7L, 0x22B65633L, 0xD0DDD530L,
- 0x0417B1DBL, 0xF67C32D8L, 0xE52CC12CL, 0x1747422FL,
- 0x49547E0BL, 0xBB3FFD08L, 0xA86F0EFCL, 0x5A048DFFL,
- 0x8ECEE914L, 0x7CA56A17L, 0x6FF599E3L, 0x9D9E1AE0L,
- 0xD3D3E1ABL, 0x21B862A8L, 0x32E8915CL, 0xC083125FL,
- 0x144976B4L, 0xE622F5B7L, 0xF5720643L, 0x07198540L,
- 0x590AB964L, 0xAB613A67L, 0xB831C993L, 0x4A5A4A90L,
- 0x9E902E7BL, 0x6CFBAD78L, 0x7FAB5E8CL, 0x8DC0DD8FL,
- 0xE330A81AL, 0x115B2B19L, 0x020BD8EDL, 0xF0605BEEL,
- 0x24AA3F05L, 0xD6C1BC06L, 0xC5914FF2L, 0x37FACCF1L,
- 0x69E9F0D5L, 0x9B8273D6L, 0x88D28022L, 0x7AB90321L,
- 0xAE7367CAL, 0x5C18E4C9L, 0x4F48173DL, 0xBD23943EL,
- 0xF36E6F75L, 0x0105EC76L, 0x12551F82L, 0xE03E9C81L,
- 0x34F4F86AL, 0xC69F7B69L, 0xD5CF889DL, 0x27A40B9EL,
- 0x79B737BAL, 0x8BDCB4B9L, 0x988C474DL, 0x6AE7C44EL,
- 0xBE2DA0A5L, 0x4C4623A6L, 0x5F16D052L, 0xAD7D5351L,
-};
-
-
-
-uint32_t
-count_crc(uint8_t *buffer, uint16_t length)
-{
- unsigned int i;
- unsigned long crc32 = ~0L;
- unsigned long result;
- unsigned char byte0, byte1, byte2, byte3;
-
+__u32 count_crc(__u8 *buffer, __u16 length)
+{
+ __u32 crc32 = ~(__u32) 0;
+ __u32 i, result;
+ __u8 byte0, byte1, byte2, byte3;
+
/* Optimize this routine to be SCTP specific, knowing how
* to skip the checksum field of the SCTP header.
*/
/* Calculate CRC up to the checksum. */
- for (i = 0; i < (sizeof(struct sctphdr) - sizeof(uint32_t)); i++) {
+ for (i = 0; i < (sizeof(struct sctphdr) - sizeof(__u32)); i++)
CRC32C(crc32, buffer[i]);
- }
+
/* Skip checksum field of the header. */
- for (i = 0; i < sizeof(uint32_t); i++){
+ for (i = 0; i < sizeof(__u32); i++)
CRC32C(crc32, 0);
- }
/* Calculate the rest of the CRC. */
- for (i = sizeof(struct sctphdr); i < length ; i++){
+ for (i = sizeof(struct sctphdr); i < length ; i++)
CRC32C(crc32, buffer[i]);
- }
-
+
result = ~crc32;
/* result now holds the negated polynomial remainder;
* That is, result has the same value as if we mapped the message
* to a polyomial, computed the host-bit-order polynomial
* remainder, performed final negation, then did an end-for-end
- * bit-reversal.
+ * bit-reversal.
* Note that a 32-bit bit-reversal is identical to four inplace
* 8-bit reversals followed by an end-for-end byteswap.
* In other words, the bytes of each bit are in the right order,
* but the bytes have been byteswapped. So we now do an explicit
- * byteswap. On a little-endian machine, this byteswap and
+ * byteswap. On a little-endian machine, this byteswap and
* the final ntohl cancel out and could be elided.
*/
byte0 = result & 0xff;
byte1 = (result>>8) & 0xff;
byte2 = (result>>16) & 0xff;
byte3 = (result>>24) & 0xff;
-
+
crc32 = ((byte0 << 24) |
(byte1 << 16) |
(byte2 << 8) |
byte3);
- return(crc32);
-
-} /* count_crc() */
+ return crc32;
+}
int sctp_debug_flag = 1; /* Initially enable DEBUG */
#endif /* SCTP_DEBUG */
-
/* These are printable forms of Chunk ID's from section 3.1. */
static const char *sctp_cid_tbl[SCTP_NUM_BASE_CHUNK_TYPES] = {
"DATA",
"ECN_ECNE",
"ECN_CWR",
"SHUTDOWN_COMPLETE",
-}; /* char *sctp_cid_tbl[] */
+};
/* Lookup "chunk type" debug name. */
-const char *
-sctp_cname(const sctp_subtype_t cid)
+const char *sctp_cname(const sctp_subtype_t cid)
{
- if ( cid.chunk < 0 ) {
+ if (cid.chunk < 0)
return "illegal chunk id";
- }
-
- if ( cid.chunk <= SCTP_CID_BASE_MAX ) {
+ if (cid.chunk <= SCTP_CID_BASE_MAX)
return sctp_cid_tbl[cid.chunk];
- }
- switch ( cid.chunk ) {
- case SCTP_CID_ASCONF : return "ASCONF";
- case SCTP_CID_ASCONF_ACK : return "ASCONF_ACK";
+ switch (cid.chunk) {
+ case SCTP_CID_ASCONF:
+ return "ASCONF";
+
+ case SCTP_CID_ASCONF_ACK:
+ return "ASCONF_ACK";
+
default:
return "unknown chunk";
- }
-
+ };
return "unknown chunk";
+}
-} /* sctp_cname() */
-
/* These are printable form of variable-length parameters. */
-const char *sctp_param_tbl[SCTP_PARAM_ECN_CAPABLE+1] = {
+const char *sctp_param_tbl[SCTP_PARAM_ECN_CAPABLE + 1] = {
"",
"PARAM_HEATBEAT_INFO",
"",
"",
"PARAM_HOST_NAME_ADDRESS",
"PARAM_SUPPORTED_ADDRESS_TYPES",
-}; /* char *sctp_param_tbl[] */
-
+};
/* These are printable forms of the states. */
const char *sctp_state_tbl[SCTP_STATE_NUM_STATES] = {
"STATE_SHUTDOWN_SENT",
"STATE_SHUTDOWN_RECEIVED",
"STATE_SHUTDOWN_ACK_SENT",
-}; /* char *sctp_state_tbl[] */
+};
/* Events that could change the state of an association. */
const char *sctp_evttype_tbl[] = {
};
/* Lookup primitive debug name. */
-const char *
-sctp_pname(const sctp_subtype_t id)
+const char *sctp_pname(const sctp_subtype_t id)
{
- if ( id.primitive < 0 ) {
+ if (id.primitive < 0)
return "illegal primitive";
- }
-
- if ( id.primitive <= SCTP_EVENT_PRIMITIVE_MAX ) {
+ if (id.primitive <= SCTP_EVENT_PRIMITIVE_MAX)
return sctp_primitive_tbl[id.primitive];
- }
-
return "unknown_primitive";
-
-} /* sctp_pname() */
+}
static const char *sctp_other_tbl[] = {
"NO_PENDING_TSN",
};
/* Lookup "other" debug name. */
-const char *
-sctp_oname(const sctp_subtype_t id)
+const char *sctp_oname(const sctp_subtype_t id)
{
- if ( id.other < 0 ) {
+ if (id.other < 0)
return "illegal 'other' event";
- }
-
- if ( id.other < SCTP_EVENT_OTHER_MAX ) {
+ if (id.other < SCTP_EVENT_OTHER_MAX)
return sctp_other_tbl[id.other];
- }
-
return "unknown 'other' event";
-} /* sctp_oname() */
+}
static const char *sctp_timer_tbl[] = {
"TIMEOUT_NONE",
};
/* Lookup timer debug name. */
-const char *
-sctp_tname(const sctp_subtype_t id)
+const char *sctp_tname(const sctp_subtype_t id)
{
- if ( id.timeout < 0 ) {
+ if (id.timeout < 0)
return "illegal 'timer' event";
- }
-
- if ( id.timeout <= SCTP_EVENT_TIMEOUT_MAX ) {
+ if (id.timeout <= SCTP_EVENT_TIMEOUT_MAX)
return sctp_timer_tbl[id.timeout];
- }
-
return "unknown_timer";
-
-} /* sctp_tname() */
-
-
+}
*/
static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_endpointola.c,v 1.26 2002/08/16 19:30:49 jgrimm Exp $";
-#include <linux/config.h>
#include <linux/types.h>
#include <linux/sched.h>
#include <linux/slab.h>
#include <net/sctp/sctp.h>
#include <net/sctp/sctp_sm.h>
-
/* Forward declarations for internal helpers. */
static void sctp_endpoint_bh_rcv(sctp_endpoint_t *ep);
-
-/* Create a sctp_endpoint_t with all that boring stuff initialized.
+/* Create a sctp_endpoint_t with all that boring stuff initialized.
* Returns NULL if there isn't enough memory.
*/
-sctp_endpoint_t *
-sctp_endpoint_new(sctp_protocol_t *proto, struct sock *sk, int priority)
+sctp_endpoint_t *sctp_endpoint_new(sctp_protocol_t *proto,
+ struct sock *sk, int priority)
{
- sctp_endpoint_t *ep;
-
- /* Build a local endpoint. */
- ep = t_new(sctp_endpoint_t, priority);
- if (NULL == ep) {
- goto fail;
- }
+ sctp_endpoint_t *ep;
- if (NULL == sctp_endpoint_init(ep, proto, sk, priority)) {
+ /* Build a local endpoint. */
+ ep = t_new(sctp_endpoint_t, priority);
+ if (!ep)
+ goto fail;
+ if (!sctp_endpoint_init(ep, proto, sk, priority))
goto fail_init;
- }
ep->base.malloced = 1;
SCTP_DBG_OBJCNT_INC(ep);
return ep;
kfree(ep);
fail:
return NULL;
+}
-} /* sctp_endpoint_new() */
-
-
-/*
- * Initialize the base fields of the endpoint structure.
+/*
+ * Initialize the base fields of the endpoint structure.
*/
-sctp_endpoint_t *
-sctp_endpoint_init(sctp_endpoint_t *ep, sctp_protocol_t *proto,
- struct sock *sk, int priority)
+sctp_endpoint_t *sctp_endpoint_init(sctp_endpoint_t *ep, sctp_protocol_t *proto,
+ struct sock *sk, int priority)
{
-
- memset(ep, 0, sizeof(sctp_endpoint_t));
+ memset(ep, 0, sizeof(sctp_endpoint_t));
/* Initialize the base structure. */
- /* What type of endpoint are we? */
+ /* What type of endpoint are we? */
ep->base.type = SCTP_EP_TYPE_SOCKET;
/* Initialize the basic object fields. */
ep->base.dead = 0;
ep->base.malloced = 1;
- /* Create an input queue. */
- sctp_inqueue_init(&ep->base.inqueue);
+ /* Create an input queue. */
+ sctp_inqueue_init(&ep->base.inqueue);
- /* Set its top-half handler */
- sctp_inqueue_set_th_handler(&ep->base.inqueue,
- (void (*)(void *))sctp_endpoint_bh_rcv,
+ /* Set its top-half handler */
+ sctp_inqueue_set_th_handler(&ep->base.inqueue,
+ (void (*)(void *))sctp_endpoint_bh_rcv,
ep);
/* Initialize the bind addr area */
sctp_bind_addr_init(&ep->base.bind_addr, 0);
ep->base.addr_lock = RW_LOCK_UNLOCKED;
- /* Remember who we are attached to. */
- ep->base.sk = sk;
+ /* Remember who we are attached to. */
+ ep->base.sk = sk;
sock_hold(ep->base.sk);
- /* This pointer is useful to access the default protocol parameter
+ /* This pointer is useful to access the default protocol parameter
* values.
*/
ep->proto = proto;
- /* Create the lists of associations. */
+ /* Create the lists of associations. */
INIT_LIST_HEAD(&ep->asocs);
- /* Set up the base timeout information. */
- ep->timeouts[SCTP_EVENT_TIMEOUT_NONE] = 0;
- ep->timeouts[SCTP_EVENT_TIMEOUT_T1_COOKIE]
+ /* Set up the base timeout information. */
+ ep->timeouts[SCTP_EVENT_TIMEOUT_NONE] = 0;
+ ep->timeouts[SCTP_EVENT_TIMEOUT_T1_COOKIE]
= SCTP_DEFAULT_TIMEOUT_T1_COOKIE;
- ep->timeouts[SCTP_EVENT_TIMEOUT_T1_INIT]
+ ep->timeouts[SCTP_EVENT_TIMEOUT_T1_INIT]
= SCTP_DEFAULT_TIMEOUT_T1_INIT;
- ep->timeouts[SCTP_EVENT_TIMEOUT_T2_SHUTDOWN]
+ ep->timeouts[SCTP_EVENT_TIMEOUT_T2_SHUTDOWN]
= sctp_sk(sk)->rtoinfo.srto_initial;
- ep->timeouts[SCTP_EVENT_TIMEOUT_T3_RTX] = 0;
- ep->timeouts[SCTP_EVENT_TIMEOUT_T4_RTO] = 0;
- ep->timeouts[SCTP_EVENT_TIMEOUT_HEARTBEAT]
+ ep->timeouts[SCTP_EVENT_TIMEOUT_T3_RTX] = 0;
+ ep->timeouts[SCTP_EVENT_TIMEOUT_T4_RTO] = 0;
+ ep->timeouts[SCTP_EVENT_TIMEOUT_HEARTBEAT]
= SCTP_DEFAULT_TIMEOUT_HEARTBEAT;
- ep->timeouts[SCTP_EVENT_TIMEOUT_SACK]
+ ep->timeouts[SCTP_EVENT_TIMEOUT_SACK]
= SCTP_DEFAULT_TIMEOUT_SACK;
- ep->timeouts[SCTP_EVENT_TIMEOUT_AUTOCLOSE]
+ ep->timeouts[SCTP_EVENT_TIMEOUT_AUTOCLOSE]
= sctp_sk(sk)->autoclose * HZ;
- ep->timeouts[SCTP_EVENT_TIMEOUT_PMTU_RAISE]
+ ep->timeouts[SCTP_EVENT_TIMEOUT_PMTU_RAISE]
= SCTP_DEFAULT_TIMEOUT_PMTU_RAISE;
/* Set up the default send/receive buffer space. */
- /* FIXME - Should the min and max window size be configurable
- * sysctl parameters as opposed to be constants?
+
+ /* FIXME - Should the min and max window size be configurable
+ * sysctl parameters as opposed to be constants?
*/
sk->rcvbuf = SCTP_DEFAULT_MAXWINDOW;
sk->sndbuf = SCTP_DEFAULT_MAXWINDOW * 2;
sk->write_space = sctp_write_space;
sk->use_write_queue = 1;
- /* Initialize the secret key used with cookie. */
+ /* Initialize the secret key used with cookie. */
get_random_bytes(&ep->secret_key[0], SCTP_SECRET_SIZE);
- ep->last_key = ep->current_key = 0;
- ep->key_changed_at = jiffies;
-
- ep->debug_name = "unnamedEndpoint";
+ ep->last_key = ep->current_key = 0;
+ ep->key_changed_at = jiffies;
+ ep->debug_name = "unnamedEndpoint";
return ep;
+}
-} /* sctp_endpoint_init() */
-
-
-/* Add an association to an endpoint.
- */
-void
-sctp_endpoint_add_asoc(sctp_endpoint_t *ep, sctp_association_t *asoc)
+/* Add an association to an endpoint. */
+void sctp_endpoint_add_asoc(sctp_endpoint_t *ep, sctp_association_t *asoc)
{
/* Now just add it to our list of asocs */
list_add_tail(&asoc->asocs, &ep->asocs);
+}
-} /* sctp_endpoint_add_asoc() */
-
-
-/* Free the endpoint structure. Delay cleanup until
+/* Free the endpoint structure. Delay cleanup until
* all users have released their reference count on this structure.
*/
-void
-sctp_endpoint_free(sctp_endpoint_t *ep)
-{
- ep->base.dead = 1;
+void sctp_endpoint_free(sctp_endpoint_t *ep)
+{
+ ep->base.dead = 1;
sctp_endpoint_put(ep);
+}
-} /* sctp_free_endpoint() */
-
-/* Final destructor for endpoint. */
+/* Final destructor for endpoint. */
void sctp_endpoint_destroy(sctp_endpoint_t *ep)
{
SCTP_ASSERT(ep->base.dead, "Endpoint is not dead", return);
- /* Unlink this endpoint, so we can't find it again! */
+ /* Unlink this endpoint, so we can't find it again! */
sctp_unhash_endpoint(ep);
- /* Cleanup the inqueue. */
+ /* Cleanup the inqueue. */
sctp_inqueue_free(&ep->base.inqueue);
sctp_bind_addr_free(&ep->base.bind_addr);
- /* Remove and free the port */
- if (ep->base.sk->prev != NULL) {
+ /* Remove and free the port */
+ if (ep->base.sk->prev != NULL)
sctp_put_port(ep->base.sk);
- }
/* Give up our hold on the sock. */
- if (ep->base.sk) {
+ if (ep->base.sk)
sock_put(ep->base.sk);
- }
-
- /* Finally, free up our memory. */
- if (ep->base.malloced) {
+
+ /* Finally, free up our memory. */
+ if (ep->base.malloced) {
kfree(ep);
SCTP_DBG_OBJCNT_DEC(ep);
}
- return;
-
-} /* sctp_endpoint_destroy() */
-
+}
/* Hold a reference to an endpoint. */
void sctp_endpoint_hold(sctp_endpoint_t *ep)
{
atomic_inc(&ep->base.refcnt);
-
-} /* sctp_endpoint_hold() */
-
+}
/* Release a reference to an endpoint and clean up if there are
- * no more references.
+ * no more references.
*/
void sctp_endpoint_put(sctp_endpoint_t *ep)
{
- if (atomic_dec_and_test(&ep->base.refcnt)) {
+ if (atomic_dec_and_test(&ep->base.refcnt))
sctp_endpoint_destroy(ep);
- }
+}
-} /* sctp_endpoint_put() */
-
-
-/* Is this the endpoint we are looking for? */
-sctp_endpoint_t *
-sctp_endpoint_is_match(sctp_endpoint_t *ep, const sockaddr_storage_t *laddr)
+/* Is this the endpoint we are looking for? */
+sctp_endpoint_t *sctp_endpoint_is_match(sctp_endpoint_t *ep,
+ const sockaddr_storage_t *laddr)
{
sctp_endpoint_t *retval;
if (sctp_bind_addr_has_addr(&ep->base.bind_addr, laddr)) {
retval = ep;
goto out;
- }
- }
-
+ }
+ }
+
retval = NULL;
-
+
out:
sctp_read_unlock(&ep->base.addr_lock);
return retval;
-
-} /* sctp_endpoint_is_match() */
+}
/* Find the association that goes with this chunk.
* We do a linear search of the associations for this endpoint.
* We return the matching transport address too.
*/
-
-
-sctp_association_t *
-__sctp_endpoint_lookup_assoc(const sctp_endpoint_t *endpoint,
- const sockaddr_storage_t *paddr,
- sctp_transport_t **transport)
+sctp_association_t *__sctp_endpoint_lookup_assoc(const sctp_endpoint_t *endpoint,
+ const sockaddr_storage_t *paddr,
+ sctp_transport_t **transport)
{
- int rport;
- sctp_association_t *asoc;
+ int rport;
+ sctp_association_t *asoc;
list_t *pos;
-
- rport = paddr->v4.sin_port;
-
+
+ rport = paddr->v4.sin_port;
+
list_for_each(pos, &endpoint->asocs) {
asoc = list_entry(pos, sctp_association_t, asocs);
if (rport == asoc->peer.port) {
-
sctp_read_lock(&asoc->base.addr_lock);
*transport = sctp_assoc_lookup_paddr(asoc, paddr);
sctp_read_unlock(&asoc->base.addr_lock);
- if (*transport) {
+ if (*transport)
return asoc;
- }
}
- } /* for (all associations on endpoint) */
+ }
*transport = NULL;
- return NULL;
-
-} /* __sctp_endpoint_lookup_assoc() */
+ return NULL;
+}
-/* Lookup association on an endpoint based on a peer address. BH-safe. */
-sctp_association_t *
-sctp_endpoint_lookup_assoc(const sctp_endpoint_t *ep,
- const sockaddr_storage_t *paddr,
- sctp_transport_t **transport)
+/* Lookup association on an endpoint based on a peer address. BH-safe. */
+sctp_association_t *sctp_endpoint_lookup_assoc(const sctp_endpoint_t *ep,
+ const sockaddr_storage_t *paddr,
+ sctp_transport_t **transport)
{
sctp_association_t *asoc;
sctp_local_bh_enable();
return asoc;
+}
-} /* sctp_endpoint_lookup_assoc() */
-
-
-
-/* Do delayed input processing. This is scheduled by sctp_rcv().
- * This may be called on BH or task time.
+/* Do delayed input processing. This is scheduled by sctp_rcv().
+ * This may be called on BH or task time.
*/
-static void
-sctp_endpoint_bh_rcv(sctp_endpoint_t *ep)
+static void sctp_endpoint_bh_rcv(sctp_endpoint_t *ep)
{
- sctp_association_t *asoc;
+ sctp_association_t *asoc;
struct sock *sk;
sctp_transport_t *transport;
- sctp_chunk_t *chunk;
- sctp_inqueue_t *inqueue;
+ sctp_chunk_t *chunk;
+ sctp_inqueue_t *inqueue;
sctp_subtype_t subtype;
- sctp_state_t state;
- int error = 0;
+ sctp_state_t state;
+ int error = 0;
- if (ep->base.dead) { goto out; }
+ if (ep->base.dead)
+ goto out;
asoc = NULL;
- inqueue = &ep->base.inqueue;
+ inqueue = &ep->base.inqueue;
sk = ep->base.sk;
- while (NULL != (chunk = sctp_pop_inqueue(inqueue))) {
- subtype.chunk = chunk->chunk_hdr->type;
+ while (NULL != (chunk = sctp_pop_inqueue(inqueue))) {
+ subtype.chunk = chunk->chunk_hdr->type;
/* We might have grown an association since last we
* looked, so try again.
*
* This happens when we've just processed our
- * COOKIE-ECHO chunk.
+ * COOKIE-ECHO chunk.
*/
if (NULL == chunk->asoc) {
asoc = sctp_endpoint_lookup_assoc(ep,
sctp_source(chunk),
- &transport);
+ &transport);
chunk->asoc = asoc;
chunk->transport = transport;
- }
-
- state = asoc ? asoc->state : SCTP_STATE_CLOSED;
-
+ }
+
+ state = asoc ? asoc->state : SCTP_STATE_CLOSED;
+
/* Remember where the last DATA chunk came from so we
* know where to send the SACK.
*/
- if ( asoc != NULL && sctp_chunk_is_data(chunk) ) {
- asoc->peer.last_data_from =
- chunk->transport;
- }
+ if (asoc && sctp_chunk_is_data(chunk))
+ asoc->peer.last_data_from = chunk->transport;
- if (chunk->transport) {
+ if (chunk->transport)
chunk->transport->last_time_heard = jiffies;
- }
-
- /* FIX ME We really would rather NOT have to use
- * GFP_ATOMIC.
- */
+ /* FIX ME We really would rather NOT have to use
+ * GFP_ATOMIC.
+ */
error = sctp_do_sm(SCTP_EVENT_T_CHUNK, subtype, state,
ep, asoc, chunk, GFP_ATOMIC);
+ if (error != 0)
+ goto err_out;
- if (error != 0) { goto err_out; }
-
/* Check to see if the endpoint is freed in response to
* the incoming chunk. If so, get out of the while loop.
*/
- if (NULL == sctp_sk(sk)->ep) { goto out; }
-
- } /* while (we have more chunks to receive) */
+ if (!sctp_sk(sk)->ep)
+ goto out;
+ }
err_out:
- /* Is this the right way to pass errors up to the ULP? */
- if (error) {
- ep->base.sk->err = -error;
- }
-out:
- return;
+ /* Is this the right way to pass errors up to the ULP? */
+ if (error)
+ ep->base.sk->err = -error;
-} /* sctp_bh_rcv_ep() */
+out:
+}
#include <net/sctp/sctp.h>
#include <net/sctp/sctp_sla1.h>
-
/* SCTP Main driver.
- passing a two pointers and two lengths,
- returning a digest pointer filled. The md5 code
- was taken directly from the RFC (2104) so to understand it
- you may want to go look at the RFC referenced in the
- SCTP spec. We did modify this code to either user OUR
- implementation of SLA1 or the MD5 that comes from its
- RFC. SLA1 may have IPR issues so you need to check in
- to this if you wish to use it... Or at least that is
- what the FIP-180.1 web page says.
+ * passing a two pointers and two lengths,
+ * returning a digest pointer filled. The md5 code
+ * was taken directly from the RFC (2104) so to understand it
+ * you may want to go look at the RFC referenced in the
+ * SCTP spec. We did modify this code to either user OUR
+ * implementation of SLA1 or the MD5 that comes from its
+ * RFC. SLA1 may have IPR issues so you need to check in
+ * to this if you wish to use it... Or at least that is
+ * what the FIP-180.1 web page says.
*/
-void
-sctp_hash_digest(const char *key, const int in_key_len,
- const char *text, const int text_len,
- uint8_t *digest)
+void sctp_hash_digest(const char *key, const int in_key_len,
+ const char *text, const int text_len,
+ __u8 *digest)
{
int key_len = in_key_len;
- unsigned long *p;
- struct SLA_1_Context context;
+ struct SLA_1_Context context;
- uint8_t k_ipad[65]; /* inner padding -
- * key XORd with ipad
- */
- uint8_t k_opad[65]; /* outer padding -
- * key XORd with opad
- */
- uint8_t tk[20];
- int i;
- /* if key is longer than 64 bytes reset it to key=MD5(key) */
- if (key_len > 64) {
- struct SLA_1_Context tctx;
- SLA1_Init(&tctx);
- SLA1_Process(&tctx, key, key_len);
- SLA1_Final(&tctx,tk);
- key = tk;
- key_len = 20;
- }
-
- /*
- * the HMAC_MD5 transform looks like:
- *
- * MD5(K XOR opad, MD5(K XOR ipad, text))
- *
- * where K is an n byte key
- * ipad is the byte 0x36 repeated 64 times
- * opad is the byte 0x5c repeated 64 times
- * and text is the data being protected
- */
+ __u8 k_ipad[65]; /* inner padding -
+ * key XORd with ipad
+ */
+ __u8 k_opad[65]; /* outer padding -
+ * key XORd with opad
+ */
+ __u8 tk[20];
+ int i;
- /* start out by storing key in pads */
- memset( k_ipad, 0, sizeof k_ipad);
- memset( k_opad, 0, sizeof k_opad);
- memcpy( k_ipad, key, key_len);
- memcpy( k_opad, key, key_len);
+ /* if key is longer than 64 bytes reset it to key=MD5(key) */
+ if (key_len > 64) {
+ struct SLA_1_Context tctx;
- /* XOR key with ipad and opad values */
- for (i=0; i<64; i++) {
- k_ipad[i] ^= 0x36;
- k_opad[i] ^= 0x5c;
+ SLA1_Init(&tctx);
+ SLA1_Process(&tctx, key, key_len);
+ SLA1_Final(&tctx,tk);
+ key = tk;
+ key_len = 20;
}
- /*
- * perform inner hash
- */
- SLA1_Init(&context); /* init context for 1st
- * pass
- */
- SLA1_Process(&context, k_ipad, 64); /* start with inner pad */
- SLA1_Process(&context, text, text_len); /* then text of datagram */
- SLA1_Final(&context,digest); /* finish up 1st pass */
+ /*
+ * the HMAC_MD5 transform looks like:
+ *
+ * MD5(K XOR opad, MD5(K XOR ipad, text))
+ *
+ * where K is an n byte key
+ * ipad is the byte 0x36 repeated 64 times
+ * opad is the byte 0x5c repeated 64 times
+ * and text is the data being protected
+ */
- /*
+ /* start out by storing key in pads */
+ memset(k_ipad, 0, sizeof k_ipad);
+ memset(k_opad, 0, sizeof k_opad);
+ memcpy(k_ipad, key, key_len);
+ memcpy(k_opad, key, key_len);
+
+ /* XOR key with ipad and opad values */
+ for (i = 0; i < 64; i++) {
+ k_ipad[i] ^= 0x36;
+ k_opad[i] ^= 0x5c;
+ }
+
+ /* perform inner hash */
+ SLA1_Init(&context); /* init context for 1st
+ * pass
+ */
+ SLA1_Process(&context, k_ipad, 64); /* start with inner pad */
+ SLA1_Process(&context, text, text_len); /* then text of datagram */
+ SLA1_Final(&context,digest); /* finish up 1st pass */
+
+ /*
* perform outer hash
*/
-
- SLA1_Init(&context); /* init context for 2nd
- * pass
- */
- SLA1_Process(&context, k_opad, 64); /* start with outer pad */
- SLA1_Process(&context, digest, 20); /* then results of 1st
- * hash
- */
- SLA1_Final(&context, digest); /* finish up 2nd pass */
-
- p = (unsigned long *)digest;
+ SLA1_Init(&context); /* init context for 2nd
+ * pass
+ */
+ SLA1_Process(&context, k_opad, 64); /* start with outer pad */
+ SLA1_Process(&context, digest, 20); /* then results of 1st
+ * hash
+ */
+ SLA1_Final(&context, digest); /* finish up 2nd pass */
}
*/
static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_input.c,v 1.24 2002/07/24 12:26:20 jgrimm Exp $";
-#include <linux/config.h>
#include <linux/types.h>
#include <linux/list.h> /* For struct list_head */
#include <linux/socket.h>
#include <net/sctp/sctp_sm.h>
/* Forward declarations for internal helpers. */
-static inline int sctp_rcv_checksum(struct sk_buff *skb);
static int sctp_rcv_ootb(struct sk_buff *);
-sctp_association_t *
-__sctp_rcv_lookup(struct sk_buff *skb, const sockaddr_storage_t *laddr,
- const sockaddr_storage_t *paddr,
- sctp_transport_t **transportp);
-sctp_endpoint_t *
-__sctp_rcv_lookup_endpoint(const sockaddr_storage_t *laddr);
-
+sctp_association_t *__sctp_rcv_lookup(struct sk_buff *skb,
+ const sockaddr_storage_t *laddr,
+ const sockaddr_storage_t *paddr,
+ sctp_transport_t **transportp);
+sctp_endpoint_t *__sctp_rcv_lookup_endpoint(const sockaddr_storage_t *laddr);
/* Initialize a sockaddr_storage from in incoming skb.
* FIXME: This belongs with AF specific sctp_func_t. --jgrimm
*/
-static sockaddr_storage_t *
-sctp_sockaddr_storage_init(sockaddr_storage_t *addr, const struct sk_buff *skb,
- int is_saddr)
+static sockaddr_storage_t *sctp_sockaddr_storage_init(sockaddr_storage_t *addr,
+ const struct sk_buff *skb,
+ int is_saddr)
{
- sockaddr_storage_t *ret=NULL;
+ sockaddr_storage_t *ret = NULL;
void *to, *saddr, *daddr;
- uint16_t *port;
+ __u16 *port;
size_t len;
struct sctphdr *sh;
- switch (skb->nh.iph->version){
+ switch (skb->nh.iph->version) {
case 4:
to = &addr->v4.sin_addr.s_addr;
port = &addr->v4.sin_port;
saddr = &skb->nh.iph->saddr;
daddr = &skb->nh.iph->daddr;
len = sizeof(struct in_addr);
- addr->v4.sin_family = AF_INET;
+ addr->v4.sin_family = AF_INET;
break;
+
case 6:
SCTP_V6(
to = &addr->v6.sin6_addr;
addr->v6.sin6_scope_id = 0; /* FIXME */
break;
)
+
default:
goto out;
- }
-
- sh = (struct sctphdr *)skb->h.raw;
+ };
+ sh = (struct sctphdr *) skb->h.raw;
if (is_saddr) {
*port = ntohs(sh->source);
memcpy(to, saddr, len);
*port = ntohs(sh->dest);
memcpy(to, daddr, len);
}
-
+
ret = addr;
out:
return ret;
-
-} /* sctp_sockaddr_storage_init() */
+}
+
+/* Calculate the SCTP checksum of an SCTP packet. */
+static inline int sctp_rcv_checksum(struct sk_buff *skb)
+{
+ struct sctphdr *sh;
+ __u32 cmp, val;
+
+ sh = (struct sctphdr *) skb->h.raw;
+ cmp = ntohl(sh->checksum);
+ val = count_crc((__u8 *)sh, skb->len);
+ if (val != cmp) {
+ /* CRC failure, dump it. */
+ return -1;
+ }
+ return 0;
+}
/*
* This is the routine which IP calls when receiving an SCTP packet.
*/
-int
-sctp_rcv(struct sk_buff *skb)
+int sctp_rcv(struct sk_buff *skb)
{
struct sock *sk;
- sctp_association_t *asoc;
+ sctp_association_t *asoc;
sctp_endpoint_t *ep = NULL;
sctp_endpoint_common_t *rcvr;
- sctp_transport_t *transport = NULL;
- sctp_chunk_t *chunk;
+ sctp_transport_t *transport = NULL;
+ sctp_chunk_t *chunk;
struct sctphdr *sh;
sockaddr_storage_t src;
sockaddr_storage_t dest;
- int ret = 0;
+ int ret = 0;
- if (skb->pkt_type!=PACKET_HOST) {
+ if (skb->pkt_type!=PACKET_HOST)
goto discard_it;
- }
- sh = (struct sctphdr *)skb->h.raw;
+ sh = (struct sctphdr *) skb->h.raw;
/* Pull up the IP and SCTP headers. */
__skb_pull(skb, skb->h.raw - skb->data);
-
- if (skb->len < sizeof(struct sctphdr)) {
+ if (skb->len < sizeof(struct sctphdr))
goto bad_packet;
- }
-
- if (sctp_rcv_checksum(skb) < 0) {
+ if (sctp_rcv_checksum(skb) < 0)
goto bad_packet;
- }
-
- skb_pull(skb, sizeof(struct sctphdr));
+
+ skb_pull(skb, sizeof(struct sctphdr));
sctp_sockaddr_storage_init(&src, skb, 1);
sctp_sockaddr_storage_init(&dest, skb, 0);
-
+
/* If the packet is to or from a non-unicast address,
* silently discard the packet.
*
* This is not clearly defined in the RFC except in section
- * 8.4 - OOTB handling. However, based on the book "Stream Control
- * Transmission Protocol" 2.1, "It is important to note that the
+ * 8.4 - OOTB handling. However, based on the book "Stream Control
+ * Transmission Protocol" 2.1, "It is important to note that the
* IP address of an SCTP transport address must be a routable
* unicast address. In other words, IP multicast addresses and
* IP broadcast addresses cannot be used in an SCTP transport
* address."
*/
- if (!sctp_addr_is_valid(&src) || !sctp_addr_is_valid(&dest)) {
+ if (!sctp_addr_is_valid(&src) || !sctp_addr_is_valid(&dest))
goto discard_it;
- }
asoc = __sctp_rcv_lookup(skb, &src, &dest, &transport);
- /*
+ /*
* RFC 2960, 8.4 - Handle "Out of the blue" Packets.
- * An SCTP packet is called an "out of the blue" (OOTB)
- * packet if it is correctly formed, i.e., passed the
- * receiver's checksum check, but the receiver is not
- * able to identify the association to which this
- * packet belongs.
+ * An SCTP packet is called an "out of the blue" (OOTB)
+ * packet if it is correctly formed, i.e., passed the
+ * receiver's checksum check, but the receiver is not
+ * able to identify the association to which this
+ * packet belongs.
*/
- if (!asoc) {
+ if (!asoc) {
ep = __sctp_rcv_lookup_endpoint(&dest);
- if (sctp_rcv_ootb(skb)) {
+ if (sctp_rcv_ootb(skb))
goto discard_release;
- }
}
/* Retrieve the common input handling substructure. */
rcvr = asoc ? &asoc->base : &ep->base;
sk = rcvr->sk;
- if (!ipsec_sk_policy(sk, skb)) {
+ if (!ipsec_sk_policy(sk, skb))
goto discard_release;
- }
- /* Create an SCTP packet structure. */
+ /* Create an SCTP packet structure. */
chunk = sctp_chunkify(skb, asoc, sk);
-
if (!chunk) {
ret = -ENOMEM;
goto discard_release;
/* Remember what endpoint is to handle this packet. */
chunk->rcvr = rcvr;
- /* Remember the SCTP header. */
- chunk->sctp_hdr = sh;
-
+ /* Remember the SCTP header. */
+ chunk->sctp_hdr = sh;
+
/* Set the source address. */
sctp_init_source(chunk);
*/
sctp_bh_lock_sock(sk);
- if (__sctp_sock_busy(sk)) {
- sk_add_backlog(sk, (struct sk_buff *)chunk);
+ if (__sctp_sock_busy(sk)) {
+ sk_add_backlog(sk, (struct sk_buff *) chunk);
} else {
- sctp_backlog_rcv(sk, (struct sk_buff *)chunk);
+ sctp_backlog_rcv(sk, (struct sk_buff *) chunk);
}
- /* Release the sock and any reference counts we took in the
+ /* Release the sock and any reference counts we took in the
* lookup calls.
*/
- sctp_bh_unlock_sock(sk);
+ sctp_bh_unlock_sock(sk);
if (asoc) {
sctp_association_put(asoc);
} else {
- sctp_endpoint_put(ep);
+ sctp_endpoint_put(ep);
}
sock_put(sk);
- return ret;
+ return ret;
bad_packet:
#if 0 /* FIXME */
SCTP_INC_STATS(SctpInErrs);
#endif /* FIXME*/
-
+
discard_it:
kfree_skb(skb);
- return ret;
+ return ret;
discard_release:
/* Release any structures we may be holding. */
sctp_association_put(asoc);
} else {
sock_put(ep->base.sk);
- sctp_endpoint_put(ep);
+ sctp_endpoint_put(ep);
}
-
- goto discard_it;
-} /* sctp_rcv() */
+ goto discard_it;
+}
/* Handle second half of inbound skb processing. If the sock was busy,
* we may have need to delay processing until later when the sock is
- * released (on the backlog). If not busy, we call this routine
- * directly from the bottom half.
+ * released (on the backlog). If not busy, we call this routine
+ * directly from the bottom half.
*/
-int
-sctp_backlog_rcv(struct sock *sk, struct sk_buff *skb)
+int sctp_backlog_rcv(struct sock *sk, struct sk_buff *skb)
{
sctp_chunk_t *chunk;
sctp_inqueue_t *inqueue;
/* One day chunk will live inside the skb, but for
* now this works.
*/
- chunk = (sctp_chunk_t *)skb;
+ chunk = (sctp_chunk_t *) skb;
inqueue = &chunk->rcvr->inqueue;
sctp_push_inqueue(inqueue, chunk);
-
return 0;
-
-} /* sctp_backlog_rcv() */
+}
/*
* This routine is called by the ICMP module when it gets some
* is probably better.
*
*/
-void
-sctp_v4_err(struct sk_buff *skb, u32 info)
+void sctp_v4_err(struct sk_buff *skb, u32 info)
{
- /* This should probably involve a call to SCTPhandleICMP(). */
-} /* void sctp_v4_err(struct sk_buff *skb, u32 info) */
-
-
-/* Calculate the SCTP checksum of an SCTP packet. */
-static inline int
-sctp_rcv_checksum(struct sk_buff *skb)
-{
- uint32_t cmp,val;
- struct sctphdr *sh;
-
- sh = (struct sctphdr *)skb->h.raw;
- cmp = ntohl(sh->checksum);
- val = count_crc((uint8_t *)sh, skb->len);
- if (val != cmp) {
- /* CRC failure, dump it. */
- return -1;
- }
- return 0;
-
-} /* sctp_rcv_checksum() */
-
-
+ /* This should probably involve a call to SCTPhandleICMP(). */
+}
/*
* RFC 2960, 8.4 - Handle "Out of the blue" Packets.
* Return 0 - If further processing is needed.
* Return 1 - If the packet can be discarded right away.
*/
-int
-sctp_rcv_ootb(struct sk_buff *skb)
+int sctp_rcv_ootb(struct sk_buff *skb)
{
sctp_chunkhdr_t *ch;
- uint8_t *ch_end;
+ __u8 *ch_end;
- ch = (sctp_chunkhdr_t *)skb->data;
+ ch = (sctp_chunkhdr_t *) skb->data;
/* Scan through all the chunks in the packet. */
do {
- ch_end = ((uint8_t *)ch) + WORD_ROUND(ntohs(ch->length));
+ ch_end = ((__u8 *) ch) + WORD_ROUND(ntohs(ch->length));
- /* RFC 8.4, 2) If the OOTB packet contains an ABORT chunk, the
- * receiver MUST silently discard the OOTB packet and take no
+ /* RFC 8.4, 2) If the OOTB packet contains an ABORT chunk, the
+ * receiver MUST silently discard the OOTB packet and take no
* further action.
*/
-
- if (SCTP_CID_ABORT == ch->type) {
+ if (SCTP_CID_ABORT == ch->type)
goto discard;
- }
-
- /* RFC 8.4, 6) If the packet contains a SHUTDOWN COMPLETE
- * chunk, the receiver should silently discard the packet
+
+ /* RFC 8.4, 6) If the packet contains a SHUTDOWN COMPLETE
+ * chunk, the receiver should silently discard the packet
* and take no further action.
*/
-
- if (ch->type == SCTP_CID_SHUTDOWN_COMPLETE) {
+ if (ch->type == SCTP_CID_SHUTDOWN_COMPLETE)
goto discard;
- }
-
- /* RFC 8.4, 7) If the packet contains a "Stale cookie" ERROR
- * or a COOKIE ACK the SCTP Packet should be silently
- * discarded.
- */
- if (ch->type == SCTP_CID_COOKIE_ACK) {
+ /* RFC 8.4, 7) If the packet contains a "Stale cookie" ERROR
+ * or a COOKIE ACK the SCTP Packet should be silently
+ * discarded.
+ */
+ if (ch->type == SCTP_CID_COOKIE_ACK)
goto discard;
- }
if (ch->type == SCTP_CID_ERROR) {
/* FIXME - Need to check the "Stale cookie" ERROR. */
goto discard;
}
- ch = (sctp_chunkhdr_t *)ch_end;
-
+ ch = (sctp_chunkhdr_t *) ch_end;
} while (ch_end < skb->tail);
return 0;
-
+
discard:
return 1;
-
-} /* sctp_rcv_ootb() */
-
+}
/* Insert endpoint into the hash table. */
void __sctp_hash_endpoint(sctp_endpoint_t *ep)
{
sctp_endpoint_common_t **epp;
sctp_endpoint_common_t *epb;
- sctp_hashbucket_t *head;
-
+ sctp_hashbucket_t *head;
+
epb = &ep->base;
-
+
epb->hashent = sctp_ep_hashfn(epb->bind_addr.port);
head = &sctp_proto.ep_hashbucket[epb->hashent];
-
+
sctp_write_lock(&head->lock);
epp = &head->chain;
epb->next = *epp;
- if (epb->next) {
+ if (epb->next)
(*epp)->pprev = &epb->next;
- }
*epp = epb;
epb->pprev = epp;
sctp_write_unlock(&head->lock);
-
-} /* __sctp_hash_endpoint() */
+}
/* Add an endpoint to the hash. Local BH-safe. */
void sctp_hash_endpoint(sctp_endpoint_t *ep)
sctp_local_bh_disable();
__sctp_hash_endpoint(ep);
sctp_local_bh_enable();
-
-} /* __sctp_hash_endpoint() */
-
+}
/* Remove endpoint from the hash table. */
void __sctp_unhash_endpoint(sctp_endpoint_t *ep)
epb = &ep->base;
epb->hashent = sctp_ep_hashfn(epb->bind_addr.port);
-
+
head = &sctp_proto.ep_hashbucket[epb->hashent];
sctp_write_lock(&head->lock);
if (epb->pprev) {
- if (epb->next) {
+ if (epb->next)
epb->next->pprev = epb->pprev;
- }
*epb->pprev = epb->next;
epb->pprev = NULL;
}
sctp_write_unlock(&head->lock);
-
-} /* __sctp_unhash_endpoint() */
+}
/* Remove endpoint from the hash. Local BH-safe. */
void sctp_unhash_endpoint(sctp_endpoint_t *ep)
sctp_local_bh_disable();
__sctp_unhash_endpoint(ep);
sctp_local_bh_enable();
-
-} /* sctp_unhash_endpoint() */
-
+}
/* Look up an endpoint. */
-sctp_endpoint_t *
-__sctp_rcv_lookup_endpoint(const sockaddr_storage_t *laddr)
+sctp_endpoint_t *__sctp_rcv_lookup_endpoint(const sockaddr_storage_t *laddr)
{
sctp_hashbucket_t *head;
sctp_endpoint_common_t *epb;
read_lock(&head->lock);
for (epb = head->chain; epb; epb = epb->next) {
ep = sctp_ep(epb);
- if (sctp_endpoint_is_match(ep, laddr)) { goto hit; }
- }
-
+ if (sctp_endpoint_is_match(ep, laddr))
+ goto hit;
+ }
+
ep = sctp_sk((sctp_get_ctl_sock()))->ep;
epb = &ep->base;
-hit:
+hit:
sctp_endpoint_hold(ep);
sock_hold(epb->sk);
read_unlock(&head->lock);
return ep;
-
-} /* __sctp_rcv_lookup_endpoint() */
-
+}
/* Add an association to the hash. Local BH-safe. */
void sctp_hash_established(sctp_association_t *asoc)
sctp_local_bh_disable();
__sctp_hash_established(asoc);
sctp_local_bh_enable();
-
-} /* sctp_hash_association() */
+}
/* Insert association into the hash table. */
void __sctp_hash_established(sctp_association_t *asoc)
sctp_endpoint_common_t **epp;
sctp_endpoint_common_t *epb;
sctp_hashbucket_t *head;
-
+
epb = &asoc->base;
/* Calculate which chain this entry will belong to. */
epb->hashent = sctp_assoc_hashfn(epb->bind_addr.port, asoc->peer.port);
-
+
head = &sctp_proto.assoc_hashbucket[epb->hashent];
-
+
sctp_write_lock(&head->lock);
epp = &head->chain;
epb->next = *epp;
- if (epb->next) {
+ if (epb->next)
(*epp)->pprev = &epb->next;
- }
*epp = epb;
epb->pprev = epp;
sctp_write_unlock(&head->lock);
-
-} /* __sctp_hash_established() */
+}
/* Remove association from the hash table. Local BH-safe. */
void sctp_unhash_established(sctp_association_t *asoc)
sctp_local_bh_disable();
__sctp_unhash_established(asoc);
sctp_local_bh_enable();
+}
-} /* sctp_unhash_established() */
-
/* Remove association from the hash table. */
void __sctp_unhash_established(sctp_association_t *asoc)
{
epb->hashent = sctp_assoc_hashfn(epb->bind_addr.port,
asoc->peer.port);
-
+
head = &sctp_proto.assoc_hashbucket[epb->hashent];
sctp_write_lock(&head->lock);
if (epb->pprev) {
- if (epb->next) {
+ if (epb->next)
epb->next->pprev = epb->pprev;
- }
*epb->pprev = epb->next;
epb->pprev = NULL;
}
sctp_write_unlock(&head->lock);
-
-} /* __sctp_unhash_established() */
-
+}
/* Look up an association. */
-sctp_association_t *
-__sctp_rcv_lookup_association(const sockaddr_storage_t *laddr,
- const sockaddr_storage_t *paddr,
- sctp_transport_t **transportp)
+sctp_association_t *__sctp_rcv_lookup_association(const sockaddr_storage_t *laddr,
+ const sockaddr_storage_t *paddr,
+ sctp_transport_t **transportp)
{
sctp_hashbucket_t *head;
sctp_endpoint_common_t *epb;
int hash;
/* Optimize here for direct hit, only listening connections can
- * have wildcards anyways.
+ * have wildcards anyways.
*/
-
hash = sctp_assoc_hashfn(laddr->v4.sin_port, paddr->v4.sin_port);
head = &sctp_proto.assoc_hashbucket[hash];
read_lock(&head->lock);
for (epb = head->chain; epb; epb = epb->next) {
asoc = sctp_assoc(epb);
transport = sctp_assoc_is_match(asoc, laddr, paddr);
- if (transport) { goto hit; }
- }
-
+ if (transport)
+ goto hit;
+ }
+
read_unlock(&head->lock);
return NULL;
sock_hold(epb->sk);
read_unlock(&head->lock);
return asoc;
-
-} /* __sctp_rcv_lookup_association() */
+}
/*
- * SCTP Implementors Guide, 2.18 Handling of address
+ * SCTP Implementors Guide, 2.18 Handling of address
* parameters within the INIT or INIT-ACK.
*
* D) When searching for a matching TCB upon reception of an INIT
* source address of the packet (containing the INIT or
* INIT-ACK) but the receiver SHOULD also use all valid
* address parameters contained within the chunk.
- *
+ *
* 2.18.3 Solution description
*
* This new text clearly specifies to an implementor the need
* in certain circumstances.
*
*/
-static sctp_association_t *
-__sctp_rcv_initack_lookup(struct sk_buff *skb,
- const sockaddr_storage_t *laddr,
- sctp_transport_t **transportp)
+static sctp_association_t *__sctp_rcv_initack_lookup(struct sk_buff *skb,
+ const sockaddr_storage_t *laddr, sctp_transport_t **transportp)
{
sctp_association_t *asoc;
sockaddr_storage_t addr;
sockaddr_storage_t *paddr = &addr;
- struct sctphdr *sh = (struct sctphdr *)skb->h.raw;
+ struct sctphdr *sh = (struct sctphdr *) skb->h.raw;
sctp_chunkhdr_t *ch;
- uint8_t *ch_end, *data;
+ __u8 *ch_end, *data;
sctpParam_t parm;
- ch = (sctp_chunkhdr_t *)skb->data;
+ ch = (sctp_chunkhdr_t *) skb->data;
- ch_end = ((uint8_t *)ch) + WORD_ROUND(ntohs(ch->length));
+ ch_end = ((__u8 *) ch) + WORD_ROUND(ntohs(ch->length));
- if (SCTP_CID_INIT_ACK != ch->type) {
+ if (SCTP_CID_INIT_ACK != ch->type)
return NULL;
- }
- /*
- * This code will NOT touch anything inside the chunk--it is
- * strictly READ-ONLY.
- *
- * RFC 2960 3 SCTP packet Format
- *
- * Multiple chunks can be bundled into one SCTP packet up to
- * the MTU size, except for the INIT, INIT ACK, and SHUTDOWN
- * COMPLETE chunks. These chunks MUST NOT be bundled with any
- * other chunk in a packet. See Section 6.10 for more details
- * on chunk bundling.
- */
-
- /* Find the start of the TLVs and the end of the chunk. This is
+ /*
+ * This code will NOT touch anything inside the chunk--it is
+ * strictly READ-ONLY.
+ *
+ * RFC 2960 3 SCTP packet Format
+ *
+ * Multiple chunks can be bundled into one SCTP packet up to
+ * the MTU size, except for the INIT, INIT ACK, and SHUTDOWN
+ * COMPLETE chunks. These chunks MUST NOT be bundled with any
+ * other chunk in a packet. See Section 6.10 for more details
+ * on chunk bundling.
+ */
+
+ /* Find the start of the TLVs and the end of the chunk. This is
* the region we search for address parameters.
*/
- data = skb->data + sizeof(sctp_init_chunk_t);
-
- /* See sctp_process_init() for how to go thru TLVs. */
- while ( data < ch_end ) {
-
+ data = skb->data + sizeof(sctp_init_chunk_t);
+
+ /* See sctp_process_init() for how to go thru TLVs. */
+ while (data < ch_end) {
parm.v = data;
-
- if (parm.p->length == 0) { break; }
-
+
+ if (!parm.p->length)
+ break;
+
data += WORD_ROUND(ntohs(parm.p->length));
-
- /* Note: Ignoring hostname addresses. */
- if((SCTP_PARAM_IPV4_ADDRESS != parm.p->type)
- && (SCTP_PARAM_IPV6_ADDRESS != parm.p->type)) {
+
+ /* Note: Ignoring hostname addresses. */
+ if ((SCTP_PARAM_IPV4_ADDRESS != parm.p->type) &&
+ (SCTP_PARAM_IPV6_ADDRESS != parm.p->type))
continue;
- }
-
+
sctp_param2sockaddr(paddr, parm, ntohs(sh->source));
-
+
asoc = __sctp_rcv_lookup_association(laddr, paddr, transportp);
- if (asoc) { return asoc; }
-
- } /* while (not at end of chunk) */
+ if (asoc)
+ return asoc;
+ }
return NULL;
-
-} /* __sctp_rcv_initack_lookup() */
+}
/* Lookup an association for an inbound skb. */
-sctp_association_t *
-__sctp_rcv_lookup(struct sk_buff *skb, const sockaddr_storage_t *paddr,
- const sockaddr_storage_t *laddr,
- sctp_transport_t **transportp)
+sctp_association_t *__sctp_rcv_lookup(struct sk_buff *skb,
+ const sockaddr_storage_t *paddr,
+ const sockaddr_storage_t *laddr,
+ sctp_transport_t **transportp)
{
sctp_association_t *asoc;
asoc = __sctp_rcv_lookup_association(laddr, paddr, transportp);
-
- /* Further lookup for INIT-ACK packet.
- * SCTP Implementors Guide, 2.18 Handling of address
+
+ /* Further lookup for INIT-ACK packet.
+ * SCTP Implementors Guide, 2.18 Handling of address
* parameters within the INIT or INIT-ACK.
*/
- if (!asoc) {
+ if (!asoc)
asoc = __sctp_rcv_initack_lookup(skb, laddr, transportp);
- }
-
- return asoc;
-} /* __sctp_rcv_lookup() */
+ return asoc;
+}
#include <linux/interrupt.h>
/* Initialize an SCTP_inqueue. */
-void
-sctp_inqueue_init(sctp_inqueue_t *queue)
+void sctp_inqueue_init(sctp_inqueue_t *queue)
{
- skb_queue_head_init(&queue->in);
+ skb_queue_head_init(&queue->in);
queue->in_progress = NULL;
- /* Create a task for delivering data. */
- INIT_LIST_HEAD(&queue->immediate.list);
- queue->immediate.sync = 0;
- queue->immediate.routine = NULL;
- queue->immediate.data = NULL;
+ /* Create a task for delivering data. */
+ INIT_LIST_HEAD(&queue->immediate.list);
+ queue->immediate.sync = 0;
+ queue->immediate.routine = NULL;
+ queue->immediate.data = NULL;
- queue->malloced = 0;
-
-} /* sctp_inqueue_init() */
+ queue->malloced = 0;
+}
/* Create an initialized SCTP_inqueue. */
-sctp_inqueue_t *
-sctp_inqueue_new()
+sctp_inqueue_t *sctp_inqueue_new(void)
{
- sctp_inqueue_t *retval;
-
- retval = t_new(sctp_inqueue_t, GFP_ATOMIC);
- if (NULL == retval) { return(NULL); }
-
- sctp_inqueue_init(retval);
- retval->malloced = 1;
-
- return(retval);
+ sctp_inqueue_t *retval;
-} /* sctp_inqueue_new() */
+ retval = t_new(sctp_inqueue_t, GFP_ATOMIC);
+ if (retval) {
+ sctp_inqueue_init(retval);
+ retval->malloced = 1;
+ }
+ return retval;
+}
/* Release the memory associated with an SCTP inqueue. */
-void
-sctp_inqueue_free(sctp_inqueue_t *queue)
+void sctp_inqueue_free(sctp_inqueue_t *queue)
{
sctp_chunk_t *chunk;
- /* Empty the queue. */
- while ((chunk=(sctp_chunk_t *)skb_dequeue(&queue->in))!=NULL) {
+ /* Empty the queue. */
+ while ((chunk = (sctp_chunk_t *) skb_dequeue(&queue->in)) != NULL)
sctp_free_chunk(chunk);
- }
- /* If there is a packet which is currently being worked on,
+
+ /* If there is a packet which is currently being worked on,
* free it as well.
*/
- if (queue->in_progress) {
+ if (queue->in_progress)
sctp_free_chunk(queue->in_progress);
- }
-
- if (queue->malloced) {
- /* Dump the master memory segment. */
- kfree(queue);
- }
-
-} /* sctp_inqueue_free() */
-
+ if (queue->malloced) {
+ /* Dump the master memory segment. */
+ kfree(queue);
+ }
+}
-/* Put a new packet in an SCTP inqueue.
- * We assume that packet->sctp_hdr is set and in host byte order.
+/* Put a new packet in an SCTP inqueue.
+ * We assume that packet->sctp_hdr is set and in host byte order.
*/
-void
-sctp_push_inqueue(sctp_inqueue_t *q, sctp_chunk_t *packet)
+void sctp_push_inqueue(sctp_inqueue_t *q, sctp_chunk_t *packet)
{
/* Directly call the packet handling routine. */
+
/* We are now calling this either from the soft interrupt
* or from the backlog processing.
* Eventually, we should clean up inqueue to not rely
* on the BH related data structures.
*/
- skb_queue_tail(&(q->in), (struct sk_buff *)packet);
+ skb_queue_tail(&(q->in), (struct sk_buff *) packet);
q->immediate.routine(q->immediate.data);
-
-} /* sctp_push_inqueue() */
+}
/* Extract a chunk from an SCTP inqueue.
- *
- * WARNING: If you need to put the chunk on another queue, you need to
+ *
+ * WARNING: If you need to put the chunk on another queue, you need to
* make a shallow copy (clone) of it.
*/
-sctp_chunk_t *
-sctp_pop_inqueue(sctp_inqueue_t *queue)
+sctp_chunk_t *sctp_pop_inqueue(sctp_inqueue_t *queue)
{
- sctp_chunk_t *chunk;
- sctp_chunkhdr_t *ch = NULL;
+ sctp_chunk_t *chunk;
+ sctp_chunkhdr_t *ch = NULL;
/* The assumption is that we are safe to process the chunks
- * at this time.
+ * at this time.
*/
- if (NULL != (chunk = queue->in_progress)) {
+ if ((chunk = queue->in_progress) != NULL) {
/* There is a packet that we have been working on.
- * Any post processing work to do before we move on?
- */
- if (chunk->singleton || chunk->end_of_packet
- || chunk->pdiscard) {
+ * Any post processing work to do before we move on?
+ */
+ if (chunk->singleton ||
+ chunk->end_of_packet ||
+ chunk->pdiscard) {
sctp_free_chunk(chunk);
chunk = queue->in_progress = NULL;
- } else {
- /* Nothing to do. Next chunk in the packet, please. */
- ch = (sctp_chunkhdr_t *)chunk->chunk_end;
+ } else {
+ /* Nothing to do. Next chunk in the packet, please. */
+ ch = (sctp_chunkhdr_t *) chunk->chunk_end;
/* Force chunk->skb->data to chunk->chunk_end. */
- skb_pull(chunk->skb,
+ skb_pull(chunk->skb,
chunk->chunk_end - chunk->skb->data);
- }
- }
+ }
+ }
/* Do we need to take the next packet out of the queue to process? */
- if (NULL == chunk) {
-
- /* Is the queue empty? */
- if (skb_queue_empty(&queue->in)) {
- return(NULL);
- }
+ if (!chunk) {
+ /* Is the queue empty? */
+ if (skb_queue_empty(&queue->in))
+ return NULL;
- chunk = queue->in_progress =
- (sctp_chunk_t *)skb_dequeue(&queue->in);
+ chunk = queue->in_progress =
+ (sctp_chunk_t *) skb_dequeue(&queue->in);
- /* This is the first chunk in the packet. */
- chunk->singleton = 1;
- ch = (sctp_chunkhdr_t *) chunk->skb->data;
+ /* This is the first chunk in the packet. */
+ chunk->singleton = 1;
+ ch = (sctp_chunkhdr_t *) chunk->skb->data;
}
chunk->chunk_hdr = ch;
- chunk->chunk_end = ((uint8_t *)ch)
+ chunk->chunk_end = ((__u8 *) ch)
+ WORD_ROUND(ntohs(ch->length));
- skb_pull(chunk->skb, sizeof(sctp_chunkhdr_t));
- chunk->subh.v = NULL; /* Subheader is no longer valid. */
+ skb_pull(chunk->skb, sizeof(sctp_chunkhdr_t));
+ chunk->subh.v = NULL; /* Subheader is no longer valid. */
- if (chunk->chunk_end < chunk->skb->tail) {
+ if (chunk->chunk_end < chunk->skb->tail) {
/* This is not a singleton */
- chunk->singleton = 0;
- } else {
- /* We are at the end of the packet, so mark the chunk
- * in case we need to send a SACK.
- */
- chunk->end_of_packet = 1;
- }
-
+ chunk->singleton = 0;
+ } else {
+ /* We are at the end of the packet, so mark the chunk
+ * in case we need to send a SACK.
+ */
+ chunk->end_of_packet = 1;
+ }
SCTP_DEBUG_PRINTK("+++sctp_pop_inqueue+++ chunk %p[%s],"
- " length %d, skb->len %d\n",chunk,
+ " length %d, skb->len %d\n",chunk,
sctp_cname(SCTP_ST_CHUNK(chunk->chunk_hdr->type)),
ntohs(chunk->chunk_hdr->length), chunk->skb->len);
- return(chunk);
-
-} /* sctp_pop_inqueue() */
+ return chunk;
+}
/* Set a top-half handler.
- *
+ *
* Originally, we the top-half handler was scheduled as a BH. We now
- * call the handler directly in sctp_push_inqueue() at a time that
- * we know we are lock safe.
- * The intent is that this routine will pull stuff out of the
+ * call the handler directly in sctp_push_inqueue() at a time that
+ * we know we are lock safe.
+ * The intent is that this routine will pull stuff out of the
* inqueue and process it.
*/
-void
-sctp_inqueue_set_th_handler(sctp_inqueue_t *q,
- void (*callback)(void *), void *arg)
+void sctp_inqueue_set_th_handler(sctp_inqueue_t *q,
+ void (*callback)(void *), void *arg)
{
- q->immediate.routine = callback;
- q->immediate.data = arg;
-
-} /* sctp_inqueue_set_th_handler() */
+ q->immediate.routine = callback;
+ q->immediate.data = arg;
+}
#define __NO_VERSION__
#include <linux/module.h>
-#include <linux/config.h>
#include <linux/errno.h>
#include <linux/types.h>
#include <linux/socket.h>
#include <asm/uaccess.h>
-
/* FIXME: Cleanup so we don't need TEST_FRAME here. */
#ifndef TEST_FRAME
/* FIXME: Comments. */
struct inet6_skb_parm *opt,
int type, int code, int offset, __u32 info)
{
- /* BUG. WRITE ME. */
-}/* sctp_v6_err */
+ /* BUG. WRITE ME. */
+}
/* Based on tcp_v6_xmit() in tcp_ipv6.c. */
-static inline int
-sctp_v6_xmit(struct sk_buff *skb)
+static inline int sctp_v6_xmit(struct sk_buff *skb)
{
struct sock *sk = skb->sk;
struct ipv6_pinfo *np = inet6_sk(sk);
int err = 0;
fl.proto = sk->protocol;
- fl.fl6_dst = &np->daddr;
+ fl.fl6_dst = &np->daddr;
fl.fl6_src = NULL;
fl.fl6_flowlabel = np->flow_label;
}
dst = __sk_dst_check(sk, np->dst_cookie);
-
+
if (dst == NULL) {
dst = ip6_route_output(sk, &fl);
skb->dst = dst_clone(dst);
/* FIXME: This is all temporary until real source address
- * selection is done.
+ * selection is done.
*/
if (ipv6_addr_any(&np->saddr)) {
err = ipv6_get_saddr(dst, fl.fl6_dst, &saddr);
- if (err) {
+ if (err)
printk(KERN_ERR "sctp_v6_xmit: no saddr available\n");
- }
- /* FIXME: This is a workaround until we get
+ /* FIXME: This is a workaround until we get
* real source address selection done. This is here
- * to disallow loopback when the scoping rules have
- * not bound loopback to the endpoint.
+ * to disallow loopback when the scoping rules have
+ * not bound loopback to the endpoint.
*/
- if (sctp_ipv6_addr_type(&saddr)
- & IPV6_ADDR_LOOPBACK) {
+ if (sctp_ipv6_addr_type(&saddr) & IPV6_ADDR_LOOPBACK) {
if (!(sctp_ipv6_addr_type(&np->daddr) &
IPV6_ADDR_LOOPBACK)) {
-
ipv6_addr_copy(&saddr, &np->daddr);
}
}
-
fl.fl6_src = &saddr;
} else {
fl.fl6_src = &np->saddr;
}
/* Restore final destination back after routing done */
- fl.nl_u.ip6_u.daddr = &np->daddr;
-
-
+ fl.nl_u.ip6_u.daddr = &np->daddr;
return ip6_xmit(sk, skb, &fl, np->opt);
-
-} /* sctp_v6_xmit */
+}
#endif /* TEST_FRAME */
/* Returns the mtu for the given v6 destination address. */
-int
-sctp_v6_get_dst_mtu(const sockaddr_storage_t *address)
+int sctp_v6_get_dst_mtu(const sockaddr_storage_t *address)
{
+ struct dst_entry *dst;
+ struct flowi fl;
int dst_mtu = SCTP_DEFAULT_MAXSEGMENT;
-
- struct flowi fl;
- struct dst_entry *dst;
-
fl.proto = 0;
fl.fl6_dst = (struct in6_addr *)&address->v6.sin6_addr;
fl.fl6_src = NULL;
fl.oif = 0;
fl.uli_u.ports.sport = 0;
fl.uli_u.ports.dport = 0;
-
+
dst = ip6_route_output(NULL, &fl);
if (dst) {
dst_mtu = dst->pmtu;
"%d as dst_mtu\n", dst_mtu);
}
- return (dst_mtu);
-
-} /* sctp_v6_get_dst_mtu() */
-
+ return dst_mtu;
+}
static struct proto_ops inet6_seqpacket_ops = {
.family = PF_INET6,
.release = inet6_release,
.bind = inet6_bind,
- .connect = inet_dgram_connect,
- .socketpair = sock_no_socketpair,
- .accept = inet_accept,
+ .connect = inet_dgram_connect,
+ .socketpair = sock_no_socketpair,
+ .accept = inet_accept,
.getname = inet6_getname,
- .poll = sctp_poll,
+ .poll = sctp_poll,
.ioctl = inet6_ioctl,
- .listen = sctp_inet_listen,
+ .listen = sctp_inet_listen,
.shutdown = inet_shutdown,
.setsockopt = inet_setsockopt,
- .getsockopt = inet_getsockopt,
- .sendmsg = inet_sendmsg,
- .recvmsg = inet_recvmsg,
+ .getsockopt = inet_getsockopt,
+ .sendmsg = inet_sendmsg,
+ .recvmsg = inet_recvmsg,
.mmap = sock_no_mmap,
};
-static struct inet_protosw sctpv6_protosw =
-{
- .type = SOCK_SEQPACKET,
+static struct inet_protosw sctpv6_protosw = {
+ .type = SOCK_SEQPACKET,
.protocol = IPPROTO_SCTP,
.prot = &sctp_prot,
.ops = &inet6_seqpacket_ops,
.flags = SCTP_PROTOSW_FLAG
};
-static struct inet6_protocol sctpv6_protocol =
-{
- .handler = sctp_rcv,
- .err_handler = sctp_v6_err,
- .next = NULL,
- .protocol = IPPROTO_SCTP,
- .copy = 0,
- .data = NULL,
- .name = "SCTPv6"
-};
+static struct inet6_protocol sctpv6_protocol = {
+ .handler = sctp_rcv,
+ .err_handler = sctp_v6_err,
+ .next = NULL,
+ .protocol = IPPROTO_SCTP,
+ .copy = 0,
+ .data = NULL,
+ .name = "SCTPv6"
+};
static sctp_func_t sctp_ipv6_specific = {
- .queue_xmit = sctp_v6_xmit,
- .setsockopt = ipv6_setsockopt,
- .getsockopt = ipv6_getsockopt,
+ .queue_xmit = sctp_v6_xmit,
+ .setsockopt = ipv6_setsockopt,
+ .getsockopt = ipv6_getsockopt,
.get_dst_mtu = sctp_v6_get_dst_mtu,
.net_header_len = sizeof(struct ipv6hdr),
- .sockaddr_len = sizeof(struct sockaddr_in6),
+ .sockaddr_len = sizeof(struct sockaddr_in6),
.sa_family = AF_INET6,
-};
-
+};
-/* Initialize IPv6 support and register with inet6 stack. */
+/* Initialize IPv6 support and register with inet6 stack. */
int sctp_v6_init(void)
-{
-
- /* Add SCTPv6 to inetsw6 linked list. */
+{
+ /* Add SCTPv6 to inetsw6 linked list. */
inet6_register_protosw(&sctpv6_protosw);
-
- /* Register inet6 protocol. */
- inet6_add_protocol(&sctpv6_protocol);
- /* Fill in address family info. */
+ /* Register inet6 protocol. */
+ inet6_add_protocol(&sctpv6_protocol);
+
+ /* Fill in address family info. */
INIT_LIST_HEAD(&sctp_ipv6_specific.list);
list_add_tail(&sctp_ipv6_specific.list, &sctp_proto.address_families);
return 0;
-
-} /* sctp_v6_init() */
+}
/* IPv6 specific exit support. */
void sctp_v6_exit(void)
list_del(&sctp_ipv6_specific.list);
inet6_del_protocol(&sctpv6_protocol);
inet6_unregister_protosw(&sctpv6_protosw);
-
-} /* sctp_v6_exit() */
+}
#include <net/sctp/sctp.h>
-/*
- * Global counters to count raw object allocation counts.
+/*
+ * Global counters to count raw object allocation counts.
* To add new counters, choose a unique suffix for the variable
* name as the helper macros key off this suffix to make
- * life easier for the programmer.
+ * life easier for the programmer.
*/
SCTP_DBG_OBJCNT(sock);
SCTP_DBG_OBJCNT(addr);
/* An array to make it easy to pretty print the debug information
- * to the proc fs.
+ * to the proc fs.
*/
sctp_dbg_objcnt_entry_t sctp_dbg_objcnt[] = {
SCTP_DBG_OBJCNT_ENTRY(sock),
SCTP_DBG_OBJCNT_ENTRY(addr),
};
-/* Callback from procfs to read out objcount information.
+/* Callback from procfs to read out objcount information.
* Walk through the entries in the sctp_dbg_objcnt array, dumping
* the raw object counts for each monitored type.
- *
+ *
* This code was modified from similar code in route.c
*/
-static int
-sctp_dbg_objcnt_read(char *buffer, char **start, off_t offset,
- int length, int *eof, void *data)
+static int sctp_dbg_objcnt_read(char *buffer, char **start, off_t offset,
+ int length, int *eof, void *data)
{
int len = 0;
off_t pos = 0;
/* How many entries? */
entries = sizeof(sctp_dbg_objcnt)/sizeof(sctp_dbg_objcnt[0]);
- /* Walk the entries and print out the debug information
+ /* Walk the entries and print out the debug information
* for proc fs.
*/
- for(i = 0; i < entries; i++) {
+ for (i = 0; i < entries; i++) {
pos += 128;
/* Skip ahead. */
continue;
}
/* Print out each entry. */
- sprintf(temp, "%s: %d",
+ sprintf(temp, "%s: %d",
sctp_dbg_objcnt[i].label,
atomic_read(sctp_dbg_objcnt[i].counter));
sprintf(buffer + len, "%-127s\n", temp);
len += 128;
- if (pos >= offset+length) {
+ if (pos >= offset+length)
goto done;
- }
-
}
done:
- *start = buffer + len - (pos - offset);
- len = pos - offset;
- if (len > length)
- len = length;
+ *start = buffer + len - (pos - offset);
+ len = pos - offset;
+ if (len > length)
+ len = length;
- return len;
+ return len;
+}
-} /* sctp_dbg_objcnt_read() */
-
-/* Initialize the objcount in the proc filesystem. */
+/* Initialize the objcount in the proc filesystem. */
void sctp_dbg_objcnt_init(void)
{
create_proc_read_entry("sctp_dbg_objcnt", 0, proc_net_sctp,
sctp_dbg_objcnt_read, NULL);
+}
-} /* sctp_dbg_objcnt_init() */
-
-/* Cleanup the objcount entry in the proc filesystem. */
+/* Cleanup the objcount entry in the proc filesystem. */
void sctp_dbg_objcnt_exit(void)
{
remove_proc_entry("sctp_dbg_objcount", proc_net_sctp);
-
-} /* sctp_dbg_objcnt_exit() */
+}
*/
static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_output.c,v 1.22 2002/07/12 14:39:05 jgrimm Exp $";
-#include <linux/config.h>
#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/wait.h>
#include <linux/ip.h>
#include <linux/ipv6.h>
#include <linux/init.h>
-#ifndef CONFIG_INET_ECN
-#define CONFIG_INET_ECN
-#endif /* CONFIG_INET_ECN */
#include <net/inet_ecn.h>
#include <net/icmp.h>
#include <net/sctp/sctp.h>
#include <net/sctp/sctp_sm.h>
-
/* Forward declarations for private helpers. */
-uint32_t count_crc(uint8_t *ptr, uint16_t count);
+__u32 count_crc(__u8 *ptr, __u16 count);
static void sctp_packet_reset(sctp_packet_t *packet);
-static inline sctp_xmit_t
-sctp_packet_append_data(sctp_packet_t *packet, sctp_chunk_t *chunk);
+static sctp_xmit_t sctp_packet_append_data(sctp_packet_t *packet,
+ sctp_chunk_t *chunk);
-/* Config a packet.
+/* Config a packet.
* This appears to be a followup set of initializations.)
*/
-sctp_packet_t *
-sctp_packet_config(sctp_packet_t *packet,
- uint32_t vtag,
- int ecn_capable,
- sctp_packet_phandler_t *prepend_handler)
+sctp_packet_t *sctp_packet_config(sctp_packet_t *packet,
+ __u32 vtag,
+ int ecn_capable,
+ sctp_packet_phandler_t *prepend_handler)
{
int packet_empty = (packet->size == SCTP_IP_OVERHEAD);
- packet->vtag = vtag;
+ packet->vtag = vtag;
packet->ecn_capable = ecn_capable;
packet->get_prepend_chunk = prepend_handler;
packet->has_cookie_echo = 0;
/* We might need to call the prepend_handler right away. */
- if (packet_empty) {
+ if (packet_empty)
sctp_packet_reset(packet);
- }
- return(packet);
-
-} /* sctp_packet_config() */
+ return packet;
+}
/* Initialize the packet structure. */
-sctp_packet_t *
-sctp_packet_init(sctp_packet_t *packet,
- sctp_transport_t *transport,
- uint16_t sport,
- uint16_t dport)
+sctp_packet_t *sctp_packet_init(sctp_packet_t *packet,
+ sctp_transport_t *transport,
+ __u16 sport,
+ __u16 dport)
{
packet->transport = transport;
packet->source_port = sport;
packet->ecn_capable = 0;
packet->get_prepend_chunk = NULL;
packet->has_cookie_echo = 0;
-
packet->malloced = 0;
-
sctp_packet_reset(packet);
- return packet;
-
-} /* sctp_packet_init() */
-
+ return packet;
+}
/* Free a packet. */
-void
-sctp_packet_free(sctp_packet_t *packet)
+void sctp_packet_free(sctp_packet_t *packet)
{
- sctp_chunk_t *chunk;
+ sctp_chunk_t *chunk;
while (NULL !=
- (chunk = (sctp_chunk_t *)skb_dequeue(&packet->chunks))) {
- sctp_free_chunk(chunk);
- }
-
- if (packet->malloced) {
- kfree(packet);
+ (chunk = (sctp_chunk_t *)skb_dequeue(&packet->chunks))) {
+ sctp_free_chunk(chunk);
}
-} /* sctp_packet_free() */
+ if (packet->malloced)
+ kfree(packet);
+}
/* This routine tries to append the chunk to the offered packet. If adding
* the chunk causes the packet to exceed the path MTU and COOKIE_ECHO chunk
* as it can fit in the packet, but any more data that does not fit in this
* packet can be sent only after receiving the COOKIE_ACK.
*/
-sctp_xmit_t
-sctp_packet_transmit_chunk(sctp_packet_t *packet, sctp_chunk_t *chunk)
+sctp_xmit_t sctp_packet_transmit_chunk(sctp_packet_t *packet, sctp_chunk_t *chunk)
{
- sctp_xmit_t retval;
- int error = 0;
+ sctp_xmit_t retval;
+ int error = 0;
- switch (retval = (sctp_packet_append_chunk(packet, chunk))) {
- case SCTP_XMIT_PMTU_FULL:
+ switch ((retval = (sctp_packet_append_chunk(packet, chunk)))) {
+ case SCTP_XMIT_PMTU_FULL:
if (!packet->has_cookie_echo) {
error = sctp_packet_transmit(packet);
- if (error < 0) {
+ if (error < 0)
chunk->skb->sk->err = -error;
- }
+
/* If we have an empty packet, then we can NOT ever
* return PMTU_FULL.
*/
retval = sctp_packet_append_chunk(packet, chunk);
}
- break;
- case SCTP_XMIT_MUST_FRAG:
- case SCTP_XMIT_RWND_FULL:
- case SCTP_XMIT_OK:
- break;
- }
+ break;
- return(retval);
-
-} /* sctp_packet_transmit_chunk() */
+ case SCTP_XMIT_MUST_FRAG:
+ case SCTP_XMIT_RWND_FULL:
+ case SCTP_XMIT_OK:
+ break;
+ };
+ return retval;
+}
/* Append a chunk to the offered packet reporting back any inability to do
- * so.
+ * so.
*/
-sctp_xmit_t
-sctp_packet_append_chunk(sctp_packet_t *packet, sctp_chunk_t *chunk)
+sctp_xmit_t sctp_packet_append_chunk(sctp_packet_t *packet, sctp_chunk_t *chunk)
{
- sctp_xmit_t retval = SCTP_XMIT_OK;
- uint16_t chunk_len = WORD_ROUND(ntohs(chunk->chunk_hdr->length));
- size_t psize = packet->size;
- size_t pmtu;
- int too_big;
+ sctp_xmit_t retval = SCTP_XMIT_OK;
+ __u16 chunk_len = WORD_ROUND(ntohs(chunk->chunk_hdr->length));
+ size_t psize = packet->size;
+ size_t pmtu;
+ int too_big;
- pmtu = ((packet->transport->asoc) ?
+ pmtu = ((packet->transport->asoc) ?
(packet->transport->asoc->pmtu) :
(packet->transport->pmtu));
too_big = (psize + chunk_len > pmtu);
- /* Decide if we need to fragment or resubmit later. */
- if (too_big) {
- int packet_empty = (packet->size == SCTP_IP_OVERHEAD);
-
+ /* Decide if we need to fragment or resubmit later. */
+ if (too_big) {
+ int packet_empty = (packet->size == SCTP_IP_OVERHEAD);
+
/* Both control chunks and data chunks with TSNs are
- * non-fragmentable.
- */
- int fragmentable = sctp_chunk_is_data(chunk)
+ * non-fragmentable.
+ */
+ int fragmentable = sctp_chunk_is_data(chunk)
&& (!chunk->has_tsn);
- if (packet_empty) {
- if (fragmentable) {
- retval = SCTP_XMIT_MUST_FRAG;
- goto finish;
- } else {
- /* The packet is too big but we can
- * not fragment it--we have to just
- * transmit and rely on IP
- * fragmentation.
- */
- goto append;
- }
-
- } else { /* !packet_empty */
- retval = SCTP_XMIT_PMTU_FULL;
- goto finish;
- }
-
+ if (packet_empty) {
+ if (fragmentable) {
+ retval = SCTP_XMIT_MUST_FRAG;
+ goto finish;
+ } else {
+ /* The packet is too big but we can
+ * not fragment it--we have to just
+ * transmit and rely on IP
+ * fragmentation.
+ */
+ goto append;
+ }
+ } else { /* !packet_empty */
+ retval = SCTP_XMIT_PMTU_FULL;
+ goto finish;
+ }
} else {
/* The chunk fits in the packet. */
goto append;
}
append:
- /* We believe that this chunk is OK to add to the packet (as
- * long as we have the cwnd for it).
- */
+ /* We believe that this chunk is OK to add to the packet (as
+ * long as we have the cwnd for it).
+ */
/* DATA is a special case since we must examine both rwnd and cwnd
* before we send DATA.
*/
if (sctp_chunk_is_data(chunk)) {
retval = sctp_packet_append_data(packet, chunk);
- if (SCTP_XMIT_OK != retval) {
+ if (SCTP_XMIT_OK != retval)
goto finish;
- }
- }
- else if (SCTP_CID_COOKIE_ECHO == chunk->chunk_hdr->type) {
+ } else if (SCTP_CID_COOKIE_ECHO == chunk->chunk_hdr->type) {
packet->has_cookie_echo = 1;
}
-
-
/* It is OK to send this chunk. */
- skb_queue_tail(&packet->chunks,
+ skb_queue_tail(&packet->chunks,
(struct sk_buff *)chunk);
packet->size += chunk_len;
-
finish:
- return(retval);
-
-} /* sctp_packet_append_chunk() */
+ return retval;
+}
/* All packets are sent to the network through this function from
* sctp_push_outqueue().
*
* The return value is a normal kernel error return value.
*/
-int
-sctp_packet_transmit(sctp_packet_t *packet)
+int sctp_packet_transmit(sctp_packet_t *packet)
{
sctp_transport_t *transport = packet->transport;
sctp_association_t *asoc = transport->asoc;
- struct sctphdr *sh;
- uint32_t crc32;
+ struct sctphdr *sh;
+ __u32 crc32;
struct sk_buff *nskb;
sctp_chunk_t *chunk;
- struct sock *sk;
- int err = 0;
+ struct sock *sk;
+ int err = 0;
int padding; /* How much padding do we need? */
- uint8_t packet_has_data = 0;
+ __u8 packet_has_data = 0;
- /* Do NOT generate a chunkless packet... */
- if (skb_queue_empty(&packet->chunks)) {
- return(err);
- }
+ /* Do NOT generate a chunkless packet... */
+ if (skb_queue_empty(&packet->chunks))
+ return err;
+
+ /* Set up convenience variables... */
+ chunk = (sctp_chunk_t *) (packet->chunks.next);
+ sk = chunk->skb->sk;
- /* Set up convenience variables... */
- chunk = (sctp_chunk_t *)(packet->chunks.next);
- sk = chunk->skb->sk;
-
/* Allocate the new skb. */
nskb = dev_alloc_skb(packet->size);
- if (NULL == nskb) {
+ if (!nskb) {
err = -ENOMEM;
goto out;
}
- /* Make sure the outbound skb has enough header room reserved. */
- skb_reserve(nskb, SCTP_IP_OVERHEAD);
- /* Set the owning socket so that we know where to get the
- * destination IP address.
- */
+ /* Make sure the outbound skb has enough header room reserved. */
+ skb_reserve(nskb, SCTP_IP_OVERHEAD);
+
+ /* Set the owning socket so that we know where to get the
+ * destination IP address.
+ */
skb_set_owner_w(nskb, sk);
/**
* 6.10 Bundling
- *
+ *
* An endpoint bundles chunks by simply including multiple
* chunks in one outbound SCTP packet. ...
*/
+
/**
* 3.2 Chunk Field Descriptions
*
* pad the chunk with all zero bytes and this padding is not
* included in the chunk length field. The sender should
* never pad with more than 3 bytes.
- *
+ *
* [This whole comment explains WORD_ROUND() below.]
*/
SCTP_DEBUG_PRINTK("***sctp_transmit_packet***\n");
skb_dequeue(&packet->chunks))) {
chunk->num_times_sent++;
chunk->sent_at = jiffies;
- if (sctp_chunk_is_data(chunk)) {
-
- sctp_chunk_assign_tsn(chunk);
+ if (sctp_chunk_is_data(chunk)) {
+ sctp_chunk_assign_tsn(chunk);
- /* 6.3.1 C4) When data is in flight and when allowed
- * by rule C5, a new RTT measurement MUST be made each
- * round trip. Furthermore, new RTT measurements
+ /* 6.3.1 C4) When data is in flight and when allowed
+ * by rule C5, a new RTT measurement MUST be made each
+ * round trip. Furthermore, new RTT measurements
* SHOULD be made no more than once per round-trip
* for a given destination transport address.
*/
- if ((1 == chunk->num_times_sent)
- && (!transport->rto_pending)) {
+ if ((1 == chunk->num_times_sent) &&
+ (!transport->rto_pending)) {
chunk->rtt_in_progress = 1;
transport->rto_pending = 1;
}
packet_has_data = 1;
- }
+ }
memcpy(skb_put(nskb, chunk->skb->len),
- chunk->skb->data, chunk->skb->len);
+ chunk->skb->data, chunk->skb->len);
padding = WORD_ROUND(chunk->skb->len) - chunk->skb->len;
memset(skb_put(nskb, padding), 0, padding);
SCTP_DEBUG_PRINTK("%s %p[%s] %s 0x%x, %s %d, %s %d, %s %d, "
sctp_cname(SCTP_ST_CHUNK(
chunk->chunk_hdr->type)),
chunk->has_tsn ? "TSN" : "No TSN",
- chunk->has_tsn ?
+ chunk->has_tsn ?
ntohl(chunk->subh.data_hdr->tsn) : 0,
"length", ntohs(chunk->chunk_hdr->length),
- "chunk->skb->len", chunk->skb->len,
- "num_times_sent", chunk->num_times_sent,
+ "chunk->skb->len", chunk->skb->len,
+ "num_times_sent", chunk->num_times_sent,
"rtt_in_progress", chunk->rtt_in_progress);
- /*
- * If this is a control chunk, this is our last
- * reference. Free data chunks after they've been
+ /*
+ * If this is a control chunk, this is our last
+ * reference. Free data chunks after they've been
* acknowledged or have failed.
*/
- if (!sctp_chunk_is_data(chunk)) {
+ if (!sctp_chunk_is_data(chunk))
sctp_free_chunk(chunk);
- }
}
- /* Build the SCTP header. */
- sh = (struct sctphdr *)skb_push(nskb, sizeof(struct sctphdr));
- sh->source = htons(packet->source_port);
- sh->dest = htons(packet->destination_port);
-
- /* From 6.8 Adler-32 Checksum Calculation:
- * After the packet is constructed (containing the SCTP common
- * header and one or more control or DATA chunks), the
- * transmitter shall:
- *
- * 1) Fill in the proper Verification Tag in the SCTP common
- * header and initialize the checksum field to 0's.
- */
- sh->vtag = htonl(packet->vtag);
- sh->checksum = 0;
-
- /* 2) Calculate the Adler-32 checksum of the whole packet,
- * including the SCTP common header and all the
- * chunks.
- *
+ /* Build the SCTP header. */
+ sh = (struct sctphdr *) skb_push(nskb, sizeof(struct sctphdr));
+ sh->source = htons(packet->source_port);
+ sh->dest = htons(packet->destination_port);
+
+ /* From 6.8 Adler-32 Checksum Calculation:
+ * After the packet is constructed (containing the SCTP common
+ * header and one or more control or DATA chunks), the
+ * transmitter shall:
+ *
+ * 1) Fill in the proper Verification Tag in the SCTP common
+ * header and initialize the checksum field to 0's.
+ */
+ sh->vtag = htonl(packet->vtag);
+ sh->checksum = 0;
+
+ /* 2) Calculate the Adler-32 checksum of the whole packet,
+ * including the SCTP common header and all the
+ * chunks.
+ *
* Note: Adler-32 is no longer applicable, as has been replaced
- * by CRC32-C as described in <draft-ietf-tsvwg-sctpcsum-02.txt>.
- */
- crc32 = count_crc((uint8_t *)sh, nskb->len);
-
- /* 3) Put the resultant value into the checksum field in the
- * common header, and leave the rest of the bits unchanged.
- */
- sh->checksum = htonl(crc32);
-
- switch (transport->ipaddr.sa.sa_family){
- case AF_INET:
- inet_sk(sk)->daddr = transport->ipaddr.v4.sin_addr.s_addr;
- break;
- case AF_INET6:
- SCTP_V6(inet6_sk(sk)->daddr = transport->ipaddr.v6.sin6_addr;)
- break;
- default:
- /* This is bogus address type, just bail. */
- break;
- }
-
- /* IP layer ECN support
+ * by CRC32-C as described in <draft-ietf-tsvwg-sctpcsum-02.txt>.
+ */
+ crc32 = count_crc((__u8 *)sh, nskb->len);
+
+ /* 3) Put the resultant value into the checksum field in the
+ * common header, and leave the rest of the bits unchanged.
+ */
+ sh->checksum = htonl(crc32);
+
+ switch (transport->ipaddr.sa.sa_family) {
+ case AF_INET:
+ inet_sk(sk)->daddr = transport->ipaddr.v4.sin_addr.s_addr;
+ break;
+
+ case AF_INET6:
+ SCTP_V6(inet6_sk(sk)->daddr = transport->ipaddr.v6.sin6_addr;)
+ break;
+
+ default:
+ /* This is bogus address type, just bail. */
+ break;
+ };
+
+ /* IP layer ECN support
* From RFC 2481
- * "The ECN-Capable Transport (ECT) bit would be set by the
- * data sender to indicate that the end-points of the
+ * "The ECN-Capable Transport (ECT) bit would be set by the
+ * data sender to indicate that the end-points of the
* transport protocol are ECN-capable."
*
* If ECN capable && negotiated && it makes sense for
* this packet to support it (e.g. post ECN negotiation)
- * then lets set the ECT bit
+ * then lets set the ECT bit
*
* FIXME: Need to do something else for IPv6
*/
-
if (packet->ecn_capable) {
INET_ECN_xmit(nskb->sk);
} else {
INET_ECN_dontxmit(nskb->sk);
}
+ /* Set up the IP options. */
+ /* BUG: not implemented
+ * For v4 this all lives somewhere in sk->opt...
+ */
- /* Set up the IP options. */
- /* BUG: not implemented
- * For v4 this all lives somewhere in sk->opt...
- */
-
- /* Dump that on IP! */
- if ( asoc != NULL
- && asoc->peer.last_sent_to != transport ) {
-
+ /* Dump that on IP! */
+ if (asoc && asoc->peer.last_sent_to != transport) {
/* Considering the multiple CPU scenario, this is a
* "correcter" place for last_sent_to. --xguo
*/
asoc->peer.last_sent_to = transport;
- }
+ }
/* Hey, before Linux changes, here's what we have to
* do to force IP routing to recognize the change of
* dest addr. --xguo
*/
- if ( sk->dst_cache != NULL ) { sk->dst_cache->obsolete = 1; }
-
+ if (sk->dst_cache)
+ sk->dst_cache->obsolete = 1;
+
if (packet_has_data) {
- struct timer_list *timer;
- unsigned long timeout;
+ struct timer_list *timer;
+ unsigned long timeout;
transport->last_time_used = jiffies;
/* Restart the AUTOCLOSE timer when sending data. */
- if ((SCTP_STATE_ESTABLISHED == asoc->state) &&
+ if ((SCTP_STATE_ESTABLISHED == asoc->state) &&
(asoc->autoclose)) {
timer = &asoc->timers[SCTP_EVENT_TIMEOUT_AUTOCLOSE];
timeout = asoc->timeouts[SCTP_EVENT_TIMEOUT_AUTOCLOSE];
- if (!mod_timer(timer, jiffies + timeout)) {
+ if (!mod_timer(timer, jiffies + timeout))
sctp_association_hold(asoc);
- }
}
}
- SCTP_DEBUG_PRINTK("***sctp_transmit_packet*** skb length %d\n",
+ SCTP_DEBUG_PRINTK("***sctp_transmit_packet*** skb length %d\n",
nskb->len);
(*transport->af_specific->queue_xmit)(nskb);
out:
- packet->size = SCTP_IP_OVERHEAD;
- return err;
-
-} /* sctp_packet_transmit() */
+ packet->size = SCTP_IP_OVERHEAD;
+ return err;
+}
/********************************************************************
* 2nd Level Abstractions
********************************************************************/
-/*
+/*
* This private function resets the packet to a fresh state.
*/
-static void
-sctp_packet_reset(sctp_packet_t *packet)
+static void sctp_packet_reset(sctp_packet_t *packet)
{
sctp_chunk_t *chunk = NULL;
packet->size = SCTP_IP_OVERHEAD;
- if ( NULL != packet->get_prepend_chunk ) {
+ if (packet->get_prepend_chunk)
chunk = packet->get_prepend_chunk(packet->transport->asoc);
- }
- /* If there a is a prepend chunk stick it on the list before
+ /* If there a is a prepend chunk stick it on the list before
* any other chunks get appended.
*/
- if (chunk) {
+ if (chunk)
sctp_packet_append_chunk(packet, chunk);
- }
-} /* sctp_packet_reset() */
+}
-/* This private function handles the specifics of appending DATA chunks. */
-static inline sctp_xmit_t
-sctp_packet_append_data(sctp_packet_t *packet, sctp_chunk_t *chunk)
+/* This private function handles the specifics of appending DATA chunks. */
+static sctp_xmit_t sctp_packet_append_data(sctp_packet_t *packet, sctp_chunk_t *chunk)
{
- sctp_xmit_t retval = SCTP_XMIT_OK;
+ sctp_xmit_t retval = SCTP_XMIT_OK;
size_t datasize, rwnd, inflight;
sctp_transport_t *transport = packet->transport;
- uint32_t max_burst_bytes;
-
- /* RFC 2960 6.1 Transmission of DATA Chunks
- *
- * A) At any given time, the data sender MUST NOT transmit new data to
- * any destination transport address if its peer's rwnd indicates
- * that the peer has no buffer space (i.e. rwnd is 0, see Section
- * 6.2.1). However, regardless of the value of rwnd (including if it
- * is 0), the data sender can always have one DATA chunk in flight to
- * the receiver if allowed by cwnd (see rule B below). This rule
- * allows the sender to probe for a change in rwnd that the sender
- * missed due to the SACK having been lost in transit from the data
- * receiver to the data sender.
- */
+ __u32 max_burst_bytes;
+ /* RFC 2960 6.1 Transmission of DATA Chunks
+ *
+ * A) At any given time, the data sender MUST NOT transmit new data to
+ * any destination transport address if its peer's rwnd indicates
+ * that the peer has no buffer space (i.e. rwnd is 0, see Section
+ * 6.2.1). However, regardless of the value of rwnd (including if it
+ * is 0), the data sender can always have one DATA chunk in flight to
+ * the receiver if allowed by cwnd (see rule B below). This rule
+ * allows the sender to probe for a change in rwnd that the sender
+ * missed due to the SACK having been lost in transit from the data
+ * receiver to the data sender.
+ */
rwnd = transport->asoc->peer.rwnd;
inflight = transport->asoc->outqueue.outstanding_bytes;
datasize = sctp_data_size(chunk);
-
+
if (datasize > rwnd) {
- if (inflight > 0){
- /* We have (at least) one data chunk in flight,
- * so we can't fall back to rule 6.1 B).
- */
+ if (inflight > 0) {
+ /* We have (at least) one data chunk in flight,
+ * so we can't fall back to rule 6.1 B).
+ */
retval = SCTP_XMIT_RWND_FULL;
goto finish;
}
}
-
- /* sctpimpguide-05 2.14.2 D) When the time comes for the sender to
- * transmit new DATA chunks, the protocol parameter Max.Burst MUST
- * first be applied to limit how many new DATA chunks may be sent.
+ /* sctpimpguide-05 2.14.2 D) When the time comes for the sender to
+ * transmit new DATA chunks, the protocol parameter Max.Burst MUST
+ * first be applied to limit how many new DATA chunks may be sent.
* The limit is applied by adjusting cwnd as follows:
* if((flightsize + Max.Burst*MTU) < cwnd)
* cwnd = flightsize + Max.Burst*MTU
- */
+ */
max_burst_bytes = transport->asoc->max_burst * transport->asoc->pmtu;
if ((transport->flight_size + max_burst_bytes) < transport->cwnd) {
transport->cwnd = transport->flight_size + max_burst_bytes;
SCTP_DEBUG_PRINTK("%s: cwnd limited by max_burst: "
- "transport: %p, cwnd: %d, "
+ "transport: %p, cwnd: %d, "
"ssthresh: %d, flight_size: %d, "
"pba: %d\n",
- __FUNCTION__, transport,
- transport->cwnd,
- transport->ssthresh,
- transport->flight_size,
+ __FUNCTION__, transport,
+ transport->cwnd,
+ transport->ssthresh,
+ transport->flight_size,
transport->partial_bytes_acked);
- }
-
- /* RFC 2960 6.1 Transmission of DATA Chunks
- *
- * B) At any given time, the sender MUST NOT transmit new data
- * to a given transport address if it has cwnd or more bytes
- * of data outstanding to that transport address.
- */
+ }
+
+ /* RFC 2960 6.1 Transmission of DATA Chunks
+ *
+ * B) At any given time, the sender MUST NOT transmit new data
+ * to a given transport address if it has cwnd or more bytes
+ * of data outstanding to that transport address.
+ */
if (transport->flight_size >= transport->cwnd) {
retval = SCTP_XMIT_RWND_FULL;
goto finish;
}
-
+
/* Keep track of how many bytes are in flight over this transport. */
transport->flight_size += datasize;
transport->asoc->peer.rwnd = rwnd;
finish:
- return(retval);
-
-} /* sctp_packet_append_data() */
+ return retval;
+}
*/
static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_outqueue.c,v 1.35 2002/08/05 02:58:05 jgrimm Exp $";
-#include <linux/config.h>
#include <linux/types.h>
#include <linux/list.h> /* For struct list_head */
#include <linux/socket.h>
#include <net/sctp/sctp.h>
/* Declare internal functions here. */
-static int sctp_acked(sctp_sackhdr_t *sack, uint32_t tsn);
+static int sctp_acked(sctp_sackhdr_t *sack, __u32 tsn);
static void sctp_check_transmitted(sctp_outqueue_t *q,
struct list_head *transmitted_queue,
sctp_transport_t *transport,
sctp_sackhdr_t *sack);
-
/* Generate a new outqueue. */
-sctp_outqueue_t *
-sctp_outqueue_new(sctp_association_t *asoc)
+sctp_outqueue_t *sctp_outqueue_new(sctp_association_t *asoc)
{
- sctp_outqueue_t *q;
-
- q = t_new(sctp_outqueue_t, GFP_KERNEL);
- if (q) {
- sctp_outqueue_init(asoc, q);
- q->malloced = 1;
- }
- return q;
-
-} /* sctp_outqueue_new() */
+ sctp_outqueue_t *q;
+ q = t_new(sctp_outqueue_t, GFP_KERNEL);
+ if (q) {
+ sctp_outqueue_init(asoc, q);
+ q->malloced = 1;
+ }
+ return q;
+}
/* Initialize an existing SCTP_outqueue. This does the boring stuff.
* You still need to define handlers if you really want to DO
* something with this structure...
*/
-void
-sctp_outqueue_init(sctp_association_t *asoc, sctp_outqueue_t *q)
+void sctp_outqueue_init(sctp_association_t *asoc, sctp_outqueue_t *q)
{
- q->asoc = asoc;
- skb_queue_head_init(&q->out);
- skb_queue_head_init(&q->control);
- INIT_LIST_HEAD(&q->retransmit);
- INIT_LIST_HEAD(&q->sacked);
-
- q->init_output = NULL;
- q->config_output = NULL;
- q->append_output = NULL;
- q->build_output = NULL;
- q->force_output = NULL;
-
- q->outstanding_bytes = 0;
- q->empty = 1;
-
- q->malloced = 0;
-
-} /* sctp_outqueue_init() */
-
-
-/* Free the outqueue structure and any related pending chunks.
- * FIXME: Add SEND_FAILED support.
- */
-void
-sctp_outqueue_teardown(sctp_outqueue_t *q)
+ q->asoc = asoc;
+ skb_queue_head_init(&q->out);
+ skb_queue_head_init(&q->control);
+ INIT_LIST_HEAD(&q->retransmit);
+ INIT_LIST_HEAD(&q->sacked);
+
+ q->init_output = NULL;
+ q->config_output = NULL;
+ q->append_output = NULL;
+ q->build_output = NULL;
+ q->force_output = NULL;
+
+ q->outstanding_bytes = 0;
+ q->empty = 1;
+
+ q->malloced = 0;
+}
+
+/* Free the outqueue structure and any related pending chunks.
+ * FIXME: Add SEND_FAILED support.
+ */
+void sctp_outqueue_teardown(sctp_outqueue_t *q)
{
sctp_transport_t *transport;
- list_t *lchunk, *pos;
- sctp_chunk_t *chunk;
+ list_t *lchunk, *pos;
+ sctp_chunk_t *chunk;
/* Throw away unacknowledged chunks. */
- list_for_each(pos, &q->asoc->peer.transport_addr_list) {
+ list_for_each(pos, &q->asoc->peer.transport_addr_list) {
transport = list_entry(pos, sctp_transport_t, transports);
while ((lchunk = sctp_list_dequeue(&transport->transmitted))) {
- chunk = list_entry(lchunk, sctp_chunk_t,
+ chunk = list_entry(lchunk, sctp_chunk_t,
transmitted_list);
- sctp_free_chunk(chunk);
- }
- } /* For all transports. */
+ sctp_free_chunk(chunk);
+ }
+ }
/* Throw away any leftover chunks. */
- while((chunk = (sctp_chunk_t *)skb_dequeue(&q->out))) {
+ while ((chunk = (sctp_chunk_t *) skb_dequeue(&q->out)))
sctp_free_chunk(chunk);
- }
-
-} /* sctp_outqueue_teardown() */
+}
-/* Free the outqueue structure and any related pending chunks. */
-void
-sctp_outqueue_free(sctp_outqueue_t *q)
+/* Free the outqueue structure and any related pending chunks. */
+void sctp_outqueue_free(sctp_outqueue_t *q)
{
/* Throw away leftover chunks. */
sctp_outqueue_teardown(q);
- /* If we were kmalloc()'d, free the memory. */
- if (q->malloced) {
- kfree(q);
- }
-
-} /* sctp_outqueue_free() */
+ /* If we were kmalloc()'d, free the memory. */
+ if (q->malloced)
+ kfree(q);
+}
/* Transmit any pending partial chunks. */
-void
-sctp_force_outqueue(sctp_outqueue_t *q)
+void sctp_force_outqueue(sctp_outqueue_t *q)
{
/* Do we really need this? */
- /* BUG */
-} /* sctp_force_outqueue() */
+ /* BUG */
+}
-/* Put a new chunk in an SCTP_outqueue. */
-int
-sctp_push_outqueue(sctp_outqueue_t *q, sctp_chunk_t *chunk)
+/* Put a new chunk in an SCTP_outqueue. */
+int sctp_push_outqueue(sctp_outqueue_t *q, sctp_chunk_t *chunk)
{
- int error = 0;
+ int error = 0;
SCTP_DEBUG_PRINTK("sctp_push_outqueue(%p, %p[%s])\n",
q, chunk, chunk && chunk->chunk_hdr ?
sctp_cname(SCTP_ST_CHUNK(chunk->chunk_hdr->type))
- : "Illegal Chunk");
-
- /* If it is data, queue it up, otherwise, send it
- * immediately.
- */
- if (SCTP_CID_DATA == chunk->chunk_hdr->type) {
- /* Is it OK to queue data chunks? */
- /* From 9. Termination of Association
- *
- * When either endpoint performs a shutdown, the
- * association on each peer will stop accepting new
- * data from its user and only deliver data in queue
- * at the time of sending or receiving the SHUTDOWN
- * chunk.
- */
- switch(q->asoc->state) {
- case SCTP_STATE_EMPTY:
- case SCTP_STATE_CLOSED:
- case SCTP_STATE_SHUTDOWN_PENDING:
- case SCTP_STATE_SHUTDOWN_SENT:
- case SCTP_STATE_SHUTDOWN_RECEIVED:
- case SCTP_STATE_SHUTDOWN_ACK_SENT:
- /* Cannot send after transport endpoint shutdown */
- error = -ESHUTDOWN;
- break;
- default:
+ : "Illegal Chunk");
+
+ /* If it is data, queue it up, otherwise, send it
+ * immediately.
+ */
+ if (SCTP_CID_DATA == chunk->chunk_hdr->type) {
+ /* Is it OK to queue data chunks? */
+ /* From 9. Termination of Association
+ *
+ * When either endpoint performs a shutdown, the
+ * association on each peer will stop accepting new
+ * data from its user and only deliver data in queue
+ * at the time of sending or receiving the SHUTDOWN
+ * chunk.
+ */
+ switch (q->asoc->state) {
+ case SCTP_STATE_EMPTY:
+ case SCTP_STATE_CLOSED:
+ case SCTP_STATE_SHUTDOWN_PENDING:
+ case SCTP_STATE_SHUTDOWN_SENT:
+ case SCTP_STATE_SHUTDOWN_RECEIVED:
+ case SCTP_STATE_SHUTDOWN_ACK_SENT:
+ /* Cannot send after transport endpoint shutdown */
+ error = -ESHUTDOWN;
+ break;
+
+ default:
SCTP_DEBUG_PRINTK("outqueueing (%p, %p[%s])\n",
- q, chunk,
- chunk && chunk->chunk_hdr ?
- sctp_cname(SCTP_ST_CHUNK(
- chunk->chunk_hdr->type))
- : "Illegal Chunk");
+ q, chunk,
+ chunk && chunk->chunk_hdr ?
+ sctp_cname(SCTP_ST_CHUNK(chunk->chunk_hdr->type))
+ : "Illegal Chunk");
- skb_queue_tail(&q->out,
- (struct sk_buff *)chunk);
- q->empty = 0;
- break;
- } /* switch() (case It is OK to queue data chunks) */
- } else {
- skb_queue_tail(&q->control, (struct sk_buff *)chunk);
- }
- if (error < 0) { return error; }
+ skb_queue_tail(&q->out, (struct sk_buff *) chunk);
+ q->empty = 0;
+ break;
+ };
+ } else {
+ skb_queue_tail(&q->control, (struct sk_buff *) chunk);
+ }
+ if (error < 0)
+ return error;
error = sctp_flush_outqueue(q, 0);
- return error;
-
-} /* sctp_push_outqueue() */
+ return error;
+}
-/* Mark all the eligible packets on a transport for retransmission and force
+/* Mark all the eligible packets on a transport for retransmission and force
* one packet out.
*/
-void
-sctp_retransmit(sctp_outqueue_t *q, sctp_transport_t *transport,
- uint8_t fast_retransmit)
+void sctp_retransmit(sctp_outqueue_t *q, sctp_transport_t *transport,
+ __u8 fast_retransmit)
{
struct list_head *lchunk;
sctp_chunk_t *chunk;
INIT_LIST_HEAD(&tlist);
- while( !list_empty(&transport->transmitted)) {
+ while (!list_empty(&transport->transmitted)) {
lchunk = sctp_list_dequeue(&transport->transmitted);
-
chunk = list_entry(lchunk, sctp_chunk_t, transmitted_list);
- /* If we are doing retransmission due to a fast retransmit,
- * only the chunk's that are marked for fast retransmit
- * should be added to the retransmit queue. If we are doing
- * retransmission due to a timeout, only the chunks that are
+ /* If we are doing retransmission due to a fast retransmit,
+ * only the chunk's that are marked for fast retransmit
+ * should be added to the retransmit queue. If we are doing
+ * retransmission due to a timeout, only the chunks that are
* not yet acked should be added to the retransmit queue.
*/
-
- if ((fast_retransmit && !chunk->fast_retransmit)
- || (!fast_retransmit && chunk->tsn_gap_acked)) {
+ if ((fast_retransmit && !chunk->fast_retransmit) ||
+ (!fast_retransmit && chunk->tsn_gap_acked)) {
list_add_tail(lchunk, &tlist);
} else {
/* RFC 2960 6.2.1 Processing a Received SACK
- *
- * C) Any time a DATA chunk is marked for
- * retransmission (via either T3-rtx timer expiration
- * (Section 6.3.3) or via fast retransmit
- * (Section 7.2.4)), add the data size of those
- * chunks to the rwnd.
- */
+ *
+ * C) Any time a DATA chunk is marked for
+ * retransmission (via either T3-rtx timer expiration
+ * (Section 6.3.3) or via fast retransmit
+ * (Section 7.2.4)), add the data size of those
+ * chunks to the rwnd.
+ */
q->asoc->peer.rwnd += sctp_data_size(chunk);
- q->outstanding_bytes -= sctp_data_size(chunk);
+ q->outstanding_bytes -= sctp_data_size(chunk);
transport->flight_size -= sctp_data_size(chunk);
- /* sctpimpguide-05 Section 2.8.2
- * M5) If a T3-rtx timer expires, the
- * 'TSN.Missing.Report' of all affected TSNs is set
+ /* sctpimpguide-05 Section 2.8.2
+ * M5) If a T3-rtx timer expires, the
+ * 'TSN.Missing.Report' of all affected TSNs is set
* to 0.
*/
chunk->tsn_missing_report = 0;
/* If a chunk that is being used for RTT measurement
* has to be retransmitted, we cannot use this chunk
- * anymore for RTT measurements. Reset rto_pending so
- * that a new RTT measurement is started when a new
+ * anymore for RTT measurements. Reset rto_pending so
+ * that a new RTT measurement is started when a new
* data chunk is sent.
*/
if (chunk->rtt_in_progress) {
chunk->rtt_in_progress = 0;
transport->rto_pending = 0;
- }
-
+ }
list_add_tail(lchunk, &q->retransmit);
}
+ }
- } /* while (we have stuff left on transport->transmitted) */
-
- /* Reconstruct the transmitted queue with chunks that are not
- * eligible for retransmission.
+ /* Reconstruct the transmitted queue with chunks that are not
+ * eligible for retransmission.
*/
- while( NULL != (lchunk = sctp_list_dequeue(&tlist))) {
+ while (NULL != (lchunk = sctp_list_dequeue(&tlist)))
list_add_tail(lchunk, &transport->transmitted);
- }
-
+
SCTP_DEBUG_PRINTK(__FUNCTION__": transport: %p, fast_retransmit: %d, "
"cwnd: %d, ssthresh: %d, flight_size: %d, "
- "pba: %d\n",transport, fast_retransmit,
- transport->cwnd, transport->ssthresh,
- transport->flight_size,
+ "pba: %d\n",transport, fast_retransmit,
+ transport->cwnd, transport->ssthresh,
+ transport->flight_size,
transport->partial_bytes_acked);
- error = sctp_flush_outqueue(q, /* rtx_timeout */ 1);
-
- if (error) {
- q->asoc->base.sk->err = -error;
- }
-
-} /* sctp_retransmit() */
+ error = sctp_flush_outqueue(q, /* rtx_timeout */ 1);
+ if (error)
+ q->asoc->base.sk->err = -error;
+}
-/*
+/*
* Transmit DATA chunks on the retransmit queue. Upon return from
* sctp_flush_retran_queue() the packet 'pkt' may contain chunks which
* need to be transmitted by the caller.
*
* The return value is a normal kernel error return value.
*/
-static int
-sctp_flush_retran_queue(sctp_outqueue_t *q, sctp_packet_t *pkt,
- int rtx_timeout, int *start_timer)
+static int sctp_flush_retran_queue(sctp_outqueue_t *q, sctp_packet_t *pkt,
+ int rtx_timeout, int *start_timer)
{
-
struct list_head *lqueue;
struct list_head *lchunk;
sctp_transport_t *transport = pkt->transport;
asoc = q->asoc;
lqueue = &q->retransmit;
-
+
/* RFC 2960 6.3.3 Handle T3-rtx Expiration
*
* E3) Determine how many of the earliest (i.e., lowest TSN)
* because a timeout just happened, we should send only ONE
* packet of retransmitted data.]
*/
-
-
lchunk = sctp_list_dequeue(lqueue);
- while (NULL != lchunk) {
-
+ while (lchunk) {
chunk = list_entry(lchunk, sctp_chunk_t, transmitted_list);
-
#if 0
/* If a chunk has been tried for more than SCTP_DEF_MAX_SEND
* times, discard it, and check the empty flag of the outqueue.
*
* --xguo
*/
-
- if ( chunk->snd_count > SCTP_DEF_MAX_SEND ) {
+ if (chunk->snd_count > SCTP_DEF_MAX_SEND) {
sctp_free_chunk(chunk);
continue;
}
#endif
/* Attempt to append this chunk to the packet. */
status = (*q->append_output)(pkt, chunk);
-
+
switch (status) {
case SCTP_XMIT_PMTU_FULL:
-
/* Send this packet. */
- if ( (error = (*q->force_output)(pkt)) == 0 ) {
+ if ((error = (*q->force_output)(pkt)) == 0)
*start_timer = 1;
- }
/* If we are retransmitting, we should only
* send a single packet.
list_add(lchunk, lqueue);
lchunk = NULL;
}
-
+
/* Bundle lchunk in the next round. */
break;
- case SCTP_XMIT_RWND_FULL:
+ case SCTP_XMIT_RWND_FULL:
/* Send this packet. */
- if ( (error = (*q->force_output)(pkt)) == 0 ) {
+ if ((error = (*q->force_output)(pkt)) == 0)
*start_timer = 1;
- }
/* Stop sending DATA as there is no more room
* at the reciever.
list_add(lchunk, lqueue);
lchunk = NULL;
break;
+
default:
/* The append was successful, so add this chunk to
* the transmitted list.
*start_timer = 1;
q->empty = 0;
- /* Retrieve a new chunk to bundle. */
+ /* Retrieve a new chunk to bundle. */
lchunk = sctp_list_dequeue(lqueue);
break;
- }
-
-
- } /* while (more retransmit chunks) */
-
- return(error);
+ };
+ }
-} /* sctp_flush_retran_queue() */
+ return error;
+}
-/* This routine either transmits the fragment or puts it on the output
+/* This routine either transmits the fragment or puts it on the output
* queue. 'pos' points to the next chunk in the output queue after the
* chunk that is currently in the process of fragmentation.
*/
-void
-sctp_xmit_frag(sctp_outqueue_t *q, struct sk_buff *pos, sctp_packet_t *packet,
- sctp_chunk_t *frag, uint32_t tsn)
+void sctp_xmit_frag(sctp_outqueue_t *q, struct sk_buff *pos,
+ sctp_packet_t *packet,
+ sctp_chunk_t *frag, __u32 tsn)
{
- sctp_transport_t *transport = packet->transport;
+ sctp_transport_t *transport = packet->transport;
struct sk_buff_head *queue = &q->out;
sctp_xmit_t status;
int error;
frag->subh.data_hdr->tsn = htonl(tsn);
frag->has_tsn = 1;
- /* An inner fragment may be smaller than the earlier one and may get
- * in if we call q->build_output. This ensures that all the fragments
- * are sent in order.
+ /* An inner fragment may be smaller than the earlier one and may get
+ * in if we call q->build_output. This ensures that all the fragments
+ * are sent in order.
*/
- if (!skb_queue_empty(queue)) {
+ if (!skb_queue_empty(queue)) {
SCTP_DEBUG_PRINTK("sctp_xmit_frag: q not empty. "
- "adding 0x%x to outqueue\n",
+ "adding 0x%x to outqueue\n",
ntohl(frag->subh.data_hdr->tsn));
if (pos) {
- skb_insert(pos, (struct sk_buff *)frag);
+ skb_insert(pos, (struct sk_buff *) frag);
} else {
- skb_queue_tail(queue, (struct sk_buff *)frag);
+ skb_queue_tail(queue, (struct sk_buff *) frag);
}
return;
}
case SCTP_XMIT_RWND_FULL:
/* RWND is full, so put the chunk in the output queue. */
SCTP_DEBUG_PRINTK("sctp_xmit_frag: rwnd full. "
- "adding 0x%x to outqueue\n",
+ "adding 0x%x to outqueue\n",
ntohl(frag->subh.data_hdr->tsn));
if (pos) {
- skb_insert(pos, (struct sk_buff *)frag);
+ skb_insert(pos, (struct sk_buff *) frag);
} else {
- skb_queue_tail(queue, (struct sk_buff *)frag);
+ skb_queue_tail(queue, (struct sk_buff *) frag);
}
- break;
+ break;
+
case SCTP_XMIT_OK:
error = (*q->force_output)(packet);
if (error < 0) {
- /* Packet could not be transmitted, put the chunk in
- * the output queue
- */
+ /* Packet could not be transmitted, put the chunk in
+ * the output queue
+ */
SCTP_DEBUG_PRINTK("sctp_xmit_frag: force output "
- "failed. adding 0x%x to outqueue\n",
+ "failed. adding 0x%x to outqueue\n",
ntohl(frag->subh.data_hdr->tsn));
if (pos) {
- skb_insert(pos, (struct sk_buff *)frag);
+ skb_insert(pos, (struct sk_buff *) frag);
} else {
- skb_queue_tail(queue, (struct sk_buff *)frag);
+ skb_queue_tail(queue, (struct sk_buff *) frag);
}
} else {
SCTP_DEBUG_PRINTK("sctp_xmit_frag: force output "
- "success. 0x%x sent\n",
+ "success. 0x%x sent\n",
ntohl(frag->subh.data_hdr->tsn));
list_add_tail(&frag->transmitted_list,
- &transport->transmitted);
+ &transport->transmitted);
sctp_transport_reset_timers(transport);
- }
+ }
break;
+
default:
BUG();
- }
-
-} /* sctp_xmit_frag() */
+ };
+}
-/* This routine calls sctp_xmit_frag() for all the fragments of a message.
- * The argument 'frag' point to the first fragment and it holds the list
+/* This routine calls sctp_xmit_frag() for all the fragments of a message.
+ * The argument 'frag' point to the first fragment and it holds the list
* of all the other fragments in the 'frag_list' field.
- */
-void
-sctp_xmit_fragmented_chunks(sctp_outqueue_t *q, sctp_packet_t *packet,
- sctp_chunk_t *frag)
+ */
+void sctp_xmit_fragmented_chunks(sctp_outqueue_t *q, sctp_packet_t *packet,
+ sctp_chunk_t *frag)
{
sctp_association_t *asoc = frag->asoc;
struct list_head *lfrag, *frag_list;
- uint32_t tsn;
+ __u32 tsn;
int nfrags = 1;
struct sk_buff *pos;
nfrags++;
}
- /* Get a TSN block of nfrags TSNs. */
+ /* Get a TSN block of nfrags TSNs. */
tsn = __sctp_association_get_tsn_block(asoc, nfrags);
pos = skb_peek(&q->out);
- /* Transmit the first fragment. */
+ /* Transmit the first fragment. */
sctp_xmit_frag(q, pos, packet, frag, tsn++);
- /* Transmit the rest of fragments. */
+ /* Transmit the rest of fragments. */
frag_list = &frag->frag_list;
- list_for_each(lfrag, frag_list) {
+ list_for_each(lfrag, frag_list) {
frag = list_entry(lfrag, sctp_chunk_t, frag_list);
sctp_xmit_frag(q, pos, packet, frag, tsn++);
}
-
-} /* sctp_xmit_fragmented_chunks() */
-
-/* This routine breaks the given chunk into 'max_frag_data_len' size
- * fragments. It returns the first fragment with the frag_list field holding
- * the remaining fragments.
- */
-sctp_chunk_t *
-sctp_fragment_chunk(sctp_chunk_t *chunk, size_t max_frag_data_len)
+}
+
+/* This routine breaks the given chunk into 'max_frag_data_len' size
+ * fragments. It returns the first fragment with the frag_list field holding
+ * the remaining fragments.
+ */
+sctp_chunk_t *sctp_fragment_chunk(sctp_chunk_t *chunk, size_t max_frag_data_len)
{
sctp_association_t *asoc = chunk->asoc;
void *data_ptr = chunk->subh.data_hdr;
struct sctp_sndrcvinfo *sinfo = &chunk->sinfo;
- uint16_t chunk_data_len = sctp_data_size(chunk);
- uint16_t ssn = ntohs(chunk->subh.data_hdr->ssn);
+ __u16 chunk_data_len = sctp_data_size(chunk);
+ __u16 ssn = ntohs(chunk->subh.data_hdr->ssn);
sctp_chunk_t *first_frag, *frag;
struct list_head *frag_list;
- int nfrags;
+ int nfrags;
/* nfrags = no. of max size fragments + any smaller last fragment. */
- nfrags = ((chunk_data_len / max_frag_data_len) +
- ((chunk_data_len % max_frag_data_len) ? 1 : 0));
+ nfrags = ((chunk_data_len / max_frag_data_len) +
+ ((chunk_data_len % max_frag_data_len) ? 1 : 0));
/* Start of the data in the chunk. */
data_ptr += sizeof(sctp_datahdr_t);
/* Make the first fragment. */
- first_frag = sctp_make_datafrag(asoc, sinfo, max_frag_data_len,
+ first_frag = sctp_make_datafrag(asoc, sinfo, max_frag_data_len,
data_ptr, SCTP_DATA_FIRST_FRAG, ssn);
- if (!first_frag) { goto err; }
+ if (!first_frag)
+ goto err;
/* All the fragments are added to the frag_list of the first chunk. */
frag_list = &first_frag->frag_list;
/* Make the middle fragments. */
while (chunk_data_len > max_frag_data_len) {
+ frag = sctp_make_datafrag(asoc, sinfo, max_frag_data_len,
+ data_ptr, SCTP_DATA_MIDDLE_FRAG, ssn);
+ if (!frag)
+ goto err;
- frag = sctp_make_datafrag(asoc, sinfo, max_frag_data_len,
- data_ptr, SCTP_DATA_MIDDLE_FRAG, ssn);
- if (!frag) { goto err; }
-
- /* Add the middle fragment to the first fragment's frag_list.*/
- list_add_tail(&frag->frag_list, frag_list);
+ /* Add the middle fragment to the first fragment's frag_list. */
+ list_add_tail(&frag->frag_list, frag_list);
chunk_data_len -= max_frag_data_len;
data_ptr += max_frag_data_len;
}
/* Make the last fragment. */
- frag = sctp_make_datafrag(asoc, sinfo, chunk_data_len, data_ptr,
- SCTP_DATA_LAST_FRAG, ssn);
- if (!frag) { goto err; }
+ frag = sctp_make_datafrag(asoc, sinfo, chunk_data_len, data_ptr,
+ SCTP_DATA_LAST_FRAG, ssn);
+ if (!frag)
+ goto err;
/* Add the last fragment to the first fragment's frag_list. */
- list_add_tail(&frag->frag_list, frag_list);
+ list_add_tail(&frag->frag_list, frag_list);
/* Free the original chunk. */
- sctp_free_chunk(chunk);
+ sctp_free_chunk(chunk);
- return (first_frag);
+ return first_frag;
err:
- /* Free any fragments that are created before the failure. */
+ /* Free any fragments that are created before the failure. */
if (first_frag) {
struct list_head *flist, *lfrag;
-
- /* Free all the fragments off the first one. */
+
+ /* Free all the fragments off the first one. */
flist = &first_frag->frag_list;
while (NULL != (lfrag = sctp_list_dequeue(flist))) {
frag = list_entry(lfrag, sctp_chunk_t, frag_list);
/* Free the first fragment. */
sctp_free_chunk(first_frag);
}
-
- return(NULL);
-
-} /* sctp_fragment_chunk() */
+ return NULL;
+}
/*
* sctp_flush_outqueue - Try to flush an outqueue.
- *
+ *
* Description: Send everything in q which we legally can, subject to
* congestion limitations.
- *
+ *
* Note: This function can be called from multiple contexts so appropriate
* locking concerns must be made. Today we use the sock lock to protect
- * this function.
+ * this function.
*/
-int
-sctp_flush_outqueue(sctp_outqueue_t *q, int rtx_timeout)
+int sctp_flush_outqueue(sctp_outqueue_t *q, int rtx_timeout)
{
sctp_packet_t *packet;
sctp_packet_t singleton;
sctp_association_t *asoc = q->asoc;
int ecn_capable = asoc->peer.ecn_capable;
- uint16_t sport = asoc->base.bind_addr.port;
- uint16_t dport = asoc->peer.port;
- uint32_t vtag = asoc->peer.i.init_tag;
+ __u16 sport = asoc->base.bind_addr.port;
+ __u16 dport = asoc->peer.port;
+ __u32 vtag = asoc->peer.i.init_tag;
/* This is the ECNE handler for singleton packets. */
sctp_packet_phandler_t *s_ecne_handler = NULL;
sctp_packet_phandler_t *ecne_handler = NULL;
sctp_chunk_t *chunk;
sctp_xmit_t status;
int error = 0;
- int start_timer = 0;
+ int start_timer = 0;
sctp_ulpevent_t *event;
-
/* These transports have chunks to send. */
struct list_head transport_list;
struct list_head *ltransport;
-
INIT_LIST_HEAD(&transport_list);
packet = NULL;
* within a SCTP packet in increasing order of TSN.
* ...
*/
-
- if ( ecn_capable ) {
+ if (ecn_capable) {
s_ecne_handler = &sctp_get_no_prepend;
ecne_handler = &sctp_get_ecne_prepend;
}
queue = &q->control;
while (NULL != (chunk = (sctp_chunk_t *)skb_dequeue(queue))) {
-
/* Pick the right transport to use. */
new_transport = chunk->transport;
- if (new_transport == NULL ) {
+ if (!new_transport) {
new_transport = asoc->peer.active_path;
} else if (!new_transport->state.active) {
-
/* If the chunk is Heartbeat, send it to
* chunk->transport, even it's inactive.
- */
- if ( chunk->chunk_hdr->type != SCTP_CID_HEARTBEAT) {
+ */
+ if (chunk->chunk_hdr->type != SCTP_CID_HEARTBEAT)
new_transport = asoc->peer.active_path;
- }
}
/* Are we switching transports?
* Take care of transport locks.
*/
- if ( new_transport != transport ) {
-
+ if (new_transport != transport) {
transport = new_transport;
-
- if ( list_empty(&transport->send_ready) ) {
+ if (list_empty(&transport->send_ready)) {
list_add_tail(&transport->send_ready,
&transport_list);
}
ecn_capable, ecne_handler);
}
- switch(chunk->chunk_hdr->type) {
+ switch (chunk->chunk_hdr->type) {
/*
* 6.10 Bundling
* ...
(*q->init_output)(&singleton, transport, sport, dport);
(*q->config_output)(&singleton, vtag, ecn_capable,
s_ecne_handler);
- (void) (*q->build_output)(&singleton, chunk);
+ (void) (*q->build_output)(&singleton, chunk);
error = (*q->force_output)(&singleton);
- if (error < 0) {return(error);}
+ if (error < 0)
+ return(error);
break;
+
case SCTP_CID_ABORT:
case SCTP_CID_SACK:
case SCTP_CID_HEARTBEAT:
case SCTP_CID_COOKIE_ECHO:
case SCTP_CID_COOKIE_ACK:
case SCTP_CID_ECN_ECNE:
- case SCTP_CID_ECN_CWR:
- (void) (*q->build_output)(packet, chunk);
+ case SCTP_CID_ECN_CWR:
+ (void) (*q->build_output)(packet, chunk);
break;
+
case SCTP_CID_ASCONF:
case SCTP_CID_ASCONF_ACK:
- (void) (*q->build_output)(packet, chunk);
+ (void) (*q->build_output)(packet, chunk);
break;
+
default:
/* We built a chunk with an illegal type! */
BUG();
- } /* switch(type) */
- } /* while (more chunks in control queue) */
-
- /* Is it OK to send data chunks? */
- switch (asoc->state) {
+ };
+ }
- case SCTP_STATE_COOKIE_ECHOED:
+ /* Is it OK to send data chunks? */
+ switch (asoc->state) {
+ case SCTP_STATE_COOKIE_ECHOED:
/* Only allow bundling, if this packet has a COOKIE-ECHO
* chunk.
*/
- if (packet && !packet->has_cookie_echo) {
+ if (packet && !packet->has_cookie_echo)
break;
- }
- case SCTP_STATE_ESTABLISHED:
- case SCTP_STATE_SHUTDOWN_PENDING:
- case SCTP_STATE_SHUTDOWN_RECEIVED:
+ /* fallthru */
+ case SCTP_STATE_ESTABLISHED:
+ case SCTP_STATE_SHUTDOWN_PENDING:
+ case SCTP_STATE_SHUTDOWN_RECEIVED:
/*
* RFC 2960 6.1 Transmission of DATA Chunks
*
* are marked for retransmission (limited by the
* current cwnd).
*/
-
- if ( !list_empty(&q->retransmit) ) {
- if ( transport == asoc->peer.retran_path ) {
+ if (!list_empty(&q->retransmit)) {
+ if (transport == asoc->peer.retran_path)
goto retran;
- }
-
+
/* Switch transports & prepare the packet. */
transport = asoc->peer.retran_path;
- if ( list_empty(&transport->send_ready) ) {
+ if (list_empty(&transport->send_ready)) {
list_add_tail(&transport->send_ready,
&transport_list);
}
-
+
packet = &transport->packet;
(*q->config_output)(packet, vtag,
ecn_capable, ecne_handler);
rtx_timeout,
&start_timer);
- if (start_timer) {
+ if (start_timer)
sctp_transport_reset_timers(transport);
- }
-
- } /* if (we have chunks for retransmission) */
+ }
/* Finally, transmit new packets. */
- start_timer = 0;
- queue = &q->out;
- while (NULL != (chunk = (sctp_chunk_t *)
- skb_dequeue(queue))) {
-
+ start_timer = 0;
+ queue = &q->out;
+ while (NULL != (chunk = (sctp_chunk_t *) skb_dequeue(queue))) {
/* RFC 2960 6.5 Every DATA chunk MUST carry a valid
- * stream identifier.
+ * stream identifier.
*/
- if (chunk->sinfo.sinfo_stream >=
- asoc->c.sinit_num_ostreams) {
-
+ if (chunk->sinfo.sinfo_stream >=
+ asoc->c.sinit_num_ostreams) {
/* Generate a SEND FAILED event. */
event = sctp_ulpevent_make_send_failed(asoc,
- chunk, SCTP_DATA_UNSENT,
- SCTP_ERROR_INV_STRM,
+ chunk, SCTP_DATA_UNSENT,
+ SCTP_ERROR_INV_STRM,
GFP_ATOMIC);
if (event) {
- sctp_ulpqueue_tail_event(&asoc->ulpq,
- event);
+ sctp_ulpqueue_tail_event(&asoc->ulpq,
+ event);
}
- /* Free the chunk. This chunk is not on any
- * list yet, just free it. */
+ /* Free the chunk. This chunk is not on any
+ * list yet, just free it.
+ */
sctp_free_chunk(chunk);
-
continue;
-
}
+
/* If there is a specified transport, use it.
* Otherwise, we want to use the active path.
*/
new_transport = chunk->transport;
-
- if ( new_transport == NULL
- || !new_transport->state.active ) {
+ if (new_transport == NULL ||
+ !new_transport->state.active)
new_transport = asoc->peer.active_path;
- }
/* Change packets if necessary. */
- if ( new_transport != transport ) {
-
+ if (new_transport != transport) {
transport = new_transport;
/* Schedule to have this transport's
* packet flushed.
*/
- if ( list_empty(&transport->send_ready) ) {
+ if (list_empty(&transport->send_ready)) {
list_add_tail(&transport->send_ready,
&transport_list);
}
(*q->config_output)(packet, vtag,
ecn_capable, ecne_handler);
}
-
+
SCTP_DEBUG_PRINTK("sctp_transmit_packet(%p, %p[%s]), ",
- q, chunk,
- chunk && chunk->chunk_hdr ?
+ q, chunk,
+ chunk && chunk->chunk_hdr ?
sctp_cname(SCTP_ST_CHUNK(
chunk->chunk_hdr->type))
: "Illegal Chunk");
SCTP_DEBUG_PRINTK("TX TSN 0x%x skb->head "
"%p skb->users %d.\n",
ntohl(chunk->subh.data_hdr->tsn),
- chunk->skb?chunk->skb->head:0,
- chunk->skb?
- (int)atomic_read(&chunk->skb->users)
- :-1);
+ chunk->skb ?chunk->skb->head : 0,
+ chunk->skb ?
+ atomic_read(&chunk->skb->users) :
+ -1);
/* Add the chunk to the packet. */
status = (*q->build_output)(packet, chunk);
skb_queue_head(queue, (struct sk_buff *)chunk);
goto sctp_flush_out;
break;
- case SCTP_XMIT_MUST_FRAG:
- {
+
+ case SCTP_XMIT_MUST_FRAG: {
sctp_chunk_t *frag;
- frag = sctp_fragment_chunk(chunk,
+ frag = sctp_fragment_chunk(chunk,
packet->transport->asoc->frag_point);
if (!frag) {
/* We could not fragment due to out of
- * memory condition. Free the original
- * chunk and return ENOMEM.
+ * memory condition. Free the original
+ * chunk and return ENOMEM.
*/
sctp_free_chunk(chunk);
error = -ENOMEM;
- return (error);
+ return error;
}
sctp_xmit_fragmented_chunks(q, packet, frag);
goto sctp_flush_out;
break;
}
+
case SCTP_XMIT_OK:
break;
+
default:
BUG();
- }
-
+ };
+
/* BUG: We assume that the (*q->force_output())
- * call below will succeed all the time and add the
- * chunk to the transmitted list and restart the
- * timers.
- * It is possible that the call can fail under OOM
+ * call below will succeed all the time and add the
+ * chunk to the transmitted list and restart the
+ * timers.
+ * It is possible that the call can fail under OOM
* conditions.
*
* Is this really a problem? Won't this behave
sctp_transport_reset_timers(transport);
q->empty = 0;
- }
- break;
- default:
- /* Do nothing. */
- break;
- } /* switch ()(case It is OK to send data chunks:) */
-
- sctp_flush_out:
+ }
+ break;
+
+ default:
+ /* Do nothing. */
+ break;
+ };
+
+sctp_flush_out:
/* Before returning, examine all the transports touched in
* this call. Right now, we bluntly force clear all the
* transports. Things might change after we implement Nagle.
* But such an examination is still required.
- *
+ *
* --xguo
*/
- while ( (ltransport = sctp_list_dequeue(&transport_list)) != NULL ) {
+ while ((ltransport = sctp_list_dequeue(&transport_list)) != NULL ) {
sctp_transport_t *t = list_entry(ltransport,
sctp_transport_t, send_ready);
- if ( t != transport ) {
+ if (t != transport)
transport = t;
- }
packet = &transport->packet;
- if ( packet->size != SCTP_IP_OVERHEAD ) {
+ if (packet->size != SCTP_IP_OVERHEAD)
error = (*q->force_output)(packet);
- }
- }
-
- return(error);
+ }
-} /* sctp_flush_outqueue() */
+ return error;
+}
-/* Set the various output handling callbacks. */
-int
-sctp_outqueue_set_output_handlers(sctp_outqueue_t *q,
- sctp_outqueue_ohandler_init_t init,
- sctp_outqueue_ohandler_config_t config,
- sctp_outqueue_ohandler_t append,
- sctp_outqueue_ohandler_t build,
- sctp_outqueue_ohandler_force_t force)
+/* Set the various output handling callbacks. */
+int sctp_outqueue_set_output_handlers(sctp_outqueue_t *q,
+ sctp_outqueue_ohandler_init_t init,
+ sctp_outqueue_ohandler_config_t config,
+ sctp_outqueue_ohandler_t append,
+ sctp_outqueue_ohandler_t build,
+ sctp_outqueue_ohandler_force_t force)
{
- q->init_output = init;
- q->config_output = config;
- q->append_output = append;
- q->build_output = build;
- q->force_output = force;
+ q->init_output = init;
+ q->config_output = config;
+ q->append_output = append;
+ q->build_output = build;
+ q->force_output = force;
return 0;
-
-} /* sctp_outqueue_set_out_handler() */
-
+}
/* Update unack_data based on the incoming SACK chunk */
-static void
-sctp_sack_update_unack_data(sctp_association_t *assoc, sctp_sackhdr_t *sack)
+static void sctp_sack_update_unack_data(sctp_association_t *assoc,
+ sctp_sackhdr_t *sack)
{
sctp_sack_variable_t *frags;
- uint16_t unack_data;
+ __u16 unack_data;
int i;
unack_data = assoc->next_tsn - assoc->ctsn_ack_point - 1;
frags = sack->variable;
for (i = 0; i < ntohs(sack->num_gap_ack_blocks); i++) {
- unack_data -= ((ntohs(frags[i].gab.end) -
+ unack_data -= ((ntohs(frags[i].gab.end) -
ntohs(frags[i].gab.start) + 1));
}
* Process the sack against the outqueue. Mostly, this just frees
* things off the transmitted queue.
*/
-int
-sctp_sack_outqueue(sctp_outqueue_t *q, sctp_sackhdr_t *sack)
+int sctp_sack_outqueue(sctp_outqueue_t *q, sctp_sackhdr_t *sack)
{
- sctp_chunk_t *tchunk;
- list_t *lchunk, *transport_list, *pos;
- uint32_t tsn;
- uint32_t sack_ctsn;
- uint32_t ctsn;
+ sctp_chunk_t *tchunk;
+ list_t *lchunk, *transport_list, *pos;
+ __u32 tsn;
+ __u32 sack_ctsn;
+ __u32 ctsn;
sctp_transport_t *transport;
int outstanding;
- uint32_t sack_a_rwnd;
+ __u32 sack_a_rwnd;
/* Grab the association's destination address list. */
transport_list = &q->asoc->peer.transport_addr_list;
-
+
/* Run through the retransmit queue. Credit bytes received
- * and free those chunks that we can.
+ * and free those chunks that we can.
*/
sctp_check_transmitted(q, &q->retransmit, NULL, sack);
- /* Run through the transmitted queue.
- * Credit bytes received and free those chunks which we can.
- *
- * This is a MASSIVE candidate for optimization.
- */
- list_for_each(pos, transport_list) {
- transport = list_entry(pos, sctp_transport_t, transports);
- sctp_check_transmitted(q, &transport->transmitted,
- transport, sack);
- }
-
- /* Move the Cumulative TSN Ack Point if appropriate. */
- sack_ctsn = ntohl(sack->cum_tsn_ack);
- if (TSN_lt(q->asoc->ctsn_ack_point, sack_ctsn)) {
- q->asoc->ctsn_ack_point = sack_ctsn;
- }
+ /* Run through the transmitted queue.
+ * Credit bytes received and free those chunks which we can.
+ *
+ * This is a MASSIVE candidate for optimization.
+ */
+ list_for_each(pos, transport_list) {
+ transport = list_entry(pos, sctp_transport_t, transports);
+ sctp_check_transmitted(q, &transport->transmitted,
+ transport, sack);
+ }
+
+ /* Move the Cumulative TSN Ack Point if appropriate. */
+ sack_ctsn = ntohl(sack->cum_tsn_ack);
+ if (TSN_lt(q->asoc->ctsn_ack_point, sack_ctsn))
+ q->asoc->ctsn_ack_point = sack_ctsn;
/* Update unack_data field in the assoc. */
sctp_sack_update_unack_data(q->asoc, sack);
- ctsn = q->asoc->ctsn_ack_point;
+ ctsn = q->asoc->ctsn_ack_point;
SCTP_DEBUG_PRINTK(__FUNCTION__ ": sack Cumulative TSN Ack is 0x%x.\n",
sack_ctsn);
SCTP_DEBUG_PRINTK(__FUNCTION__ ": Cumulative TSN Ack of association "
- "%p is 0x%x.\n",q->asoc, ctsn);
-
- /* Throw away stuff rotting on the sack queue. */
- list_for_each(lchunk, &q->sacked) {
- tchunk = list_entry(lchunk, sctp_chunk_t, transmitted_list);
- tsn = ntohl(tchunk->subh.data_hdr->tsn);
- if (TSN_lte(tsn, ctsn)){
- lchunk = lchunk->prev;
- sctp_free_chunk(tchunk);
- }
- }
-
- /* ii) Set rwnd equal to the newly received a_rwnd minus the
- * number of bytes still outstanding after processing the
- * Cumulative TSN Ack and the Gap Ack Blocks.
- */
-
- sack_a_rwnd = ntohl(sack->a_rwnd);
+ "%p is 0x%x.\n",q->asoc, ctsn);
+
+ /* Throw away stuff rotting on the sack queue. */
+ list_for_each(lchunk, &q->sacked) {
+ tchunk = list_entry(lchunk, sctp_chunk_t, transmitted_list);
+ tsn = ntohl(tchunk->subh.data_hdr->tsn);
+ if (TSN_lte(tsn, ctsn)) {
+ lchunk = lchunk->prev;
+ sctp_free_chunk(tchunk);
+ }
+ }
+
+ /* ii) Set rwnd equal to the newly received a_rwnd minus the
+ * number of bytes still outstanding after processing the
+ * Cumulative TSN Ack and the Gap Ack Blocks.
+ */
+
+ sack_a_rwnd = ntohl(sack->a_rwnd);
outstanding = q->outstanding_bytes;
-
- if (outstanding < sack_a_rwnd) {
+
+ if (outstanding < sack_a_rwnd) {
sack_a_rwnd -= outstanding;
} else {
sack_a_rwnd = 0;
q->asoc->peer.rwnd = sack_a_rwnd;
-
- /* See if all chunks are acked.
- * Make sure the empty queue handler will get run later.
- */
-
- q->empty = skb_queue_empty(&q->out) && list_empty(&q->retransmit);
- if (!q->empty) { goto finish; }
+ /* See if all chunks are acked.
+ * Make sure the empty queue handler will get run later.
+ */
+ q->empty = skb_queue_empty(&q->out) && list_empty(&q->retransmit);
+ if (!q->empty)
+ goto finish;
list_for_each(pos, transport_list) {
transport = list_entry(pos, sctp_transport_t, transports);
q->empty = q->empty && list_empty(&transport->transmitted);
- if (!q->empty) { goto finish; }
+ if (!q->empty)
+ goto finish;
}
SCTP_DEBUG_PRINTK("sack queue is empty.\n");
-
- finish:
- return(q->empty);
-
-} /* sctp_sack_outqueue() */
+finish:
+ return q->empty;
+}
/* Is the outqueue empty? */
-int
-sctp_outqueue_is_empty(const sctp_outqueue_t *q)
+int sctp_outqueue_is_empty(const sctp_outqueue_t *q)
{
- return(q->empty);
-
-} /* sctp_outqueue_is_empty() */
+ return q->empty;
+}
/********************************************************************
* 2nd Level Abstractions
/* Go through a transport's transmitted list or the assocication's retransmit
* list and move chunks that are acked by the Cumulative TSN Ack to q->sacked.
* The retransmit list will not have an associated transport. In case of a
- * transmitted list with a transport, the transport's congestion, rto and fast
+ * transmitted list with a transport, the transport's congestion, rto and fast
* retransmit parameters are also updated and if needed a fast retransmit
- * process is started.
+ * process is started.
*
* I added coherent debug information output. --xguo
*
* transmitted_queue, we print a range: SACKED: TSN1-TSN2, TSN3, TSN4-TSN5.
* KEPT TSN6-TSN7, etc.
*/
-static void
-sctp_check_transmitted(sctp_outqueue_t *q,
- struct list_head *transmitted_queue,
- sctp_transport_t *transport,
- sctp_sackhdr_t *sack)
+static void sctp_check_transmitted(sctp_outqueue_t *q,
+ struct list_head *transmitted_queue,
+ sctp_transport_t *transport,
+ sctp_sackhdr_t *sack)
{
- struct list_head *lchunk;
- sctp_chunk_t *tchunk;
- struct list_head tlist;
- uint32_t tsn;
- uint32_t sack_ctsn;
- uint32_t rtt;
- uint32_t highest_new_tsn_in_sack;
- uint8_t restart_timer = 0;
- uint8_t do_fast_retransmit = 0;
+ struct list_head *lchunk;
+ sctp_chunk_t *tchunk;
+ struct list_head tlist;
+ __u32 tsn;
+ __u32 sack_ctsn;
+ __u32 rtt;
+ __u32 highest_new_tsn_in_sack;
+ __u8 restart_timer = 0;
+ __u8 do_fast_retransmit = 0;
int bytes_acked = 0;
/* These state variables are for coherent debug output. --xguo */
#if SCTP_DEBUG
- uint32_t dbg_ack_tsn = 0; /* An ACKed TSN range starts here... */
- uint32_t dbg_last_ack_tsn = 0; /* ...and finishes here. */
- uint32_t dbg_kept_tsn = 0; /* An un-ACKed range starts here... */
- uint32_t dbg_last_kept_tsn = 0; /* ...and finishes here. */
+ __u32 dbg_ack_tsn = 0; /* An ACKed TSN range starts here... */
+ __u32 dbg_last_ack_tsn = 0; /* ...and finishes here. */
+ __u32 dbg_kept_tsn = 0; /* An un-ACKed range starts here... */
+ __u32 dbg_last_kept_tsn = 0; /* ...and finishes here. */
/* 0 : The last TSN was ACKed.
* 1 : The last TSN was NOT ACKed (i.e. KEPT).
sack_ctsn = ntohl(sack->cum_tsn_ack);
highest_new_tsn_in_sack = sack_ctsn;
- INIT_LIST_HEAD(&tlist);
-
- /* The while loop will skip empty transmitted queues. */
- while (NULL != (lchunk = sctp_list_dequeue(transmitted_queue))) {
-
- tchunk = list_entry(lchunk, sctp_chunk_t, transmitted_list);
-
- tsn = ntohl(tchunk->subh.data_hdr->tsn);
+ INIT_LIST_HEAD(&tlist);
- if (sctp_acked(sack, tsn)) {
+ /* The while loop will skip empty transmitted queues. */
+ while (NULL != (lchunk = sctp_list_dequeue(transmitted_queue))) {
+ tchunk = list_entry(lchunk, sctp_chunk_t, transmitted_list);
+ tsn = ntohl(tchunk->subh.data_hdr->tsn);
+ if (sctp_acked(sack, tsn)) {
/* If this queue is the retransmit queue, the
* retransmit timer has already reclaimed
* the outstanding bytes for this chunk, so only
* count bytes associated with a transport.
*/
if (transport) {
-
- /* If this chunk is being used for RTT
- * measurement, calculate the RTT and update
- * the RTO using this value.
- *
- * 6.3.1 C5) Karn's algorithm: RTT measurements
- * MUST NOT be made using packets that were
- * retransmitted (and thus for which it is
- * ambiguous whether the reply was for the first
- * instance of the packet or a later instance).
- */
+ /* If this chunk is being used for RTT
+ * measurement, calculate the RTT and update
+ * the RTO using this value.
+ *
+ * 6.3.1 C5) Karn's algorithm: RTT measurements
+ * MUST NOT be made using packets that were
+ * retransmitted (and thus for which it is
+ * ambiguous whether the reply was for the first
+ * instance of the packet or a later instance).
+ */
if ((!tchunk->tsn_gap_acked) &&
(1 == tchunk->num_times_sent) &&
(tchunk->rtt_in_progress)) {
rtt = jiffies - tchunk->sent_at;
- sctp_transport_update_rto(transport,
+ sctp_transport_update_rto(transport,
rtt);
}
- }
-
+ }
if (TSN_lte(tsn, sack_ctsn)) {
- /* RFC 2960 6.3.2 Retransmission Timer Rules
- *
- * R3) Whenever a SACK is received
- * that acknowledges the DATA chunk
- * with the earliest outstanding TSN
- * for that address, restart T3-rtx
- * timer for that address with its
- * current RTO.
- */
- restart_timer = 1;
-
+ /* RFC 2960 6.3.2 Retransmission Timer Rules
+ *
+ * R3) Whenever a SACK is received
+ * that acknowledges the DATA chunk
+ * with the earliest outstanding TSN
+ * for that address, restart T3-rtx
+ * timer for that address with its
+ * current RTO.
+ */
+ restart_timer = 1;
+
if (!tchunk->tsn_gap_acked) {
tchunk->tsn_gap_acked = 1;
bytes_acked += sctp_data_size(tchunk);
}
- list_add_tail(&tchunk->transmitted_list,
+ list_add_tail(&tchunk->transmitted_list,
&q->sacked);
- } else {
- /* RFC2960 7.2.4, sctpimpguide-05 2.8.2
- * M2) Each time a SACK arrives reporting
- * 'Stray DATA chunk(s)' record the highest TSN
- * reported as newly acknowledged, call this
- * value 'HighestTSNinSack'. A newly
- * acknowledged DATA chunk is one not previously
- * acknowledged in a SACK.
- *
- * When the SCTP sender of data receives a SACK
- * chunk that acknowledges, for the first time,
- * the receipt of a DATA chunk, all the still
- * unacknowledged DATA chunks whose TSN is older
- * than that newly acknowledged DATA chunk, are
- * qualified as 'Stray DATA chunks'.
- */
+ } else {
+ /* RFC2960 7.2.4, sctpimpguide-05 2.8.2
+ * M2) Each time a SACK arrives reporting
+ * 'Stray DATA chunk(s)' record the highest TSN
+ * reported as newly acknowledged, call this
+ * value 'HighestTSNinSack'. A newly
+ * acknowledged DATA chunk is one not previously
+ * acknowledged in a SACK.
+ *
+ * When the SCTP sender of data receives a SACK
+ * chunk that acknowledges, for the first time,
+ * the receipt of a DATA chunk, all the still
+ * unacknowledged DATA chunks whose TSN is older
+ * than that newly acknowledged DATA chunk, are
+ * qualified as 'Stray DATA chunks'.
+ */
if (!tchunk->tsn_gap_acked) {
tchunk->tsn_gap_acked = 1;
bytes_acked += sctp_data_size(tchunk);
- if (TSN_lt(highest_new_tsn_in_sack,
+ if (TSN_lt(highest_new_tsn_in_sack,
tsn)) {
highest_new_tsn_in_sack = tsn;
}
}
-
- list_add_tail(lchunk, &tlist);
+ list_add_tail(lchunk, &tlist);
}
-
+
#if SCTP_DEBUG
- switch ( dbg_prt_state ) {
+ switch (dbg_prt_state) {
case 0: /* last TSN was ACKed */
- if ( dbg_last_ack_tsn + 1 == tsn ) {
+ if (dbg_last_ack_tsn + 1 == tsn) {
/* This TSN belongs to the
* current ACK range.
- */
+ */
break;
}
- if ( dbg_last_ack_tsn != dbg_ack_tsn ) {
+ if (dbg_last_ack_tsn != dbg_ack_tsn) {
/* Display the end of the
* current range.
- */
+ */
SCTP_DEBUG_PRINTK("-%08x",
dbg_last_ack_tsn);
}
/* Start a new range. */
- SCTP_DEBUG_PRINTK(",%08x",tsn);
+ SCTP_DEBUG_PRINTK(",%08x", tsn);
dbg_ack_tsn = tsn;
-
break;
+
case 1: /* The last TSN was NOT ACKed. */
- if ( dbg_last_kept_tsn != dbg_kept_tsn ) {
+ if (dbg_last_kept_tsn != dbg_kept_tsn) {
/* Display the end of current range. */
SCTP_DEBUG_PRINTK("-%08x",
dbg_last_kept_tsn);
SCTP_DEBUG_PRINTK("\n");
/* FALL THROUGH... */
- default:
- /* This is the first-ever TSN we examined. */
- /* Start a new range of ACK-ed TSNs. */
- SCTP_DEBUG_PRINTK("ACKed: %08x",tsn);
+ default:
+ /* This is the first-ever TSN we examined. */
+ /* Start a new range of ACK-ed TSNs. */
+ SCTP_DEBUG_PRINTK("ACKed: %08x", tsn);
dbg_prt_state = 0;
dbg_ack_tsn = tsn;
- }
+ };
+
dbg_last_ack_tsn = tsn;
-#endif /* SCTP_DEBUG */
+#endif /* SCTP_DEBUG */
- } else {
+ } else {
if (tchunk->tsn_gap_acked) {
- SCTP_DEBUG_PRINTK(__FUNCTION__
+ SCTP_DEBUG_PRINTK(__FUNCTION__
": Receiver reneged on data "
"TSN: 0x%x\n", tsn);
tchunk->tsn_gap_acked = 0;
-
- bytes_acked -= sctp_data_size(tchunk);
- /* RFC 2960 6.3.2 Retransmission Timer Rules
- *
- * R4) Whenever a SACK is received missing a TSN
- * that was previously acknowledged via a Gap Ack
- * Block, start T3-rtx for the destination
- * address to which the DATA chunk was originally
- * transmitted if it is not already running.
- */
+ bytes_acked -= sctp_data_size(tchunk);
+
+ /* RFC 2960 6.3.2 Retransmission Timer Rules
+ *
+ * R4) Whenever a SACK is received missing a TSN
+ * that was previously acknowledged via a Gap Ack
+ * Block, start T3-rtx for the destination
+ * address to which the DATA chunk was originally
+ * transmitted if it is not already running.
+ */
restart_timer = 1;
}
#if SCTP_DEBUG
/* See the above comments on ACK-ed TSNs. */
- switch ( dbg_prt_state ) {
+ switch (dbg_prt_state) {
case 1:
- if ( dbg_last_kept_tsn + 1 == tsn ) { break; }
+ if (dbg_last_kept_tsn + 1 == tsn)
+ break;
- if ( dbg_last_kept_tsn != dbg_kept_tsn )
+ if (dbg_last_kept_tsn != dbg_kept_tsn)
SCTP_DEBUG_PRINTK("-%08x",
dbg_last_kept_tsn);
- SCTP_DEBUG_PRINTK(",%08x",tsn);
+ SCTP_DEBUG_PRINTK(",%08x", tsn);
dbg_kept_tsn = tsn;
break;
+
case 0:
- if ( dbg_last_ack_tsn != dbg_ack_tsn )
+ if (dbg_last_ack_tsn != dbg_ack_tsn)
SCTP_DEBUG_PRINTK("-%08x",
dbg_last_ack_tsn);
SCTP_DEBUG_PRINTK("\n");
SCTP_DEBUG_PRINTK("KEPT: %08x",tsn);
dbg_prt_state = 1;
dbg_kept_tsn = tsn;
- }
+ };
+
dbg_last_kept_tsn = tsn;
#endif /* SCTP_DEBUG */
-
- } /* if (TSN has just been ACKd) */
- }; /* while (transmitted queue is not empty) */
+ }
+ }
#if SCTP_DEBUG
- /* Finish off the last range, displaying its ending TSN. */
- switch ( dbg_prt_state ) {
- case 0: if ( dbg_last_ack_tsn != dbg_ack_tsn ) {
- SCTP_DEBUG_PRINTK("-%08x\n",dbg_last_ack_tsn);
- } else { SCTP_DEBUG_PRINTK("\n"); }
+ /* Finish off the last range, displaying its ending TSN. */
+ switch (dbg_prt_state) {
+ case 0:
+ if (dbg_last_ack_tsn != dbg_ack_tsn) {
+ SCTP_DEBUG_PRINTK("-%08x\n", dbg_last_ack_tsn);
+ } else {
+ SCTP_DEBUG_PRINTK("\n");
+ }
break;
- case 1: if ( dbg_last_kept_tsn != dbg_kept_tsn ) {
- SCTP_DEBUG_PRINTK("-%08x\n",dbg_last_kept_tsn);
- } else { SCTP_DEBUG_PRINTK("\n"); }
- }
+ case 1:
+ if (dbg_last_kept_tsn != dbg_kept_tsn) {
+ SCTP_DEBUG_PRINTK("-%08x\n", dbg_last_kept_tsn);
+ } else {
+ SCTP_DEBUG_PRINTK("\n");
+ }
+ };
#endif /* SCTP_DEBUG */
-
if (transport) {
if (bytes_acked) {
- /* 8.2. When an outstanding TSN is acknowledged,
- * the endpoint shall clear the error counter of
- * the destination transport address to which the
+ /* 8.2. When an outstanding TSN is acknowledged,
+ * the endpoint shall clear the error counter of
+ * the destination transport address to which the
* DATA chunk was last sent.
- * The association's overall error counter is
+ * The association's overall error counter is
* also cleared.
- */
+ */
transport->error_count = 0;
transport->asoc->overall_error_count = 0;
- /* Mark the destination transport address as
+ /* Mark the destination transport address as
* active if it is not so marked.
*/
if (!transport->state.active) {
- sctp_assoc_control_transport(
- transport->asoc,
- transport,
- SCTP_TRANSPORT_UP,
- SCTP_RECEIVED_SACK);
+ sctp_assoc_control_transport(transport->asoc,
+ transport,
+ SCTP_TRANSPORT_UP,
+ SCTP_RECEIVED_SACK);
}
sctp_transport_raise_cwnd(transport, sack_ctsn,
transport->flight_size -= bytes_acked;
q->outstanding_bytes -= bytes_acked;
} else {
- /* RFC 2960 6.1, sctpimpguide-06 2.15.2
- * When a sender is doing zero window probing, it
- * should not timeout the association if it continues
- * to receive new packets from the receiver. The
- * reason is that the receiver MAY keep its window
+ /* RFC 2960 6.1, sctpimpguide-06 2.15.2
+ * When a sender is doing zero window probing, it
+ * should not timeout the association if it continues
+ * to receive new packets from the receiver. The
+ * reason is that the receiver MAY keep its window
* closed for an indefinite time.
- * A sender is doing zero window probing when the
+ * A sender is doing zero window probing when the
* receiver's advertised window is zero, and there is
- * only one data chunk in flight to the receiver.
- */
- if ((0 == q->asoc->peer.rwnd)
- && (!list_empty(&tlist))
- && (sack_ctsn+2 == q->asoc->next_tsn)) {
+ * only one data chunk in flight to the receiver.
+ */
+ if ((0 == q->asoc->peer.rwnd) &&
+ (!list_empty(&tlist)) &&
+ (sack_ctsn+2 == q->asoc->next_tsn)) {
SCTP_DEBUG_PRINTK("%s: SACK received for zero "
- "window probe: %u\n",
+ "window probe: %u\n",
__FUNCTION__, sack_ctsn);
q->asoc->overall_error_count = 0;
transport->error_count = 0;
}
}
- /* RFC 2960 6.3.2 Retransmission Timer Rules
- *
- * R2) Whenever all outstanding data sent to an address have
- * been acknowledged, turn off the T3-rtx timer of that
- * address.
- */
- if (!transport->flight_size) {
+ /* RFC 2960 6.3.2 Retransmission Timer Rules
+ *
+ * R2) Whenever all outstanding data sent to an address have
+ * been acknowledged, turn off the T3-rtx timer of that
+ * address.
+ */
+ if (!transport->flight_size) {
if (timer_pending(&transport->T3_rtx_timer) &&
del_timer(&transport->T3_rtx_timer)) {
sctp_transport_put(transport);
}
} else if (restart_timer) {
if (!mod_timer(&transport->T3_rtx_timer,
- jiffies + transport->rto)) {
+ jiffies + transport->rto))
sctp_transport_hold(transport);
- }
}
-
}
- /* Reconstruct the transmitted list with chunks that are not yet
+ /* Reconstruct the transmitted list with chunks that are not yet
* acked by the Cumulative TSN Ack.
*/
- while( NULL != (lchunk = sctp_list_dequeue(&tlist))) {
- tchunk = list_entry(lchunk, sctp_chunk_t, transmitted_list);
- tsn = ntohl(tchunk->subh.data_hdr->tsn);
+ while (NULL != (lchunk = sctp_list_dequeue(&tlist))) {
+ tchunk = list_entry(lchunk, sctp_chunk_t, transmitted_list);
+ tsn = ntohl(tchunk->subh.data_hdr->tsn);
- /* RFC 2960 7.2.4, sctpimpguide-05 2.8.2 M3) Examine all
- * 'Unacknowledged TSN's', if the TSN number of an
+ /* RFC 2960 7.2.4, sctpimpguide-05 2.8.2 M3) Examine all
+ * 'Unacknowledged TSN's', if the TSN number of an
* 'Unacknowledged TSN' is smaller than the 'HighestTSNinSack'
- * value, increment the 'TSN.Missing.Report' count on that
- * chunk if it has NOT been fast retransmitted or marked for
+ * value, increment the 'TSN.Missing.Report' count on that
+ * chunk if it has NOT been fast retransmitted or marked for
* fast retransmit already.
*
- * M4) If any DATA chunk is found to have a
- * 'TSN.Missing.Report'
- * value larger than or equal to 4, mark that chunk for
- * retransmission and start the fast retransmit procedure.
+ * M4) If any DATA chunk is found to have a
+ * 'TSN.Missing.Report'
+ * value larger than or equal to 4, mark that chunk for
+ * retransmission and start the fast retransmit procedure.
*/
- if ((!tchunk->fast_retransmit)
- && (!tchunk->tsn_gap_acked)
- && (TSN_lt(tsn, highest_new_tsn_in_sack))) {
+ if ((!tchunk->fast_retransmit) &&
+ (!tchunk->tsn_gap_acked) &&
+ (TSN_lt(tsn, highest_new_tsn_in_sack))) {
tchunk->tsn_missing_report++;
SCTP_DEBUG_PRINTK("%s: TSN 0x%x missing counter: %d\n",
- __FUNCTION__, tsn,
+ __FUNCTION__, tsn,
tchunk->tsn_missing_report);
}
if (tchunk->tsn_missing_report >= 4) {
}
if (transport) {
- if (do_fast_retransmit) {
+ if (do_fast_retransmit)
sctp_retransmit(q, transport, do_fast_retransmit);
- }
+
SCTP_DEBUG_PRINTK(__FUNCTION__ ": transport: %p, cwnd: %d, "
- "ssthresh: %d, flight_size: %d, pba: %d\n",
- transport, transport->cwnd,
- transport->ssthresh, transport->flight_size,
+ "ssthresh: %d, flight_size: %d, pba: %d\n",
+ transport, transport->cwnd,
+ transport->ssthresh, transport->flight_size,
transport->partial_bytes_acked);
}
-
-} /* sctp_check_transmitted() */
+}
/* Is the given TSN acked by this packet? */
-static int
-sctp_acked(sctp_sackhdr_t *sack, uint32_t tsn)
+static int sctp_acked(sctp_sackhdr_t *sack, __u32 tsn)
{
- int i;
- sctp_sack_variable_t *frags;
- uint16_t gap;
- uint32_t ctsn = ntohl(sack->cum_tsn_ack);
+ int i;
+ sctp_sack_variable_t *frags;
+ __u16 gap;
+ __u32 ctsn = ntohl(sack->cum_tsn_ack);
- if (TSN_lte(tsn, ctsn)) {
+ if (TSN_lte(tsn, ctsn))
goto pass;
- }
-
- /* 3.3.4 Selective Acknowledgement (SACK) (3):
- *
- * Gap Ack Blocks:
- * These fields contain the Gap Ack Blocks. They are repeated
- * for each Gap Ack Block up to the number of Gap Ack Blocks
- * defined in the Number of Gap Ack Blocks field. All DATA
- * chunks with TSNs greater than or equal to (Cumulative TSN
- * Ack + Gap Ack Block Start) and less than or equal to
- * (Cumulative TSN Ack + Gap Ack Block End) of each Gap Ack
- * Block are assumed to have been received correctly.
- */
-
- frags = sack->variable;
- gap = tsn - ctsn;
- for (i = 0; i < ntohs(sack->num_gap_ack_blocks); ++i) {
- if (TSN_lte(ntohs(frags[i].gab.start), gap)
- && TSN_lte(gap, ntohs(frags[i].gab.end))) {
- goto pass;
- }
- }
-
- return 0;
-pass:
- return 1;
-} /* sctp_acked() */
+ /* 3.3.4 Selective Acknowledgement (SACK) (3):
+ *
+ * Gap Ack Blocks:
+ * These fields contain the Gap Ack Blocks. They are repeated
+ * for each Gap Ack Block up to the number of Gap Ack Blocks
+ * defined in the Number of Gap Ack Blocks field. All DATA
+ * chunks with TSNs greater than or equal to (Cumulative TSN
+ * Ack + Gap Ack Block Start) and less than or equal to
+ * (Cumulative TSN Ack + Gap Ack Block End) of each Gap Ack
+ * Block are assumed to have been received correctly.
+ */
+ frags = sack->variable;
+ gap = tsn - ctsn;
+ for (i = 0; i < ntohs(sack->num_gap_ack_blocks); ++i) {
+ if (TSN_lte(ntohs(frags[i].gab.start), gap) &&
+ TSN_lte(gap, ntohs(frags[i].gab.end)))
+ goto pass;
+ }
+ return 0;
+pass:
+ return 1;
+}
*/
static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_primitive.c,v 1.6 2002/08/21 18:34:04 jgrimm Exp $";
-#include <linux/config.h>
#include <linux/types.h>
#include <linux/list.h> /* For struct list_head */
#include <linux/socket.h>
#include <net/sctp/sctp.h>
#include <net/sctp/sctp_sm.h>
-
#define DECLARE_PRIMITIVE(name) \
/* This is called in the code as sctp_primitive_ ## name. */ \
-int \
-sctp_primitive_ ## name(sctp_association_t *asoc, \
- void *arg) { \
- int error = 0; \
- sctp_event_t event_type; sctp_subtype_t subtype; \
+int sctp_primitive_ ## name(sctp_association_t *asoc, \
+ void *arg) { \
+ int error = 0; \
+ sctp_event_t event_type; sctp_subtype_t subtype; \
sctp_state_t state; \
- sctp_endpoint_t *ep; \
+ sctp_endpoint_t *ep; \
+ \
+ event_type = SCTP_EVENT_T_PRIMITIVE; \
+ subtype = SCTP_ST_PRIMITIVE(SCTP_PRIMITIVE_ ## name); \
+ state = asoc ? asoc->state : SCTP_STATE_CLOSED; \
+ ep = asoc ? asoc->ep : NULL; \
\
- event_type = SCTP_EVENT_T_PRIMITIVE; \
- subtype = SCTP_ST_PRIMITIVE(SCTP_PRIMITIVE_ ## name); \
- state = asoc ? asoc->state : SCTP_STATE_CLOSED; \
- ep = asoc ? asoc->ep : NULL; \
- \
- error = sctp_do_sm(event_type, subtype, state, ep, asoc, arg, GFP_KERNEL); \
- return error; \
-} /* sctp_primitive_ ## name() */
+ error = sctp_do_sm(event_type, subtype, state, ep, asoc, \
+ arg, GFP_KERNEL); \
+ return error; \
+}
/* 10.1 ULP-to-SCTP
* B) Associate
- *
- * Format: ASSOCIATE(local SCTP instance name, destination transport addr,
- * outbound stream count)
+ *
+ * Format: ASSOCIATE(local SCTP instance name, destination transport addr,
+ * outbound stream count)
* -> association id [,destination transport addr list] [,outbound stream
- * count]
- *
+ * count]
+ *
* This primitive allows the upper layer to initiate an association to a
- * specific peer endpoint.
+ * specific peer endpoint.
*
* This version assumes that asoc is fully populated with the initial
* parameters. We then return a traditional kernel indicator of
- * success or failure.
+ * success or failure.
*/
/* This is called in the code as sctp_primitive_ASSOCIATE. */
/* 10.1 ULP-to-SCTP
* C) Shutdown
- *
+ *
* Format: SHUTDOWN(association id)
- * -> result
- *
+ * -> result
+ *
* Gracefully closes an association. Any locally queued user data
* will be delivered to the peer. The association will be terminated only
* after the peer acknowledges all the SCTP packets sent. A success code
/* 10.1 ULP-to-SCTP
* C) Abort
- *
+ *
* Format: Abort(association id [, cause code])
- * -> result
- *
+ * -> result
+ *
* Ungracefully closes an association. Any locally queued user data
* will be discarded and an ABORT chunk is sent to the peer. A success
* code will be returned on successful abortion of the association. If
/* 10.1 ULP-to-SCTP
* E) Send
- *
+ *
* Format: SEND(association id, buffer address, byte count [,context]
- * [,stream id] [,life time] [,destination transport address]
+ * [,stream id] [,life time] [,destination transport address]
* [,unorder flag] [,no-bundle flag] [,payload protocol-id] )
* -> result
- *
- * This is the main method to send user data via SCTP.
- *
+ *
+ * This is the main method to send user data via SCTP.
+ *
* Mandatory attributes:
- *
+ *
* o association id - local handle to the SCTP association
- *
+ *
* o buffer address - the location where the user message to be
* transmitted is stored;
- *
+ *
* o byte count - The size of the user data in number of bytes;
- *
+ *
* Optional attributes:
- *
+ *
* o context - an optional 32 bit integer that will be carried in the
* sending failure notification to the ULP if the transportation of
* this User Message fails.
- *
+ *
* o stream id - to indicate which stream to send the data on. If not
* specified, stream 0 will be used.
- *
+ *
* o life time - specifies the life time of the user data. The user data
* will not be sent by SCTP after the life time expires. This
* parameter can be used to avoid efforts to transmit stale
* user messages. SCTP notifies the ULP if the data cannot be
* initiated to transport (i.e. sent to the destination via SCTP's
* send primitive) within the life time variable. However, the
- * user data will be transmitted if SCTP has attempted to transmit a
+ * user data will be transmitted if SCTP has attempted to transmit a
* chunk before the life time expired.
- *
+ *
* o destination transport address - specified as one of the destination
* transport addresses of the peer endpoint to which this packet
* should be sent. Whenever possible, SCTP should use this destination
* transport address for sending the packets, instead of the current
- * primary path.
- *
+ * primary path.
+ *
* o unorder flag - this flag, if present, indicates that the user
- * would like the data delivered in an unordered fashion to the peer
- * (i.e., the U flag is set to 1 on all DATA chunks carrying this
+ * would like the data delivered in an unordered fashion to the peer
+ * (i.e., the U flag is set to 1 on all DATA chunks carrying this
* message).
- *
+ *
* o no-bundle flag - instructs SCTP not to bundle this user data with
- * other outbound DATA chunks. SCTP MAY still bundle even when
+ * other outbound DATA chunks. SCTP MAY still bundle even when
* this flag is present, when faced with network congestion.
- *
- * o payload protocol-id - A 32 bit unsigned integer that is to be
- * passed to the peer indicating the type of payload protocol data
+ *
+ * o payload protocol-id - A 32 bit unsigned integer that is to be
+ * passed to the peer indicating the type of payload protocol data
* being transmitted. This value is passed as opaque data by SCTP.
*/
DECLARE_PRIMITIVE(SEND);
-
/* COMMENT BUG. Find out where this is mentioned in the spec. */
-int
-sctp_other_icmp_unreachfrag(sctp_association_t *asoc, void *arg)
+int sctp_other_icmp_unreachfrag(sctp_association_t *asoc, void *arg)
{
- int error = 0;
- sctp_event_t event_type;
+ int error = 0;
+ sctp_event_t event_type;
sctp_subtype_t subtype;
sctp_state_t state;
- sctp_endpoint_t *ep;
-
- event_type = SCTP_EVENT_T_OTHER;
- subtype = SCTP_ST_OTHER(SCTP_EVENT_ICMP_UNREACHFRAG);
- state = asoc ? asoc->state : SCTP_STATE_CLOSED;
- ep = asoc ? asoc->ep : NULL;
-
- error = sctp_do_sm(event_type, subtype, state, ep, asoc, arg,
- GFP_ATOMIC);
+ sctp_endpoint_t *ep;
- return error;
+ event_type = SCTP_EVENT_T_OTHER;
+ subtype = SCTP_ST_OTHER(SCTP_EVENT_ICMP_UNREACHFRAG);
+ state = asoc ? asoc->state : SCTP_STATE_CLOSED;
+ ep = asoc ? asoc->ep : NULL;
-} /* sctp_other_icmp_unreachfrag() */
+ error = sctp_do_sm(event_type, subtype, state, ep,
+ asoc, arg, GFP_ATOMIC);
+ return error;
+}
extern struct net_proto_family inet_family_ops;
-
/* Return the address of the control sock. */
struct sock *sctp_get_ctl_sock(void)
{
- return(sctp_ctl_socket->sk);
-
-} /* sctp_get_ctl_sock() */
+ return sctp_ctl_socket->sk;
+}
/* Set up the proc fs entry for the SCTP protocol. */
-void
-sctp_proc_init(void)
-{
+void sctp_proc_init(void)
+{
if (!proc_net_sctp) {
struct proc_dir_entry *ent;
ent = proc_mkdir("net/sctp", 0);
if (ent) {
ent->owner = THIS_MODULE;
- proc_net_sctp= ent;
+ proc_net_sctp = ent;
}
}
-
-} /* sctp_proc_init() */
+}
/* Clean up the proc fs entry for the SCTP protocol. */
-void
-sctp_proc_exit(void)
+void sctp_proc_exit(void)
{
if (proc_net_sctp) {
proc_net_sctp= NULL;
remove_proc_entry("net/sctp", 0);
}
-
-} /* sctp_proc_exit() */
-
-
+}
/* Private helper to extract ipv4 address and stash them in
- * the protocol structure.
+ * the protocol structure.
*/
-static inline void
-sctp_v4_get_local_addr_list(sctp_protocol_t *proto,
- struct net_device *dev)
+static inline void sctp_v4_get_local_addr_list(sctp_protocol_t *proto,
+ struct net_device *dev)
{
struct in_device *in_dev;
- struct in_ifaddr *ifa;
+ struct in_ifaddr *ifa;
struct sockaddr_storage_list *addr;
read_lock(&inetdev_lock);
for (ifa = in_dev->ifa_list; ifa; ifa = ifa->ifa_next) {
/* Add the address to the local list. */
+ /* XXX BUG: sleeping allocation with lock held -DaveM */
addr = t_new(struct sockaddr_storage_list, GFP_KERNEL);
if (addr) {
INIT_LIST_HEAD(&addr->list);
list_add_tail(&addr->list, &proto->local_addr_list);
}
}
-
+
read_unlock(&in_dev->lock);
read_unlock(&inetdev_lock);
- return;
-
-} /* sctp_v4_get_local_addr_list() */
+}
/* Private helper to extract ipv6 address and stash them in
* the protocol structure.
- * FIXME: Make this an address family function.
+ * FIXME: Make this an address family function.
*/
-static inline void
-sctp_v6_get_local_addr_list(sctp_protocol_t *proto, struct net_device *dev)
+static inline void sctp_v6_get_local_addr_list(sctp_protocol_t *proto, struct net_device *dev)
{
#ifdef SCTP_V6_SUPPORT
/* FIXME: The testframe doesn't support this function. */
-#ifndef TEST_FRAME
+#ifndef TEST_FRAME
struct inet6_dev *in6_dev;
struct inet6_ifaddr *ifp;
struct sockaddr_storage_list *addr;
-
- read_lock(&addrconf_lock);
- if ((in6_dev = __in6_dev_get(dev)) == NULL) {
+
+ read_lock(&addrconf_lock);
+ if ((in6_dev = __in6_dev_get(dev)) == NULL) {
read_unlock(&addrconf_lock);
return;
}
-
+
read_lock_bh(&in6_dev->lock);
- for (ifp=in6_dev->addr_list; ifp; ifp=ifp->if_next) {
+ for (ifp = in6_dev->addr_list; ifp; ifp = ifp->if_next) {
/* Add the address to the local list. */
+ /* XXX BUG: sleeping allocation with lock held -DaveM */
addr = t_new(struct sockaddr_storage_list, GFP_KERNEL);
if (addr) {
addr->a.v6.sin6_family = AF_INET6;
addr->a.v6.sin6_addr = ifp->addr;
INIT_LIST_HEAD(&addr->list);
list_add_tail(&addr->list, &proto->local_addr_list);
- }
+ }
}
read_unlock_bh(&in6_dev->lock);
read_unlock(&addrconf_lock);
-
#endif /* TEST_FRAME */
- return;
#endif /* SCTP_V6_SUPPORT */
-} /* sctp_v6_get_local_addr_list() */
-
+}
/* Extract our IP addresses from the system and stash them in the
* protocol structure.
*/
-static void
-__sctp_get_local_addr_list(sctp_protocol_t *proto)
+static void __sctp_get_local_addr_list(sctp_protocol_t *proto)
{
struct net_device *dev;
-
+
read_lock(&dev_base_lock);
-
- for (dev=dev_base; dev; dev = dev->next) {
+ for (dev = dev_base; dev; dev = dev->next) {
sctp_v4_get_local_addr_list(proto, dev);
sctp_v6_get_local_addr_list(proto, dev);
- }
-
+ }
read_unlock(&dev_base_lock);
+}
-} /* __sctp_get_local_addr_list() */
-
-
-static void
-sctp_get_local_addr_list(sctp_protocol_t *proto)
+static void sctp_get_local_addr_list(sctp_protocol_t *proto)
{
long flags __attribute__ ((unused));
-
- sctp_spin_lock_irqsave(&sctp_proto.local_addr_lock, flags);
- __sctp_get_local_addr_list(&sctp_proto);
- sctp_spin_unlock_irqrestore(&sctp_proto.local_addr_lock, flags);
-
-} /* sctp_get_local_addr_list() */
+ sctp_spin_lock_irqsave(&sctp_proto.local_addr_lock, flags);
+ __sctp_get_local_addr_list(&sctp_proto);
+ sctp_spin_unlock_irqrestore(&sctp_proto.local_addr_lock, flags);
+}
-/* Free the existing local addresses. */
-static void
-__sctp_free_local_addr_list(sctp_protocol_t *proto)
+/* Free the existing local addresses. */
+static void __sctp_free_local_addr_list(sctp_protocol_t *proto)
{
- struct sockaddr_storage_list *addr;
- list_t *pos, *temp;
-
+ astruct sockaddr_storage_list *addr;
+ list_t *pos, *temp;
+
list_for_each_safe(pos, temp, &proto->local_addr_list) {
- addr = list_entry(pos, struct sockaddr_storage_list, list);
- list_del(pos);
- kfree(addr);
- }
-
-} /* __sctp_free_local_addr_list() */
-
-/* Free the existing local addresses. */
-static void
-sctp_free_local_addr_list(sctp_protocol_t *proto)
+ addr = list_entry(pos, struct sockaddr_storage_list, list);
+ list_del(pos);
+ kfree(addr);
+ }
+}
+
+/* Free the existing local addresses. */
+static void sctp_free_local_addr_list(sctp_protocol_t *proto)
{
- long flags __attribute__ ((unused));
-
- sctp_spin_lock_irqsave(&proto->local_addr_lock, flags);
- __sctp_free_local_addr_list(proto);
- sctp_spin_unlock_irqrestore(&proto->local_addr_lock, flags);
-
-} /* sctp_free_local_addr_list() */
+ long flags __attribute__ ((unused));
+ sctp_spin_lock_irqsave(&proto->local_addr_lock, flags);
+ __sctp_free_local_addr_list(proto);
+ sctp_spin_unlock_irqrestore(&proto->local_addr_lock, flags);
+}
/* Copy the local addresses which are valid for 'scope' into 'bp'. */
-int
-sctp_copy_local_addr_list(sctp_protocol_t *proto, sctp_bind_addr_t *bp,
- sctp_scope_t scope, int priority, int copy_flags)
+int sctp_copy_local_addr_list(sctp_protocol_t *proto, sctp_bind_addr_t *bp,
+ sctp_scope_t scope, int priority, int copy_flags)
{
-
struct sockaddr_storage_list *addr;
int error = 0;
list_t *pos;
long flags __attribute__ ((unused));
-
+
sctp_spin_lock_irqsave(&proto->local_addr_lock, flags);
list_for_each(pos, &proto->local_addr_list) {
addr = list_entry(pos, struct sockaddr_storage_list, list);
if (sctp_in_scope(&addr->a, scope)) {
- /* Now that the address is in scope, check to see if
+ /* Now that the address is in scope, check to see if
* the address type is really supported by the local
* sock as well as the remote peer.
*/
- if ((((AF_INET == addr->a.sa.sa_family)
- && (copy_flags & SCTP_ADDR4_PEERSUPP)))
- || (((AF_INET6 == addr->a.sa.sa_family)
- && (copy_flags & SCTP_ADDR6_ALLOWED)
- && (copy_flags & SCTP_ADDR6_PEERSUPP)))) {
-
- error = sctp_add_bind_addr(bp,
- &addr->a,
+ if ((((AF_INET == addr->a.sa.sa_family) &&
+ (copy_flags & SCTP_ADDR4_PEERSUPP))) ||
+ (((AF_INET6 == addr->a.sa.sa_family) &&
+ (copy_flags & SCTP_ADDR6_ALLOWED) &&
+ (copy_flags & SCTP_ADDR6_PEERSUPP)))) {
+ error = sctp_add_bind_addr(bp,
+ &addr->a,
priority);
- if (0 != error) { goto end_copy; }
+ if (error)
+ goto end_copy;
}
- }
- }
+ }
+ }
end_copy:
sctp_spin_unlock_irqrestore(&proto->local_addr_lock, flags);
- return(error);
-
-} /* sctp_copy_local_addr_list() */
-
+ return error;
+}
-/* Returns the mtu for the given v4 destination address. */
-int
-sctp_v4_get_dst_mtu(const sockaddr_storage_t *address)
+/* Returns the mtu for the given v4 destination address. */
+int sctp_v4_get_dst_mtu(const sockaddr_storage_t *address)
{
int dst_mtu = SCTP_DEFAULT_MAXSEGMENT;
-
struct rtable *rt;
- struct rt_key key = {
+ struct rt_key key = {
.dst = address->v4.sin_addr.s_addr,
.src = 0,
.iif = 0,
.tos = 0,
.scope = 0
};
-
+
if (ip_route_output_key(&rt, &key)) {
SCTP_DEBUG_PRINTK("sctp_v4_get_dst_mtu:ip_route_output_key"
- " failed, returning %d as dst_mtu\n",
+ " failed, returning %d as dst_mtu\n",
dst_mtu);
} else {
dst_mtu = rt->u.dst.pmtu;
ip_rt_put(rt);
}
- return (dst_mtu);
-
-} /* sctp_v4_get_dst_mtu() */
+ return dst_mtu;
+}
-
-/* Event handler for inet device events.
+/* Event handler for inet device events.
* Basically, whenever there is an event, we re-build our local address list.
*/
-static int
-sctp_netdev_event(struct notifier_block *this, unsigned long event, void *ptr)
+static int sctp_netdev_event(struct notifier_block *this, unsigned long event, void *ptr)
{
long flags __attribute__ ((unused));
sctp_spin_unlock_irqrestore(&sctp_proto.local_addr_lock, flags);
return NOTIFY_DONE;
-
-} /* sctp_netdev_event() */
+}
/*
* Initialize the control inode/socket with a control endpoint data
* structure. This endpoint is reserved exclusively for the OOTB processing.
*/
-int
-sctp_ctl_sock_init(void)
+int sctp_ctl_sock_init(void)
{
int err = 0;
int family = PF_INET;
SCTP_V6(family = PF_INET6;)
- err = sock_create(family, SOCK_SEQPACKET, IPPROTO_SCTP,
+ err = sock_create(family, SOCK_SEQPACKET, IPPROTO_SCTP,
&sctp_ctl_socket);
if (err < 0) {
- printk(KERN_ERR
+ printk(KERN_ERR
"SCTP: Failed to create the SCTP control socket.\n");
return err;
}
inet_sk(sctp_ctl_socket->sk)->ttl = MAXTTL;
return 0;
-
-} /* sctp_ctl_sock_init() */
+}
/* Get the table of functions for manipulating a particular address
* family.
*/
-sctp_func_t *
-sctp_get_af_specific(const sockaddr_storage_t *address)
+sctp_func_t *sctp_get_af_specific(const sockaddr_storage_t *address)
{
list_t *pos;
- sctp_protocol_t *proto = sctp_get_protocol();
- sctp_func_t *retval, *af;
-
+ sctp_protocol_t *proto = sctp_get_protocol();
+ sctp_func_t *retval, *af;
+
retval = NULL;
- /* Cycle through all AF specific functions looking for a
+ /* Cycle through all AF specific functions looking for a
* match.
*/
list_for_each(pos, &proto->address_families) {
af = list_entry(pos, sctp_func_t, list);
+ if (address->sa.sa_family == af->sa_family) {
+ retval = af;
+ break;
+ }
+ }
- if (address->sa.sa_family == af->sa_family) {
- retval = af;
- break;
- }
- }
-
- return retval;
-
-} /* sctp_get_af_specific() */
+ return retval;
+}
-/* Registration for netdev events. */
+/* Registration for netdev events. */
struct notifier_block sctp_netdev_notifier = {
.notifier_call = sctp_netdev_event,
};
-/* Socket operations. */
+/* Socket operations. */
struct proto_ops inet_seqpacket_ops = {
.family = PF_INET,
.release = inet_release, /* Needs to be wrapped... */
.bind = inet_bind,
- .connect = inet_dgram_connect,
+ .connect = inet_dgram_connect,
.socketpair = sock_no_socketpair,
.accept = inet_accept,
.getname = inet_getname, /* Semantics are different. */
- .poll = sctp_poll,
+ .poll = sctp_poll,
.ioctl = inet_ioctl,
.listen = sctp_inet_listen,
.shutdown = inet_shutdown, /* Looks harmless. */
.setsockopt = inet_setsockopt, /* IP_SOL IP_OPTION is a problem. */
- .getsockopt = inet_getsockopt,
+ .getsockopt = inet_getsockopt,
.sendmsg = inet_sendmsg,
.recvmsg = inet_recvmsg,
.mmap = sock_no_mmap,
.sendpage = sock_no_sendpage,
};
-/* Registration with AF_INET family. */
-struct inet_protosw sctp_protosw =
-{
- .type = SOCK_SEQPACKET,
+/* Registration with AF_INET family. */
+struct inet_protosw sctp_protosw = {
+ .type = SOCK_SEQPACKET,
.protocol = IPPROTO_SCTP,
.prot = &sctp_prot,
.ops = &inet_seqpacket_ops,
.flags = SCTP_PROTOSW_FLAG
};
-/* Register with IP layer. */
-static struct inet_protocol sctp_protocol =
-{
+/* Register with IP layer. */
+static struct inet_protocol sctp_protocol = {
.handler = sctp_rcv, /* SCTP input handler. */
.err_handler = sctp_v4_err, /* SCTP error control */
.protocol = IPPROTO_SCTP, /* protocol ID */
.name = "SCTP" /* name */
};
-/* IPv4 address related functions. */
+/* IPv4 address related functions. */
sctp_func_t sctp_ipv4_specific = {
- .queue_xmit = ip_queue_xmit,
- .setsockopt = ip_setsockopt,
- .getsockopt = ip_getsockopt,
+ .queue_xmit = ip_queue_xmit,
+ .setsockopt = ip_setsockopt,
+ .getsockopt = ip_getsockopt,
.get_dst_mtu = sctp_v4_get_dst_mtu,
.net_header_len = sizeof(struct iphdr),
- .sockaddr_len = sizeof(struct sockaddr_in),
+ .sockaddr_len = sizeof(struct sockaddr_in),
.sa_family = AF_INET,
-};
-
+};
/* Initialize the universe into something sensible. */
-int
-sctp_init(void)
+int sctp_init(void)
{
int i;
int status = 0;
-
- /* Add SCTP to inetsw linked list. */
+
+ /* Add SCTP to inetsw linked list. */
inet_register_protosw(&sctp_protosw);
-
- /* Add SCTP to inet_protos hash table. */
+
+ /* Add SCTP to inet_protos hash table. */
inet_add_protocol(&sctp_protocol);
- /* Initialize proc fs directory. */
+ /* Initialize proc fs directory. */
sctp_proc_init();
- /* Initialize object count debugging. */
+ /* Initialize object count debugging. */
sctp_dbg_objcnt_init();
/*
* 14. Suggested SCTP Protocol Parameter Values
*/
- /* The following protocol parameters are RECOMMENDED: */
- /* RTO.Initial - 3 seconds */
- sctp_proto.rto_initial = SCTP_RTO_INITIAL;
- /* RTO.Min - 1 second */
- sctp_proto.rto_min = SCTP_RTO_MIN;
- /* RTO.Max - 60 seconds */
- sctp_proto.rto_max = SCTP_RTO_MAX;
- /* RTO.Alpha - 1/8 */
- sctp_proto.rto_alpha = SCTP_RTO_ALPHA;
- /* RTO.Beta - 1/4 */
- sctp_proto.rto_beta = SCTP_RTO_BETA;
-
- /* Valid.Cookie.Life - 60 seconds
- */
- sctp_proto.valid_cookie_life = 60 * HZ;
+ /* The following protocol parameters are RECOMMENDED: */
+ /* RTO.Initial - 3 seconds */
+ sctp_proto.rto_initial = SCTP_RTO_INITIAL;
+ /* RTO.Min - 1 second */
+ sctp_proto.rto_min = SCTP_RTO_MIN;
+ /* RTO.Max - 60 seconds */
+ sctp_proto.rto_max = SCTP_RTO_MAX;
+ /* RTO.Alpha - 1/8 */
+ sctp_proto.rto_alpha = SCTP_RTO_ALPHA;
+ /* RTO.Beta - 1/4 */
+ sctp_proto.rto_beta = SCTP_RTO_BETA;
+
+ /* Valid.Cookie.Life - 60 seconds */
+ sctp_proto.valid_cookie_life = 60 * HZ;
/* Max.Burst - 4 */
sctp_proto.max_burst = SCTP_MAX_BURST;
- /* Association.Max.Retrans - 10 attempts
- * Path.Max.Retrans - 5 attempts (per destination address)
- * Max.Init.Retransmits - 8 attempts
- */
- sctp_proto.max_retrans_association = 10;
- sctp_proto.max_retrans_path = 5;
- sctp_proto.max_retrans_init = 8;
+ /* Association.Max.Retrans - 10 attempts
+ * Path.Max.Retrans - 5 attempts (per destination address)
+ * Max.Init.Retransmits - 8 attempts
+ */
+ sctp_proto.max_retrans_association = 10;
+ sctp_proto.max_retrans_path = 5;
+ sctp_proto.max_retrans_init = 8;
+
+ /* HB.interval - 30 seconds */
+ sctp_proto.hb_interval = 30 * HZ;
- /* HB.interval - 30 seconds */
- sctp_proto.hb_interval = 30 * HZ;
-
/* Implementation specific variables. */
- /* Initialize default stream count setup information.
- * Note: today the stream accounting data structures are very
- * fixed size, so one really does need to make sure that these have
- * upper/lower limits when changing.
+ /* Initialize default stream count setup information.
+ * Note: today the stream accounting data structures are very
+ * fixed size, so one really does need to make sure that these have
+ * upper/lower limits when changing.
*/
- sctp_proto.max_instreams = SCTP_MAX_STREAM;
+ sctp_proto.max_instreams = SCTP_MAX_STREAM;
sctp_proto.max_outstreams = SCTP_MAX_STREAM;
- /* Allocate and initialize the association hash table. */
+ /* Allocate and initialize the association hash table. */
sctp_proto.assoc_hashsize = 4096;
sctp_proto.assoc_hashbucket = (sctp_hashbucket_t *)
kmalloc(4096 * sizeof(sctp_hashbucket_t), GFP_KERNEL);
if (!sctp_proto.assoc_hashbucket) {
- printk (KERN_ERR "SCTP: Failed association hash alloc.\n");
+ printk (KERN_ERR "SCTP: Failed association hash alloc.\n");
status = -ENOMEM;
- goto err_ahash_alloc;
+ goto err_ahash_alloc;
}
for (i = 0; i < sctp_proto.assoc_hashsize; i++) {
sctp_proto.assoc_hashbucket[i].lock = RW_LOCK_UNLOCKED;
sctp_proto.assoc_hashbucket[i].chain = NULL;
}
- /* Allocate and initialize the endpoint hash table. */
+ /* Allocate and initialize the endpoint hash table. */
sctp_proto.ep_hashsize = 64;
sctp_proto.ep_hashbucket = (sctp_hashbucket_t *)
kmalloc(64 * sizeof(sctp_hashbucket_t), GFP_KERNEL);
if (!sctp_proto.ep_hashbucket) {
- printk (KERN_ERR "SCTP: Failed endpoint_hash alloc.\n");
+ printk (KERN_ERR "SCTP: Failed endpoint_hash alloc.\n");
status = -ENOMEM;
- goto err_ehash_alloc;
+ goto err_ehash_alloc;
}
for (i = 0; i < sctp_proto.ep_hashsize; i++) {
sctp_proto.port_hashtable = (sctp_bind_hashbucket_t *)
kmalloc(4096 * sizeof(sctp_bind_hashbucket_t), GFP_KERNEL);
if (!sctp_proto.port_hashtable) {
- printk (KERN_ERR "SCTP: Failed bind hash alloc.");
+ printk (KERN_ERR "SCTP: Failed bind hash alloc.");
status = -ENOMEM;
- goto err_bhash_alloc;
+ goto err_bhash_alloc;
}
-
sctp_proto.port_alloc_lock = SPIN_LOCK_UNLOCKED;
sctp_proto.port_rover = sysctl_local_port_range[0] - 1;
INIT_LIST_HEAD(&sctp_proto.address_families);
INIT_LIST_HEAD(&sctp_ipv4_specific.list);
list_add_tail(&sctp_ipv4_specific.list, &sctp_proto.address_families);
-
+
status = sctp_v6_init();
- if (status != 0) {
+ if (status)
goto err_v6_init;
- }
- /* Initialize the control inode/socket for handling OOTB packets. */
- if (0 != (status = sctp_ctl_sock_init())) {
- printk (KERN_ERR
+ /* Initialize the control inode/socket for handling OOTB packets. */
+ if ((status = sctp_ctl_sock_init())) {
+ printk (KERN_ERR
"SCTP: Failed to initialize the SCTP control sock.\n");
- goto err_ctl_sock_init;
+ goto err_ctl_sock_init;
}
-
+
/* Initialize the local address list. */
INIT_LIST_HEAD(&sctp_proto.local_addr_list);
sctp_proto.local_addr_lock = SPIN_LOCK_UNLOCKED;
inet_del_protocol(&sctp_protocol);
inet_unregister_protosw(&sctp_protosw);
return status;
+}
-} /* sctp_init() */
-
-
-/* Exit handler for the SCTP protocol. */
-void
-sctp_exit(void)
+/* Exit handler for the SCTP protocol. */
+void sctp_exit(void)
{
- /* BUG. This should probably do something useful like clean
- * up all the remaining associations and all that memory.
- */
-
- /* Free the local address list. */
+ /* BUG. This should probably do something useful like clean
+ * up all the remaining associations and all that memory.
+ */
+
+ /* Free the local address list. */
unregister_inetaddr_notifier(&sctp_netdev_notifier);
sctp_free_local_addr_list(&sctp_proto);
- /* Free the control endpoint. */
+ /* Free the control endpoint. */
sock_release(sctp_ctl_socket);
sctp_v6_exit();
sctp_sysctl_unregister();
- list_del(&sctp_ipv4_specific.list);
+ list_del(&sctp_ipv4_specific.list);
kfree(sctp_proto.assoc_hashbucket);
kfree(sctp_proto.ep_hashbucket);
kfree(sctp_proto.port_hashtable);
-
+
sctp_dbg_objcnt_exit();
sctp_proc_exit();
inet_del_protocol(&sctp_protocol);
inet_unregister_protosw(&sctp_protosw);
-
-} /* sctp_exit() */
+}
module_init(sctp_init);
module_exit(sctp_exit);
*/
static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_sla1.c,v 1.4 2002/07/19 22:00:33 jgrimm Exp $";
-
-#include <linux/config.h>
#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/fcntl.h>
#include <net/sctp/sctp_sla1.h>
-void
-SLA1_Init(struct SLA_1_Context *ctx)
+void SLA1_Init(struct SLA_1_Context *ctx)
{
- /* Init the SLA-1 context structure */
- ctx->A = 0;
- ctx->B = 0;
- ctx->C = 0;
- ctx->D = 0;
- ctx->E = 0;
- ctx->H0 = H0INIT;
- ctx->H1 = H1INIT;
- ctx->H2 = H2INIT;
- ctx->H3 = H3INIT;
- ctx->H4 = H4INIT;
- ctx->TEMP = 0;
- memset(ctx->words,0,sizeof(ctx->words));
- ctx->howManyInBlock = 0;
- ctx->runningTotal = 0;
+ /* Init the SLA-1 context structure. */
+ ctx->A = 0;
+ ctx->B = 0;
+ ctx->C = 0;
+ ctx->D = 0;
+ ctx->E = 0;
+ ctx->H0 = H0INIT;
+ ctx->H1 = H1INIT;
+ ctx->H2 = H2INIT;
+ ctx->H3 = H3INIT;
+ ctx->H4 = H4INIT;
+ ctx->TEMP = 0;
+ memset(ctx->words, 0, sizeof(ctx->words));
+ ctx->howManyInBlock = 0;
+ ctx->runningTotal = 0;
}
-void
-SLA1processABlock(struct SLA_1_Context *ctx,unsigned int *block)
+void SLA1processABlock(struct SLA_1_Context *ctx,unsigned int *block)
{
- int i;
- /* init the W0-W15 to the block of words being
- * hashed.
- */
- /* step a) */
+ int i;
+ /* init the W0-W15 to the block of words being
+ * hashed.
+ */
+ /* step a) */
- for(i=0;i<16;i++){
- ctx->words[i] = ntohl(block[i]);
- }
- /* now init the rest based on the SLA-1 formula, step b) */
- for(i=16;i<80;i++){
- ctx->words[i] = CSHIFT(1,((ctx->words[(i-3)]) ^ (ctx->words[(i-8)]) ^ (ctx->words[(i-14)]) ^ (ctx->words[(i-16)])));
- }
- /* step c) */
- ctx->A = ctx->H0;
- ctx->B = ctx->H1;
- ctx->C = ctx->H2;
- ctx->D = ctx->H3;
- ctx->E = ctx->H4;
+ for (i = 0; i < 16; i++)
+ ctx->words[i] = ntohl(block[i]);
- /* step d) */
- for(i=0;i<80;i++){
- if(i < 20){
- ctx->TEMP = ((CSHIFT(5,ctx->A)) +
- (F1(ctx->B,ctx->C,ctx->D)) +
- (ctx->E) +
- ctx->words[i] +
- K1
- );
- }else if(i < 40){
- ctx->TEMP = ((CSHIFT(5,ctx->A)) +
- (F2(ctx->B,ctx->C,ctx->D)) +
- (ctx->E) +
- (ctx->words[i]) +
- K2
- );
- }else if(i < 60){
- ctx->TEMP = ((CSHIFT(5,ctx->A)) +
- (F3(ctx->B,ctx->C,ctx->D)) +
- (ctx->E) +
- (ctx->words[i]) +
- K3
- );
- }else{
- ctx->TEMP = ((CSHIFT(5,ctx->A)) +
- (F4(ctx->B,ctx->C,ctx->D)) +
- (ctx->E) +
- (ctx->words[i]) +
- K4
- );
- }
- ctx->E = ctx->D;
- ctx->D = ctx->C;
- ctx->C = CSHIFT(30,ctx->B);
- ctx->B = ctx->A;
- ctx->A = ctx->TEMP;
- }
- /* step e) */
- ctx->H0 = (ctx->H0) + (ctx->A);
- ctx->H1 = (ctx->H1) + (ctx->B);
- ctx->H2 = (ctx->H2) + (ctx->C);
- ctx->H3 = (ctx->H3) + (ctx->D);
- ctx->H4 = (ctx->H4) + (ctx->E);
-}
+ /* now init the rest based on the SLA-1 formula, step b) */
+ for (i = 16; i < 80; i++)
+ ctx->words[i] =
+ CSHIFT(1, ((ctx->words[(i-3)]) ^
+ (ctx->words[(i-8)]) ^
+ (ctx->words[(i-14)]) ^
+ (ctx->words[(i-16)])));
+ /* step c) */
+ ctx->A = ctx->H0;
+ ctx->B = ctx->H1;
+ ctx->C = ctx->H2;
+ ctx->D = ctx->H3;
+ ctx->E = ctx->H4;
-void
-SLA1_Process(struct SLA_1_Context *ctx, const unsigned char *ptr, int siz)
-{
- int numberLeft,leftToFill;
- numberLeft = siz;
- while(numberLeft > 0){
- leftToFill = sizeof(ctx->SLAblock) - ctx->howManyInBlock;
- if(leftToFill > numberLeft){
- /* can only partially fill up this one */
- memcpy(&ctx->SLAblock[ctx->howManyInBlock],ptr,numberLeft);
- ctx->howManyInBlock += siz;
- ctx->runningTotal += siz;
- break;
- }else{
- /* block is now full, process it */
- memcpy(&ctx->SLAblock[ctx->howManyInBlock],ptr,leftToFill);
- SLA1processABlock(ctx,(unsigned int *)ctx->SLAblock);
- numberLeft -= leftToFill;
- ctx->runningTotal += leftToFill;
- ctx->howManyInBlock = 0;
- }
- }
+ /* step d) */
+ for (i = 0; i < 80; i++) {
+ if (i < 20) {
+ ctx->TEMP = ((CSHIFT(5, ctx->A)) +
+ (F1(ctx->B, ctx->C, ctx->D)) +
+ (ctx->E) +
+ ctx->words[i] +
+ K1
+ );
+ } else if (i < 40) {
+ ctx->TEMP = ((CSHIFT(5, ctx->A)) +
+ (F2(ctx->B, ctx->C, ctx->D)) +
+ (ctx->E) +
+ (ctx->words[i]) +
+ K2
+ );
+ } else if (i < 60) {
+ ctx->TEMP = ((CSHIFT(5, ctx->A)) +
+ (F3(ctx->B, ctx->C, ctx->D)) +
+ (ctx->E) +
+ (ctx->words[i]) +
+ K3
+ );
+ } else {
+ ctx->TEMP = ((CSHIFT(5, ctx->A)) +
+ (F4(ctx->B, ctx->C, ctx->D)) +
+ (ctx->E) +
+ (ctx->words[i]) +
+ K4
+ );
+ }
+ ctx->E = ctx->D;
+ ctx->D = ctx->C;
+ ctx->C = CSHIFT(30, ctx->B);
+ ctx->B = ctx->A;
+ ctx->A = ctx->TEMP;
+ }
+
+ /* step e) */
+ ctx->H0 = (ctx->H0) + (ctx->A);
+ ctx->H1 = (ctx->H1) + (ctx->B);
+ ctx->H2 = (ctx->H2) + (ctx->C);
+ ctx->H3 = (ctx->H3) + (ctx->D);
+ ctx->H4 = (ctx->H4) + (ctx->E);
}
-void
-SLA1_Final(struct SLA_1_Context *ctx,unsigned char *digestBuf)
+void SLA1_Process(struct SLA_1_Context *ctx, const unsigned char *ptr, int siz)
{
- /* if any left in block fill with padding
- * and process. Then transfer the digest to
- * the pointer. At the last block some special
- * rules need to apply. We must add a 1 bit
- * following the message, then we pad with
- * 0's. The total size is encoded as a 64 bit
- * number at the end. Now if the last buffer has
- * more than 55 octets in it we cannot fit
- * the 64 bit number + 10000000 pad on the end
- * and must add the 10000000 pad, pad the rest
- * of the message with 0's and then create a
- * all 0 message with just the 64 bit size
- * at the end and run this block through by itself.
- * Also the 64 bit int must be in network byte
- * order.
- */
- int i,leftToFill;
- unsigned int *ptr;
- if(ctx->howManyInBlock > 55){
- /* special case, we need to process two
- * blocks here. One for the current stuff
- * plus possibly the pad. The other for
- * the size.
- */
- leftToFill = sizeof(ctx->SLAblock) - ctx->howManyInBlock;
- if(leftToFill == 0){
- /* Should not really happen but I am paranoid */
- /* Not paranoid enough! It is possible for leftToFill to become
- * negative! AAA!!!! This is another reason to pick MD5 :-)...
- */
- SLA1processABlock(ctx,(unsigned int *)ctx->SLAblock);
- /* init last block, a bit different then the rest :-) */
- ctx->SLAblock[0] = 0x80;
- for(i=1;i<sizeof(ctx->SLAblock);i++){
- ctx->SLAblock[i] = 0x0;
- }
- }else if(leftToFill == 1){
- ctx->SLAblock[ctx->howManyInBlock] = 0x80;
- SLA1processABlock(ctx,(unsigned int *)ctx->SLAblock);
- /* init last block */
- memset(ctx->SLAblock,0,sizeof(ctx->SLAblock));
- }else{
- ctx->SLAblock[ctx->howManyInBlock] = 0x80;
- for(i=(ctx->howManyInBlock+1);i<sizeof(ctx->SLAblock);i++){
- ctx->SLAblock[i] = 0x0;
- }
- SLA1processABlock(ctx,(unsigned int *)ctx->SLAblock);
- /* init last block */
- memset(ctx->SLAblock,0,sizeof(ctx->SLAblock));
- }
- /* This is in bits so multiply by 8 */
- ctx->runningTotal *= 8;
- ptr = (unsigned int *)&ctx->SLAblock[60];
- *ptr = htonl(ctx->runningTotal);
- SLA1processABlock(ctx,(unsigned int *)ctx->SLAblock);
- }else{
- /* easy case, we just pad this
- * message to size - end with 0
- * add the magic 0x80 to the next
- * word and then put the network byte
- * order size in the last spot and
- * process the block.
- */
- ctx->SLAblock[ctx->howManyInBlock] = 0x80;
- for(i=(ctx->howManyInBlock+1);i<sizeof(ctx->SLAblock);i++){
- ctx->SLAblock[i] = 0x0;
- }
- /* get last int spot */
- ctx->runningTotal *= 8;
- ptr = (unsigned int *)&ctx->SLAblock[60];
- *ptr = htonl(ctx->runningTotal);
- SLA1processABlock(ctx,(unsigned int *)ctx->SLAblock);
- }
- /* Now at this point all we need do is transfer the
- * digest back to the user
- */
- digestBuf[3] = (ctx->H0 & 0xff);
- digestBuf[2] = ((ctx->H0 >> 8) & 0xff);
- digestBuf[1] = ((ctx->H0 >> 16) & 0xff);
- digestBuf[0] = ((ctx->H0 >> 24) & 0xff);
-
- digestBuf[7] = (ctx->H1 & 0xff);
- digestBuf[6] = ((ctx->H1 >> 8) & 0xff);
- digestBuf[5] = ((ctx->H1 >> 16) & 0xff);
- digestBuf[4] = ((ctx->H1 >> 24) & 0xff);
-
- digestBuf[11] = (ctx->H2 & 0xff);
- digestBuf[10] = ((ctx->H2 >> 8) & 0xff);
- digestBuf[9] = ((ctx->H2 >> 16) & 0xff);
- digestBuf[8] = ((ctx->H2 >> 24) & 0xff);
-
- digestBuf[15] = (ctx->H3 & 0xff);
- digestBuf[14] = ((ctx->H3 >> 8) & 0xff);
- digestBuf[13] = ((ctx->H3 >> 16) & 0xff);
- digestBuf[12] = ((ctx->H3 >> 24) & 0xff);
-
- digestBuf[19] = (ctx->H4 & 0xff);
- digestBuf[18] = ((ctx->H4 >> 8) & 0xff);
- digestBuf[17] = ((ctx->H4 >> 16) & 0xff);
- digestBuf[16] = ((ctx->H4 >> 24) & 0xff);
+ int numberLeft, leftToFill;
+ numberLeft = siz;
+ while (numberLeft > 0) {
+ leftToFill = sizeof(ctx->SLAblock) - ctx->howManyInBlock;
+ if (leftToFill > numberLeft) {
+ /* can only partially fill up this one */
+ memcpy(&ctx->SLAblock[ctx->howManyInBlock],
+ ptr, numberLeft);
+ ctx->howManyInBlock += siz;
+ ctx->runningTotal += siz;
+ break;
+ } else {
+ /* block is now full, process it */
+ memcpy(&ctx->SLAblock[ctx->howManyInBlock],
+ ptr, leftToFill);
+ SLA1processABlock(ctx, (unsigned int *) ctx->SLAblock);
+ numberLeft -= leftToFill;
+ ctx->runningTotal += leftToFill;
+ ctx->howManyInBlock = 0;
+ }
+ }
}
+void SLA1_Final(struct SLA_1_Context *ctx, unsigned char *digestBuf)
+{
+ /* if any left in block fill with padding
+ * and process. Then transfer the digest to
+ * the pointer. At the last block some special
+ * rules need to apply. We must add a 1 bit
+ * following the message, then we pad with
+ * 0's. The total size is encoded as a 64 bit
+ * number at the end. Now if the last buffer has
+ * more than 55 octets in it we cannot fit
+ * the 64 bit number + 10000000 pad on the end
+ * and must add the 10000000 pad, pad the rest
+ * of the message with 0's and then create a
+ * all 0 message with just the 64 bit size
+ * at the end and run this block through by itself.
+ * Also the 64 bit int must be in network byte
+ * order.
+ */
+ int i, leftToFill;
+ unsigned int *ptr;
+ if (ctx->howManyInBlock > 55) {
+ /* special case, we need to process two
+ * blocks here. One for the current stuff
+ * plus possibly the pad. The other for
+ * the size.
+ */
+ leftToFill = sizeof(ctx->SLAblock) - ctx->howManyInBlock;
+ if (leftToFill == 0) {
+ /* Should not really happen but I am paranoid */
+ /* Not paranoid enough! It is possible for leftToFill
+ * to become negative! AAA!!!! This is another reason
+ * to pick MD5 :-)...
+ */
+ SLA1processABlock(ctx, (unsigned int *) ctx->SLAblock);
+ /* init last block, a bit different then the rest :-) */
+ ctx->SLAblock[0] = 0x80;
+ for (i = 1; i < sizeof(ctx->SLAblock); i++) {
+ ctx->SLAblock[i] = 0x0;
+ }
+ } else if (leftToFill == 1) {
+ ctx->SLAblock[ctx->howManyInBlock] = 0x80;
+ SLA1processABlock(ctx, (unsigned int *) ctx->SLAblock);
+ /* init last block */
+ memset(ctx->SLAblock, 0, sizeof(ctx->SLAblock));
+ } else {
+ ctx->SLAblock[ctx->howManyInBlock] = 0x80;
+ for (i = (ctx->howManyInBlock + 1);
+ i < sizeof(ctx->SLAblock);
+ i++) {
+ ctx->SLAblock[i] = 0x0;
+ }
+ SLA1processABlock(ctx, (unsigned int *) ctx->SLAblock);
+ /* init last block */
+ memset(ctx->SLAblock, 0, sizeof(ctx->SLAblock));
+ }
+ /* This is in bits so multiply by 8 */
+ ctx->runningTotal *= 8;
+ ptr = (unsigned int *) &ctx->SLAblock[60];
+ *ptr = htonl(ctx->runningTotal);
+ SLA1processABlock(ctx, (unsigned int *) ctx->SLAblock);
+ } else {
+ /* easy case, we just pad this
+ * message to size - end with 0
+ * add the magic 0x80 to the next
+ * word and then put the network byte
+ * order size in the last spot and
+ * process the block.
+ */
+ ctx->SLAblock[ctx->howManyInBlock] = 0x80;
+ for (i = (ctx->howManyInBlock + 1);
+ i < sizeof(ctx->SLAblock);
+ i++) {
+ ctx->SLAblock[i] = 0x0;
+ }
+ /* get last int spot */
+ ctx->runningTotal *= 8;
+ ptr = (unsigned int *) &ctx->SLAblock[60];
+ *ptr = htonl(ctx->runningTotal);
+ SLA1processABlock(ctx, (unsigned int *) ctx->SLAblock);
+ }
+ /* Now at this point all we need do is transfer the
+ * digest back to the user
+ */
+ digestBuf[3] = (ctx->H0 & 0xff);
+ digestBuf[2] = ((ctx->H0 >> 8) & 0xff);
+ digestBuf[1] = ((ctx->H0 >> 16) & 0xff);
+ digestBuf[0] = ((ctx->H0 >> 24) & 0xff);
+ digestBuf[7] = (ctx->H1 & 0xff);
+ digestBuf[6] = ((ctx->H1 >> 8) & 0xff);
+ digestBuf[5] = ((ctx->H1 >> 16) & 0xff);
+ digestBuf[4] = ((ctx->H1 >> 24) & 0xff);
+ digestBuf[11] = (ctx->H2 & 0xff);
+ digestBuf[10] = ((ctx->H2 >> 8) & 0xff);
+ digestBuf[9] = ((ctx->H2 >> 16) & 0xff);
+ digestBuf[8] = ((ctx->H2 >> 24) & 0xff);
+ digestBuf[15] = (ctx->H3 & 0xff);
+ digestBuf[14] = ((ctx->H3 >> 8) & 0xff);
+ digestBuf[13] = ((ctx->H3 >> 16) & 0xff);
+ digestBuf[12] = ((ctx->H3 >> 24) & 0xff);
-
+ digestBuf[19] = (ctx->H4 & 0xff);
+ digestBuf[18] = ((ctx->H4 >> 8) & 0xff);
+ digestBuf[17] = ((ctx->H4 >> 16) & 0xff);
+ digestBuf[16] = ((ctx->H4 >> 24) & 0xff);
+}
*/
static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_sm_make_chunk.c,v 1.38 2002/07/26 22:52:32 jgrimm Exp $";
-#include <linux/config.h>
#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/ip.h>
* of this parameter indicates that the sending endpoint can
* support any address type.
*/
-static const sctp_supported_addrs_param_t sat_param= {
+static const sctp_supported_addrs_param_t sat_param = {
{
SCTP_PARAM_SUPPORTED_ADDRESS_TYPES,
__constant_htons(SCTP_SAT_LEN),
SCTP_PARAM_IPV4_ADDRESS,
SCTP_V6(SCTP_PARAM_IPV6_ADDRESS,)
}
-};
+};
/* RFC 2960 3.3.2 Initiation (INIT) (1)
*
* Note 2: The ECN capable field is reserved for future use of
* Explicit Congestion Notification.
*/
-
-static const sctp_ecn_capable_param_t ecap_param = {{
- SCTP_PARAM_ECN_CAPABLE,
- __constant_htons(sizeof(sctp_ecn_capable_param_t)),
-}};
-
+static const sctp_ecn_capable_param_t ecap_param = {
+ {
+ SCTP_PARAM_ECN_CAPABLE,
+ __constant_htons(sizeof(sctp_ecn_capable_param_t)),
+ }
+};
/* A helper to initilize to initilize an op error inside a
* provided chunk, as most cause codes will be embedded inside an
* abort chunk.
*/
-void
-sctp_init_cause(sctp_chunk_t *chunk, uint16_t cause_code, const void *payload,
- size_t paylen)
+void sctp_init_cause(sctp_chunk_t *chunk, __u16 cause_code,
+ const void *payload, size_t paylen)
{
sctp_errhdr_t err;
int padlen;
- uint16_t len;
+ __u16 len;
- /* Cause code constants are now defined in network order. */
+ /* Cause code constants are now defined in network order. */
err.cause = cause_code;
len = sizeof(sctp_errhdr_t) + paylen;
padlen = len % 4;
err.length = htons(len);
sctp_addto_chunk(chunk, sizeof(sctp_errhdr_t), &err);
chunk->subh.err_hdr = sctp_addto_chunk(chunk, paylen, payload);
-
- return;
-
-} /* sctp_init_cause() */
-
+}
/* 3.3.2 Initiation (INIT) (1)
- *
+ *
* This chunk is used to initiate a SCTP association between two
* endpoints. The format of the INIT chunk is shown below:
- *
+ *
* 0 1 2 3
* 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* / Optional/Variable-Length Parameters /
* \ \
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
- *
- *
- * The INIT chunk contains the following parameters. Unless otherwise
+ *
+ *
+ * The INIT chunk contains the following parameters. Unless otherwise
* noted, each parameter MUST only be included once in the INIT chunk.
- *
+ *
* Fixed Parameters Status
* ----------------------------------------------
* Initiate Tag Mandatory
* Number of Outbound Streams Mandatory
* Number of Inbound Streams Mandatory
* Initial TSN Mandatory
- *
+ *
* Variable Parameters Status Type Value
* -------------------------------------------------------------
* IPv4 Address (Note 1) Optional 5
* Host Name Address (Note 3) Optional 11
* Supported Address Types (Note 4) Optional 12
*/
-
-sctp_chunk_t *
-sctp_make_init(const sctp_association_t *asoc, const sctp_bind_addr_t *bp,
- int priority)
+sctp_chunk_t *sctp_make_init(const sctp_association_t *asoc,
+ const sctp_bind_addr_t *bp,
+ int priority)
{
- sctp_inithdr_t init;
- sctpParam_t addrs;
+ sctp_inithdr_t init;
+ sctpParam_t addrs;
size_t chunksize;
- sctp_chunk_t *retval = NULL;
- int addrs_len = 0;
+ sctp_chunk_t *retval = NULL;
+ int addrs_len = 0;
/* RFC 2960 3.3.2 Initiation (INIT) (1)
*
* Note 1: The INIT chunks can contain multiple addresses that
- * can be IPv4 and/or IPv6 in any combination.
- */
+ * can be IPv4 and/or IPv6 in any combination.
+ */
retval = NULL;
addrs.v = NULL;
/* Convert the provided bind address list to raw format */
addrs = sctp_bind_addrs_to_raw(bp, &addrs_len, priority);
- if (NULL == addrs.v) { goto nodata; }
-
- init.init_tag = htonl(asoc->c.my_vtag);
- init.a_rwnd = htonl(asoc->rwnd);
- init.num_outbound_streams = htons(asoc->c.sinit_num_ostreams);
- init.num_inbound_streams = htons(asoc->c.sinit_max_instreams);
- init.initial_tsn = htonl(asoc->c.initial_tsn);
+ if (!addrs.v)
+ goto nodata;
+
+ init.init_tag = htonl(asoc->c.my_vtag);
+ init.a_rwnd = htonl(asoc->rwnd);
+ init.num_outbound_streams = htons(asoc->c.sinit_num_ostreams);
+ init.num_inbound_streams = htons(asoc->c.sinit_max_instreams);
+ init.initial_tsn = htonl(asoc->c.initial_tsn);
chunksize = sizeof(init) + addrs_len + SCTP_SAT_LEN;
chunksize += sizeof(ecap_param);
-
/* RFC 2960 3.3.2 Initiation (INIT) (1)
*
* Note 3: An INIT chunk MUST NOT contain more than one Host
- * Name address parameter. Moreover, the sender of the INIT
- * MUST NOT combine any other address types with the Host Name
- * address in the INIT. The receiver of INIT MUST ignore any
- * other address types if the Host Name address parameter is
- * present in the received INIT chunk.
+ * Name address parameter. Moreover, the sender of the INIT
+ * MUST NOT combine any other address types with the Host Name
+ * address in the INIT. The receiver of INIT MUST ignore any
+ * other address types if the Host Name address parameter is
+ * present in the received INIT chunk.
*
- * PLEASE DO NOT FIXME [This version does not support Host Name.]
- */
+ * PLEASE DO NOT FIXME [This version does not support Host Name.]
+ */
- retval = sctp_make_chunk(asoc, SCTP_CID_INIT, 0, chunksize);
- if (NULL == retval) { goto nodata; }
+ retval = sctp_make_chunk(asoc, SCTP_CID_INIT, 0, chunksize);
+ if (!retval)
+ goto nodata;
- retval->subh.init_hdr =
- sctp_addto_chunk(retval, sizeof(init), &init);
- retval->param_hdr.v =
- sctp_addto_chunk(retval, addrs_len, addrs.v);
- sctp_addto_chunk(retval, SCTP_SAT_LEN, &sat_param);
+ retval->subh.init_hdr =
+ sctp_addto_chunk(retval, sizeof(init), &init);
+ retval->param_hdr.v =
+ sctp_addto_chunk(retval, addrs_len, addrs.v);
+ sctp_addto_chunk(retval, SCTP_SAT_LEN, &sat_param);
sctp_addto_chunk(retval, sizeof(ecap_param), &ecap_param);
-
nodata:
- if (NULL != addrs.v) { kfree(addrs.v); }
- return retval;
-
-}; /* sctp_chunk_t *sctp_make_init() */
+ if (addrs.v)
+ kfree(addrs.v);
+ return retval;
+}
-sctp_chunk_t *
-sctp_make_init_ack(const sctp_association_t *asoc, const sctp_chunk_t *chunk,
- int priority)
+sctp_chunk_t *sctp_make_init_ack(const sctp_association_t *asoc,
+ const sctp_chunk_t *chunk,
+ int priority)
{
- sctp_inithdr_t initack;
- sctp_chunk_t *retval;
- sctpParam_t addrs;
- int addrs_len;
- sctp_cookie_param_t *cookie;
- int cookie_len;
+ sctp_inithdr_t initack;
+ sctp_chunk_t *retval;
+ sctpParam_t addrs;
+ int addrs_len;
+ sctp_cookie_param_t *cookie;
+ int cookie_len;
size_t chunksize;
int error;
sctp_scope_t scope;
sctp_bind_addr_t *bp = NULL;
int flags;
-
-
+
retval = NULL;
/* Build up the bind address list for the association based on
* info from the local endpoint and the remote peer.
*/
bp = sctp_bind_addr_new(priority);
- if (NULL == bp) { goto nomem_bindaddr; }
+ if (!bp)
+ goto nomem_bindaddr;
- /* Look for supported address types parameter and then build
+ /* Look for supported address types parameter and then build
* our address list based on that.
*/
scope = sctp_scope(&asoc->peer.active_path->ipaddr);
flags = (PF_INET6 == asoc->base.sk->family) ? SCTP_ADDR6_ALLOWED : 0;
- if (asoc->peer.ipv4_address) {
+ if (asoc->peer.ipv4_address)
flags |= SCTP_ADDR4_PEERSUPP;
- }
- if (asoc->peer.ipv6_address) {
+ if (asoc->peer.ipv6_address)
flags |= SCTP_ADDR6_PEERSUPP;
- }
error = sctp_bind_addr_copy(bp, &asoc->ep->base.bind_addr,
scope, priority, flags);
- if (0 != error) { goto nomem_copyaddr; }
+ if (error)
+ goto nomem_copyaddr;
addrs = sctp_bind_addrs_to_raw(bp, &addrs_len, priority);
- if (NULL == addrs.v) { goto nomem_rawaddr; }
-
- initack.init_tag = htonl(asoc->c.my_vtag);
- initack.a_rwnd = htonl(asoc->rwnd);
- initack.num_outbound_streams = htons(asoc->c.sinit_num_ostreams);
- initack.num_inbound_streams = htons(asoc->c.sinit_max_instreams);
- initack.initial_tsn = htonl(asoc->c.initial_tsn);
-
+ if (!addrs.v)
+ goto nomem_rawaddr;
+
+ initack.init_tag = htonl(asoc->c.my_vtag);
+ initack.a_rwnd = htonl(asoc->rwnd);
+ initack.num_outbound_streams = htons(asoc->c.sinit_num_ostreams);
+ initack.num_inbound_streams = htons(asoc->c.sinit_max_instreams);
+ initack.initial_tsn = htonl(asoc->c.initial_tsn);
+
/* FIXME: We really ought to build the cookie right
- * into the packet instead of allocating more fresh memory.
+ * into the packet instead of allocating more fresh memory.
*/
- cookie = sctp_pack_cookie(asoc->ep, asoc, chunk, &cookie_len,
+ cookie = sctp_pack_cookie(asoc->ep, asoc, chunk, &cookie_len,
addrs.v, addrs_len);
- if (NULL == cookie) { goto nomem_cookie; }
+ if (!cookie)
+ goto nomem_cookie;
chunksize = sizeof(initack) + addrs_len + cookie_len;
- /* Tell peer that we'll do ECN only if peer advertised such cap. */
- if (asoc->peer.ecn_capable) {
+ /* Tell peer that we'll do ECN only if peer advertised such cap. */
+ if (asoc->peer.ecn_capable)
chunksize += sizeof(ecap_param);
- }
- /* Now allocate and fill out the chunk. */
+ /* Now allocate and fill out the chunk. */
+ retval = sctp_make_chunk(asoc, SCTP_CID_INIT_ACK, 0, chunksize);
+ if (!retval)
+ goto nomem_chunk;
- retval = sctp_make_chunk(asoc, SCTP_CID_INIT_ACK, 0, chunksize);
- if (NULL == retval) { goto nomem_chunk; }
-
/* Per the advice in RFC 2960 6.4, send this reply to
* the source of the INIT packet.
- */
- retval->transport = chunk->transport;
- retval->subh.init_hdr =
- sctp_addto_chunk(retval, sizeof(initack), &initack);
- retval->param_hdr.v = sctp_addto_chunk(retval, addrs_len, addrs.v);
- sctp_addto_chunk(retval, cookie_len, cookie);
- if (asoc->peer.ecn_capable) {
+ */
+ retval->transport = chunk->transport;
+ retval->subh.init_hdr =
+ sctp_addto_chunk(retval, sizeof(initack), &initack);
+ retval->param_hdr.v = sctp_addto_chunk(retval, addrs_len, addrs.v);
+ sctp_addto_chunk(retval, cookie_len, cookie);
+ if (asoc->peer.ecn_capable)
sctp_addto_chunk(retval, sizeof(ecap_param), &ecap_param);
- }
/* We need to remove the const qualifier at this point. */
- retval->asoc = (sctp_association_t *) asoc;
+ retval->asoc = (sctp_association_t *) asoc;
/* RFC 2960 6.4 Multi-homed SCTP Endpoints
- *
+ *
* An endpoint SHOULD transmit reply chunks (e.g., SACK,
* HEARTBEAT ACK, * etc.) to the same destination transport
* address from which it received the DATA or control chunk
*
* [INIT ACK back to where the INIT came from.]
*/
- if (NULL != chunk) {
+ if (chunk)
retval->transport = chunk->transport;
- }
-
+
nomem_chunk:
- kfree(cookie);
+ kfree(cookie);
nomem_cookie:
kfree(addrs.v);
nomem_rawaddr:
nomem_copyaddr:
sctp_bind_addr_free(bp);
nomem_bindaddr:
- return retval;
-
-} /* sctp_make_init_ack() */
+ return retval;
+}
/* 3.3.11 Cookie Echo (COOKIE ECHO) (10):
- *
+ *
* This chunk is used only during the initialization of an association.
* It is sent by the initiator of an association to its peer to complete
- * the initialization process. This chunk MUST precede any DATA chunk
+ * the initialization process. This chunk MUST precede any DATA chunk
* sent within the association, but MAY be bundled with one or more DATA
* chunks in the same packet.
- *
+ *
* 0 1 2 3
* 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* / Cookie /
* \ \
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
- *
+ *
* Chunk Flags: 8 bit
- *
+ *
* Set to zero on transmit and ignored on receipt.
- *
+ *
* Length: 16 bits (unsigned integer)
- *
+ *
* Set to the size of the chunk in bytes, including the 4 bytes of
* the chunk header and the size of the Cookie.
- *
+ *
* Cookie: variable size
- *
- * This field must contain the exact cookie received in the
+ *
+ * This field must contain the exact cookie received in the
* State Cookie parameter from the previous INIT ACK.
- *
+ *
* An implementation SHOULD make the cookie as small as possible
* to insure interoperability.
*/
-sctp_chunk_t *
-sctp_make_cookie_echo(const sctp_association_t *asoc,
- const sctp_chunk_t *chunk)
+sctp_chunk_t *sctp_make_cookie_echo(const sctp_association_t *asoc,
+ const sctp_chunk_t *chunk)
{
- sctp_chunk_t *retval;
+ sctp_chunk_t *retval;
void *cookie;
int cookie_len;
cookie = asoc->peer.cookie;
cookie_len = asoc->peer.cookie_len;
- /* Build a cookie echo chunk. */
- retval = sctp_make_chunk(asoc, SCTP_CID_COOKIE_ECHO, 0, cookie_len);
- if (NULL == retval) { goto nodata; }
- retval->subh.cookie_hdr =
- sctp_addto_chunk(retval, cookie_len, cookie);
-
+ /* Build a cookie echo chunk. */
+ retval = sctp_make_chunk(asoc, SCTP_CID_COOKIE_ECHO, 0, cookie_len);
+ if (!retval)
+ goto nodata;
+ retval->subh.cookie_hdr =
+ sctp_addto_chunk(retval, cookie_len, cookie);
+
/* RFC 2960 6.4 Multi-homed SCTP Endpoints
- *
+ *
* An endpoint SHOULD transmit reply chunks (e.g., SACK,
* HEARTBEAT ACK, * etc.) to the same destination transport
* address from which it * received the DATA or control chunk
*
* [COOKIE ECHO back to where the INIT ACK came from.]
*/
- if (NULL != chunk) {
+ if (chunk)
retval->transport = chunk->transport;
- }
nodata:
- return retval;
-
-} /* sctp_make_cookie_echo() */
-
+ return retval;
+}
/* 3.3.12 Cookie Acknowledgement (COOKIE ACK) (11):
*
* This chunk is used only during the initialization of an
- * association. It is used to acknowledge the receipt of a COOKIE
+ * association. It is used to acknowledge the receipt of a COOKIE
* ECHO chunk. This chunk MUST precede any DATA or SACK chunk sent
* within the association, but MAY be bundled with one or more DATA
* chunks or SACK chunk in the same SCTP packet.
- *
+ *
* 0 1 2 3
* 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Type = 11 |Chunk Flags | Length = 4 |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
- *
+ *
* Chunk Flags: 8 bits
- *
- *
+ *
* Set to zero on transmit and ignored on receipt.
*/
-sctp_chunk_t *
-sctp_make_cookie_ack(const sctp_association_t *asoc, const sctp_chunk_t *chunk)
+sctp_chunk_t *sctp_make_cookie_ack(const sctp_association_t *asoc,
+ const sctp_chunk_t *chunk)
{
- sctp_chunk_t *retval;
+ sctp_chunk_t *retval;
- retval = sctp_make_chunk(asoc, SCTP_CID_COOKIE_ACK, 0, 0);
+ retval = sctp_make_chunk(asoc, SCTP_CID_COOKIE_ACK, 0, 0);
/* RFC 2960 6.4 Multi-homed SCTP Endpoints
- *
+ *
* An endpoint SHOULD transmit reply chunks (e.g., SACK,
* HEARTBEAT ACK, * etc.) to the same destination transport
* address from which it * received the DATA or control chunk
*
* [COOKIE ACK back to where the COOKIE ECHO came from.]
*/
- if (NULL != retval && NULL != chunk) {
+ if (retval && chunk)
retval->transport = chunk->transport;
- }
-
- return retval;
-} /* sctp_make_cookie_ack() */
+ return retval;
+}
-/*
- * Appendix A: Explicit Congestion Notification:
+/*
+ * Appendix A: Explicit Congestion Notification:
* CWR:
- *
+ *
* RFC 2481 details a specific bit for a sender to send in the header of
* its next outbound TCP segment to indicate to its peer that it has
* reduced its congestion window. This is termed the CWR bit. For
*
* Note: The CWR is considered a Control chunk.
*/
-sctp_chunk_t *
-sctp_make_cwr(const sctp_association_t *asoc, const uint32_t lowest_tsn,
- const sctp_chunk_t *chunk)
+sctp_chunk_t *sctp_make_cwr(const sctp_association_t *asoc,
+ const __u32 lowest_tsn,
+ const sctp_chunk_t *chunk)
{
- sctp_chunk_t *retval;
+ sctp_chunk_t *retval;
sctp_cwrhdr_t cwr;
-
+
cwr.lowest_tsn = htonl(lowest_tsn);
- retval = sctp_make_chunk(asoc, SCTP_CID_ECN_CWR, 0,
- sizeof(sctp_cwrhdr_t));
+ retval = sctp_make_chunk(asoc, SCTP_CID_ECN_CWR, 0,
+ sizeof(sctp_cwrhdr_t));
+
+ if (!retval)
+ goto nodata;
- if (NULL == retval) { goto nodata; }
- retval->subh.ecn_cwr_hdr =
- sctp_addto_chunk(retval, sizeof(cwr), &cwr);
+ retval->subh.ecn_cwr_hdr =
+ sctp_addto_chunk(retval, sizeof(cwr), &cwr);
/* RFC 2960 6.4 Multi-homed SCTP Endpoints
- *
+ *
* An endpoint SHOULD transmit reply chunks (e.g., SACK,
* HEARTBEAT ACK, * etc.) to the same destination transport
* address from which it * received the DATA or control chunk
* [Report a reduced congestion window back to where the ECNE
* came from.]
*/
- if (NULL != chunk) {
+ if (chunk)
retval->transport = chunk->transport;
- }
-
- nodata:
- return retval;
-} /* sctp_make_cwr() */
+nodata:
+ return retval;
+}
/* Make an ECNE chunk. This is a congestion experienced report. */
-sctp_chunk_t *
-sctp_make_ecne(const sctp_association_t *asoc, const uint32_t lowest_tsn)
+sctp_chunk_t *sctp_make_ecne(const sctp_association_t *asoc,
+ const __u32 lowest_tsn)
{
- sctp_chunk_t *retval;
+ sctp_chunk_t *retval;
sctp_ecnehdr_t ecne;
-
+
ecne.lowest_tsn = htonl(lowest_tsn);
- retval = sctp_make_chunk(asoc, SCTP_CID_ECN_ECNE, 0,
- sizeof(sctp_ecnehdr_t));
- if (NULL == retval) { goto nodata; }
- retval->subh.ecne_hdr =
- sctp_addto_chunk(retval, sizeof(ecne), &ecne);
+ retval = sctp_make_chunk(asoc, SCTP_CID_ECN_ECNE, 0,
+ sizeof(sctp_ecnehdr_t));
+ if (!retval)
+ goto nodata;
+ retval->subh.ecne_hdr =
+ sctp_addto_chunk(retval, sizeof(ecne), &ecne);
nodata:
- return retval;
-
-} /* sctp_make_ecne() */
-
+ return retval;
+}
-/* Make a DATA chunk for the given association from the provided
- * parameters. However, do not populate the data payload.
+/* Make a DATA chunk for the given association from the provided
+ * parameters. However, do not populate the data payload.
*/
-sctp_chunk_t *
-sctp_make_datafrag_empty(sctp_association_t *asoc,
- const struct sctp_sndrcvinfo *sinfo,
- int data_len, uint8_t flags, uint16_t ssn)
+sctp_chunk_t *sctp_make_datafrag_empty(sctp_association_t *asoc,
+ const struct sctp_sndrcvinfo *sinfo,
+ int data_len, __u8 flags, __u16 ssn)
{
sctp_chunk_t *retval;
sctp_datahdr_t dp;
int chunk_len;
-
- retval = NULL;
-
+
/* We assign the TSN as LATE as possible, not here when
* creating the chunk.
*/
dp.ppid = htonl(sinfo->sinfo_ppid);
dp.ssn = htons(ssn);
- /* Set the flags for an unordered send. */
- if (sinfo->sinfo_flags & MSG_UNORDERED) {
+ /* Set the flags for an unordered send. */
+ if (sinfo->sinfo_flags & MSG_UNORDERED)
flags |= SCTP_DATA_UNORDERED;
- }
-
+
chunk_len = sizeof(dp) + data_len;
-
retval = sctp_make_chunk(asoc, SCTP_CID_DATA, flags, chunk_len);
- if (NULL == retval) { goto nodata; }
-
+ if (!retval)
+ goto nodata;
+
retval->subh.data_hdr = sctp_addto_chunk(retval, sizeof(dp), &dp);
- memcpy(&retval->sinfo, sinfo, sizeof(struct sctp_sndrcvinfo));
-
+ memcpy(&retval->sinfo, sinfo, sizeof(struct sctp_sndrcvinfo));
+
nodata:
- return retval;
-
-} /* sctp_make_datafrag_empty() */
+ return retval;
+}
/* Make a DATA chunk for the given association. Populate the data
* payload.
*/
-sctp_chunk_t *
-sctp_make_datafrag(sctp_association_t *asoc,
- const struct sctp_sndrcvinfo *sinfo,
- int data_len, const uint8_t *data,
- uint8_t flags, uint16_t ssn)
+sctp_chunk_t *sctp_make_datafrag(sctp_association_t *asoc,
+ const struct sctp_sndrcvinfo *sinfo,
+ int data_len, const __u8 *data,
+ __u8 flags, __u16 ssn)
{
- sctp_chunk_t *retval;
-
- retval = sctp_make_datafrag_empty(asoc, sinfo, data_len, flags, ssn);
- if (NULL != retval) {
- sctp_addto_chunk(retval, data_len, data);
- }
+ sctp_chunk_t *retval;
+
+ retval = sctp_make_datafrag_empty(asoc, sinfo, data_len, flags, ssn);
+ if (retval)
+ sctp_addto_chunk(retval, data_len, data);
- return retval;
-
-} /* sctp_make_datafrag() */
+ return retval;
+}
/* Make a DATA chunk for the given association to ride on stream id
* 'stream', with a payload id of 'payload', and a body of 'data'.
*/
-sctp_chunk_t *
-sctp_make_data(sctp_association_t *asoc,
- const struct sctp_sndrcvinfo *sinfo,
- int data_len, const uint8_t *data)
+sctp_chunk_t *sctp_make_data(sctp_association_t *asoc,
+ const struct sctp_sndrcvinfo *sinfo,
+ int data_len, const __u8 *data)
{
- sctp_chunk_t *retval = NULL;
-
- retval = sctp_make_data_empty(asoc, sinfo, data_len);
-
- if (NULL != retval) {
- sctp_addto_chunk(retval, data_len, data);
- }
+ sctp_chunk_t *retval = NULL;
+ retval = sctp_make_data_empty(asoc, sinfo, data_len);
+ if (retval)
+ sctp_addto_chunk(retval, data_len, data);
return retval;
-
-} /* sctp_make_data() */
+}
/* Make a DATA chunk for the given association to ride on stream id
* 'stream', with a payload id of 'payload', and a body big enough to
* hold 'data_len' octets of data. We use this version when we need
* to build the message AFTER allocating memory.
*/
-sctp_chunk_t *
-sctp_make_data_empty(sctp_association_t *asoc,
- const struct sctp_sndrcvinfo *sinfo,
- int data_len)
+sctp_chunk_t *sctp_make_data_empty(sctp_association_t *asoc,
+ const struct sctp_sndrcvinfo *sinfo,
+ int data_len)
{
- sctp_chunk_t *retval;
- uint16_t ssn;
- uint8_t flags = SCTP_DATA_NOT_FRAG;
-
+ sctp_chunk_t *retval;
+ __u16 ssn;
+ __u8 flags = SCTP_DATA_NOT_FRAG;
+
/* Sockets API Extensions for SCTP 5.2.2
* MSG_UNORDERED - This flag requests the un-ordered delivery of the
- * message. If this flag is clear, the datagram is considered an
- * ordered send and a new ssn is generated. The flags field is set
- * in the inner routine - sctp_make_datafrag_empty().
+ * message. If this flag is clear, the datagram is considered an
+ * ordered send and a new ssn is generated. The flags field is set
+ * in the inner routine - sctp_make_datafrag_empty().
*/
if (sinfo->sinfo_flags & MSG_UNORDERED) {
ssn = 0;
} else {
- ssn = __sctp_association_get_next_ssn(asoc,
+ ssn = __sctp_association_get_next_ssn(asoc,
sinfo->sinfo_stream);
}
-
- retval = sctp_make_datafrag_empty(asoc, sinfo, data_len, flags, ssn);
- if (NULL == retval) { goto nodata; }
-
-nodata:
- return retval;
-} /* sctp_make_data_empty() */
+ return sctp_make_datafrag_empty(asoc, sinfo, data_len, flags, ssn);
+}
/* Create a selective ackowledgement (SACK) for the given
* association. This reports on which TSN's we've seen to date,
* including duplicates and gaps.
*/
-sctp_chunk_t *
-sctp_make_sack(const sctp_association_t *asoc)
+sctp_chunk_t *sctp_make_sack(const sctp_association_t *asoc)
{
- sctp_chunk_t *retval;
- sctp_sackhdr_t sack;
+ sctp_chunk_t *retval;
+ sctp_sackhdr_t sack;
sctp_gap_ack_block_t gab;
- int length;
- uint32_t ctsn;
+ int length;
+ __u32 ctsn;
sctp_tsnmap_iter_t iter;
- uint16_t num_gabs;
- uint16_t num_dup_tsns = asoc->peer.next_dup_tsn;
+ __u16 num_gabs;
+ __u16 num_dup_tsns = asoc->peer.next_dup_tsn;
const sctp_tsnmap_t *map = &asoc->peer.tsn_map;
ctsn = sctp_tsnmap_get_ctsn(map);
/* Count the number of Gap Ack Blocks. */
sctp_tsnmap_iter_init(map, &iter);
- for (num_gabs = 0;
- sctp_tsnmap_next_gap_ack(map, &iter, &gab.start, &gab.end);
+ for (num_gabs = 0;
+ sctp_tsnmap_next_gap_ack(map, &iter, &gab.start, &gab.end);
num_gabs++) {
/* Do nothing. */
}
-
- /* Initialize the SACK header. */
- sack.cum_tsn_ack = htonl(ctsn);
- sack.a_rwnd = htonl(asoc->rwnd);
- sack.num_gap_ack_blocks = htons(num_gabs);
- sack.num_dup_tsns = htons(num_dup_tsns);
+ /* Initialize the SACK header. */
+ sack.cum_tsn_ack = htonl(ctsn);
+ sack.a_rwnd = htonl(asoc->rwnd);
+ sack.num_gap_ack_blocks = htons(num_gabs);
+ sack.num_dup_tsns = htons(num_dup_tsns);
- length = sizeof(sack)
- + sizeof(sctp_gap_ack_block_t) * num_gabs
+ length = sizeof(sack)
+ + sizeof(sctp_gap_ack_block_t) * num_gabs
+ sizeof(sctp_dup_tsn_t) * num_dup_tsns;
-
- /* Create the chunk. */
- retval = sctp_make_chunk(asoc, SCTP_CID_SACK, 0, length);
- if (NULL == retval) { goto nodata; }
+
+ /* Create the chunk. */
+ retval = sctp_make_chunk(asoc, SCTP_CID_SACK, 0, length);
+ if (!retval)
+ goto nodata;
/* RFC 2960 6.4 Multi-homed SCTP Endpoints
- *
+ *
* An endpoint SHOULD transmit reply chunks (e.g., SACK,
* HEARTBEAT ACK, etc.) to the same destination transport
* address from which it received the DATA or control chunk to
* which it is replying. This rule should also be followed if
* the endpoint is bundling DATA chunks together with the
* reply chunk.
- *
+ *
* However, when acknowledging multiple DATA chunks received
* in packets from different source addresses in a single
* SACK, the SACK chunk may be transmitted to one of the
* Perhaps we should remember the last transport we used for a
* SACK and avoid that (if possible) if we have seen any
* duplicates. --piggy]
- *
+ *
* When a receiver of a duplicate DATA chunk sends a SACK to a
* multi- homed endpoint it MAY be beneficial to vary the
* destination address and not use the source address of the
* from a multi-homed endpoint might indicate that the return
* path (as specified in the source address of the DATA chunk)
* for the SACK is broken.
- *
- *
+ *
* [Send to the address from which we last received a DATA chunk.]
*/
retval->transport = asoc->peer.last_data_from;
- retval->subh.sack_hdr =
- sctp_addto_chunk(retval, sizeof(sack), &sack);
-
- /* Put the Gap Ack Blocks into the chunk. */
+ retval->subh.sack_hdr =
+ sctp_addto_chunk(retval, sizeof(sack), &sack);
+
+ /* Put the Gap Ack Blocks into the chunk. */
sctp_tsnmap_iter_init(map, &iter);
while(sctp_tsnmap_next_gap_ack(map, &iter, &gab.start, &gab.end)) {
gab.start = htons(gab.start);
&gab);
}
- /* Register the duplicates. */
+ /* Register the duplicates. */
sctp_addto_chunk(retval,
sizeof(sctp_dup_tsn_t) * num_dup_tsns,
&asoc->peer.dup_tsns);
- nodata:
- return retval;
-} /* sctp_make_sack() */
+nodata:
+ return retval;
+}
-sctp_chunk_t *
-sctp_make_shutdown(const sctp_association_t *asoc)
+sctp_chunk_t *sctp_make_shutdown(const sctp_association_t *asoc)
{
- sctp_chunk_t *retval;
- sctp_shutdownhdr_t shut;
- uint32_t ctsn;
+ sctp_chunk_t *retval;
+ sctp_shutdownhdr_t shut;
+ __u32 ctsn;
ctsn = sctp_tsnmap_get_ctsn(&asoc->peer.tsn_map);
- shut.cum_tsn_ack = htonl(ctsn);
+ shut.cum_tsn_ack = htonl(ctsn);
- retval = sctp_make_chunk(asoc, SCTP_CID_SHUTDOWN, 0,
+ retval = sctp_make_chunk(asoc, SCTP_CID_SHUTDOWN, 0,
sizeof(sctp_shutdownhdr_t));
- if (NULL == retval) { goto nodata; }
-
- retval->subh.shutdown_hdr =
- sctp_addto_chunk(retval, sizeof(shut), &shut);
+ if (!retval)
+ goto nodata;
- nodata:
- return retval;
+ retval->subh.shutdown_hdr =
+ sctp_addto_chunk(retval, sizeof(shut), &shut);
-} /* sctp_make_shutdown() */
+nodata:
+ return retval;
+}
-sctp_chunk_t *
-sctp_make_shutdown_ack(const sctp_association_t *asoc,
- const sctp_chunk_t *chunk)
+sctp_chunk_t *sctp_make_shutdown_ack(const sctp_association_t *asoc,
+ const sctp_chunk_t *chunk)
{
- sctp_chunk_t *retval;
+ sctp_chunk_t *retval;
- retval = sctp_make_chunk(asoc, SCTP_CID_SHUTDOWN_ACK, 0, 0);
+ retval = sctp_make_chunk(asoc, SCTP_CID_SHUTDOWN_ACK, 0, 0);
/* RFC 2960 6.4 Multi-homed SCTP Endpoints
- *
+ *
* An endpoint SHOULD transmit reply chunks (e.g., SACK,
* HEARTBEAT ACK, * etc.) to the same destination transport
* address from which it * received the DATA or control chunk
*
* [ACK back to where the SHUTDOWN came from.]
*/
- if (NULL != retval && NULL != chunk) {
+ if (retval && chunk)
retval->transport = chunk->transport;
- }
- return retval;
-
-} /* sctp_make_shutdown_ack() */
+ return retval;
+}
-sctp_chunk_t *
-sctp_make_shutdown_complete(const sctp_association_t *asoc,
- const sctp_chunk_t *chunk)
+sctp_chunk_t *sctp_make_shutdown_complete(const sctp_association_t *asoc,
+ const sctp_chunk_t *chunk)
{
- sctp_chunk_t *retval;
- uint8_t flags = 0;
+ sctp_chunk_t *retval;
+ __u8 flags = 0;
- /* Maybe set the T-bit if we have no association. */
- flags |= asoc ? 0 : SCTP_CHUNK_FLAG_T;
+ /* Maybe set the T-bit if we have no association. */
+ flags |= asoc ? 0 : SCTP_CHUNK_FLAG_T;
- retval = sctp_make_chunk(asoc, SCTP_CID_SHUTDOWN_COMPLETE, flags, 0);
+ retval = sctp_make_chunk(asoc, SCTP_CID_SHUTDOWN_COMPLETE, flags, 0);
/* RFC 2960 6.4 Multi-homed SCTP Endpoints
- *
+ *
* An endpoint SHOULD transmit reply chunks (e.g., SACK,
* HEARTBEAT ACK, * etc.) to the same destination transport
* address from which it * received the DATA or control chunk
* [Report SHUTDOWN COMPLETE back to where the SHUTDOWN ACK
* came from.]
*/
- if (NULL != retval && NULL != chunk) {
+ if (retval && chunk)
retval->transport = chunk->transport;
- }
return retval;
-
-} /* sctp_make_shutdown_complete() */
+}
/* Create an ABORT. Note that we set the T bit if we have no
* association.
*/
-sctp_chunk_t *
-sctp_make_abort(const sctp_association_t *asoc, const sctp_chunk_t *chunk,
- const size_t hint)
+sctp_chunk_t *sctp_make_abort(const sctp_association_t *asoc,
+ const sctp_chunk_t *chunk,
+ const size_t hint)
{
- sctp_chunk_t *retval;
- uint8_t flags = 0;
+ sctp_chunk_t *retval;
+ __u8 flags = 0;
- /* Maybe set the T-bit if we have no association. */
- flags |= asoc ? 0 : SCTP_CHUNK_FLAG_T;
+ /* Maybe set the T-bit if we have no association. */
+ flags |= asoc ? 0 : SCTP_CHUNK_FLAG_T;
- retval = sctp_make_chunk(asoc, SCTP_CID_ABORT, flags, hint);
+ retval = sctp_make_chunk(asoc, SCTP_CID_ABORT, flags, hint);
/* RFC 2960 6.4 Multi-homed SCTP Endpoints
- *
+ *
* An endpoint SHOULD transmit reply chunks (e.g., SACK,
* HEARTBEAT ACK, * etc.) to the same destination transport
* address from which it * received the DATA or control chunk
*
* [ABORT back to where the offender came from.]
*/
- if (NULL != retval && NULL != chunk) {
+ if (retval && chunk)
retval->transport = chunk->transport;
- }
-
- return retval;
-} /* sctp_make_abort() */
+ return retval;
+}
-/* Helper to create ABORT with a NO_USER_DATA error. */
-sctp_chunk_t *
-sctp_make_abort_no_data(const sctp_association_t *asoc,
- const sctp_chunk_t *chunk, uint32_t tsn)
+/* Helper to create ABORT with a NO_USER_DATA error. */
+sctp_chunk_t *sctp_make_abort_no_data(const sctp_association_t *asoc,
+ const sctp_chunk_t *chunk, __u32 tsn)
{
- sctp_chunk_t *retval;
- uint32_t payload;
+ sctp_chunk_t *retval;
+ __u32 payload;
- retval = sctp_make_abort(asoc, chunk, sizeof(sctp_errhdr_t)
+ retval = sctp_make_abort(asoc, chunk, sizeof(sctp_errhdr_t)
+ sizeof(tsn));
- if (!retval) { goto no_mem; }
+ if (!retval)
+ goto no_mem;
- /* Put the tsn back into network byte order. */
+ /* Put the tsn back into network byte order. */
payload = htonl(tsn);
- sctp_init_cause(retval, SCTP_ERROR_NO_DATA, (const void *)&payload,
+ sctp_init_cause(retval, SCTP_ERROR_NO_DATA, (const void *)&payload,
sizeof(payload));
/* RFC 2960 6.4 Multi-homed SCTP Endpoints
- *
+ *
* An endpoint SHOULD transmit reply chunks (e.g., SACK,
* HEARTBEAT ACK, * etc.) to the same destination transport
* address from which it * received the DATA or control chunk
*
* [ABORT back to where the offender came from.]
*/
- if (chunk) {
+ if (chunk)
retval->transport = chunk->transport;
- }
no_mem:
- return retval;
-
-} /* sctp_make_abort_no_data() */
-
+ return retval;
+}
-/* Make a HEARTBEAT chunk. */
-sctp_chunk_t *
-sctp_make_heartbeat(const sctp_association_t *asoc,
- const sctp_transport_t *transport,
- const void *payload, const size_t paylen)
+/* Make a HEARTBEAT chunk. */
+sctp_chunk_t *sctp_make_heartbeat(const sctp_association_t *asoc,
+ const sctp_transport_t *transport,
+ const void *payload, const size_t paylen)
{
- sctp_chunk_t *retval;
- retval = sctp_make_chunk(asoc, SCTP_CID_HEARTBEAT, 0, paylen);
- if (NULL == retval) { goto nodata; }
+ sctp_chunk_t *retval = sctp_make_chunk(asoc, SCTP_CID_HEARTBEAT,
+ 0, paylen);
+
+ if (!retval)
+ goto nodata;
/* Cast away the 'const', as this is just telling the chunk
* what transport it belongs to.
*/
- retval->transport = (sctp_transport_t *)transport;
+ retval->transport = (sctp_transport_t *) transport;
retval->subh.hbs_hdr = sctp_addto_chunk(retval, paylen, payload);
- nodata:
- return retval;
-} /* sctp_make_heartbeat() */
+nodata:
+ return retval;
+}
-sctp_chunk_t *
-sctp_make_heartbeat_ack(const sctp_association_t *asoc,
- const sctp_chunk_t *chunk,
- const void *payload, const size_t paylen)
+sctp_chunk_t *sctp_make_heartbeat_ack(const sctp_association_t *asoc,
+ const sctp_chunk_t *chunk,
+ const void *payload, const size_t paylen)
{
- sctp_chunk_t *retval;
- retval = sctp_make_chunk(asoc, SCTP_CID_HEARTBEAT_ACK, 0, paylen);
- if (NULL == retval) { goto nodata; }
+ sctp_chunk_t *retval = sctp_make_chunk(asoc, SCTP_CID_HEARTBEAT_ACK,
+ 0, paylen);
+
+ if (!retval)
+ goto nodata;
retval->subh.hbs_hdr = sctp_addto_chunk(retval, paylen, payload);
/* RFC 2960 6.4 Multi-homed SCTP Endpoints
- *
+ *
* An endpoint SHOULD transmit reply chunks (e.g., SACK,
* HEARTBEAT ACK, * etc.) to the same destination transport
* address from which it * received the DATA or control chunk
*
* [HBACK back to where the HEARTBEAT came from.]
*/
- if (NULL != chunk) {
+ if (chunk)
retval->transport = chunk->transport;
- }
-
- nodata:
- return retval;
-
-} /* sctp_make_heartbeat_ack() */
-
+nodata:
+ return retval;
+}
-/* Create an Operation Error chunk. */
-sctp_chunk_t *
-sctp_make_op_error(const sctp_association_t *asoc, const sctp_chunk_t *chunk,
- uint16_t cause_code, const void *payload, size_t paylen)
+/* Create an Operation Error chunk. */
+sctp_chunk_t *sctp_make_op_error(const sctp_association_t *asoc,
+ const sctp_chunk_t *chunk,
+ __u16 cause_code, const void *payload,
+ size_t paylen)
{
- sctp_chunk_t *retval;
-
- retval = sctp_make_chunk(asoc, SCTP_CID_ERROR, 0,
- sizeof(sctp_errhdr_t) + paylen);
- if (NULL == retval) { goto nodata; }
-
- sctp_init_cause(retval, cause_code, payload, paylen);
+ sctp_chunk_t *retval = sctp_make_chunk(asoc, SCTP_CID_ERROR, 0,
+ sizeof(sctp_errhdr_t) + paylen);
+
+ if (!retval)
+ goto nodata;
+ sctp_init_cause(retval, cause_code, payload, paylen);
/* RFC 2960 6.4 Multi-homed SCTP Endpoints
- *
+ *
* An endpoint SHOULD transmit reply chunks (e.g., SACK,
* HEARTBEAT ACK, * etc.) to the same destination transport
* address from which it * received the DATA or control chunk
* to which it is replying.
- *
*/
- if (NULL != chunk) {
+ if (chunk)
retval->transport = chunk->transport;
- }
-
- nodata:
- return retval;
-} /* sctp_make_op_error() */
+nodata:
+ return retval;
+}
/********************************************************************
* 2nd Level Abstractions
********************************************************************/
-
-/* Turn an skb into a chunk.
+/* Turn an skb into a chunk.
* FIXME: Eventually move the structure directly inside the skb->cb[].
*/
-sctp_chunk_t *
-sctp_chunkify(struct sk_buff *skb, const sctp_association_t *asoc,
- struct sock *sk)
+sctp_chunk_t *sctp_chunkify(struct sk_buff *skb, const sctp_association_t *asoc,
+ struct sock *sk)
{
- sctp_chunk_t *retval;
-
- retval = t_new(sctp_chunk_t, GFP_ATOMIC);
- if (NULL == retval) { goto nodata; }
- memset(retval, 0, sizeof(sctp_chunk_t));
-
- if (!sk) {
- SCTP_DEBUG_PRINTK("chunkifying skb %p w/o an sk\n", skb);
- }
-
- retval->skb = skb;
- retval->asoc = (sctp_association_t *)asoc;
- retval->num_times_sent = 0;
- retval->has_tsn = 0;
- retval->rtt_in_progress = 0;
- retval->sent_at = jiffies;
- retval->singleton = 1;
- retval->end_of_packet = 0;
- retval->ecn_ce_done = 0;
- retval->pdiscard = 0;
-
- /* sctpimpguide-05.txt Section 2.8.2
- * M1) Each time a new DATA chunk is transmitted
+ sctp_chunk_t *retval = t_new(sctp_chunk_t, GFP_ATOMIC);
+
+ if (!retval)
+ goto nodata;
+ memset(retval, 0, sizeof(sctp_chunk_t));
+
+ if (!sk) {
+ SCTP_DEBUG_PRINTK("chunkifying skb %p w/o an sk\n", skb);
+ }
+
+ retval->skb = skb;
+ retval->asoc = (sctp_association_t *) asoc;
+ retval->num_times_sent = 0;
+ retval->has_tsn = 0;
+ retval->rtt_in_progress = 0;
+ retval->sent_at = jiffies;
+ retval->singleton = 1;
+ retval->end_of_packet = 0;
+ retval->ecn_ce_done = 0;
+ retval->pdiscard = 0;
+
+ /* sctpimpguide-05.txt Section 2.8.2
+ * M1) Each time a new DATA chunk is transmitted
* set the 'TSN.Missing.Report' count for that TSN to 0. The
- * 'TSN.Missing.Report' count will be used to determine missing chunks
+ * 'TSN.Missing.Report' count will be used to determine missing chunks
* and when to fast retransmit.
*/
retval->tsn_missing_report = 0;
retval->tsn_gap_acked = 0;
retval->fast_retransmit = 0;
- /* Polish the bead hole. */
- INIT_LIST_HEAD(&retval->transmitted_list);
+ /* Polish the bead hole. */
+ INIT_LIST_HEAD(&retval->transmitted_list);
INIT_LIST_HEAD(&retval->frag_list);
SCTP_DBG_OBJCNT_INC(chunk);
- nodata:
- return retval;
-
-} /* sctp_chunkify() */
+nodata:
+ return retval;
+}
/* Set chunk->source based on the IP header in chunk->skb. */
-void
-sctp_init_source(sctp_chunk_t *chunk)
+void sctp_init_source(sctp_chunk_t *chunk)
{
- sockaddr_storage_t *source;
- struct sk_buff *skb;
- struct sctphdr *sh;
- struct iphdr *ih4;
+ sockaddr_storage_t *source;
+ struct sk_buff *skb;
+ struct sctphdr *sh;
+ struct iphdr *ih4;
struct ipv6hdr *ih6;
-
- source = &chunk->source;
- skb = chunk->skb;
- ih4 = skb->nh.iph;
- ih6 = skb->nh.ipv6h;
- sh = chunk->sctp_hdr;
+
+ source = &chunk->source;
+ skb = chunk->skb;
+ ih4 = skb->nh.iph;
+ ih6 = skb->nh.ipv6h;
+ sh = chunk->sctp_hdr;
switch (ih4->version) {
case 4:
source->v4.sin_port = ntohs(sh->source);
source->v4.sin_addr.s_addr = ih4->saddr;
break;
+
case 6:
SCTP_V6(
source->v6.sin6_family = AF_INET6;
source->v6.sin6_port = ntohs(sh->source);
source->v6.sin6_addr = ih6->saddr;
- /* FIXME: What do we do with scope, etc. ?
- */
+ /* FIXME: What do we do with scope, etc. ? */
break;
)
+
default:
/* This is a bogus address type, just bail. */
break;
- } /* switch (IP version of header) */
-
-} /* sctp_init_source() */
+ };
+}
/* Extract the source address from a chunk. */
-const sockaddr_storage_t *
-sctp_source(const sctp_chunk_t *chunk)
+const sockaddr_storage_t *sctp_source(const sctp_chunk_t *chunk)
{
-
/* If we have a known transport, use that. */
if (chunk->transport) {
return &chunk->transport->ipaddr;
- } else {
- /* Otherwise, extract it from the IP header. */
+ } else {
+ /* Otherwise, extract it from the IP header. */
return &chunk->source;
}
-
-} /* sctp_source() */
+}
/* Create a new chunk, setting the type and flags headers from the
* arguments, reserving enough space for a 'paylen' byte payload.
*/
-sctp_chunk_t *
-sctp_make_chunk(const sctp_association_t *asoc, uint8_t type, uint8_t flags,
- int paylen)
+sctp_chunk_t *sctp_make_chunk(const sctp_association_t *asoc,
+ __u8 type, __u8 flags, int paylen)
{
- sctp_chunk_t *retval;
- sctp_chunkhdr_t *chunk_hdr;
- struct sk_buff *skb;
- struct sock *sk;
-
- skb = dev_alloc_skb(WORD_ROUND(sizeof(sctp_chunkhdr_t) + paylen));
- if (NULL == skb) { goto nodata; }
-
- /* Make room for the chunk header. */
- chunk_hdr = (sctp_chunkhdr_t *)skb_put(skb, sizeof(sctp_chunkhdr_t));
- skb_pull(skb, sizeof(sctp_chunkhdr_t));
-
- chunk_hdr->type = type;
- chunk_hdr->flags = flags;
- chunk_hdr->length = htons(sizeof(sctp_chunkhdr_t));
-
- /* Move the data pointer back up to the start of the chunk. */
- skb_push(skb, sizeof(sctp_chunkhdr_t));
-
- sk = asoc ? asoc->base.sk : NULL;
- retval = sctp_chunkify(skb, asoc, sk);
- if (NULL == retval) {
+ sctp_chunk_t *retval;
+ sctp_chunkhdr_t *chunk_hdr;
+ struct sk_buff *skb;
+ struct sock *sk;
+
+ skb = dev_alloc_skb(WORD_ROUND(sizeof(sctp_chunkhdr_t) + paylen));
+ if (!skb)
+ goto nodata;
+
+ /* Make room for the chunk header. */
+ chunk_hdr = (sctp_chunkhdr_t *)skb_put(skb, sizeof(sctp_chunkhdr_t));
+ skb_pull(skb, sizeof(sctp_chunkhdr_t));
+
+ chunk_hdr->type = type;
+ chunk_hdr->flags = flags;
+ chunk_hdr->length = htons(sizeof(sctp_chunkhdr_t));
+
+ /* Move the data pointer back up to the start of the chunk. */
+ skb_push(skb, sizeof(sctp_chunkhdr_t));
+
+ sk = asoc ? asoc->base.sk : NULL;
+ retval = sctp_chunkify(skb, asoc, sk);
+ if (!retval) {
dev_kfree_skb(skb);
- goto nodata;
- }
-
- retval->chunk_hdr = chunk_hdr;
- retval->chunk_end = ((uint8_t *)chunk_hdr)
- + sizeof(sctp_chunkhdr_t);
+ goto nodata;
+ }
+
+ retval->chunk_hdr = chunk_hdr;
+ retval->chunk_end = ((__u8 *)chunk_hdr) + sizeof(sctp_chunkhdr_t);
/* Set the skb to the belonging sock for accounting. */
skb->sk = sk;
- return retval;
- nodata:
- return NULL;
+ return retval;
-} /* sctp_make_chunk() */
+nodata:
+ return NULL;
+}
/* Release the memory occupied by a chunk. */
-void
-sctp_free_chunk(sctp_chunk_t *chunk)
+void sctp_free_chunk(sctp_chunk_t *chunk)
{
- /* Make sure that we are not on any list. */
- skb_unlink((struct sk_buff *)chunk);
- list_del(&chunk->transmitted_list);
+ /* Make sure that we are not on any list. */
+ skb_unlink((struct sk_buff *) chunk);
+ list_del(&chunk->transmitted_list);
/* Free the chunk skb data and the SCTP_chunk stub itself. */
- dev_kfree_skb(chunk->skb);
+ dev_kfree_skb(chunk->skb);
kfree(chunk);
SCTP_DBG_OBJCNT_DEC(chunk);
-
-} /* sctp_free_chunk() */
+}
/* Do a deep copy of a chunk. */
-sctp_chunk_t *
-sctp_copy_chunk(sctp_chunk_t *chunk, const int priority)
+sctp_chunk_t *sctp_copy_chunk(sctp_chunk_t *chunk, const int priority)
{
- sctp_chunk_t *retval;
- long offset;
+ sctp_chunk_t *retval;
+ long offset;
- retval = t_new(sctp_chunk_t, priority);
- if (NULL == retval) { goto nodata; }
+ retval = t_new(sctp_chunk_t, priority);
+ if (!retval)
+ goto nodata;
- /* Do the shallow copy. */
- *retval = *chunk;
+ /* Do the shallow copy. */
+ *retval = *chunk;
- /* Make sure that the copy does NOT think it is on any lists. */
- retval->next = NULL;
- retval->prev = NULL;
- retval->list = NULL;
- INIT_LIST_HEAD(&retval->transmitted_list);
+ /* Make sure that the copy does NOT think it is on any lists. */
+ retval->next = NULL;
+ retval->prev = NULL;
+ retval->list = NULL;
+ INIT_LIST_HEAD(&retval->transmitted_list);
INIT_LIST_HEAD(&retval->frag_list);
- /* Now we copy the deep structure. */
- retval->skb = skb_copy(chunk->skb, priority);
- if (NULL == retval->skb) {
- kfree(retval);
- goto nodata;
- }
-
- /* Move the copy headers to point into the new skb. */
- offset = ((uint8_t *)retval->skb->head)
- - ((uint8_t *)chunk->skb->head);
-
- if (NULL != retval->param_hdr.v) {
- retval->param_hdr.v += offset;
- }
- if (NULL != retval->subh.v) {
- (retval->subh.v) += offset;
- }
- if (NULL != retval->chunk_end) {
- ((uint8_t *)retval->chunk_end) += offset;
- }
- if (NULL != retval->chunk_hdr) {
- ((uint8_t *)retval->chunk_hdr) += offset;
- }
- if (NULL != retval->sctp_hdr) {
- ((uint8_t *)retval->sctp_hdr) += offset;
- }
- SCTP_DBG_OBJCNT_INC(chunk);
- return retval;
- nodata:
- return NULL;
+ /* Now we copy the deep structure. */
+ retval->skb = skb_copy(chunk->skb, priority);
+ if (!retval->skb) {
+ kfree(retval);
+ goto nodata;
+ }
-} /* sctp_copy_chunk() */
+ /* Move the copy headers to point into the new skb. */
+ offset = ((__u8 *)retval->skb->head)
+ - ((__u8 *)chunk->skb->head);
+
+ if (retval->param_hdr.v)
+ retval->param_hdr.v += offset;
+ if (retval->subh.v)
+ retval->subh.v += offset;
+ if (retval->chunk_end)
+ ((__u8 *) retval->chunk_end) += offset;
+ if (retval->chunk_hdr)
+ ((__u8 *) retval->chunk_hdr) += offset;
+ if (retval->sctp_hdr)
+ ((__u8 *) retval->sctp_hdr) += offset;
+ SCTP_DBG_OBJCNT_INC(chunk);
+ return retval;
+nodata:
+ return NULL;
+}
/* Append bytes to the end of a chunk. Will panic if chunk is not big
* enough.
*/
-void *
-sctp_addto_chunk(sctp_chunk_t *chunk, int len, const void *data)
+void *sctp_addto_chunk(sctp_chunk_t *chunk, int len, const void *data)
{
- void *target;
- void *padding;
+ void *target;
+ void *padding;
int chunklen = ntohs(chunk->chunk_hdr->length);
int padlen = chunklen % 4;
padding = skb_put(chunk->skb, padlen);
- target = skb_put(chunk->skb, len);
-
+ target = skb_put(chunk->skb, len);
+
memset(padding, 0, padlen);
- memcpy(target, data, len);
- /* Adjust the chunk length field. */
- chunk->chunk_hdr->length = htons(chunklen + padlen + len);
- chunk->chunk_end = chunk->skb->tail;
+ memcpy(target, data, len);
- return target;
+ /* Adjust the chunk length field. */
+ chunk->chunk_hdr->length = htons(chunklen + padlen + len);
+ chunk->chunk_end = chunk->skb->tail;
-} /* sctp_addto_chunk() */
+ return target;
+}
/* Append bytes from user space to the end of a chunk. Will panic if
* chunk is not big enough.
* Returns a kernel err value.
*/
-int
-sctp_user_addto_chunk(sctp_chunk_t *chunk, int len, struct iovec *data)
+int sctp_user_addto_chunk(sctp_chunk_t *chunk, int len, struct iovec *data)
{
- uint8_t *target;
- int err = 0;
-
- /* Make room in chunk for data */
+ __u8 *target;
+ int err = 0;
+
+ /* Make room in chunk for data. */
target = skb_put(chunk->skb, len);
/* Copy data (whole iovec) into chunk */
- if ((err = memcpy_fromiovec(target, data, len))) {
+ if ((err = memcpy_fromiovec(target, data, len)))
goto out;
- }
-
- /* Adjust the chunk length field. */
- chunk->chunk_hdr->length =
- htons(ntohs(chunk->chunk_hdr->length) + len);
- chunk->chunk_end = chunk->skb->tail;
-out:
- return err;
+ /* Adjust the chunk length field. */
+ chunk->chunk_hdr->length =
+ htons(ntohs(chunk->chunk_hdr->length) + len);
+ chunk->chunk_end = chunk->skb->tail;
-} /* sctp_user_addto_chunk() */
+out:
+ return err;
+}
/* Helper function to assign a TSN if needed. This assumes that both
- * the data_hdr and association have already been assigned.
+ * the data_hdr and association have already been assigned.
*/
-void
-sctp_chunk_assign_tsn(sctp_chunk_t *chunk)
+void sctp_chunk_assign_tsn(sctp_chunk_t *chunk)
{
if (!chunk->has_tsn) {
/* This is the last possible instant to
* assign a TSN.
*/
- chunk->subh.data_hdr->tsn =
+ chunk->subh.data_hdr->tsn =
htonl(__sctp_association_get_next_tsn(chunk->asoc));
chunk->has_tsn = 1;
}
-
}
-
/* Create a CLOSED association to use with an incoming packet. */
-sctp_association_t *
-sctp_make_temp_asoc(const sctp_endpoint_t *ep, sctp_chunk_t *chunk,
- int priority)
+sctp_association_t *sctp_make_temp_asoc(const sctp_endpoint_t *ep,
+ sctp_chunk_t *chunk,
+ int priority)
{
- sctp_association_t *asoc;
+ sctp_association_t *asoc;
sctp_scope_t scope;
- /* Create the bare association. */
+ /* Create the bare association. */
scope = sctp_scope(sctp_source(chunk));
- asoc = sctp_association_new(ep, ep->base.sk, scope, priority);
- if (NULL == asoc) { goto nodata; }
-
- /* Create an entry for the source address of the packet. */
- switch (chunk->skb->nh.iph->version) {
- case 4:
- asoc->c.peer_addr.v4.sin_family = AF_INET;
- asoc->c.peer_addr.v4.sin_port = ntohs(chunk->sctp_hdr->source);
- asoc->c.peer_addr.v4.sin_addr.s_addr =
+ asoc = sctp_association_new(ep, ep->base.sk, scope, priority);
+ if (!asoc)
+ goto nodata;
+
+ /* Create an entry for the source address of the packet. */
+ switch (chunk->skb->nh.iph->version) {
+ case 4:
+ asoc->c.peer_addr.v4.sin_family = AF_INET;
+ asoc->c.peer_addr.v4.sin_port = ntohs(chunk->sctp_hdr->source);
+ asoc->c.peer_addr.v4.sin_addr.s_addr =
chunk->skb->nh.iph->saddr;
break;
- case 6:
- asoc->c.peer_addr.v6.sin6_family = AF_INET6;
- asoc->c.peer_addr.v6.sin6_port
+
+ case 6:
+ asoc->c.peer_addr.v6.sin6_family = AF_INET6;
+ asoc->c.peer_addr.v6.sin6_port
= ntohs(chunk->sctp_hdr->source);
- asoc->c.peer_addr.v6.sin6_flowinfo = 0; /* BUG BUG BUG */
- asoc->c.peer_addr.v6.sin6_addr = chunk->skb->nh.ipv6h->saddr;
- asoc->c.peer_addr.v6.sin6_scope_id = 0; /* BUG BUG BUG */
- break;
+ asoc->c.peer_addr.v6.sin6_flowinfo = 0; /* BUG BUG BUG */
+ asoc->c.peer_addr.v6.sin6_addr = chunk->skb->nh.ipv6h->saddr;
+ asoc->c.peer_addr.v6.sin6_scope_id = 0; /* BUG BUG BUG */
+ break;
+
default:
- /* Yikes! I never heard of this kind of address. */
+ /* Yikes! I never heard of this kind of address. */
goto fail;
- }
+ };
nodata:
- return asoc;
+ return asoc;
+
fail:
sctp_association_free(asoc);
return NULL;
-
-} /* sctp_make_temp_asoc() */
-
+}
/* Build a cookie representing asoc.
* This INCLUDES the param header needed to put the cookie in the INIT ACK.
*/
-sctp_cookie_param_t *
-sctp_pack_cookie(const sctp_endpoint_t *ep, const sctp_association_t *asoc,
- const sctp_chunk_t *init_chunk, int *cookie_len,
- const uint8_t *raw_addrs, int addrs_len)
-
+sctp_cookie_param_t *sctp_pack_cookie(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_chunk_t *init_chunk,
+ int *cookie_len,
+ const __u8 *raw_addrs, int addrs_len)
{
- sctp_cookie_param_t *retval;
+ sctp_cookie_param_t *retval;
sctp_signed_cookie_t *cookie;
- int headersize, bodysize;
+ int headersize, bodysize;
- headersize = sizeof(sctp_paramhdr_t) + SCTP_SECRET_SIZE;
- bodysize = sizeof(sctp_cookie_t)
+ headersize = sizeof(sctp_paramhdr_t) + SCTP_SECRET_SIZE;
+ bodysize = sizeof(sctp_cookie_t)
+ ntohs(init_chunk->chunk_hdr->length) + addrs_len;
-
/* Pad out the cookie to a multiple to make the signature
* functions simpler to write.
*/
- if (bodysize % SCTP_COOKIE_MULTIPLE) {
+ if (bodysize % SCTP_COOKIE_MULTIPLE)
bodysize += SCTP_COOKIE_MULTIPLE
- (bodysize % SCTP_COOKIE_MULTIPLE);
- }
- *cookie_len = headersize + bodysize;
-
- retval = (sctp_cookie_param_t *)
- kmalloc(*cookie_len, GFP_ATOMIC);
+ *cookie_len = headersize + bodysize;
- if (NULL == retval) {
- *cookie_len = 0;
+ retval = (sctp_cookie_param_t *)
+ kmalloc(*cookie_len, GFP_ATOMIC);
+ if (!retval) {
+ *cookie_len = 0;
goto nodata;
- }
+ }
/* Clear this memory since we are sending this data structure
* out on the network.
*/
-
memset(retval, 0x00, *cookie_len);
+ cookie = (sctp_signed_cookie_t *) retval->body;
- cookie = (sctp_signed_cookie_t *)retval->body;
-
- /* Set up the parameter header. */
- retval->p.type = SCTP_PARAM_STATE_COOKIE;
- retval->p.length = htons(*cookie_len);
+ /* Set up the parameter header. */
+ retval->p.type = SCTP_PARAM_STATE_COOKIE;
+ retval->p.length = htons(*cookie_len);
- /* Copy the cookie part of the association itself. */
- cookie->c = asoc->c;
+ /* Copy the cookie part of the association itself. */
+ cookie->c = asoc->c;
/* Save the raw address list length in the cookie. */
cookie->c.raw_addr_list_len = addrs_len;
- /* Set an expiration time for the cookie. */
- do_gettimeofday(&cookie->c.expiration);
- tv_add(&asoc->cookie_life, &cookie->c.expiration);
-
- /* Copy the peer's init packet. */
- memcpy(&cookie->c.peer_init[0], init_chunk->chunk_hdr,
- ntohs(init_chunk->chunk_hdr->length));
+ /* Set an expiration time for the cookie. */
+ do_gettimeofday(&cookie->c.expiration);
+ tv_add(&asoc->cookie_life, &cookie->c.expiration);
+
+ /* Copy the peer's init packet. */
+ memcpy(&cookie->c.peer_init[0], init_chunk->chunk_hdr,
+ ntohs(init_chunk->chunk_hdr->length));
/* Copy the raw local address list of the association. */
- memcpy((uint8_t *)&cookie->c.peer_init[0] +
- ntohs(init_chunk->chunk_hdr->length), raw_addrs,
- addrs_len);
-
- /* Sign the message. */
- sctp_hash_digest(ep->secret_key[ep->current_key], SCTP_SECRET_SIZE,
- (uint8_t *)&cookie->c, bodysize, cookie->signature);
-
-nodata:
- return retval;
+ memcpy((__u8 *)&cookie->c.peer_init[0] +
+ ntohs(init_chunk->chunk_hdr->length), raw_addrs,
+ addrs_len);
-} /* sctp_pack_cookie() */
+ /* Sign the message. */
+ sctp_hash_digest(ep->secret_key[ep->current_key], SCTP_SECRET_SIZE,
+ (__u8 *) &cookie->c, bodysize, cookie->signature);
+
+nodata:
+ return retval;
+}
/* Unpack the cookie from COOKIE ECHO chunk, recreating the association. */
-sctp_association_t *
-sctp_unpack_cookie(const sctp_endpoint_t *ep, const sctp_association_t *asoc,
- sctp_chunk_t *chunk, int priority, int *error)
+sctp_association_t *sctp_unpack_cookie(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ sctp_chunk_t *chunk, int priority,
+ int *error)
{
- sctp_association_t *retval = NULL;
- sctp_signed_cookie_t *cookie;
- sctp_cookie_t *bear_cookie;
- int headersize, bodysize;
+ sctp_association_t *retval = NULL;
+ sctp_signed_cookie_t *cookie;
+ sctp_cookie_t *bear_cookie;
+ int headersize, bodysize;
int fixed_size, var_size1, var_size2, var_size3;
- uint8_t digest_buf[SCTP_SIGNATURE_SIZE];
- int secret;
+ __u8 digest_buf[SCTP_SIGNATURE_SIZE];
+ int secret;
sctp_scope_t scope;
- uint8_t *raw_addr_list;
-
- headersize = sizeof(sctp_chunkhdr_t) + SCTP_SECRET_SIZE;
- bodysize = ntohs(chunk->chunk_hdr->length) - headersize;
+ __u8 *raw_addr_list;
+
+ headersize = sizeof(sctp_chunkhdr_t) + SCTP_SECRET_SIZE;
+ bodysize = ntohs(chunk->chunk_hdr->length) - headersize;
fixed_size = headersize + sizeof(sctp_cookie_t);
- /* Verify that the chunk looks like it even has a cookie.
+ /* Verify that the chunk looks like it even has a cookie.
* There must be enough room for our cookie and our peer's
* INIT chunk.
*/
- if (ntohs(chunk->chunk_hdr->length) <
- (fixed_size + sizeof(sctp_chunkhdr_t))) {
+ if (ntohs(chunk->chunk_hdr->length) <
+ (fixed_size + sizeof(sctp_chunkhdr_t)))
goto malformed;
- }
/* Verify that the cookie has been padded out. */
- if (bodysize % SCTP_COOKIE_MULTIPLE) {
+ if (bodysize % SCTP_COOKIE_MULTIPLE)
goto malformed;
- }
-
/* Process the cookie. */
- cookie = chunk->subh.cookie_hdr;
- bear_cookie = &cookie->c;
+ cookie = chunk->subh.cookie_hdr;
+ bear_cookie = &cookie->c;
var_size1 = ntohs(chunk->chunk_hdr->length) - fixed_size;
var_size2 = ntohs(bear_cookie->peer_init->chunk_hdr.length);
var_size3 = bear_cookie->raw_addr_list_len;
-
-
- /* Check the signature. */
- secret = ep->current_key;
- sctp_hash_digest(ep->secret_key[secret], SCTP_SECRET_SIZE,
- (uint8_t *)bear_cookie, bodysize,
- digest_buf);
- if (memcmp(digest_buf, cookie->signature,
- SCTP_SIGNATURE_SIZE)) {
-
- /* Try the previous key. */
- secret = ep->last_key;
- sctp_hash_digest(ep->secret_key[secret], SCTP_SECRET_SIZE,
- (uint8_t *)bear_cookie, bodysize, digest_buf);
- if (memcmp(digest_buf, cookie->signature,
- SCTP_SIGNATURE_SIZE)) {
- /* Yikes! Still bad signature! */
- *error = -SCTP_IERROR_BAD_SIG;
+
+ /* Check the signature. */
+ secret = ep->current_key;
+ sctp_hash_digest(ep->secret_key[secret], SCTP_SECRET_SIZE,
+ (__u8 *) bear_cookie, bodysize,
+ digest_buf);
+ if (memcmp(digest_buf, cookie->signature, SCTP_SIGNATURE_SIZE)) {
+ /* Try the previous key. */
+ secret = ep->last_key;
+ sctp_hash_digest(ep->secret_key[secret], SCTP_SECRET_SIZE,
+ (__u8 *) bear_cookie, bodysize, digest_buf);
+ if (memcmp(digest_buf, cookie->signature, SCTP_SIGNATURE_SIZE)) {
+ /* Yikes! Still bad signature! */
+ *error = -SCTP_IERROR_BAD_SIG;
goto fail;
- }
- } /* if (the current secret failed) */
+ }
+ }
/* Check to see if the cookie is stale. If there is already
* an association, there is no need to check cookie's expiration
* for init collision case of lost COOKIE ACK.
*/
- if (!asoc && tv_lt(bear_cookie->expiration, chunk->skb->stamp)) {
- *error = -SCTP_IERROR_STALE_COOKIE;
+ if (!asoc && tv_lt(bear_cookie->expiration, chunk->skb->stamp)) {
+ *error = -SCTP_IERROR_STALE_COOKIE;
goto fail;
- }
+ }
- /* Make a new base association. */
+ /* Make a new base association. */
scope = sctp_scope(sctp_source(chunk));
- retval = sctp_association_new(ep, ep->base.sk, scope, priority);
- if (NULL == retval) {
- *error = -SCTP_IERROR_NOMEM;
+ retval = sctp_association_new(ep, ep->base.sk, scope, priority);
+ if (!retval) {
+ *error = -SCTP_IERROR_NOMEM;
goto fail;
- }
+ }
/* Set up our peer's port number. */
retval->peer.port = ntohs(chunk->sctp_hdr->source);
- /* Populate the association from the cookie. */
- retval->c = *bear_cookie;
- /* Build the bind address list based on the cookie */
- raw_addr_list = (uint8_t *)bear_cookie +
+ /* Populate the association from the cookie. */
+ retval->c = *bear_cookie;
+
+ /* Build the bind address list based on the cookie. */
+ raw_addr_list = (__u8 *) bear_cookie +
sizeof(sctp_cookie_t) + var_size2;
- if (sctp_raw_to_bind_addrs(&retval->base.bind_addr, raw_addr_list,
- var_size3, retval->base.bind_addr.port,
+ if (sctp_raw_to_bind_addrs(&retval->base.bind_addr, raw_addr_list,
+ var_size3, retval->base.bind_addr.port,
priority)) {
*error = -SCTP_IERROR_NOMEM;
goto fail;
}
retval->next_tsn = retval->c.initial_tsn;
- retval->ctsn_ack_point = retval->next_tsn - 1;
+ retval->ctsn_ack_point = retval->next_tsn - 1;
/* The INIT stuff will be done by the side effects. */
+ return retval;
- return retval;
- fail:
- if (retval) {
+fail:
+ if (retval)
sctp_association_free(retval);
- }
return NULL;
- malformed:
+malformed:
/* Yikes! The packet is either corrupt or deliberately
* malformed.
*/
*error = -SCTP_IERROR_MALFORMED;
goto fail;
-
-} /* sctp_unpack_cookie() */
-
+}
/********************************************************************
* 3rd Level Abstractions
/* Unpack the parameters in an INIT packet.
* FIXME: There is no return status to allow callers to do
- * error handling.
+ * error handling.
*/
-void
-sctp_process_init(sctp_association_t *asoc, sctp_cid_t cid,
- const sockaddr_storage_t *peer_addr,
- sctp_init_chunk_t *peer_init,
- int priority)
+void sctp_process_init(sctp_association_t *asoc, sctp_cid_t cid,
+ const sockaddr_storage_t *peer_addr,
+ sctp_init_chunk_t *peer_init,
+ int priority)
{
sctpParam_t param;
- uint8_t *end;
+ __u8 *end;
sctp_transport_t *transport;
list_t *pos, *temp;
- /* We must include the address that the INIT packet came from.
+ /* We must include the address that the INIT packet came from.
* This is the only address that matters for an INIT packet.
* When processing a COOKIE ECHO, we retrieve the from address
* of the INIT from the cookie.
* added as the primary transport. The source address seems to
* be a a better choice than any of the embedded addresses.
*/
- if (NULL != peer_addr) {
+ if (peer_addr)
sctp_assoc_add_peer(asoc, peer_addr, priority);
- }
-
- /* Process the initialization parameters. */
-
- end = ((uint8_t *)peer_init
- + ntohs(peer_init->chunk_hdr.length));
-
- for (param.v = peer_init->init_hdr.params;
- param.v < end;
- param.v += WORD_ROUND(ntohs(param.p->length))) {
+ /* Process the initialization parameters. */
+ end = ((__u8 *)peer_init + ntohs(peer_init->chunk_hdr.length));
+ for (param.v = peer_init->init_hdr.params;
+ param.v < end;
+ param.v += WORD_ROUND(ntohs(param.p->length))) {
if (!sctp_process_param(asoc, param, peer_addr, cid,
- priority)) {
+ priority))
goto clean_up;
- }
+ }
- } /* for (loop through all parameters) */
-
-
/* The fixed INIT headers are always in network byte
* order.
*/
- asoc->peer.i.init_tag =
+ asoc->peer.i.init_tag =
ntohl(peer_init->init_hdr.init_tag);
asoc->peer.i.a_rwnd =
ntohl(peer_init->init_hdr.a_rwnd);
- asoc->peer.i.num_outbound_streams =
+ asoc->peer.i.num_outbound_streams =
ntohs(peer_init->init_hdr.num_outbound_streams);
- asoc->peer.i.num_inbound_streams =
+ asoc->peer.i.num_inbound_streams =
ntohs(peer_init->init_hdr.num_inbound_streams);
- asoc->peer.i.initial_tsn =
+ asoc->peer.i.initial_tsn =
ntohl(peer_init->init_hdr.initial_tsn);
- /* Apply the upper bounds for output streams based on peer's
- * number of inbound streams.
+ /* Apply the upper bounds for output streams based on peer's
+ * number of inbound streams.
*/
- if (asoc->c.sinit_num_ostreams
- > ntohs(peer_init->init_hdr.num_inbound_streams)) {
- asoc->c.sinit_num_ostreams =
+ if (asoc->c.sinit_num_ostreams >
+ ntohs(peer_init->init_hdr.num_inbound_streams)) {
+ asoc->c.sinit_num_ostreams =
ntohs(peer_init->init_hdr.num_inbound_streams);
- }
+ }
- /* Copy Initiation tag from INIT to VT_peer in cookie */
+ /* Copy Initiation tag from INIT to VT_peer in cookie. */
asoc->c.peer_vtag = asoc->peer.i.init_tag;
- /* Peer Rwnd : Current calculated value of the peer's rwnd. */
+ /* Peer Rwnd : Current calculated value of the peer's rwnd. */
asoc->peer.rwnd = asoc->peer.i.a_rwnd;
- /* RFC 2960 7.2.1 The initial value of ssthresh MAY be arbitrarily
+ /* RFC 2960 7.2.1 The initial value of ssthresh MAY be arbitrarily
* high (for example, implementations MAY use the size of the receiver
* advertised window).
*/
asoc->peer.i.initial_tsn);
/* ADDIP Section 4.1 ASCONF Chunk Procedures
- *
+ *
* When an endpoint has an ASCONF signaled change to be sent to the
* remote endpoint it should do the following:
* ...
* association to the same value as the Initial TSN.
*/
asoc->peer.addip_serial = asoc->peer.i.initial_tsn - 1;
-
return;
- clean_up:
- /* Release the transport structures. */
- list_for_each_safe(pos, temp, &asoc->peer.transport_addr_list) {
+
+clean_up:
+ /* Release the transport structures. */
+ list_for_each_safe(pos, temp, &asoc->peer.transport_addr_list) {
transport = list_entry(pos, sctp_transport_t, transports);
list_del(pos);
sctp_transport_free(transport);
}
- return;
-
-} /* sctp_process_init() */
-
+}
/* Update asoc with the option described in param.
- *
+ *
* RFC2960 3.3.2.1 Optional/Variable Length Parameters in INIT
*
* asoc is the association to update.
* work we do. In particular, we should not build transport
* structures for the addresses.
*/
-int
-sctp_process_param(sctp_association_t *asoc, sctpParam_t param,
- const sockaddr_storage_t *peer_addr,
- sctp_cid_t cid, int priority)
+int sctp_process_param(sctp_association_t *asoc, sctpParam_t param,
+ const sockaddr_storage_t *peer_addr,
+ sctp_cid_t cid, int priority)
{
sockaddr_storage_t addr;
int j;
int retval = 1;
sctp_scope_t scope;
-/* We maintain all INIT parameters in network byte order all the
- * time. This allows us to not worry about whether the parameters
- * came from a fresh INIT, and INIT ACK, or were stored in a cookie.
- */
-
+ /* We maintain all INIT parameters in network byte order all the
+ * time. This allows us to not worry about whether the parameters
+ * came from a fresh INIT, and INIT ACK, or were stored in a cookie.
+ */
switch (param.p->type) {
case SCTP_PARAM_IPV4_ADDRESS:
if (SCTP_CID_INIT != cid) {
sctp_param2sockaddr(&addr, param, asoc->peer.port);
scope = sctp_scope(peer_addr);
- if (sctp_in_scope(&addr, scope)) {
+ if (sctp_in_scope(&addr, scope))
sctp_assoc_add_peer(asoc, &addr, priority);
- }
}
break;
+
case SCTP_PARAM_IPV6_ADDRESS:
if (SCTP_CID_INIT != cid) {
if (PF_INET6 == asoc->base.sk->family) {
- sctp_param2sockaddr(&addr, param,
+ sctp_param2sockaddr(&addr, param,
asoc->peer.port);
scope = sctp_scope(peer_addr);
- if (sctp_in_scope(&addr, scope)) {
- sctp_assoc_add_peer(asoc, &addr,
+ if (sctp_in_scope(&addr, scope))
+ sctp_assoc_add_peer(asoc, &addr,
priority);
- }
}
}
break;
+
case SCTP_PARAM_COOKIE_PRESERVATIVE:
- asoc->cookie_preserve
- = ntohl(param.bht->lifespan_increment);
+ asoc->cookie_preserve =
+ ntohl(param.bht->lifespan_increment);
break;
+
case SCTP_PARAM_HOST_NAME_ADDRESS:
SCTP_DEBUG_PRINTK("unimplmented SCTP_HOST_NAME_ADDRESS\n");
- break;
+ break;
+
case SCTP_PARAM_SUPPORTED_ADDRESS_TYPES:
/* Turn off the default values first so we'll know which
- * ones are really set by the peer.
+ * ones are really set by the peer.
*/
asoc->peer.ipv4_address = 0;
asoc->peer.ipv6_address = 0;
- j = (ntohs(param.p->length)
- - sizeof(sctp_paramhdr_t))
- / sizeof(uint16_t);
+ j = (ntohs(param.p->length) -
+ sizeof(sctp_paramhdr_t)) /
+ sizeof(__u16);
for (i = 0; i < j; ++i) {
switch (param.sat->types[i]) {
case SCTP_PARAM_IPV4_ADDRESS:
asoc->peer.ipv4_address = 1;
break;
+
case SCTP_PARAM_IPV6_ADDRESS:
asoc->peer.ipv6_address = 1;
break;
+
case SCTP_PARAM_HOST_NAME_ADDRESS:
asoc->peer.hostname_address = 1;
break;
+
default: /* Just ignore anything else. */
break;
- }
+ };
}
break;
+
case SCTP_PARAM_STATE_COOKIE:
asoc->peer.cookie_len =
- ntohs(param.p->length)
- - sizeof(sctp_paramhdr_t);
+ ntohs(param.p->length) =
+ sizeof(sctp_paramhdr_t);
asoc->peer.cookie = param.cookie->body;
break;
+
case SCTP_PARAM_HEATBEAT_INFO:
SCTP_DEBUG_PRINTK("unimplmented "
"SCTP_PARAM_HEATBEAT_INFO\n");
break;
+
case SCTP_PARAM_UNRECOGNIZED_PARAMETERS:
SCTP_DEBUG_PRINTK("unimplemented "
"SCTP_PARAM_UNRECOGNIZED_PARAMETERS\n");
break;
+
case SCTP_PARAM_ECN_CAPABLE:
- asoc->peer.ecn_capable = 1;
+ asoc->peer.ecn_capable = 1;
break;
+
default:
SCTP_DEBUG_PRINTK("Ignoring param: %d for association %p.\n",
ntohs(param.p->type), asoc);
retval = 1;
break;
- }
- return retval;
+ };
-} /* sctp_process_param() */
+ return retval;
+}
/* Select a new verification tag. */
-uint32_t
-sctp_generate_tag(const sctp_endpoint_t *ep)
+__u32 sctp_generate_tag(const sctp_endpoint_t *ep)
{
- /* I believe that this random number generator complies with RFC1750.
+ /* I believe that this random number generator complies with RFC1750.
* A tag of 0 is reserved for special cases (e.g. INIT).
*/
- uint32_t x;
- do { get_random_bytes(&x, sizeof(uint32_t)); } while( x==0 );
- return x;
+ __u32 x;
-} /* sctp_generate_tag() */
+ do {
+ get_random_bytes(&x, sizeof(__u32));
+ } while (x == 0);
+
+ return x;
+}
/* Select an initial TSN to send during startup. */
-uint32_t
-sctp_generate_tsn(const sctp_endpoint_t *ep)
+__u32 sctp_generate_tsn(const sctp_endpoint_t *ep)
{
- /* I believe that this random number generator complies with RFC1750.
- */
-
- uint32_t retval;
- get_random_bytes(&retval, sizeof(uint32_t));
+ /* I believe that this random number generator complies with RFC1750. */
+ __u32 retval;
- return retval;
-
-} /* sctp_generate_tsn() */
+ get_random_bytes(&retval, sizeof(__u32));
+ return retval;
+}
/********************************************************************
* 4th Level Abstractions
********************************************************************/
/* Convert from an SCTP IP parameter to a sockaddr_storage_t. */
-void
-sctp_param2sockaddr(sockaddr_storage_t *addr, sctpParam_t param,
- uint16_t port)
+void sctp_param2sockaddr(sockaddr_storage_t *addr, sctpParam_t param, __u16 port)
{
switch(param.p->type) {
case SCTP_PARAM_IPV4_ADDRESS:
addr->v4.sin_port = port;
addr->v4.sin_addr.s_addr = param.v4->addr.s_addr;
break;
+
case SCTP_PARAM_IPV6_ADDRESS:
addr->v6.sin6_family = AF_INET6;
addr->v6.sin6_port = port;
addr->v6.sin6_addr = param.v6->addr;
addr->v6.sin6_scope_id = 0; /* BUG */
break;
+
default:
SCTP_DEBUG_PRINTK("Illegal address type %d\n",
ntohs(param.p->type));
break;
- }
-} /* sctp_param2sockaddr() */
+ };
+}
/* Convert an IP address in an SCTP param into a sockaddr_in. */
/* Returns true if a valid conversion was possible. */
-int
-sctp_addr2sockaddr(sctpParam_t p, sockaddr_storage_t *sa)
+int sctp_addr2sockaddr(sctpParam_t p, sockaddr_storage_t *sa)
{
- if (NULL == p.v) {
- return 0;
- }
- switch (p.p->type) {
- case SCTP_PARAM_IPV4_ADDRESS:
- sa->v4.sin_addr = *((struct in_addr *)&p.v4->addr);
- sa->v4.sin_family = AF_INET;
- break;
- case SCTP_PARAM_IPV6_ADDRESS:
- *((struct in6_addr *)&sa->v4.sin_addr)
- = p.v6->addr;
- sa->v4.sin_family = AF_INET6;
- break;
+ if (!p.v)
+ return 0;
+
+ switch (p.p->type) {
+ case SCTP_PARAM_IPV4_ADDRESS:
+ sa->v4.sin_addr = *((struct in_addr *)&p.v4->addr);
+ sa->v4.sin_family = AF_INET;
+ break;
+
+ case SCTP_PARAM_IPV6_ADDRESS:
+ *((struct in6_addr *)&sa->v4.sin_addr)
+ = p.v6->addr;
+ sa->v4.sin_family = AF_INET6;
+ break;
+
default:
return 0;
- }
- return 1;
+ };
-} /* sctp_addr2sockaddr() */
+ return 1;
+}
/* Convert from an IP version number to an Address Family symbol. */
-int ipver2af(uint8_t ipver)
+int ipver2af(__u8 ipver)
{
- int family;
- switch (ipver) {
- case 4:
- family = AF_INET;
- break;
- case 6:
- family = AF_INET6;
- break;
- default:
- family = 0;
- break;
- } /* switch (ipversion) */
+ int family;
+
+ switch (ipver) {
+ case 4:
+ family = AF_INET;
+ break;
+
+ case 6:
+ family = AF_INET6;
+ break;
- return family;
+ default:
+ family = 0;
+ break;
+ };
-} /* ipver2af() */
+ return family;
+}
/* Convert a sockaddr_in to IP address in an SCTP para. */
/* Returns true if a valid conversion was possible. */
-int
-sockaddr2sctp_addr(const sockaddr_storage_t *sa, sctpParam_t p)
+int sockaddr2sctp_addr(const sockaddr_storage_t *sa, sctpParam_t p)
{
- int len = 0;
-
- switch (sa->v4.sin_family) {
- case AF_INET:
- p.p->type = SCTP_PARAM_IPV4_ADDRESS;
- p.p->length = ntohs(sizeof(sctp_ipv4addr_param_t));
- len = sizeof(sctp_ipv4addr_param_t);
- p.v4->addr.s_addr = sa->v4.sin_addr.s_addr;
- break;
- case AF_INET6:
- p.p->type = SCTP_PARAM_IPV6_ADDRESS;
- p.p->length = ntohs(sizeof(sctp_ipv6addr_param_t));
- len = sizeof(sctp_ipv6addr_param_t);
- p.v6->addr = *(&sa->v6.sin6_addr);
- break;
- default:
+ int len = 0;
+
+ switch (sa->v4.sin_family) {
+ case AF_INET:
+ p.p->type = SCTP_PARAM_IPV4_ADDRESS;
+ p.p->length = ntohs(sizeof(sctp_ipv4addr_param_t));
+ len = sizeof(sctp_ipv4addr_param_t);
+ p.v4->addr.s_addr = sa->v4.sin_addr.s_addr;
+ break;
+
+ case AF_INET6:
+ p.p->type = SCTP_PARAM_IPV6_ADDRESS;
+ p.p->length = ntohs(sizeof(sctp_ipv6addr_param_t));
+ len = sizeof(sctp_ipv6addr_param_t);
+ p.v6->addr = *(&sa->v6.sin6_addr);
+ break;
+
+ default:
printk(KERN_WARNING "sockaddr2sctp_addr: Illegal family %d.\n",
sa->v4.sin_family);
- return 0;
- }
-
- return len;
+ return 0;
+ };
-} /* sockaddr2sctp_addr() */
+ return len;
+}
static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_sm_sideeffect.c,v 1.44 2002/08/16 19:30:50 jgrimm Exp $";
#include <linux/skbuff.h>
-#include <linux/config.h>
#include <linux/types.h>
#include <linux/socket.h>
#include <linux/ip.h>
#include <net/sctp/sctp.h>
#include <net/sctp/sctp_sm.h>
-
/* Do forward declarations of static functions. */
static void sctp_do_ecn_ce_work(sctp_association_t *asoc,
- uint32_t lowest_tsn);
-static
-sctp_chunk_t *sctp_do_ecn_ecne_work(sctp_association_t *asoc,
- uint32_t lowest_tsn,
- sctp_chunk_t *);
-static void sctp_do_ecn_cwr_work(sctp_association_t *asoc,
- uint32_t lowest_tsn);
+ __u32 lowest_tsn);
+static sctp_chunk_t *sctp_do_ecn_ecne_work(sctp_association_t *asoc,
+ __u32 lowest_tsn,
+ sctp_chunk_t *);
+static void sctp_do_ecn_cwr_work(sctp_association_t *asoc,
+ __u32 lowest_tsn);
static void sctp_do_8_2_transport_strike(sctp_association_t *asoc,
sctp_transport_t *transport);
-static void
-sctp_cmd_init_failed(sctp_cmd_seq_t *, sctp_association_t *asoc);
-
-static void
-sctp_cmd_assoc_failed(sctp_cmd_seq_t *, sctp_association_t *asoc);
-
-static void
-sctp_cmd_process_init(sctp_cmd_seq_t *, sctp_association_t *asoc,
- sctp_chunk_t *chunk, sctp_init_chunk_t *peer_init,
- int priority);
-static void
-sctp_cmd_hb_timers_start(sctp_cmd_seq_t *, sctp_association_t *);
-static void
-sctp_cmd_set_bind_addrs(sctp_cmd_seq_t *, sctp_association_t *,
- sctp_bind_addr_t *);
-
-static void
-sctp_cmd_transport_reset(sctp_cmd_seq_t *, sctp_association_t *,
- sctp_transport_t *);
-static void
-sctp_cmd_transport_on(sctp_cmd_seq_t *, sctp_association_t *,
- sctp_transport_t *, sctp_chunk_t *);
-static int
-sctp_cmd_process_sack(sctp_cmd_seq_t *, sctp_association_t *,
- sctp_sackhdr_t *);
-
-static void
-sctp_cmd_setup_t2(sctp_cmd_seq_t *, sctp_association_t *, sctp_chunk_t *);
-
+static void sctp_cmd_init_failed(sctp_cmd_seq_t *, sctp_association_t *asoc);
+static void sctp_cmd_assoc_failed(sctp_cmd_seq_t *, sctp_association_t *asoc);
+static void sctp_cmd_process_init(sctp_cmd_seq_t *, sctp_association_t *asoc,
+ sctp_chunk_t *chunk,
+ sctp_init_chunk_t *peer_init,
+ int priority);
+static void sctp_cmd_hb_timers_start(sctp_cmd_seq_t *, sctp_association_t *);
+static void sctp_cmd_set_bind_addrs(sctp_cmd_seq_t *, sctp_association_t *,
+ sctp_bind_addr_t *);
+static void sctp_cmd_transport_reset(sctp_cmd_seq_t *, sctp_association_t *,
+ sctp_transport_t *);
+static void sctp_cmd_transport_on(sctp_cmd_seq_t *, sctp_association_t *,
+ sctp_transport_t *, sctp_chunk_t *);
+static int sctp_cmd_process_sack(sctp_cmd_seq_t *, sctp_association_t *,
+ sctp_sackhdr_t *);
+static void sctp_cmd_setup_t2(sctp_cmd_seq_t *, sctp_association_t *,
+ sctp_chunk_t *);
/* These three macros allow us to pull the debugging code out of the
* main flow of sctp_do_sm() to keep attention focused on the real
SCTP_DEBUG_PRINTK("sctp_do_sm postfn: " \
"asoc %p, status: %s\n", \
asoc, sctp_status_tbl[status])
-
+
#define DEBUG_POST_SFX \
SCTP_DEBUG_PRINTK("sctp_do_sm post sfx: error %d, asoc %p[%s]\n", \
error, asoc, \
sctp_state_tbl[sctp_id2assoc(ep->base.sk, \
sctp_assoc2id(asoc))?asoc->state:SCTP_STATE_CLOSED])
-
/*
* This is the master state machine processing function.
*
* If you want to understand all of lksctp, this is a
* good place to start.
*/
-int
-sctp_do_sm(sctp_event_t event_type, sctp_subtype_t subtype,
- sctp_state_t state,
- sctp_endpoint_t *ep,
- sctp_association_t *asoc,
- void *event_arg,
- int priority)
+int sctp_do_sm(sctp_event_t event_type, sctp_subtype_t subtype,
+ sctp_state_t state,
+ sctp_endpoint_t *ep,
+ sctp_association_t *asoc,
+ void *event_arg,
+ int priority)
{
sctp_cmd_seq_t commands;
- sctp_sm_table_entry_t *state_fn;
- sctp_disposition_t status;
- int error = 0;
+ sctp_sm_table_entry_t *state_fn;
+ sctp_disposition_t status;
+ int error = 0;
typedef const char *(printfn_t)(sctp_subtype_t);
static printfn_t *table[] = {
/* Look up the state function, run it, and then process the
* side effects. These three steps are the heart of lksctp.
*/
+ state_fn = sctp_sm_lookup_event(event_type, state, subtype);
- state_fn = sctp_sm_lookup_event(event_type, state, subtype);
-
sctp_init_cmd_seq(&commands);
DEBUG_PRE;
status = (*state_fn->fn)(ep, asoc, subtype, event_arg, &commands);
DEBUG_POST;
-
+
error = sctp_side_effects(event_type, subtype, state,
ep, asoc, event_arg,
status, &commands,
priority);
DEBUG_POST_SFX;
-
- return error;
-} /* sctp_do_sm() */
+ return error;
+}
#undef DEBUG_PRE
#undef DEBUG_POST
/*****************************************************************
* This the master state function side effect processing function.
*****************************************************************/
-int
-sctp_side_effects(sctp_event_t event_type, sctp_subtype_t subtype,
- sctp_state_t state,
- sctp_endpoint_t *ep,
- sctp_association_t *asoc,
- void *event_arg,
- sctp_disposition_t status,
- sctp_cmd_seq_t *commands,
- int priority)
+int sctp_side_effects(sctp_event_t event_type, sctp_subtype_t subtype,
+ sctp_state_t state,
+ sctp_endpoint_t *ep,
+ sctp_association_t *asoc,
+ void *event_arg,
+ sctp_disposition_t status,
+ sctp_cmd_seq_t *commands,
+ int priority)
{
- int error;
- sctp_chunk_t *chunk;
- chunk = (sctp_chunk_t *)event_arg;
-
- /* FIXME - Most of the dispositions left today would be categorized
- * as "exceptional" dispositions. For those dispositions, it
+ sctp_chunk_t *chunk = (sctp_chunk_t *) event_arg;
+ int error;
+
+ /* FIXME - Most of the dispositions left today would be categorized
+ * as "exceptional" dispositions. For those dispositions, it
* may not be proper to run through any of the commands at all.
* For example, the command interpreter might be run only with
* disposition SCTP_DISPOSITION_CONSUME.
if (0 != (error = sctp_cmd_interpreter(event_type, subtype, state,
ep, asoc,
event_arg, status,
- commands, priority))) {
+ commands, priority)))
goto bail;
- }
- switch (status) {
- case SCTP_DISPOSITION_DISCARD:
+ switch (status) {
+ case SCTP_DISPOSITION_DISCARD:
SCTP_DEBUG_PRINTK("Ignored sctp protocol event - state %d, "
"event_type %d, event_id %d\n",
state, event_type, subtype.chunk);
- break;
+ break;
+
case SCTP_DISPOSITION_NOMEM:
/* We ran out of memory, so we need to discard this
* packet.
* reneging...
*/
break;
+
case SCTP_DISPOSITION_DELETE_TCB:
/* This should now be a command. */
break;
- case SCTP_DISPOSITION_CONSUME:
- case SCTP_DISPOSITION_ABORT:
- /*
- * We should no longer have much work to do here as the
+
+ case SCTP_DISPOSITION_CONSUME:
+ case SCTP_DISPOSITION_ABORT:
+ /*
+ * We should no longer have much work to do here as the
* real work has been done as explicit commands above.
*/
break;
- case SCTP_DISPOSITION_VIOLATION:
+
+ case SCTP_DISPOSITION_VIOLATION:
printk(KERN_ERR "sctp protocol violation state %d "
"chunkid %d\n", state, subtype.chunk);
- break;
- case SCTP_DISPOSITION_NOT_IMPL:
+ break;
+
+ case SCTP_DISPOSITION_NOT_IMPL:
printk(KERN_WARNING "sctp unimplemented feature in state %d, "
"event_type %d, event_id %d\n",
state, event_type, subtype.chunk);
-
- break;
+ break;
+
case SCTP_DISPOSITION_BUG:
printk(KERN_ERR "sctp bug in state %d, "
"event_type %d, event_id %d\n",
state, event_type, subtype.chunk);
BUG();
break;
- default:
+
+ default:
printk(KERN_ERR "sctp impossible disposition %d "
"in state %d, event_type %d, event_id %d\n",
status, state, event_type, subtype.chunk);
BUG();
- break;
- } /* switch (status) */
-
- bail:
- return error;
+ break;
+ };
-} /* sctp_side_effects() */
+bail:
+ return error;
+}
/********************************************************************
* 2nd Level Abstractions
********************************************************************/
/* This is the side-effect interpreter. */
-int
-sctp_cmd_interpreter(sctp_event_t event_type, sctp_subtype_t subtype,
- sctp_state_t state, sctp_endpoint_t *ep,
- sctp_association_t *asoc, void *event_arg,
- sctp_disposition_t status, sctp_cmd_seq_t *commands,
- int priority)
+int sctp_cmd_interpreter(sctp_event_t event_type, sctp_subtype_t subtype,
+ sctp_state_t state, sctp_endpoint_t *ep,
+ sctp_association_t *asoc, void *event_arg,
+ sctp_disposition_t status, sctp_cmd_seq_t *commands,
+ int priority)
{
- int error = 0;
+ int error = 0;
int force;
sctp_cmd_t *command;
sctp_chunk_t *new_obj;
sctp_chunk_t *chunk;
sctp_packet_t *packet;
- struct timer_list *timer;
- unsigned long timeout;
- sctp_transport_t *t;
- sctp_sackhdr_t sackh;
-
+ struct timer_list *timer;
+ unsigned long timeout;
+ sctp_transport_t *t;
+ sctp_sackhdr_t sackh;
- chunk = (sctp_chunk_t *)event_arg;
+ chunk = (sctp_chunk_t *) event_arg;
- /* Note: This whole file is a huge candidate for rework.
- * For example, each command could either have its own handler, so
- * the loop would look like:
+ /* Note: This whole file is a huge candidate for rework.
+ * For example, each command could either have its own handler, so
+ * the loop would look like:
* while (cmds)
* cmd->handle(x, y, z)
* --jgrimm
*/
-
while (NULL != (command = sctp_next_cmd(commands))) {
switch (command->verb) {
case SCTP_CMD_NOP:
case SCTP_CMD_PURGE_OUTQUEUE:
sctp_outqueue_teardown(&asoc->outqueue);
break;
-
+
case SCTP_CMD_DELETE_TCB:
/* Delete the current association. */
sctp_unhash_established(asoc);
break;
case SCTP_CMD_NEW_STATE:
- /* Enter a new state. */
+ /* Enter a new state. */
asoc->state = command->obj.state;
asoc->state_timestamp = jiffies;
break;
case SCTP_CMD_PROCESS_SACK:
/* Process an inbound SACK. */
- error = sctp_cmd_process_sack(commands, asoc,
+ error = sctp_cmd_process_sack(commands, asoc,
command->obj.ptr);
break;
-
case SCTP_CMD_GEN_INIT_ACK:
/* Generate an INIT ACK chunk. */
new_obj = sctp_make_init_ack(asoc, chunk, GFP_ATOMIC);
- if (!new_obj) { goto nomem; }
+ if (!new_obj)
+ goto nomem;
- sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
+ sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
SCTP_CHUNK(new_obj));
break;
case SCTP_CMD_PEER_INIT:
- /* Process a unified INIT from the peer. */
- sctp_cmd_process_init(commands,
+ /* Process a unified INIT from the peer. */
+ sctp_cmd_process_init(commands,
asoc, chunk, command->obj.ptr,
priority);
break;
case SCTP_CMD_GEN_COOKIE_ECHO:
- /* Generate a COOKIE ECHO chunk. */
+ /* Generate a COOKIE ECHO chunk. */
new_obj = sctp_make_cookie_echo(asoc, chunk);
- if (!new_obj) { goto nomem; }
- sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
+ if (!new_obj)
+ goto nomem;
+ sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
SCTP_CHUNK(new_obj));
break;
case SCTP_CMD_GEN_SHUTDOWN:
- /* Generate SHUTDOWN when in SHUTDOWN_SENT state.
- * Reset error counts.
+ /* Generate SHUTDOWN when in SHUTDOWN_SENT state.
+ * Reset error counts.
*/
asoc->overall_error_count = 0;
-
- /* Generate a SHUTDOWN chunk. */
+ /* Generate a SHUTDOWN chunk. */
new_obj = sctp_make_shutdown(asoc);
- if (!new_obj) { goto nomem; }
- sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
+ if (!new_obj)
+ goto nomem;
+ sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
SCTP_CHUNK(new_obj));
break;
break;
case SCTP_CMD_EVENT_ULP:
- /* Send a notification to the sockets layer. */
+ /* Send a notification to the sockets layer. */
SCTP_DEBUG_PRINTK("sm_sideff: %s %p, %s %p.\n",
"event_up:",
command->obj.ptr,
"ulpq:",
&asoc->ulpq);
- sctp_ulpqueue_tail_event(&asoc->ulpq,
+ sctp_ulpqueue_tail_event(&asoc->ulpq,
command->obj.ptr);
break;
case SCTP_CMD_REPLY:
/* Send a chunk to our peer. */
- error = sctp_push_outqueue(&asoc->outqueue,
+ error = sctp_push_outqueue(&asoc->outqueue,
command->obj.ptr);
break;
case SCTP_CMD_SEND_PKT:
/* Send a full packet to our peer. */
packet = command->obj.ptr;
- sctp_packet_transmit(packet);
- sctp_transport_free(packet->transport);
- sctp_packet_free(packet);
+ sctp_packet_transmit(packet);
+ sctp_transport_free(packet->transport);
+ sctp_packet_free(packet);
break;
case SCTP_CMD_RETRAN:
sctp_cmd_setup_t2(commands, asoc, command->obj.ptr);
break;
- case SCTP_CMD_TIMER_START:
+ case SCTP_CMD_TIMER_START:
timer = &asoc->timers[command->obj.to];
timeout = asoc->timeouts[command->obj.to];
- if (0 == timeout) { BUG(); }
+ if (!timeout)
+ BUG();
- timer->expires = jiffies + timeout;
+ timer->expires = jiffies + timeout;
sctp_association_hold(asoc);
- add_timer(timer);
- break;
+ add_timer(timer);
+ break;
- case SCTP_CMD_TIMER_RESTART:
+ case SCTP_CMD_TIMER_RESTART:
timer = &asoc->timers[command->obj.to];
timeout = asoc->timeouts[command->obj.to];
-
- if (!mod_timer(timer, jiffies + timeout)) {
+ if (!mod_timer(timer, jiffies + timeout))
sctp_association_hold(asoc);
- }
- break;
+ break;
- case SCTP_CMD_TIMER_STOP:
+ case SCTP_CMD_TIMER_STOP:
timer = &asoc->timers[command->obj.to];
- if (timer_pending(timer) && del_timer(timer)) {
+ if (timer_pending(timer) && del_timer(timer))
sctp_association_put(asoc);
- }
- break;
+ break;
case SCTP_CMD_INIT_RESTART:
/* Do the needed accounting and updates
* associated with restarting an initialization
- * timer.
+ * timer.
*/
asoc->counters[SCTP_COUNTER_INIT_ERROR]++;
asoc->timeouts[command->obj.to] *= 2;
-
- if (asoc->timeouts[command->obj.to]
- > asoc->max_init_timeo) {
- asoc->timeouts[command->obj.to]
- = asoc->max_init_timeo;
+ if (asoc->timeouts[command->obj.to] >
+ asoc->max_init_timeo) {
+ asoc->timeouts[command->obj.to] =
+ asoc->max_init_timeo;
}
-
+
sctp_add_cmd_sf(commands,
SCTP_CMD_TIMER_RESTART,
SCTP_TO(command->obj.to));
-
break;
case SCTP_CMD_INIT_FAILED:
case SCTP_CMD_COUNTER_INC:
asoc->counters[command->obj.counter]++;
break;
-
+
case SCTP_CMD_COUNTER_RESET:
- asoc->counters[command->obj.counter]=0;
+ asoc->counters[command->obj.counter] = 0;
break;
case SCTP_CMD_REPORT_DUP:
if (asoc->peer.next_dup_tsn < SCTP_MAX_DUP_TSNS) {
- asoc->peer.dup_tsns[asoc->peer.next_dup_tsn++]
- = ntohl(command->obj.u32);
+ asoc->peer.dup_tsns[asoc->peer.next_dup_tsn++] =
+ ntohl(command->obj.u32);
}
break;
&asoc->peer.tsn_map),
command->obj.u32);
break;
+
case SCTP_CMD_REPORT_BAD_TAG:
SCTP_DEBUG_PRINTK("vtag mismatch!\n");
break;
case SCTP_CMD_TRANSPORT_RESET:
t = command->obj.transport;
sctp_cmd_transport_reset(commands, asoc, t);
- break;
-
+ break;
+
case SCTP_CMD_TRANSPORT_ON:
- t = command->obj.transport;
+ t = command->obj.transport;
sctp_cmd_transport_on(commands, asoc, t, chunk);
- break;
-
+ break;
+
case SCTP_CMD_HB_TIMERS_START:
- sctp_cmd_hb_timers_start(commands, asoc);
+ sctp_cmd_hb_timers_start(commands, asoc);
break;
-
+
case SCTP_CMD_REPORT_ERROR:
error = command->obj.error;
break;
sackh.a_rwnd = 0;
sackh.num_gap_ack_blocks = 0;
sackh.num_dup_tsns = 0;
- sctp_add_cmd_sf(commands,
- SCTP_CMD_PROCESS_SACK,
+ sctp_add_cmd_sf(commands,
+ SCTP_CMD_PROCESS_SACK,
SCTP_SACKH(&sackh));
break;
-
+
case SCTP_CMD_DISCARD_PACKET:
/* We need to discard the whole packet. */
chunk->pdiscard = 1;
break;
-
+
default:
printk(KERN_WARNING "Impossible command: %u, %p\n",
command->verb, command->obj.ptr);
break;
- } /* switch (cmd->verb) */
- } /* while (more commands) */
+ };
+ }
return error;
nomem:
error = -ENOMEM;
return error;
-
-} /* sctp_cmd_interpreter() */
-
+}
/* A helper function for delayed processing of INET ECN CE bit. */
-static void
-sctp_do_ecn_ce_work(sctp_association_t *asoc, uint32_t lowest_tsn)
+static void sctp_do_ecn_ce_work(sctp_association_t *asoc, __u32 lowest_tsn)
{
- /*
+ /*
* Save the TSN away for comparison when we receive CWR
- * Note: dp->TSN is expected in host endian
+ * Note: dp->TSN is expected in host endian
*/
asoc->last_ecne_tsn = lowest_tsn;
asoc->need_ecne = 1;
-
-} /* void sctp_do_ecn_ce_work(asoc, command) */
-
+}
/* Helper function for delayed processing of SCTP ECNE chunk. */
/* RFC 2960 Appendix A
*
- * RFC 2481 details a specific bit for a sender to send in
- * the header of its next outbound TCP segment to indicate to
- * its peer that it has reduced its congestion window. This
- * is termed the CWR bit. For SCTP the same indication is made
- * by including the CWR chunk. This chunk contains one data
- * element, i.e. the TSN number that was sent in the ECNE chunk.
- * This element represents the lowest TSN number in the datagram
+ * RFC 2481 details a specific bit for a sender to send in
+ * the header of its next outbound TCP segment to indicate to
+ * its peer that it has reduced its congestion window. This
+ * is termed the CWR bit. For SCTP the same indication is made
+ * by including the CWR chunk. This chunk contains one data
+ * element, i.e. the TSN number that was sent in the ECNE chunk.
+ * This element represents the lowest TSN number in the datagram
* that was originally marked with the CE bit.
*/
-static sctp_chunk_t *
-sctp_do_ecn_ecne_work(sctp_association_t *asoc,
- uint32_t lowest_tsn,
- sctp_chunk_t *chunk)
+static sctp_chunk_t *sctp_do_ecn_ecne_work(sctp_association_t *asoc,
+ __u32 lowest_tsn,
+ sctp_chunk_t *chunk)
{
-
sctp_chunk_t *repl;
sctp_transport_t *transport;
-
/* Our previously transmitted packet ran into some congestion
* so we should take action by reducing cwnd and ssthresh
- * and then ACK our peer that we we've done so by
+ * and then ACK our peer that we we've done so by
* sending a CWR.
*/
-
-
- /* Find which transport's congestion variables
+
+ /* Find which transport's congestion variables
* need to be adjusted.
*/
-
+
transport = sctp_assoc_lookup_tsn(asoc, lowest_tsn);
/* Update the congestion variables. */
-
- if (transport) {
+ if (transport)
sctp_transport_lower_cwnd(transport, SCTP_LOWER_CWND_ECNE);
- }
-
+
/* Save away a rough idea of when we last sent out a CWR.
- * We compare against this value (see above) to decide if
- * this is a fairly new request.
- * Note that this is not a perfect solution. We may
+ * We compare against this value (see above) to decide if
+ * this is a fairly new request.
+ * Note that this is not a perfect solution. We may
* have moved beyond the window (several times) by the
* next time we get an ECNE. However, it is cute. This idea
* came from Randy's reference code.
* what we do.
*
* RFC 2960 Appendix A
- *
+ *
* CWR:
*
* RFC 2481 details a specific bit for a sender to send in
* chunk contains one data element, i.e. the TSN number
* that was sent in the ECNE chunk. This element
* represents the lowest TSN number in the datagram that
- * was originally marked with the CE bit.
+ * was originally marked with the CE bit.
*/
asoc->last_cwr_tsn = asoc->next_tsn - 1;
- repl = sctp_make_cwr(asoc, asoc->last_cwr_tsn, chunk);
+ repl = sctp_make_cwr(asoc, asoc->last_cwr_tsn, chunk);
/* If we run out of memory, it will look like a lost CWR. We'll
* get back in sync eventually.
*/
-
return repl;
+}
-} /* sctp_chunk_t *sctp_do_ecn_ecne_work(asoc, command) */
-
-/* Helper function to do delayed processing of ECN CWR chunk */
-static void
-sctp_do_ecn_cwr_work(sctp_association_t *asoc,
- uint32_t lowest_tsn)
+/* Helper function to do delayed processing of ECN CWR chunk. */
+static void sctp_do_ecn_cwr_work(sctp_association_t *asoc,
+ __u32 lowest_tsn)
{
-
/* Turn off ECNE getting auto-prepended to every outgoing
* packet
*/
-
asoc->need_ecne = 0;
+}
-} /* void sctp_do_ecn_cwr_work(asoc, command) */
-
-/* This macro is to compress the text a bit... */
+/* This macro is to compress the text a bit... */
#define AP(v) asoc->peer.v
/* Generate SACK if necessary. We call this at the end of a packet. */
-int
-sctp_gen_sack(sctp_association_t *asoc, int force, sctp_cmd_seq_t *commands)
+int sctp_gen_sack(sctp_association_t *asoc, int force, sctp_cmd_seq_t *commands)
{
- uint32_t ctsn, max_tsn_seen;
+ __u32 ctsn, max_tsn_seen;
sctp_chunk_t *sack;
int error = 0;
- if (force) {
- asoc->peer.sack_needed = 1;
- }
+ if (force)
+ asoc->peer.sack_needed = 1;
ctsn = sctp_tsnmap_get_ctsn(&asoc->peer.tsn_map);
max_tsn_seen = sctp_tsnmap_get_max_tsn_seen(&asoc->peer.tsn_map);
- /* From 12.2 Parameters necessary per association (i.e. the TCB):
- *
- * Ack State : This flag indicates if the next received packet
- * : is to be responded to with a SACK. ...
- * : When DATA chunks are out of order, SACK's
- * : are not delayed (see Section 6).
- *
- * [This is actually not mentioned in Section 6, but we
- * implement it here anyway. --piggy]
- */
- if (max_tsn_seen != ctsn) {
- asoc->peer.sack_needed = 1;
- }
-
- /* From 6.2 Acknowledgement on Reception of DATA Chunks:
- *
- * Section 4.2 of [RFC2581] SHOULD be followed. Specifically,
- * an acknowledgement SHOULD be generated for at least every
- * second packet (not every second DATA chunk) received, and
- * SHOULD be generated within 200 ms of the arrival of any
- * unacknowledged DATA chunk. ...
- */
- if (!asoc->peer.sack_needed) {
+ /* From 12.2 Parameters necessary per association (i.e. the TCB):
+ *
+ * Ack State : This flag indicates if the next received packet
+ * : is to be responded to with a SACK. ...
+ * : When DATA chunks are out of order, SACK's
+ * : are not delayed (see Section 6).
+ *
+ * [This is actually not mentioned in Section 6, but we
+ * implement it here anyway. --piggy]
+ */
+ if (max_tsn_seen != ctsn)
+ asoc->peer.sack_needed = 1;
+
+ /* From 6.2 Acknowledgement on Reception of DATA Chunks:
+ *
+ * Section 4.2 of [RFC2581] SHOULD be followed. Specifically,
+ * an acknowledgement SHOULD be generated for at least every
+ * second packet (not every second DATA chunk) received, and
+ * SHOULD be generated within 200 ms of the arrival of any
+ * unacknowledged DATA chunk. ...
+ */
+ if (!asoc->peer.sack_needed) {
/* We will need a SACK for the next packet. */
- asoc->peer.sack_needed = 1;
+ asoc->peer.sack_needed = 1;
goto out;
- } else {
+ } else {
sack = sctp_make_sack(asoc);
- if (NULL == sack) { goto nomem; }
+ if (!sack)
+ goto nomem;
- asoc->peer.sack_needed = 0;
+ asoc->peer.sack_needed = 0;
asoc->peer.next_dup_tsn = 0;
- error = sctp_push_outqueue(&asoc->outqueue, sack);
-
- /* Stop the SACK timer. */
- sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
- SCTP_TO(SCTP_EVENT_TIMEOUT_SACK));
+ error = sctp_push_outqueue(&asoc->outqueue, sack);
- } /* else (we needed a sack) */
+ /* Stop the SACK timer. */
+ sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
+ SCTP_TO(SCTP_EVENT_TIMEOUT_SACK));
+ }
- out:
+out:
return error;
- nomem:
+
+nomem:
error = -ENOMEM;
return error;
-
-} /* sctp_gen_sack() */
+}
/* Handle a duplicate TSN. */
-void
-sctp_do_TSNdup(sctp_association_t *asoc, sctp_chunk_t *chunk,
- long gap)
+void sctp_do_TSNdup(sctp_association_t *asoc, sctp_chunk_t *chunk, long gap)
{
#if 0
- sctp_chunk_t *sack;
+ sctp_chunk_t *sack;
/* Caution: gap < 2 * SCTP_TSN_MAP_SIZE
* so gap can be negative.
* --xguo
*/
- /* Count this TSN. */
- if (gap < SCTP_TSN_MAP_SIZE) {
- asoc->peer.tsn_map[gap]++;
- } else {
- asoc->peer.tsn_map_overflow[gap - SCTP_TSN_MAP_SIZE]++;
- }
-
- /* From 6.2 Acknowledgement on Reception of DATA Chunks
- *
- * When a packet arrives with duplicate DATA chunk(s)
- * and with no new DATA chunk(s), the endpoint MUST
- * immediately send a SACK with no delay. If a packet
- * arrives with duplicate DATA chunk(s) bundled with
- * new DATA chunks, the endpoint MAY immediately send a
- * SACK. Normally receipt of duplicate DATA chunks
- * will occur when the original SACK chunk was lost and
- * the peer's RTO has expired. The duplicate TSN
- * number(s) SHOULD be reported in the SACK as
- * duplicate.
- */
- asoc->counters[SctpCounterAckState] = 2;
+ /* Count this TSN. */
+ if (gap < SCTP_TSN_MAP_SIZE) {
+ asoc->peer.tsn_map[gap]++;
+ } else {
+ asoc->peer.tsn_map_overflow[gap - SCTP_TSN_MAP_SIZE]++;
+ }
+
+ /* From 6.2 Acknowledgement on Reception of DATA Chunks
+ *
+ * When a packet arrives with duplicate DATA chunk(s)
+ * and with no new DATA chunk(s), the endpoint MUST
+ * immediately send a SACK with no delay. If a packet
+ * arrives with duplicate DATA chunk(s) bundled with
+ * new DATA chunks, the endpoint MAY immediately send a
+ * SACK. Normally receipt of duplicate DATA chunks
+ * will occur when the original SACK chunk was lost and
+ * the peer's RTO has expired. The duplicate TSN
+ * number(s) SHOULD be reported in the SACK as
+ * duplicate.
+ */
+ asoc->counters[SctpCounterAckState] = 2;
#endif /* 0 */
} /* sctp_do_TSNdup() */
/* When the T3-RTX timer expires, it calls this function to create the
* relevant state machine event.
*/
-void
-sctp_generate_t3_rtx_event(unsigned long peer)
+void sctp_generate_t3_rtx_event(unsigned long peer)
{
- int error;
- sctp_transport_t *transport = (sctp_transport_t *)peer;
+ int error;
+ sctp_transport_t *transport = (sctp_transport_t *) peer;
sctp_association_t *asoc = transport->asoc;
- /* Check whether a task is in the sock. */
+ /* Check whether a task is in the sock. */
sctp_bh_lock_sock(asoc->base.sk);
-
if (__sctp_sock_busy(asoc->base.sk)) {
SCTP_DEBUG_PRINTK(__FUNCTION__ ":Sock is busy.\n");
- /* Try again later. */
- if (!mod_timer(&transport->T3_rtx_timer,
- jiffies + (HZ/20))) {
+ /* Try again later. */
+ if (!mod_timer(&transport->T3_rtx_timer, jiffies + (HZ/20)))
sctp_transport_hold(transport);
- }
goto out_unlock;
}
/* Is this transport really dead and just waiting around for
* the timer to let go of the reference?
- */
- if (transport->dead) {
+ */
+ if (transport->dead)
goto out_unlock;
- }
-
/* Run through the state machine. */
- error = sctp_do_sm(SCTP_EVENT_T_TIMEOUT,
+ error = sctp_do_sm(SCTP_EVENT_T_TIMEOUT,
SCTP_ST_TIMEOUT(SCTP_EVENT_TIMEOUT_T3_RTX),
- asoc->state,
- asoc->ep, asoc,
- transport, GFP_ATOMIC);
+ asoc->state,
+ asoc->ep, asoc,
+ transport, GFP_ATOMIC);
- if (error) {
- asoc->base.sk->err = -error;
- }
+ if (error)
+ asoc->base.sk->err = -error;
out_unlock:
sctp_bh_unlock_sock(asoc->base.sk);
sctp_transport_put(transport);
-
-} /* sctp_generate_t3_rtx_event() */
-
+}
/* This is a sa interface for producing timeout events. It works
* for timeouts which use the association as their parameter.
*/
-static void
-sctp_generate_timeout_event(sctp_association_t *asoc,
- sctp_event_timeout_t timeout_type)
+static void sctp_generate_timeout_event(sctp_association_t *asoc,
+ sctp_event_timeout_t timeout_type)
{
- int error = 0;
-
- sctp_bh_lock_sock(asoc->base.sk);
+ int error = 0;
+ sctp_bh_lock_sock(asoc->base.sk);
if (__sctp_sock_busy(asoc->base.sk)) {
SCTP_DEBUG_PRINTK(__FUNCTION__ "Sock is busy: timer %d\n",
timeout_type);
- /* Try again later. */
- if (!mod_timer(&asoc->timers[timeout_type],
- jiffies + (HZ/20))) {
+
+ /* Try again later. */
+ if (!mod_timer(&asoc->timers[timeout_type], jiffies + (HZ/20)))
sctp_association_hold(asoc);
- }
goto out_unlock;
- }
+ }
/* Is this association really dead and just waiting around for
* the timer to let go of the reference?
*/
- if (asoc->base.dead) {
+ if (asoc->base.dead)
goto out_unlock;
- }
- /* Run through the state machine. */
+ /* Run through the state machine. */
error = sctp_do_sm(SCTP_EVENT_T_TIMEOUT,
SCTP_ST_TIMEOUT(timeout_type),
- asoc->state, asoc->ep, asoc,
+ asoc->state, asoc->ep, asoc,
(void *)timeout_type,
GFP_ATOMIC);
- if (error) { asoc->base.sk->err = -error; }
-
+ if (error)
+ asoc->base.sk->err = -error;
+
out_unlock:
sctp_bh_unlock_sock(asoc->base.sk);
sctp_association_put(asoc);
+}
-} /* sctp_generate_timeout_event() */
-
-void
-sctp_generate_t1_cookie_event(unsigned long data)
+void sctp_generate_t1_cookie_event(unsigned long data)
{
- sctp_association_t *asoc = (sctp_association_t *)data;
-
+ sctp_association_t *asoc = (sctp_association_t *) data;
sctp_generate_timeout_event(asoc, SCTP_EVENT_TIMEOUT_T1_COOKIE);
+}
-} /* sctp_generate_t1_cookie_event() */
-
-void
-sctp_generate_t1_init_event(unsigned long data)
+void sctp_generate_t1_init_event(unsigned long data)
{
- sctp_association_t *asoc = (sctp_association_t *)data;
-
- sctp_generate_timeout_event(asoc, SCTP_EVENT_TIMEOUT_T1_INIT);
+ sctp_association_t *asoc = (sctp_association_t *) data;
+ sctp_generate_timeout_event(asoc, SCTP_EVENT_TIMEOUT_T1_INIT);
+}
-} /* sctp_generate_t1_init_event() */
-
-void
-sctp_generate_t2_shutdown_event(unsigned long data)
+void sctp_generate_t2_shutdown_event(unsigned long data)
{
- sctp_association_t *asoc = (sctp_association_t *)data;
- sctp_generate_timeout_event(asoc, SCTP_EVENT_TIMEOUT_T2_SHUTDOWN);
-
-} /* sctp_generate_t2_shutdown_event() */
+ sctp_association_t *asoc = (sctp_association_t *) data;
+ sctp_generate_timeout_event(asoc, SCTP_EVENT_TIMEOUT_T2_SHUTDOWN);
+}
-void
-sctp_generate_autoclose_event(unsigned long data)
+void sctp_generate_autoclose_event(unsigned long data)
{
- sctp_association_t *asoc = (sctp_association_t *)data;
- sctp_generate_timeout_event(asoc, SCTP_EVENT_TIMEOUT_AUTOCLOSE);
+ sctp_association_t *asoc = (sctp_association_t *) data;
+ sctp_generate_timeout_event(asoc, SCTP_EVENT_TIMEOUT_AUTOCLOSE);
+}
-} /* sctp_generate_autoclose_event() */
-
-/* Generate a heart beat event. If the sock is busy, reschedule. Make
- * sure that the transport is still valid.
+/* Generate a heart beat event. If the sock is busy, reschedule. Make
+ * sure that the transport is still valid.
*/
-void
-sctp_generate_heartbeat_event(unsigned long data)
+void sctp_generate_heartbeat_event(unsigned long data)
{
- int error = 0;
- sctp_transport_t *transport = (sctp_transport_t *)data;
+ int error = 0;
+ sctp_transport_t *transport = (sctp_transport_t *) data;
sctp_association_t *asoc = transport->asoc;
sctp_bh_lock_sock(asoc->base.sk);
if (__sctp_sock_busy(asoc->base.sk)) {
SCTP_DEBUG_PRINTK(__FUNCTION__ ":Sock is busy.\n");
- /* Try again later. */
- if (!mod_timer(&transport->hb_timer,
- jiffies + (HZ/20))) {
+ /* Try again later. */
+ if (!mod_timer(&transport->hb_timer, jiffies + (HZ/20)))
sctp_transport_hold(transport);
- }
goto out_unlock;
}
- /* Is this structure just waiting around for us to actually
+ /* Is this structure just waiting around for us to actually
* get destroyed?
*/
- if (transport->dead) {
+ if (transport->dead)
goto out_unlock;
- }
-
- error = sctp_do_sm(SCTP_EVENT_T_TIMEOUT,
- SCTP_ST_TIMEOUT(SCTP_EVENT_TIMEOUT_HEARTBEAT),
- asoc->state,
- asoc->ep, asoc,
- transport, GFP_ATOMIC);
- if (error) {
-
- asoc->base.sk->err = -error;
- }
+ error = sctp_do_sm(SCTP_EVENT_T_TIMEOUT,
+ SCTP_ST_TIMEOUT(SCTP_EVENT_TIMEOUT_HEARTBEAT),
+ asoc->state,
+ asoc->ep, asoc,
+ transport, GFP_ATOMIC);
+ if (error)
+ asoc->base.sk->err = -error;
out_unlock:
sctp_bh_unlock_sock(asoc->base.sk);
sctp_transport_put(transport);
+}
-} /* sctp_generate_heartbeat_event() */
-
-/* Inject a SACK Timeout event into the state machine. */
-void
-sctp_generate_sack_event(unsigned long data)
+/* Inject a SACK Timeout event into the state machine. */
+void sctp_generate_sack_event(unsigned long data)
{
- sctp_association_t *asoc = (sctp_association_t *)data;
- sctp_generate_timeout_event(asoc, SCTP_EVENT_TIMEOUT_SACK);
+ sctp_association_t *asoc = (sctp_association_t *) data;
+ sctp_generate_timeout_event(asoc, SCTP_EVENT_TIMEOUT_SACK);
+}
-} /* sctp_generate_sack_event() */
-
-void
-sctp_generate_pmtu_raise_event(unsigned long data)
+void sctp_generate_pmtu_raise_event(unsigned long data)
{
- sctp_association_t *asoc = (sctp_association_t *)data;
- sctp_generate_timeout_event(asoc, SCTP_EVENT_TIMEOUT_PMTU_RAISE);
-} /* sctp_generate_pmtu_raise_event() */
+ sctp_association_t *asoc = (sctp_association_t *) data;
+ sctp_generate_timeout_event(asoc, SCTP_EVENT_TIMEOUT_PMTU_RAISE);
+}
sctp_timer_event_t *sctp_timer_events[SCTP_NUM_TIMEOUT_TYPES] = {
NULL,
* 3rd Level Abstractions
********************************************************************/
-
/* RFC 2960 8.2 Path Failure Detection
*
* When its peer endpoint is multi-homed, an endpoint should keep a
* error counter for each of the destination transport addresses of the
* peer endpoint.
- *
+ *
* Each time the T3-rtx timer expires on any address, or when a
* HEARTBEAT sent to an idle address is not acknowledged within a RTO,
* the error counter of that destination address will be incremented.
* 'Path.Max.Retrans' of that destination address, the endpoint should
* mark the destination transport address as inactive, and a
* notification SHOULD be sent to the upper layer.
- *
- *
+ *
*/
-
-static void
-sctp_do_8_2_transport_strike(sctp_association_t *asoc,
- sctp_transport_t *transport)
+static void sctp_do_8_2_transport_strike(sctp_association_t *asoc,
+ sctp_transport_t *transport)
{
/* The check for association's overall error counter exceeding the
* threshold is done in the state function.
*/
asoc->overall_error_count++;
- if ( transport->state.active
- && (transport->error_count++ >= transport->error_threshold) ) {
+ if (transport->state.active &&
+ (transport->error_count++ >= transport->error_threshold)) {
SCTP_DEBUG_PRINTK("transport_strike: transport "
"IP:%d.%d.%d.%d failed.\n",
NIPQUAD(transport->ipaddr.v4.sin_addr));
- sctp_assoc_control_transport(asoc, transport,
- SCTP_TRANSPORT_DOWN,
+ sctp_assoc_control_transport(asoc, transport,
+ SCTP_TRANSPORT_DOWN,
SCTP_FAILED_THRESHOLD);
}
- /* E2) For the destination address for which the timer
- * expires, set RTO <- RTO * 2 ("back off the timer"). The
- * maximum value discussed in rule C7 above (RTO.max) may be
- * used to provide an upper bound to this doubling operation.
- */
+ /* E2) For the destination address for which the timer
+ * expires, set RTO <- RTO * 2 ("back off the timer"). The
+ * maximum value discussed in rule C7 above (RTO.max) may be
+ * used to provide an upper bound to this doubling operation.
+ */
transport->rto = min((transport->rto * 2), transport->asoc->rto_max);
+}
-} /* sctp_do_8_2_transport_strike() */
-
-/* Worker routine to handle INIT command failure. */
-static void
-sctp_cmd_init_failed(sctp_cmd_seq_t *commands, sctp_association_t *asoc)
+/* Worker routine to handle INIT command failure. */
+static void sctp_cmd_init_failed(sctp_cmd_seq_t *commands,
+ sctp_association_t *asoc)
{
sctp_ulpevent_t *event;
SCTP_CANT_STR_ASSOC,
0, 0, 0,
GFP_ATOMIC);
-
- if (event) {
+
+ if (event)
sctp_add_cmd_sf(commands, SCTP_CMD_EVENT_ULP,
SCTP_ULPEVENT(event));
- }
/* FIXME: We need to handle data possibly either
* sent via COOKIE-ECHO bundling or just waiting in
* SEND_FAILED notifications.
*/
sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
-
- return;
-
-} /* sctp_cmd_init_failed() */
+}
-
-/* Worker routine to handle SCTP_CMD_ASSOC_FAILED. */
-static void
-sctp_cmd_assoc_failed(sctp_cmd_seq_t *commands, sctp_association_t *asoc)
+/* Worker routine to handle SCTP_CMD_ASSOC_FAILED. */
+static void sctp_cmd_assoc_failed(sctp_cmd_seq_t *commands,
+ sctp_association_t *asoc)
{
sctp_ulpevent_t *event;
SCTP_COMM_LOST,
0, 0, 0,
GFP_ATOMIC);
-
- if (event) {
+
+ if (event)
sctp_add_cmd_sf(commands, SCTP_CMD_EVENT_ULP,
SCTP_ULPEVENT(event));
- }
/* FIXME: We need to handle data that could not be sent or was not
* acked, if the user has enabled SEND_FAILED notifications.
*/
sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
-
- return;
-
-} /* sctp_cmd_assoc_failed() */
+}
/* Process an init chunk (may be real INIT/INIT-ACK or an embedded INIT
- * inside the cookie.
+ * inside the cookie.
*/
-static void
-sctp_cmd_process_init(sctp_cmd_seq_t *commands, sctp_association_t *asoc,
- sctp_chunk_t *chunk, sctp_init_chunk_t *peer_init,
- int priority)
+static void sctp_cmd_process_init(sctp_cmd_seq_t *commands,
+ sctp_association_t *asoc,
+ sctp_chunk_t *chunk,
+ sctp_init_chunk_t *peer_init,
+ int priority)
{
- /* The command sequence holds commands assuming that the
+ /* The command sequence holds commands assuming that the
* processing will happen successfully. If this is not the
* case, rewind the sequence and add appropriate error handling
- * to the sequence.
+ * to the sequence.
*/
sctp_process_init(asoc, chunk->chunk_hdr->type,
sctp_source(chunk), peer_init,
priority);
-
-
-} /* sctp_cmd_process_init() */
+}
/* Helper function to break out starting up of heartbeat timers. */
-static void
-sctp_cmd_hb_timers_start(sctp_cmd_seq_t *cmds, sctp_association_t *asoc)
+static void sctp_cmd_hb_timers_start(sctp_cmd_seq_t *cmds,
+ sctp_association_t *asoc)
{
sctp_transport_t *t;
list_t *pos;
*/
list_for_each(pos, &asoc->peer.transport_addr_list) {
t = list_entry(pos, sctp_transport_t, transports);
- if (!mod_timer(&t->hb_timer,
+ if (!mod_timer(&t->hb_timer,
t->hb_interval + t->rto + jiffies)) {
sctp_transport_hold(t);
}
- } /* for (all transports) */
-
-} /* sctp_cmd_hb_timers_start() */
-
-/* Helper function to break out SCTP_CMD_SET_BIND_ADDR handling. */
-void
-sctp_cmd_set_bind_addrs(sctp_cmd_seq_t *cmds, sctp_association_t *asoc,
- sctp_bind_addr_t *bp)
+ }
+}
+
+/* Helper function to break out SCTP_CMD_SET_BIND_ADDR handling. */
+void sctp_cmd_set_bind_addrs(sctp_cmd_seq_t *cmds, sctp_association_t *asoc,
+ sctp_bind_addr_t *bp)
{
list_t *pos, *temp;
}
/* Free the temporary bind addr header, otherwise
- * there will a memory leak.
+ * there will a memory leak.
*/
sctp_bind_addr_free(bp);
-
-} /* sctp_cmd_set_bind_addrs() */
+}
-/* Helper function to handle the reception of an HEARTBEAT ACK. */
-static void
-sctp_cmd_transport_on(sctp_cmd_seq_t *cmds, sctp_association_t *asoc,
- sctp_transport_t *t, sctp_chunk_t *chunk)
+/* Helper function to handle the reception of an HEARTBEAT ACK. */
+static void sctp_cmd_transport_on(sctp_cmd_seq_t *cmds, sctp_association_t *asoc,
+ sctp_transport_t *t, sctp_chunk_t *chunk)
{
- sctp_sender_hb_info_t *hbinfo;
+ sctp_sender_hb_info_t *hbinfo;
- /* 8.3 Upon the receipt of the HEARTBEAT ACK, the sender of the
- * HEARTBEAT should clear the error counter of the destination
- * transport address to which the HEARTBEAT was sent.
+ /* 8.3 Upon the receipt of the HEARTBEAT ACK, the sender of the
+ * HEARTBEAT should clear the error counter of the destination
+ * transport address to which the HEARTBEAT was sent.
* The association's overall error count is also cleared.
*/
t->error_count = 0;
- t->asoc->overall_error_count = 0;
+ t->asoc->overall_error_count = 0;
- /* Mark the destination transport address as active if it is not so
- * marked.
+ /* Mark the destination transport address as active if it is not so
+ * marked.
*/
- if (!t->state.active) {
+ if (!t->state.active)
sctp_assoc_control_transport(asoc, t, SCTP_TRANSPORT_UP,
SCTP_HEARTBEAT_SUCCESS);
- }
-
+
/* The receiver of the HEARTBEAT ACK should also perform an
* RTT measurement for that destination transport address
* using the time value carried in the HEARTBEAT ACK chunk.
*/
- hbinfo = (sctp_sender_hb_info_t *)chunk->skb->data;
+ hbinfo = (sctp_sender_hb_info_t *) chunk->skb->data;
sctp_transport_update_rto(t, (jiffies - hbinfo->sent_at));
+}
-} /* sctp_cmd_transport_on() */
-
-/* Helper function to do a transport reset at the expiry of the hearbeat
- * timer.
- */
-static void
-sctp_cmd_transport_reset(sctp_cmd_seq_t *cmds, sctp_association_t *asoc,
- sctp_transport_t *t)
+/* Helper function to do a transport reset at the expiry of the hearbeat
+ * timer.
+ */
+static void sctp_cmd_transport_reset(sctp_cmd_seq_t *cmds,
+ sctp_association_t *asoc,
+ sctp_transport_t *t)
{
sctp_transport_lower_cwnd(t, SCTP_LOWER_CWND_INACTIVE);
/* Mark one strike against a transport. */
sctp_do_8_2_transport_strike(asoc, t);
- /* Update the heartbeat timer. */
- if (!mod_timer(&t->hb_timer, t->hb_interval + t->rto + jiffies)) {
+ /* Update the heartbeat timer. */
+ if (!mod_timer(&t->hb_timer, t->hb_interval + t->rto + jiffies))
sctp_transport_hold(t);
- }
+}
-} /* sctp_cmd_transport_reset() */
-
-/* Helper function to process the process SACK command. */
-static int
-sctp_cmd_process_sack(sctp_cmd_seq_t *cmds, sctp_association_t *asoc,
- sctp_sackhdr_t *sackh)
+/* Helper function to process the process SACK command. */
+static int sctp_cmd_process_sack(sctp_cmd_seq_t *cmds, sctp_association_t *asoc,
+ sctp_sackhdr_t *sackh)
{
int err;
if (sctp_sack_outqueue(&asoc->outqueue, sackh)) {
- /* There are no more TSNs awaiting SACK. */
+ /* There are no more TSNs awaiting SACK. */
err = sctp_do_sm(SCTP_EVENT_T_OTHER,
SCTP_ST_OTHER(SCTP_EVENT_NO_PENDING_TSN),
asoc->state, asoc->ep, asoc, NULL,
}
return err;
-
-} /* sctp_cmd_process_sack() */
-
+}
/* Helper function to set the timeout value for T2-SHUTDOWN timer and to set
- * the transport for a shutdown chunk.
+ * the transport for a shutdown chunk.
*/
-static void
-sctp_cmd_setup_t2(sctp_cmd_seq_t *cmds, sctp_association_t *asoc,
- sctp_chunk_t *chunk)
+static void sctp_cmd_setup_t2(sctp_cmd_seq_t *cmds, sctp_association_t *asoc,
+ sctp_chunk_t *chunk)
{
sctp_transport_t *t;
asoc->shutdown_last_sent_to = t;
asoc->timeouts[SCTP_EVENT_TIMEOUT_T2_SHUTDOWN] = t->rto;
chunk->transport = t;
-
- return;
-
-} /* sctp_cmd_setup_t2() */
-
+}
*/
static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_sm_statefuns.c,v 1.49 2002/08/21 18:34:04 jgrimm Exp $";
-#include <linux/config.h>
#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/ip.h>
#include <linux/net.h>
#include <linux/inet.h>
#include <net/sock.h>
-#ifndef CONFIG_INET_ECN
-#define CONFIG_INET_ECN
-#endif /* CONFIG_INET_ECN */
#include <net/inet_ecn.h>
#include <linux/skbuff.h>
#include <net/sctp/sctp.h>
#include <net/sctp/sctp_sm.h>
#include <net/sctp/sctp_structs.h>
-
/**********************************************************
* These are the state functions for handling chunk events.
**********************************************************/
/*
* Process the final SHUTDOWN COMPLETE.
- *
+ *
* Section: 4 (C) (diagram), 9.2
* Upon reception of the SHUTDOWN COMPLETE chunk the endpoint will verify
* that it is in SHUTDOWN-ACK-SENT state, if it is not the chunk should be
* ...
* - The receiver of a SHUTDOWN COMPLETE shall accept the packet if the
* Verification Tag field of the packet matches its own tag OR it is
- * set to its peer's tag and the T bit is set in the Chunk Flags.
- * Otherwise, the receiver MUST silently discard the packet and take
- * no further action. An endpoint MUST ignore the SHUTDOWN COMPLETE if
- * it is not in the SHUTDOWN-ACK-SENT state.
+ * set to its peer's tag and the T bit is set in the Chunk Flags.
+ * Otherwise, the receiver MUST silently discard the packet and take
+ * no further action. An endpoint MUST ignore the SHUTDOWN COMPLETE if
+ * it is not in the SHUTDOWN-ACK-SENT state.
*
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_do_4_C(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- sctp_chunk_t *chunk = arg;
+sctp_disposition_t sctp_sf_do_4_C(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ sctp_chunk_t *chunk = arg;
sctp_ulpevent_t *ev;
/* RFC 2960 6.10 Bundling
* An endpoint MUST NOT bundle INIT, INIT ACK or
* SHUTDOWN COMPLETE with any other chunks.
*/
- if (!chunk->singleton) {
+ if (!chunk->singleton)
return SCTP_DISPOSITION_VIOLATION;
- }
- /* RFC 2960 8.5.1 Exceptions in Verification Tag Rules
- *
+ /* RFC 2960 8.5.1 Exceptions in Verification Tag Rules
+ *
* (C) The receiver of a SHUTDOWN COMPLETE shall accept the
* packet if the Verification Tag field of the packet
* matches its own tag OR it is set to its peer's tag and
- * the T bit is set in the Chunk Flags. Otherwise, the
- * receiver MUST silently discard the packet and take no
+ * the T bit is set in the Chunk Flags. Otherwise, the
+ * receiver MUST silently discard the packet and take no
* further action....
*/
- if ((ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag)
- && !(sctp_test_T_bit(chunk)
- || (ntohl(chunk->sctp_hdr->vtag)
- != asoc->peer.i.init_tag))) {
- return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
- }
+ if ((ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag) &&
+ !(sctp_test_T_bit(chunk) ||
+ (ntohl(chunk->sctp_hdr->vtag) != asoc->peer.i.init_tag)))
+ return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
- /* RFC 2960 10.2 SCTP-to-ULP
+ /* RFC 2960 10.2 SCTP-to-ULP
*
* H) SHUTDOWN COMPLETE notification
- *
- * When SCTP completes the shutdown procedures (section 9.2) this
- * notification is passed to the upper layer.
- */
+ *
+ * When SCTP completes the shutdown procedures (section 9.2) this
+ * notification is passed to the upper layer.
+ */
ev = sctp_ulpevent_make_assoc_change(asoc, 0, SCTP_SHUTDOWN_COMP,
- 0, 0, 0, GFP_ATOMIC);
- if (!ev) { goto nomem; }
-
- sctp_add_cmd_sf(commands, SCTP_CMD_EVENT_ULP, SCTP_ULPEVENT(ev));
+ 0, 0, 0, GFP_ATOMIC);
+ if (!ev)
+ goto nomem;
+ sctp_add_cmd_sf(commands, SCTP_CMD_EVENT_ULP, SCTP_ULPEVENT(ev));
- /* Upon reception of the SHUTDOWN COMPLETE chunk the endpoint
- * will verify that it is in SHUTDOWN-ACK-SENT state, if it is
- * not the chunk should be discarded. If the endpoint is in
- * the SHUTDOWN-ACK-SENT state the endpoint should stop the
- * T2-shutdown timer and remove all knowledge of the
- * association (and thus the association enters the CLOSED
- * state).
- */
- sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
+ /* Upon reception of the SHUTDOWN COMPLETE chunk the endpoint
+ * will verify that it is in SHUTDOWN-ACK-SENT state, if it is
+ * not the chunk should be discarded. If the endpoint is in
+ * the SHUTDOWN-ACK-SENT state the endpoint should stop the
+ * T2-shutdown timer and remove all knowledge of the
+ * association (and thus the association enters the CLOSED
+ * state).
+ */
+ sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
SCTP_TO(SCTP_EVENT_TIMEOUT_T2_SHUTDOWN));
sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
return SCTP_DISPOSITION_DELETE_TCB;
- nomem:
+nomem:
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_do_4_C() */
-
+}
/*
* Discard the whole packet.
- *
+ *
* Section: 8.4 2)
*
- * 2) If the OOTB packet contains an ABORT chunk, the receiver MUST
- * silently discard the OOTB packet and take no further action.
+ * 2) If the OOTB packet contains an ABORT chunk, the receiver MUST
+ * silently discard the OOTB packet and take no further action.
* Otherwise,
*
* Verification Tag: No verification necessary
*
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_pdiscard(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_pdiscard(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
-
sctp_add_cmd_sf(commands, SCTP_CMD_DISCARD_PACKET, SCTP_NULL());
-
return SCTP_DISPOSITION_CONSUME;
-
-} /* sctp_sf_pdiscard() */
-
+}
/*
* Respond to a normal INIT chunk.
* We are the side that is being asked for an association.
- *
+ *
* Section: 5.1 Normal Establishment of an Association, B
* B) "Z" shall respond immediately with an INIT ACK chunk. The
- * destination IP address of the INIT ACK MUST be set to the source
- * IP address of the INIT to which this INIT ACK is responding. In
+ * destination IP address of the INIT ACK MUST be set to the source
+ * IP address of the INIT to which this INIT ACK is responding. In
* the response, besides filling in other parameters, "Z" must set the
- * Verification Tag field to Tag_A, and also provide its own
- * Verification Tag (Tag_Z) in the Initiate Tag field.
+ * Verification Tag field to Tag_A, and also provide its own
+ * Verification Tag (Tag_Z) in the Initiate Tag field.
*
* Verification Tag: No checking.
*
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_do_5_1B_init(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_do_5_1B_init(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
- sctp_chunk_t *chunk = arg;
+ sctp_chunk_t *chunk = arg;
sctp_chunk_t *repl;
sctp_association_t *new_asoc;
- /* If the packet is an OOTB packet which is temporarily on the
+ /* If the packet is an OOTB packet which is temporarily on the
* control endpoint, responding with an ABORT.
*/
-
- if (ep == sctp_sk((sctp_get_ctl_sock()))->ep) {
- return sctp_sf_ootb(ep, asoc, type, arg, commands);
- }
+ if (ep == sctp_sk((sctp_get_ctl_sock()))->ep)
+ return sctp_sf_ootb(ep, asoc, type, arg, commands);
/* 6.10 Bundling
* An endpoint MUST NOT bundle INIT, INIT ACK or
* SHUTDOWN COMPLETE with any other chunks.
*/
- if (!chunk->singleton) {
+ if (!chunk->singleton)
return SCTP_DISPOSITION_VIOLATION;
- }
/* Grab the INIT header. */
- chunk->subh.init_hdr = (sctp_inithdr_t *)chunk->skb->data;
+ chunk->subh.init_hdr = (sctp_inithdr_t *)chunk->skb->data;
- /* Tag the variable length parameters. */
+ /* Tag the variable length parameters. */
chunk->param_hdr.v =
skb_pull(chunk->skb, sizeof(sctp_inithdr_t));
new_asoc = sctp_make_temp_asoc(ep, chunk, GFP_ATOMIC);
- if (NULL == new_asoc) { goto nomem; }
+ if (!new_asoc)
+ goto nomem;
- /* FIXME: sctp_process_init can fail, but there is no
- * status nor handling.
+ /* FIXME: sctp_process_init can fail, but there is no
+ * status nor handling.
*/
sctp_process_init(new_asoc, chunk->chunk_hdr->type,
- sctp_source(chunk),
+ sctp_source(chunk),
(sctp_init_chunk_t *)chunk->chunk_hdr,
GFP_ATOMIC);
sctp_add_cmd_sf(commands, SCTP_CMD_NEW_ASOC, SCTP_ASOC(new_asoc));
- /* B) "Z" shall respond immediately with an INIT ACK chunk.
- */
-
+ /* B) "Z" shall respond immediately with an INIT ACK chunk. */
repl = sctp_make_init_ack(new_asoc, chunk, GFP_ATOMIC);
- if (!repl) { goto nomem_ack; }
-
+ if (!repl)
+ goto nomem_ack;
+
sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(repl));
/*
* new association. Otherwise, "Z" will be vulnerable to resource
* attacks.
*/
-
- sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
+ sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
return SCTP_DISPOSITION_DELETE_TCB;
sctp_association_free(new_asoc);
nomem:
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_do_5_1B_init() */
+}
/*
* Respond to a normal INIT ACK chunk.
* We are the side that is initiating the association.
- *
+ *
* Section: 5.1 Normal Establishment of an Association, C
* C) Upon reception of the INIT ACK from "Z", "A" shall stop the T1-init
- * timer and leave COOKIE-WAIT state. "A" shall then send the State
- * Cookie received in the INIT ACK chunk in a COOKIE ECHO chunk, start
+ * timer and leave COOKIE-WAIT state. "A" shall then send the State
+ * Cookie received in the INIT ACK chunk in a COOKIE ECHO chunk, start
* the T1-cookie timer, and enter the COOKIE-ECHOED state.
- *
+ *
* Note: The COOKIE ECHO chunk can be bundled with any pending outbound
* DATA chunks, but it MUST be the first chunk in the packet and
* until the COOKIE ACK is returned the sender MUST NOT send any
* other packets to the peer.
- *
+ *
* Verification Tag: 3.3.3
- * If the value of the Initiate Tag in a received INIT ACK chunk is
+ * If the value of the Initiate Tag in a received INIT ACK chunk is
* found to be 0, the receiver MUST treat it as an error and close the
* association by transmitting an ABORT.
*
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_do_5_1C_ack(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_do_5_1C_ack(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
- sctp_chunk_t *chunk = arg;
+ sctp_chunk_t *chunk = arg;
sctp_init_chunk_t *initchunk;
- uint32_t init_tag;
+ __u32 init_tag;
/* 6.10 Bundling
* An endpoint MUST NOT bundle INIT, INIT ACK or
* SHUTDOWN COMPLETE with any other chunks.
*/
- if (!chunk->singleton) {
+ if (!chunk->singleton)
return SCTP_DISPOSITION_VIOLATION;
- }
- /* Grab the INIT header.
- */
- chunk->subh.init_hdr = (sctp_inithdr_t *)chunk->skb->data;
+ /* Grab the INIT header. */
+ chunk->subh.init_hdr = (sctp_inithdr_t *) chunk->skb->data;
- init_tag = ntohl(chunk->subh.init_hdr->init_tag);
+ init_tag = ntohl(chunk->subh.init_hdr->init_tag);
- /* Verification Tag: 3.3.3
- * If the value of the Initiate Tag in a received INIT ACK
- * chunk is found to be 0, the receiver MUST treat it as an
- * error and close the association by transmitting an ABORT.
- */
- if (0 == init_tag) {
- sctp_chunk_t *reply;
- reply = sctp_make_abort(asoc, chunk, 0);
- if (!reply) { goto nomem; }
+ /* Verification Tag: 3.3.3
+ * If the value of the Initiate Tag in a received INIT ACK
+ * chunk is found to be 0, the receiver MUST treat it as an
+ * error and close the association by transmitting an ABORT.
+ */
+ if (!init_tag) {
+ sctp_chunk_t *reply = sctp_make_abort(asoc, chunk, 0);
+ if (!reply)
+ goto nomem;
sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
SCTP_CHUNK(reply));
- sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
+ sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
SCTP_STATE(SCTP_STATE_CLOSED));
sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB,
SCTP_NULL());
- return SCTP_DISPOSITION_DELETE_TCB;
- }
+ return SCTP_DISPOSITION_DELETE_TCB;
+ }
- /* Tag the variable length paramters. Note that we never
+ /* Tag the variable length paramters. Note that we never
* convert the parameters in an INIT chunk.
*/
chunk->param_hdr.v =
skb_pull(chunk->skb, sizeof(sctp_inithdr_t));
- initchunk = (sctp_init_chunk_t *)chunk->chunk_hdr;
+ initchunk = (sctp_init_chunk_t *) chunk->chunk_hdr;
- sctp_add_cmd_sf(commands, SCTP_CMD_PEER_INIT,
+ sctp_add_cmd_sf(commands, SCTP_CMD_PEER_INIT,
SCTP_PEER_INIT(initchunk));
-
- /* 5.1 C) "A" shall stop the T1-init timer and leave
- * COOKIE-WAIT state. "A" shall then ... start the T1-cookie
- * timer, and enter the COOKIE-ECHOED state.
- */
- sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
+ /* 5.1 C) "A" shall stop the T1-init timer and leave
+ * COOKIE-WAIT state. "A" shall then ... start the T1-cookie
+ * timer, and enter the COOKIE-ECHOED state.
+ */
+ sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
SCTP_TO(SCTP_EVENT_TIMEOUT_T1_INIT));
sctp_add_cmd_sf(commands, SCTP_CMD_COUNTER_RESET,
SCTP_COUNTER(SCTP_COUNTER_INIT_ERROR));
- sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_START,
+ sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_START,
SCTP_TO(SCTP_EVENT_TIMEOUT_T1_COOKIE));
- sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
+ sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
SCTP_STATE(SCTP_STATE_COOKIE_ECHOED));
-
- /* 5.1 C) "A" shall then send the State Cookie received in the
- * INIT ACK chunk in a COOKIE ECHO chunk, ...
- */
- sctp_add_cmd_sf(commands, SCTP_CMD_GEN_COOKIE_ECHO, SCTP_NULL());
+ /* 5.1 C) "A" shall then send the State Cookie received in the
+ * INIT ACK chunk in a COOKIE ECHO chunk, ...
+ */
+ sctp_add_cmd_sf(commands, SCTP_CMD_GEN_COOKIE_ECHO, SCTP_NULL());
return SCTP_DISPOSITION_CONSUME;
- nomem:
- return SCTP_DISPOSITION_NOMEM;
-} /* sctp_sf_do_5_1C_ack() */
+nomem:
+ return SCTP_DISPOSITION_NOMEM;
+}
/*
* Respond to a normal COOKIE ECHO chunk.
* We are the side that is being asked for an association.
- *
+ *
* Section: 5.1 Normal Establishment of an Association, D
- * D) Upon reception of the COOKIE ECHO chunk, Endpoint "Z" will reply
+ * D) Upon reception of the COOKIE ECHO chunk, Endpoint "Z" will reply
* with a COOKIE ACK chunk after building a TCB and moving to
* the ESTABLISHED state. A COOKIE ACK chunk may be bundled with
* any pending DATA chunks (and/or SACK chunks), but the COOKIE ACK
* chunk MUST be the first chunk in the packet.
- *
+ *
* IMPLEMENTATION NOTE: An implementation may choose to send the
* Communication Up notification to the SCTP user upon reception
* of a valid COOKIE ECHO chunk.
- *
+ *
* Verification Tag: 8.5.1 Exceptions in Verification Tag Rules
* D) Rules for packet carrying a COOKIE ECHO
- *
+ *
* - When sending a COOKIE ECHO, the endpoint MUST use the value of the
* Initial Tag received in the INIT ACK.
- *
+ *
* - The receiver of a COOKIE ECHO follows the procedures in Section 5.
- *
+ *
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_do_5_1D_ce(const sctp_endpoint_t *ep, const sctp_association_t *asoc,
- const sctp_subtype_t type, void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_do_5_1D_ce(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type, void *arg,
+ sctp_cmd_seq_t *commands)
{
- sctp_chunk_t *chunk = arg;
+ sctp_chunk_t *chunk = arg;
sctp_association_t *new_asoc;
sctp_init_chunk_t *peer_init;
sctp_chunk_t *repl;
sctp_ulpevent_t *ev;
int error = 0;
- /* If the packet is an OOTB packet which is temporarily on the
+ /* If the packet is an OOTB packet which is temporarily on the
* control endpoint, responding with an ABORT.
*/
-
- if (ep == sctp_sk((sctp_get_ctl_sock()))->ep) {
+ if (ep == sctp_sk((sctp_get_ctl_sock()))->ep)
return sctp_sf_ootb(ep, asoc, type, arg, commands);
- }
- /* "Decode" the chunk. We have no optional parameters so we
- * are in good shape.
- */
+ /* "Decode" the chunk. We have no optional parameters so we
+ * are in good shape.
+ */
chunk->subh.cookie_hdr =
(sctp_signed_cookie_t *)chunk->skb->data;
- skb_pull(chunk->skb,
+ skb_pull(chunk->skb,
ntohs(chunk->chunk_hdr->length) - sizeof(sctp_chunkhdr_t));
- /* 5.1 D) Upon reception of the COOKIE ECHO chunk, Endpoint
- * "Z" will reply with a COOKIE ACK chunk after building a TCB
- * and moving to the ESTABLISHED state.
- */
+ /* 5.1 D) Upon reception of the COOKIE ECHO chunk, Endpoint
+ * "Z" will reply with a COOKIE ACK chunk after building a TCB
+ * and moving to the ESTABLISHED state.
+ */
new_asoc = sctp_unpack_cookie(ep, asoc, chunk, GFP_ATOMIC, &error);
-
- /* FIXME:
+
+ /* FIXME:
* If the re-build failed, what is the proper error path
* from here?
*
* [We should abort the association. --piggy]
*/
-
- if (!new_asoc) {
+ if (!new_asoc) {
/* FIXME: Several errors are possible. A bad cookie should
* be silently discarded, but think about logging it too.
*/
switch (error) {
case -SCTP_IERROR_NOMEM:
goto nomem;
+
case -SCTP_IERROR_BAD_SIG:
default:
- return sctp_sf_pdiscard(ep, asoc, type,
+ return sctp_sf_pdiscard(ep, asoc, type,
arg, commands);
- }
-
- }
+ };
+ }
sctp_add_cmd_sf(commands, SCTP_CMD_NEW_ASOC, SCTP_ASOC(new_asoc));
-
sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
SCTP_STATE(SCTP_STATE_ESTABLISHED));
-
- sctp_add_cmd_sf(commands, SCTP_CMD_HB_TIMERS_START, SCTP_NULL());
-
- if (new_asoc->autoclose) {
- sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_START,
+ sctp_add_cmd_sf(commands, SCTP_CMD_HB_TIMERS_START, SCTP_NULL());
+
+ if (new_asoc->autoclose)
+ sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_START,
SCTP_TO(SCTP_EVENT_TIMEOUT_AUTOCLOSE));
- }
sctp_add_cmd_sf(commands, SCTP_CMD_TRANSMIT, SCTP_NULL());
- /* Re-build the bind address for the association is done in
- * the sctp_unpack_cookie() already.
+ /* Re-build the bind address for the association is done in
+ * the sctp_unpack_cookie() already.
*/
-
/* This is a brand-new association, so these are not yet side
* effects--it is safe to run them here.
*/
&chunk->subh.cookie_hdr->c.peer_addr, peer_init,
GFP_ATOMIC);
- repl = sctp_make_cookie_ack(new_asoc, chunk);
- if (!repl) { goto nomem_repl; }
+ repl = sctp_make_cookie_ack(new_asoc, chunk);
+ if (!repl)
+ goto nomem_repl;
sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(repl));
- /* RFC 2960 5.1 Normal Establishment of an Association
+ /* RFC 2960 5.1 Normal Establishment of an Association
*
* D) IMPLEMENTATION NOTE: An implementation may choose to
- * send the Communication Up notification to the SCTP user
- * upon reception of a valid COOKIE ECHO chunk.
+ * send the Communication Up notification to the SCTP user
+ * upon reception of a valid COOKIE ECHO chunk.
*/
- ev = sctp_ulpevent_make_assoc_change(new_asoc, 0, SCTP_COMM_UP, 0,
+ ev = sctp_ulpevent_make_assoc_change(new_asoc, 0, SCTP_COMM_UP, 0,
new_asoc->c.sinit_num_ostreams,
new_asoc->c.sinit_max_instreams,
GFP_ATOMIC);
- if (!ev) { goto nomem_ev; }
-
+ if (!ev)
+ goto nomem_ev;
+
sctp_add_cmd_sf(commands, SCTP_CMD_EVENT_ULP, SCTP_ULPEVENT(ev));
return SCTP_DISPOSITION_CONSUME;
nomem_ev:
sctp_free_chunk(repl);
+
nomem_repl:
sctp_association_free(new_asoc);
+
nomem:
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_do_5_1D_ce() */
-
+}
/*
* Respond to a normal COOKIE ACK chunk.
* We are the side that is being asked for an association.
- *
+ *
* RFC 2960 5.1 Normal Establishment of an Association
- *
- * E) Upon reception of the COOKIE ACK, endpoint "A" will move from the
+ *
+ * E) Upon reception of the COOKIE ACK, endpoint "A" will move from the
* COOKIE-ECHOED state to the ESTABLISHED state, stopping the T1-cookie
* timer. It may also notify its ULP about the successful
- * establishment of the association with a Communication Up
- * notification (see Section 10).
- *
+ * establishment of the association with a Communication Up
+ * notification (see Section 10).
+ *
* Verification Tag:
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_do_5_1E_ca(const sctp_endpoint_t *ep, const sctp_association_t *asoc,
- const sctp_subtype_t type, void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_do_5_1E_ca(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type, void *arg,
+ sctp_cmd_seq_t *commands)
{
sctp_ulpevent_t *ev;
-
+
/* RFC 2960 5.1 Normal Establishment of an Association
- *
+ *
* E) Upon reception of the COOKIE ACK, endpoint "A" will move
- * from the COOKIE-ECHOED state to the ESTABLISHED state,
- * stopping the T1-cookie timer.
+ * from the COOKIE-ECHOED state to the ESTABLISHED state,
+ * stopping the T1-cookie timer.
*/
-
- sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
+ sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
SCTP_TO(SCTP_EVENT_TIMEOUT_T1_COOKIE));
sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
SCTP_STATE(SCTP_STATE_ESTABLISHED));
-
- sctp_add_cmd_sf(commands, SCTP_CMD_HB_TIMERS_START, SCTP_NULL());
-
- if (asoc->autoclose) {
- sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_START,
+ sctp_add_cmd_sf(commands, SCTP_CMD_HB_TIMERS_START, SCTP_NULL());
+ if (asoc->autoclose)
+ sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_START,
SCTP_TO(SCTP_EVENT_TIMEOUT_AUTOCLOSE));
- }
-
sctp_add_cmd_sf(commands, SCTP_CMD_TRANSMIT, SCTP_NULL());
- /* It may also notify its ULP about the successful
- * establishment of the association with a Communication Up
- * notification (see Section 10).
- */
+ /* It may also notify its ULP about the successful
+ * establishment of the association with a Communication Up
+ * notification (see Section 10).
+ */
ev = sctp_ulpevent_make_assoc_change(asoc, 0, SCTP_COMM_UP,
0, asoc->c.sinit_num_ostreams,
asoc->c.sinit_max_instreams,
GFP_ATOMIC);
- if (!ev) { goto nomem; }
-
+ if (!ev)
+ goto nomem;
+
sctp_add_cmd_sf(commands, SCTP_CMD_EVENT_ULP, SCTP_ULPEVENT(ev));
return SCTP_DISPOSITION_CONSUME;
- nomem:
- return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_do_5_1E_ca() */
+nomem:
+ return SCTP_DISPOSITION_NOMEM;
+}
/* Generate a HEARTBEAT packet on the given transport. */
-sctp_disposition_t
-sctp_sf_sendbeat_8_3(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_sendbeat_8_3(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
- sctp_transport_t *transport = (sctp_transport_t *) arg;
+ sctp_transport_t *transport = (sctp_transport_t *) arg;
sctp_chunk_t *reply;
- sctp_sender_hb_info_t hbinfo;
- size_t paylen = 0;
+ sctp_sender_hb_info_t hbinfo;
+ size_t paylen = 0;
if (asoc->overall_error_count >= asoc->overall_error_threshold) {
/* CMD_ASSOC_FAILED calls CMD_DELETE_TCB. */
sctp_add_cmd_sf(commands, SCTP_CMD_ASSOC_FAILED, SCTP_NULL());
return SCTP_DISPOSITION_DELETE_TCB;
- }
+ }
- /* Section 3.3.5.
+ /* Section 3.3.5.
* The Sender-specific Heartbeat Info field should normally include
* information about the sender's current time when this HEARTBEAT
* chunk is sent and the destination transport address to which this
* HEARTBEAT is sent (see Section 8.3).
*/
- hbinfo.param_hdr.type = SCTP_PARAM_HEATBEAT_INFO;
- hbinfo.param_hdr.length = htons(sizeof(sctp_sender_hb_info_t));
- hbinfo.daddr = transport->ipaddr;
- hbinfo.sent_at = jiffies;
+ hbinfo.param_hdr.type = SCTP_PARAM_HEATBEAT_INFO;
+ hbinfo.param_hdr.length = htons(sizeof(sctp_sender_hb_info_t));
+ hbinfo.daddr = transport->ipaddr;
+ hbinfo.sent_at = jiffies;
/* Set rto_pending indicating that an RTT measurement is started
* with this heartbeat chunk.
- */
+ */
transport->rto_pending = 1;
- /* Send a heartbeat to our peer. */
-
- paylen = sizeof ( sctp_sender_hb_info_t );
+ /* Send a heartbeat to our peer. */
+ paylen = sizeof(sctp_sender_hb_info_t);
reply = sctp_make_heartbeat(asoc, transport, &hbinfo, paylen);
- if (NULL == reply) { goto nomem; }
+ if (!reply)
+ goto nomem;
sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
- SCTP_CHUNK(reply));
-
- /* Set transport error counter and association error counter
- * when sending heartbeat.
- */
- sctp_add_cmd_sf(commands, SCTP_CMD_TRANSPORT_RESET,
+ SCTP_CHUNK(reply));
+
+ /* Set transport error counter and association error counter
+ * when sending heartbeat.
+ */
+ sctp_add_cmd_sf(commands, SCTP_CMD_TRANSPORT_RESET,
SCTP_TRANSPORT(transport));
return SCTP_DISPOSITION_CONSUME;
-nomem:
- return SCTP_DISPOSITION_NOMEM;
-} /* sctp_sf_sendbeat_8_3() */
+nomem:
+ return SCTP_DISPOSITION_NOMEM;
+}
/*
* Process an heartbeat request.
- *
+ *
* Section: 8.3 Path Heartbeat
* The receiver of the HEARTBEAT should immediately respond with a
- * HEARTBEAT ACK that contains the Heartbeat Information field copied
+ * HEARTBEAT ACK that contains the Heartbeat Information field copied
* from the received HEARTBEAT chunk.
- *
+ *
* Verification Tag: 8.5 Verification Tag [Normal verification]
* When receiving an SCTP packet, the endpoint MUST ensure that the
* value in the Verification Tag field of the received SCTP packet
* matches its own Tag. If the received Verification Tag value does not
* match the receiver's own tag value, the receiver shall silently
- * discard the packet and shall not process it any further except for
+ * discard the packet and shall not process it any further except for
* those cases listed in Section 8.5.1 below.
- *
+ *
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_beat_8_3(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- sctp_chunk_t *chunk = arg;
+sctp_disposition_t sctp_sf_beat_8_3(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ sctp_chunk_t *chunk = arg;
sctp_chunk_t *reply;
size_t paylen = 0;
+ /* 8.5 When receiving an SCTP packet, the endpoint MUST ensure
+ * that the value in the Verification Tag field of the
+ * received SCTP packet matches its own Tag. If the received
+ * Verification Tag value does not match the receiver's own
+ * tag value, the receiver shall silently discard the packet...
+ */
+ if (ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag)
+ return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
- /* 8.5 When receiving an SCTP packet, the endpoint MUST ensure
- * that the value in the Verification Tag field of the
- * received SCTP packet matches its own Tag. If the received
- * Verification Tag value does not match the receiver's own
- * tag value, the receiver shall silently discard the packet...
- */
- if (ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag) {
- return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
- }
-
- /* 8.3 The receiver of the HEARTBEAT should immediately
- * respond with a HEARTBEAT ACK that contains the Heartbeat
- * Information field copied from the received HEARTBEAT chunk.
- */
-
- chunk->subh.hb_hdr = (sctp_heartbeathdr_t *)chunk->skb->data;
+ /* 8.3 The receiver of the HEARTBEAT should immediately
+ * respond with a HEARTBEAT ACK that contains the Heartbeat
+ * Information field copied from the received HEARTBEAT chunk.
+ */
+ chunk->subh.hb_hdr = (sctp_heartbeathdr_t *) chunk->skb->data;
paylen = ntohs(chunk->chunk_hdr->length) - sizeof(sctp_chunkhdr_t);
- skb_pull(chunk->skb, paylen);
+ skb_pull(chunk->skb, paylen);
- reply = sctp_make_heartbeat_ack(asoc, chunk,
+ reply = sctp_make_heartbeat_ack(asoc, chunk,
chunk->subh.hb_hdr, paylen);
- if (NULL == reply) { goto nomem; }
+ if (!reply)
+ goto nomem;
sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(reply));
-
return SCTP_DISPOSITION_CONSUME;
- nomem:
- return SCTP_DISPOSITION_NOMEM;
-} /* sctp_sf_beat_8_3() */
+nomem:
+ return SCTP_DISPOSITION_NOMEM;
+}
/*
* Process the returning HEARTBEAT ACK.
- *
+ *
* Section: 8.3 Path Heartbeat
* Upon the receipt of the HEARTBEAT ACK, the sender of the HEARTBEAT
* should clear the error counter of the destination transport
* HEARTBEAT ACK. The receiver of the HEARTBEAT ACK must also
* clear the association overall error count as well (as defined
* in section 8.1).
- *
+ *
* The receiver of the HEARTBEAT ACK should also perform an RTT
* measurement for that destination transport address using the time
* value carried in the HEARTBEAT ACK chunk.
- *
+ *
* Verification Tag: 8.5 Verification Tag [Normal verification]
*
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_backbeat_8_3(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_backbeat_8_3(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
- sctp_chunk_t *chunk = arg;
+ sctp_chunk_t *chunk = arg;
sockaddr_storage_t from_addr;
- sctp_transport_t *link;
+ sctp_transport_t *link;
sctp_sender_hb_info_t *hbinfo;
unsigned long max_interval;
- /* 8.5 When receiving an SCTP packet, the endpoint MUST ensure
- * that the value in the Verification Tag field of the
- * received SCTP packet matches its own Tag. ...
- */
- if (ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag) {
- return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
- }
+ /* 8.5 When receiving an SCTP packet, the endpoint MUST ensure
+ * that the value in the Verification Tag field of the
+ * received SCTP packet matches its own Tag. ...
+ */
+ if (ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag)
+ return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
- hbinfo= (sctp_sender_hb_info_t *)chunk->skb->data;
+ hbinfo = (sctp_sender_hb_info_t *) chunk->skb->data;
from_addr = hbinfo->daddr;
link = sctp_assoc_lookup_paddr(asoc, &from_addr);
-
- /* This should never happen, but lets log it if so. */
+
+ /* This should never happen, but lets log it if so. */
if (!link) {
- printk(KERN_WARNING __FUNCTION__
+ printk(KERN_WARNING __FUNCTION__
": Could not find address %d.%d.%d.%d\n",
NIPQUAD(from_addr.v4.sin_addr));
return SCTP_DISPOSITION_DISCARD;
max_interval = link->hb_interval + link->rto;
/* Check if the timestamp looks valid. */
-
- if (time_after(hbinfo->sent_at, jiffies)
- || time_after(jiffies, hbinfo->sent_at + max_interval)) {
+ if (time_after(hbinfo->sent_at, jiffies) ||
+ time_after(jiffies, hbinfo->sent_at + max_interval)) {
SCTP_DEBUG_PRINTK("%s: HEARTBEAT ACK with invalid timestamp
received for transport: %p\n",
__FUNCTION__, link);
return SCTP_DISPOSITION_DISCARD;
}
- /* 8.3 Upon the receipt of the HEARTBEAT ACK, the sender of
- * the HEARTBEAT should clear the error counter of the
- * destination transport address to which the HEARTBEAT was
- * sent and mark the destination transport address as active if
- * it is not so marked.
- */
- sctp_add_cmd_sf(commands, SCTP_CMD_TRANSPORT_ON,
- SCTP_TRANSPORT(link));
+ /* 8.3 Upon the receipt of the HEARTBEAT ACK, the sender of
+ * the HEARTBEAT should clear the error counter of the
+ * destination transport address to which the HEARTBEAT was
+ * sent and mark the destination transport address as active if
+ * it is not so marked.
+ */
+ sctp_add_cmd_sf(commands, SCTP_CMD_TRANSPORT_ON,
+ SCTP_TRANSPORT(link));
return SCTP_DISPOSITION_CONSUME;
nomem:__attribute__((unused))
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_backbeat_8_3() */
+}
/* Populate the verification/tie tags based on overlapping INIT
- * scenario.
- *
+ * scenario.
+ *
* Note: Do not use in CLOSED or SHUTDOWN-ACK-SENT state.
*/
-static void
-sctp_tietags_populate(sctp_association_t *new_asoc,
- const sctp_association_t *asoc)
+static void sctp_tietags_populate(sctp_association_t *new_asoc,
+ const sctp_association_t *asoc)
{
switch (asoc->state) {
new_asoc->c.my_ttag = asoc->c.my_vtag;
new_asoc->c.peer_ttag = 0;
break;
+
case SCTP_STATE_COOKIE_ECHOED:
new_asoc->c.my_vtag = asoc->c.my_vtag;
new_asoc->c.my_ttag = asoc->c.my_vtag;
new_asoc->c.peer_ttag = asoc->c.peer_vtag;
- break;
+ break;
/* 5.2.2 Unexpected INIT in States Other than CLOSED, COOKIE-ECHOED,
* COOKIE-WAIT and SHUTDOWN-ACK-SENT
new_asoc->c.my_ttag = asoc->c.my_vtag;
new_asoc->c.peer_ttag = asoc->c.peer_vtag;
break;
- }
-
+ };
+
/* Other parameters for the endpoint SHOULD be copied from the
- * existing parameters of the association (e.g. number of
- * outbound streams) into the INIT ACK and cookie.
- */
+ * existing parameters of the association (e.g. number of
+ * outbound streams) into the INIT ACK and cookie.
+ */
new_asoc->rwnd = asoc->rwnd;
new_asoc->c.sinit_num_ostreams = asoc->c.sinit_num_ostreams;
new_asoc->c.sinit_max_instreams = asoc->c.sinit_max_instreams;
new_asoc->c.initial_tsn = asoc->c.initial_tsn;
-
- return;
-
-} /* sctp_tietags_populate() */
+}
-
-/*
+/*
* Compare vtag/tietag values to determine unexpected COOKIE-ECHO
* handling action.
*
* RFC 2960 5.2.4 Handle a COOKIE ECHO when a TCB exists.
- *
- * Returns value representing action to be taken. These action values
- * correspond to Action/Description values in RFC 2960, Table 2.
+ *
+ * Returns value representing action to be taken. These action values
+ * correspond to Action/Description values in RFC 2960, Table 2.
*/
-static char
-sctp_tietags_compare(sctp_association_t *new_asoc,
- const sctp_association_t *asoc)
+static char sctp_tietags_compare(sctp_association_t *new_asoc,
+ const sctp_association_t *asoc)
{
-
/* In this case, the peer may have restarted. */
- if ((asoc->c.my_vtag != new_asoc->c.my_vtag)
- && (asoc->c.peer_vtag != new_asoc->c.peer_vtag)
- && (asoc->c.my_vtag == new_asoc->c.my_ttag)
- && (asoc->c.peer_vtag == new_asoc->c.peer_ttag)) {
-
+ if ((asoc->c.my_vtag != new_asoc->c.my_vtag) &&
+ (asoc->c.peer_vtag != new_asoc->c.peer_vtag) &&
+ (asoc->c.my_vtag == new_asoc->c.my_ttag) &&
+ (asoc->c.peer_vtag == new_asoc->c.peer_ttag))
return 'A';
- }
/* Collision case D.
* Note: Test case D first, otherwise it may be incorrectly
* identified as second case of B if the value of the Tie_tag is
* not filled into the state cookie.
*/
- if ((asoc->c.my_vtag == new_asoc->c.my_vtag)
- && (asoc->c.peer_vtag == new_asoc->c.peer_vtag)) {
-
+ if ((asoc->c.my_vtag == new_asoc->c.my_vtag) &&
+ (asoc->c.peer_vtag == new_asoc->c.peer_vtag))
return 'D';
- }
-
+
/* Collision case B. */
- if (( asoc->c.my_vtag == new_asoc->c.my_vtag)
- && ((asoc->c.peer_vtag != new_asoc->c.peer_vtag)
- || (!new_asoc->c.my_ttag && !new_asoc->c.peer_ttag))) {
-
+ if ((asoc->c.my_vtag == new_asoc->c.my_vtag) &&
+ ((asoc->c.peer_vtag != new_asoc->c.peer_vtag) ||
+ (!new_asoc->c.my_ttag && !new_asoc->c.peer_ttag)))
return 'B';
- }
-
+
/* Collision case C. */
- if ((asoc->c.my_vtag != new_asoc->c.my_vtag)
- && ( asoc->c.peer_vtag == new_asoc->c.peer_vtag)
- && (0 == new_asoc->c.my_ttag)
- && (0 == new_asoc->c.peer_ttag)) {
+ if ((asoc->c.my_vtag != new_asoc->c.my_vtag) &&
+ (asoc->c.peer_vtag == new_asoc->c.peer_vtag) &&
+ (0 == new_asoc->c.my_ttag) &&
+ (0 == new_asoc->c.peer_ttag))
return 'C';
- }
-
- return 'E'; /* No such case available. */
-
-} /* sctp_tietags_compare() */
+ return 'E'; /* No such case available. */
+}
/* Common helper routine for both duplicate and simulataneous INIT
* chunk handling.
*/
-static sctp_disposition_t
-sctp_sf_do_unexpected_init(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
+static sctp_disposition_t sctp_sf_do_unexpected_init(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc, const sctp_subtype_t type,
+ void *arg, sctp_cmd_seq_t *commands)
+{
sctp_chunk_t *chunk = arg;
sctp_chunk_t *repl;
- sctp_association_t *new_asoc;
+ sctp_association_t *new_asoc;
- /* 6.10 Bundling
+ /* 6.10 Bundling
* An endpoint MUST NOT bundle INIT, INIT ACK or
* SHUTDOWN COMPLETE with any other chunks.
*/
- if (!chunk->singleton) {
+ if (!chunk->singleton)
return SCTP_DISPOSITION_VIOLATION;
- }
+ /* Grab the INIT header. */
+ chunk->subh.init_hdr = (sctp_inithdr_t *) chunk->skb->data;
- /* Grab the INIT header. */
- chunk->subh.init_hdr = (sctp_inithdr_t *)chunk->skb->data;
-
- /* Tag the variable length parameters.
- */
+ /* Tag the variable length parameters. */
chunk->param_hdr.v =
skb_pull(chunk->skb, sizeof(sctp_inithdr_t));
- /*
+ /*
* Other parameters for the endpoint SHOULD be copied from the
* existing parameters of the association (e.g. number of
* outbound streams) into the INIT ACK and cookie.
- * FIXME: We are copying parameters from the endpoint not the
- * association.
+ * FIXME: We are copying parameters from the endpoint not the
+ * association.
*/
new_asoc = sctp_make_temp_asoc(ep, chunk, GFP_ATOMIC);
- if (!new_asoc) { goto nomem; }
-
+ if (!new_asoc)
+ goto nomem;
+
/* In the outbound INIT ACK the endpoint MUST copy its current
- * Verification Tag and Peers Verification tag into a reserved
- * place (local tie-tag and per tie-tag) within the state cookie.
+ * Verification Tag and Peers Verification tag into a reserved
+ * place (local tie-tag and per tie-tag) within the state cookie.
*/
sctp_process_init(new_asoc, chunk->chunk_hdr->type,
- sctp_source(chunk),
+ sctp_source(chunk),
(sctp_init_chunk_t *)chunk->chunk_hdr,
GFP_ATOMIC);
sctp_tietags_populate(new_asoc, asoc);
-
- /* B) "Z" shall respond immediately with an INIT ACK chunk.
- */
- repl = sctp_make_init_ack(new_asoc, chunk, GFP_ATOMIC);
- if (!repl) { goto nomem; }
+ /* B) "Z" shall respond immediately with an INIT ACK chunk. */
+ repl = sctp_make_init_ack(new_asoc, chunk, GFP_ATOMIC);
+ if (!repl)
+ goto nomem;
sctp_add_cmd_sf(commands, SCTP_CMD_NEW_ASOC, SCTP_ASOC(new_asoc));
- sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(repl));
-
+ sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(repl));
+
/*
* Note: After sending out INIT ACK with the State Cookie parameter,
* "Z" MUST NOT allocate any resources for this new association.
* Otherwise, "Z" will be vulnerable to resource attacks.
*/
-
- sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
-
- return SCTP_DISPOSITION_CONSUME;
+ sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
+ return SCTP_DISPOSITION_CONSUME;
nomem:
- return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_do_unexpected_init() */
+ return SCTP_DISPOSITION_NOMEM;
+}
/*
* Handle simultanous INIT.
* This means we started an INIT and then we got an INIT request from
* our peer.
- *
+ *
* Section: 5.2.1 INIT received in COOKIE-WAIT or COOKIE-ECHOED State (Item B)
* This usually indicates an initialization collision, i.e., each
* endpoint is attempting, at about the same time, to establish an
* association with the other endpoint.
- *
+ *
* Upon receipt of an INIT in the COOKIE-WAIT or COOKIE-ECHOED state, an
* endpoint MUST respond with an INIT ACK using the same parameters it
* sent in its original INIT chunk (including its Verification Tag,
* newly received INIT chunk. The endpoint shall also generate a State
* Cookie with the INIT ACK. The endpoint uses the parameters sent in its
* INIT to calculate the State Cookie.
- *
- * After that, the endpoint MUST NOT change its state, the T1-init
- * timer shall be left running and the corresponding TCB MUST NOT be
+ *
+ * After that, the endpoint MUST NOT change its state, the T1-init
+ * timer shall be left running and the corresponding TCB MUST NOT be
* destroyed. The normal procedures for handling State Cookies when
* a TCB exists will resolve the duplicate INITs to a single association.
- *
+ *
* For an endpoint that is in the COOKIE-ECHOED state it MUST populate
* its Tie-Tags with the Tag information of itself and its peer (see
* section 5.2.2 for a description of the Tie-Tags).
- *
+ *
* Verification Tag: Not explicit, but an INIT can not have a valid
* verification tag, so we skip the check.
*
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_do_5_2_1_siminit(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- sctp_disposition_t retval;
-
+sctp_disposition_t sctp_sf_do_5_2_1_siminit(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
/* Call helper to do the real work for both simulataneous and
- * duplicate INIT chunk handling.
+ * duplicate INIT chunk handling.
*/
-
- retval = sctp_sf_do_unexpected_init(ep, asoc, type, arg, commands);
-
- return retval;
-
-} /* sctp_sf_do_5_2_1_siminit() */
-
-
+ return sctp_sf_do_unexpected_init(ep, asoc, type, arg, commands);
+}
/*
- * Handle duplicated INIT messages. These are usually delayed
+ * Handle duplicated INIT messages. These are usually delayed
* restransmissions.
- *
+ *
* Section: 5.2.2 Unexpected INIT in States Other than CLOSED,
* COOKIE-ECHOED and COOKIE-WAIT
*
* (i.e. the endpoint is in a COOKIE-WAIT state), the Tie-Tags MUST be
* set to 0 (indicating that no previous TCB existed). The INIT ACK and
* State Cookie are populated as specified in section 5.2.1.
- *
+ *
* Verification Tag: Not specifed, but an INIT has no way of knowing
* what the verification tag could be, so we ignore it.
*
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
- * The return value is the disposition of the chunk. */
-sctp_disposition_t
-sctp_sf_do_5_2_2_dupinit(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+ * The return value is the disposition of the chunk.
+ */
+sctp_disposition_t sctp_sf_do_5_2_2_dupinit(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
- sctp_disposition_t retval;
-
/* Call helper to do the real work for both simulataneous and
- * duplicate INIT chunk handling.
+ * duplicate INIT chunk handling.
*/
-
- retval = sctp_sf_do_unexpected_init(ep, asoc, type, arg, commands);
- return retval;
-
-} /* sctp_sf_do_5_2_2_dupinit() */
+ return sctp_sf_do_unexpected_init(ep, asoc, type, arg, commands);
+}
/* Unexpected COOKIE-ECHO handlerfor peer restart (Table 2, action 'A')
- *
- * Section 5.2.4
+ *
+ * Section 5.2.4
* A) In this case, the peer may have restarted.
*/
-static sctp_disposition_t
-sctp_sf_do_dupcook_a(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- sctp_chunk_t *chunk,
- sctp_cmd_seq_t *commands,
- sctp_association_t *new_asoc)
+static sctp_disposition_t sctp_sf_do_dupcook_a(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ sctp_chunk_t *chunk,
+ sctp_cmd_seq_t *commands,
+ sctp_association_t *new_asoc)
{
sctp_init_chunk_t *peer_init;
sctp_ulpevent_t *ev;
sctp_chunk_t *repl;
- sctp_transport_t *new_addr, *addr;
+ sctp_transport_t *new_addr, *addr;
list_t *pos, *pos2, *temp;
int found, error;
peer_init = &chunk->subh.cookie_hdr->c.peer_init[0];
sctp_process_init(new_asoc, chunk->chunk_hdr->type,
sctp_source(chunk), peer_init, GFP_ATOMIC);
-
- /* Make sure peer is not adding new addresses. */
+ /* Make sure peer is not adding new addresses. */
found = 0;
new_addr = NULL;
list_for_each(pos, &new_asoc->peer.transport_addr_list) {
new_addr = list_entry(pos, sctp_transport_t, transports);
found = 1;
- list_for_each_safe(pos2, temp,
+ list_for_each_safe(pos2, temp,
&asoc->peer.transport_addr_list) {
addr = list_entry(pos2, sctp_transport_t, transports);
if (!sctp_cmp_addr_exact(&new_addr->ipaddr,
- &addr->ipaddr)){
+ &addr->ipaddr)) {
found = 0;
break;
}
}
- if (!found){ break; }
- } /* list_for_each(...) */
-
- if (!found){
+ if (!found)
+ break;
+ }
+
+ if (!found) {
sctp_bind_addr_t *bp;
sctpParam_t rawaddr;
int len;
bp = sctp_bind_addr_new(GFP_ATOMIC);
- if (!bp) { goto nomem; }
+ if (!bp)
+ goto nomem;
error = sctp_add_bind_addr(bp, &new_addr->ipaddr, GFP_ATOMIC);
- if (error) { goto nomem_add; }
+ if (error)
+ goto nomem_add;
rawaddr = sctp_bind_addrs_to_raw(bp, &len, GFP_ATOMIC);
- if (!rawaddr.v) { goto nomem_raw; }
+ if (!rawaddr.v)
+ goto nomem_raw;
repl = sctp_make_abort(asoc, chunk, len+sizeof(sctp_errhdr_t));
-
- if (!repl) { goto nomem_abort; }
+ if (!repl)
+ goto nomem_abort;
sctp_init_cause(repl, SCTP_ERROR_RESTART, rawaddr.v, len);
-
sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(repl));
return SCTP_DISPOSITION_CONSUME;
+
nomem_abort:
kfree(rawaddr.v);
+
nomem_raw:
nomem_add:
- sctp_bind_addr_free(bp);
- goto nomem;
+ sctp_bind_addr_free(bp);
+ goto nomem;
}
-
+
/* For now, fail any unsent/unacked data. Consider the optional
* choice of resending of this data.
*/
/* Update the content of current association. */
sctp_add_cmd_sf(commands, SCTP_CMD_UPDATE_ASSOC, SCTP_ASOC(new_asoc));
-
+
repl = sctp_make_cookie_ack(new_asoc, chunk);
- if (!repl) { goto nomem; }
-
- sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(repl));
-
+ if (!repl)
+ goto nomem;
+
+ sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(repl));
+
/* Report association restart to upper layer. */
- ev = sctp_ulpevent_make_assoc_change(asoc, 0, SCTP_RESTART, 0,
+ ev = sctp_ulpevent_make_assoc_change(asoc, 0, SCTP_RESTART, 0,
new_asoc->c.sinit_num_ostreams,
new_asoc->c.sinit_max_instreams,
GFP_ATOMIC);
- if (NULL == ev) { goto nomem_ev; }
-
- sctp_add_cmd_sf(commands, SCTP_CMD_EVENT_ULP, SCTP_ULPEVENT(ev));
+ if (!ev)
+ goto nomem_ev;
+ sctp_add_cmd_sf(commands, SCTP_CMD_EVENT_ULP, SCTP_ULPEVENT(ev));
return SCTP_DISPOSITION_CONSUME;
nomem_ev:
sctp_free_chunk(repl);
+
nomem:
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_dupcook_a() */
+}
/* Unexpected COOKIE-ECHO handler for setup collision (Table 2, action 'B')
- *
- * Section 5.2.4
+ *
+ * Section 5.2.4
* B) In this case, both sides may be attempting to start an association
* at about the same time but the peer endpoint started its INIT
* after responding to the local endpoint's INIT
*/
/* This case represents an intialization collision. */
-static sctp_disposition_t
-sctp_sf_do_dupcook_b(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- sctp_chunk_t *chunk,
- sctp_cmd_seq_t *commands,
- sctp_association_t *new_asoc)
+static sctp_disposition_t sctp_sf_do_dupcook_b(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ sctp_chunk_t *chunk,
+ sctp_cmd_seq_t *commands,
+ sctp_association_t *new_asoc)
{
sctp_init_chunk_t *peer_init;
sctp_ulpevent_t *ev;
- sctp_chunk_t *repl;
+ sctp_chunk_t *repl;
/* new_asoc is a brand-new association, so these are not yet
* side effects--it is safe to run them here.
*/
peer_init = &chunk->subh.cookie_hdr->c.peer_init[0];
- sctp_process_init(new_asoc, chunk->chunk_hdr->type,
+ sctp_process_init(new_asoc, chunk->chunk_hdr->type,
sctp_source(chunk), peer_init, GFP_ATOMIC);
-
- /* Update the content of current association. */
+
+ /* Update the content of current association. */
sctp_add_cmd_sf(commands, SCTP_CMD_UPDATE_ASSOC, SCTP_ASOC(new_asoc));
-
- sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
+ sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
SCTP_STATE(SCTP_STATE_ESTABLISHED));
-
sctp_add_cmd_sf(commands, SCTP_CMD_HB_TIMERS_START, SCTP_NULL());
-
+
repl = sctp_make_cookie_ack(new_asoc, chunk);
- if (NULL == repl) { goto nomem; }
-
- sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(repl));
+ if (!repl)
+ goto nomem;
+
+ sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(repl));
sctp_add_cmd_sf(commands, SCTP_CMD_TRANSMIT, SCTP_NULL());
-
+
/* RFC 2960 5.1 Normal Establishment of an Association
*
* D) IMPLEMENTATION NOTE: An implementation may choose to
* upon reception of a valid COOKIE ECHO chunk.
*/
ev = sctp_ulpevent_make_assoc_change(asoc, 0, SCTP_COMM_UP, 0,
- new_asoc->c.sinit_num_ostreams,
- new_asoc->c.sinit_max_instreams,
- GFP_ATOMIC);
- if (NULL == ev) { goto nomem_ev; }
-
+ new_asoc->c.sinit_num_ostreams,
+ new_asoc->c.sinit_max_instreams,
+ GFP_ATOMIC);
+ if (!ev)
+ goto nomem_ev;
+
sctp_add_cmd_sf(commands, SCTP_CMD_EVENT_ULP, SCTP_ULPEVENT(ev));
-
return SCTP_DISPOSITION_CONSUME;
+
nomem_ev:
sctp_free_chunk(repl);
nomem:
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_dupcook_b() */
+}
/* Unexpected COOKIE-ECHO handler for setup collision (Table 2, action 'C')
- *
- * Section 5.2.4
+ *
+ * Section 5.2.4
* C) In this case, the local endpoint's cookie has arrived late.
* Before it arrived, the local endpoint sent an INIT and received an
* INIT-ACK and finally sent a COOKIE ECHO with the peer's same tag
* but a new tag of its own.
*/
/* This case represents an intialization collision. */
-static sctp_disposition_t
-sctp_sf_do_dupcook_c(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- sctp_chunk_t *chunk,
- sctp_cmd_seq_t *commands,
- sctp_association_t *new_asoc)
+static sctp_disposition_t sctp_sf_do_dupcook_c(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ sctp_chunk_t *chunk,
+ sctp_cmd_seq_t *commands,
+ sctp_association_t *new_asoc)
{
- /* The cookie should be silently discarded.
+ /* The cookie should be silently discarded.
* The endpoint SHOULD NOT change states and should leave
* any timers running.
*/
return SCTP_DISPOSITION_DISCARD;
-
-} /* sctp_sf_do_dupcook_c() */
+}
/* Unexpected COOKIE-ECHO handler lost chunk (Table 2, action 'D')
- *
- * Section 5.2.4
- *
+ *
+ * Section 5.2.4
+ *
* D) When both local and remote tags match the endpoint should always
* enter the ESTABLISHED state, if it has not already done so.
*/
/* This case represents an intialization collision. */
-static sctp_disposition_t
-sctp_sf_do_dupcook_d(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- sctp_chunk_t *chunk,
- sctp_cmd_seq_t *commands,
- sctp_association_t *new_asoc)
+static sctp_disposition_t sctp_sf_do_dupcook_d(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ sctp_chunk_t *chunk,
+ sctp_cmd_seq_t *commands,
+ sctp_association_t *new_asoc)
{
sctp_ulpevent_t *ev = NULL;
sctp_chunk_t *repl;
if (SCTP_STATE_ESTABLISHED != asoc->state) {
sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
SCTP_STATE(SCTP_STATE_ESTABLISHED));
-
sctp_add_cmd_sf(commands, SCTP_CMD_HB_TIMERS_START,
SCTP_NULL());
-
+
/* RFC 2960 5.1 Normal Establishment of an Association
*
* D) IMPLEMENTATION NOTE: An implementation may choose
* SCTP user upon reception of a valid COOKIE
* ECHO chunk.
*/
- ev = sctp_ulpevent_make_assoc_change(new_asoc, 0,
- SCTP_COMM_UP, 0,
+ ev = sctp_ulpevent_make_assoc_change(new_asoc, 0,
+ SCTP_COMM_UP, 0,
new_asoc->c.sinit_num_ostreams,
new_asoc->c.sinit_max_instreams,
GFP_ATOMIC);
-
- if (NULL == ev) { goto nomem; }
-
- sctp_add_cmd_sf(commands, SCTP_CMD_EVENT_ULP,
+ if (!ev)
+ goto nomem;
+ sctp_add_cmd_sf(commands, SCTP_CMD_EVENT_ULP,
SCTP_ULPEVENT(ev));
}
-
sctp_add_cmd_sf(commands, SCTP_CMD_TRANSMIT, SCTP_NULL());
-
+
repl = sctp_make_cookie_ack(new_asoc, chunk);
- if (NULL == repl) { goto nomem; }
-
- sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(repl));
-
- sctp_add_cmd_sf(commands, SCTP_CMD_TRANSMIT, SCTP_NULL());
+ if (!repl)
+ goto nomem;
+ sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(repl));
+ sctp_add_cmd_sf(commands, SCTP_CMD_TRANSMIT, SCTP_NULL());
return SCTP_DISPOSITION_CONSUME;
+
nomem:
- if (ev) { sctp_ulpevent_free(ev); }
+ if (ev)
+ sctp_ulpevent_free(ev);
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_do_dupcook_c() */
+}
/*
* Handle a duplicate COOKIE-ECHO. This usually means a cookie-carrying
* chunk was retransmitted and then delayed in the network.
- *
+ *
* Section: 5.2.4 Handle a COOKIE ECHO when a TCB exists
- *
+ *
* Verification Tag: None. Do cookie validation.
*
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_do_5_2_4_dupcook(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_do_5_2_4_dupcook(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
- sctp_disposition_t retval;
- sctp_chunk_t *chunk = arg;
- sctp_association_t *new_asoc;
- int error = 0;
- char action;
-
- /* "Decode" the chunk. We have no optional parameters so we
- * are in good shape.
- */
- chunk->subh.cookie_hdr =
- (sctp_signed_cookie_t *)chunk->skb->data;
- skb_pull(chunk->skb,
- ntohs(chunk->chunk_hdr->length) - sizeof(sctp_chunkhdr_t));
-
- /* In RFC 2960 5.2.4 3, if both Verification Tags in the State Cookie
- * of a duplicate COOKIE ECHO match the Verification Tags of the
- * current association, consider the State Cookie valid even if
- * the lifespan is exceeded.
- */
- new_asoc = sctp_unpack_cookie(ep, asoc, chunk, GFP_ATOMIC, &error);
-
- /* FIXME:
- * If the re-build failed, what is the proper error path
- * from here?
- *
- * [We should abort the association. --piggy]
- */
-
- if (NULL == new_asoc) {
+ sctp_disposition_t retval;
+ sctp_chunk_t *chunk = arg;
+ sctp_association_t *new_asoc;
+ int error = 0;
+ char action;
+
+ /* "Decode" the chunk. We have no optional parameters so we
+ * are in good shape.
+ */
+ chunk->subh.cookie_hdr =
+ (sctp_signed_cookie_t *) chunk->skb->data;
+ skb_pull(chunk->skb,
+ ntohs(chunk->chunk_hdr->length) - sizeof(sctp_chunkhdr_t));
+
+ /* In RFC 2960 5.2.4 3, if both Verification Tags in the State Cookie
+ * of a duplicate COOKIE ECHO match the Verification Tags of the
+ * current association, consider the State Cookie valid even if
+ * the lifespan is exceeded.
+ */
+ new_asoc = sctp_unpack_cookie(ep, asoc, chunk, GFP_ATOMIC, &error);
+
+ /* FIXME:
+ * If the re-build failed, what is the proper error path
+ * from here?
+ *
+ * [We should abort the association. --piggy]
+ */
+ if (!new_asoc) {
/* FIXME: Several errors are possible. A bad cookie should
* be silently discarded, but think about logging it too.
*/
switch (error) {
case -SCTP_IERROR_NOMEM:
goto nomem;
+
case -SCTP_IERROR_BAD_SIG:
default:
- return sctp_sf_pdiscard(ep, asoc, type,
+ return sctp_sf_pdiscard(ep, asoc, type,
arg, commands);
- }
- }
-
- /* Compare the tie_tag in cookie with the verification tag of
- * current association.
- */
- action = sctp_tietags_compare(new_asoc, asoc);
-
- switch (action) {
- case 'A': /* Association restart. */
+ };
+ }
+
+ /* Compare the tie_tag in cookie with the verification tag of
+ * current association.
+ */
+ action = sctp_tietags_compare(new_asoc, asoc);
+
+ switch (action) {
+ case 'A': /* Association restart. */
retval = sctp_sf_do_dupcook_a(ep, asoc, chunk, commands,
new_asoc);
break;
- case 'B': /* Collision case B. */
- retval = sctp_sf_do_dupcook_b(ep, asoc, chunk, commands,
- new_asoc);
- break;
- case 'C': /* Collisioun case C. */
+
+ case 'B': /* Collision case B. */
+ retval = sctp_sf_do_dupcook_b(ep, asoc, chunk, commands,
+ new_asoc);
+ break;
+
+ case 'C': /* Collisioun case C. */
retval = sctp_sf_do_dupcook_c(ep, asoc, chunk, commands,
new_asoc);
break;
-
- case 'D': /* Collision case D. */
+
+ case 'D': /* Collision case D. */
retval = sctp_sf_do_dupcook_d(ep, asoc, chunk, commands,
new_asoc);
break;
- default: /* No such case, discard it. */
- printk(KERN_WARNING __FUNCTION__ ":unknown case\n");
+ default: /* No such case, discard it. */
+ printk(KERN_WARNING __FUNCTION__ ":unknown case\n");
retval = SCTP_DISPOSITION_DISCARD;
break;
- } /* switch */
-
- /* Delete the tempory new association. */
- sctp_add_cmd_sf(commands, SCTP_CMD_NEW_ASOC, SCTP_ASOC(new_asoc));
- sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
-
+ };
+
+ /* Delete the tempory new association. */
+ sctp_add_cmd_sf(commands, SCTP_CMD_NEW_ASOC, SCTP_ASOC(new_asoc));
+ sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
+
return retval;
+
nomem:
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_do_5_2_4_dupcook() */
-
+}
#if 0
/*
* Handle a Stale COOKIE Error
- *
+ *
* Section: 5.2.6 Handle Stale COOKIE Error
* If the association is in the COOKIE-ECHOED state, the endpoint may elect
- * one of the following three alternatives.
+ * one of the following three alternatives.
* ...
* 3) Send a new INIT chunk to the endpoint, adding a Cookie
* Preservative parameter requesting an extension to the lifetime of
- * the State Cookie. When calculating the time extension, an
+ * the State Cookie. When calculating the time extension, an
* implementation SHOULD use the RTT information measured based on the
- * previous COOKIE ECHO / ERROR exchange, and should add no more
- * than 1 second beyond the measured RTT, due to long State Cookie
+ * previous COOKIE ECHO / ERROR exchange, and should add no more
+ * than 1 second beyond the measured RTT, due to long State Cookie
* lifetimes making the endpoint more subject to a replay attack.
- *
+ *
* Verification Tag: Not explicit, but safe to ignore.
*
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-do_5_2_6_stale(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- sctp_chunk_t *chunk = arg;
-
- /* This is not a real chunk type. It is a subtype of the
- * ERROR chunk type. The ERROR chunk processing will bring us
- * here.
- */
+sctp_disposition_t do_5_2_6_stale(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ sctp_chunk_t *chunk = arg;
+
+ /* This is not a real chunk type. It is a subtype of the
+ * ERROR chunk type. The ERROR chunk processing will bring us
+ * here.
+ */
sctp_chunk_t *in_packet;
stp_chunk_t *reply;
- sctp_inithdr_t initack;
- uint8_t *addrs;
- int addrs_len;
+ sctp_inithdr_t initack;
+ __u8 *addrs;
+ int addrs_len;
time_t rtt;
- extern struct timespec now;
-
- struct sctpCookiePreserve bht;
+ struct sctpCookiePreserve bht;
- /* If we have gotten too many failures, give up. */
- if (1 + asoc->counters[SctpCounterInits] > asoc->max_init_attempts){
- /* FIXME: Move to new ulpevent. */
- retval->event_up = sctp_make_ulp_init_timeout(asoc);
- if (NULL == retval->event_up) { goto nomem; }
+ /* If we have gotten too many failures, give up. */
+ if (1 + asoc->counters[SctpCounterInits] > asoc->max_init_attempts) {
+ /* FIXME: Move to new ulpevent. */
+ retval->event_up = sctp_make_ulp_init_timeout(asoc);
+ if (!retval->event_up)
+ goto nomem;
sctp_add_cmd_sf(retval->commands, SCTP_CMD_DELETE_TCB,
SCTP_NULL());
- return SCTP_DISPOSITION_DELETE_TCB;
- }
-
+ return SCTP_DISPOSITION_DELETE_TCB;
+ }
+
retval->counters[0] = SCTP_COUNTER_INCR;
retval->counters[0] = SctpCounterInits;
retval->counters[1] = 0;
retval->counters[1] = 0;
-
+
/* Calculate the RTT in ms. */
- /* BUG--we should get the send time of the HEARTBEAT REQUEST. */
+ /* BUG--we should get the send time of the HEARTBEAT REQUEST. */
in_packet = chunk;
- rtt = 1000 * timeval_sub(in_packet->skb->stamp,
+ rtt = 1000 * timeval_sub(in_packet->skb->stamp,
asoc->c.state_timestamp);
- /* When calculating the time extension, an implementation
- * SHOULD use the RTT information measured based on the
- * previous COOKIE ECHO / ERROR exchange, and should add no
- * more than 1 second beyond the measured RTT, due to long
- * State Cookie lifetimes making the endpoint more subject to
- * a replay attack.
- */
- bht.p = {SCTP_COOKIE_PRESERVE, 8};
- bht.extraTime = htonl(rtt + 1000);
-
- initack.init_tag = htonl(asoc->c.my_vtag);
- initack.a_rwnd = htonl(atomic_read(&asoc->rnwd));
- initack.num_outbound_streams = htons(asoc->streamoutcnt);
- initack.num_inbound_streams = htons(asoc->streamincnt);
- initack.initial_tsn = htonl(asoc->c.initSeqNumber);
-
- sctp_get_my_addrs(asoc, &addrs, &addrs_len);
+ /* When calculating the time extension, an implementation
+ * SHOULD use the RTT information measured based on the
+ * previous COOKIE ECHO / ERROR exchange, and should add no
+ * more than 1 second beyond the measured RTT, due to long
+ * State Cookie lifetimes making the endpoint more subject to
+ * a replay attack.
+ */
+ bht.p = {SCTP_COOKIE_PRESERVE, 8};
+ bht.extraTime = htonl(rtt + 1000);
+
+ initack.init_tag = htonl(asoc->c.my_vtag);
+ initack.a_rwnd = htonl(atomic_read(&asoc->rnwd));
+ initack.num_outbound_streams = htons(asoc->streamoutcnt);
+ initack.num_inbound_streams = htons(asoc->streamincnt);
+ initack.initial_tsn = htonl(asoc->c.initSeqNumber);
+
+ sctp_get_my_addrs(asoc, &addrs, &addrs_len);
/* Build that new INIT chunk. */
- reply = sctp_make_chunk(SCTP_INITIATION, 0,
+ reply = sctp_make_chunk(SCTP_INITIATION, 0,
sizeof(initack)
+ sizeof(bht)
+ addrs_len);
- if (NULL == reply) { goto nomem; }
+ if (!reply)
+ goto nomem;
sctp_addto_chunk(reply, sizeof(initack), &initack);
sctp_addto_chunk(reply, sizeof(bht), &bht);
sctp_addto_chunk(reply, addrs_len, addrs);
sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(reply));
return SCTP_DISPOSITION_CONSUME;
+
nomem:
return SCTP_DISPOSITION_NOMEM;
-} /* sctp_disposition_t do_5_2_6_stale (...) */
+}
#endif /* 0 */
-
/*
* Process an ABORT.
- *
+ *
* Section: 9.1
- * After checking the Verification Tag, the receiving endpoint shall
- * remove the association from its record, and shall report the
+ * After checking the Verification Tag, the receiving endpoint shall
+ * remove the association from its record, and shall report the
* termination to its upper layer.
*
* Verification Tag: 8.5.1 Exceptions in Verification Tag Rules
* B) Rules for packet carrying ABORT:
- *
+ *
* - The endpoint shall always fill in the Verification Tag field of the
* outbound packet with the destination endpoint's tag value if it
- * is known.
- *
+ * is known.
+ *
* - If the ABORT is sent in response to an OOTB packet, the endpoint
* MUST follow the procedure described in Section 8.4.
- *
+ *
* - The receiver MUST accept the packet if the Verification Tag
* matches either its own tag, OR the tag of its peer. Otherwise, the
* receiver MUST silently discard the packet and take no further
- * action.
- *
+ * action.
+ *
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_do_9_1_abort(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
-
- /* Check the verification tag. */
- /* BUG: WRITE ME. */
-
- sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
+sctp_disposition_t sctp_sf_do_9_1_abort(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ /* Check the verification tag. */
+ /* BUG: WRITE ME. */
+ sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
SCTP_STATE(SCTP_STATE_CLOSED));
sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
/* BUG? This does not look complete... */
return SCTP_DISPOSITION_ABORT;
-
-} /* sctp_sf_do_9_1_abort() */
+}
/*
* Process an ABORT. (COOKIE-WAIT state)
- *
+ *
* See sctp_sf_do_9_1_abort() above.
*/
-sctp_disposition_t
-sctp_sf_cookie_wait_abort(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_cookie_wait_abort(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
-
sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
- SCTP_STATE(SCTP_STATE_CLOSED));
- sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
+ SCTP_STATE(SCTP_STATE_CLOSED));
+ sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
SCTP_TO(SCTP_EVENT_TIMEOUT_T1_INIT));
/* CMD_INIT_FAILED will DELETE_TCB. */
sctp_add_cmd_sf(commands,SCTP_CMD_INIT_FAILED, SCTP_NULL());
return SCTP_DISPOSITION_DELETE_TCB;
-
-} /* sctp_sf_cookie_wait_abort() */
+}
/*
* Process an ABORT. (COOKIE-ECHOED state)
- *
+ *
* See sctp_sf_do_9_1_abort() above.
*/
-sctp_disposition_t
-sctp_sf_cookie_echoed_abort(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_cookie_echoed_abort(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
-
/* There is a single T1 timer, so we should be able to use
- * common function with the COOKIE-WAIT state.
+ * common function with the COOKIE-WAIT state.
*/
return sctp_sf_cookie_wait_abort(ep, asoc, type, arg, commands);
-
-} /* sctp_sf_cookie_echoed_abort() */
+}
#if 0
/*
* Handle a shutdown timeout or INIT during a shutdown phase.
- *
+ *
* Section: 9.2
- * If an endpoint is in SHUTDOWN-ACK-SENT state and receives an INIT chunk
- * (e.g., if the SHUTDOWN COMPLETE was lost) with source and destination
- * transport addresses (either in the IP addresses or in the INIT chunk)
- * that belong to this association, it should discard the INIT chunk and
- * retransmit the SHUTDOWN ACK chunk.
- *...
+ * If an endpoint is in SHUTDOWN-ACK-SENT state and receives an INIT chunk
+ * (e.g., if the SHUTDOWN COMPLETE was lost) with source and destination
+ * transport addresses (either in the IP addresses or in the INIT chunk)
+ * that belong to this association, it should discard the INIT chunk and
+ * retransmit the SHUTDOWN ACK chunk.
+ *...
* While in SHUTDOWN-SENT state ... If the timer expires, the endpoint
* must re-send the SHUTDOWN ACK.
*
*
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_do_9_2_reshutack(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- sctp_chunk_t *chunk = arg;
+sctp_disposition_t sctp_do_9_2_reshutack(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ sctp_chunk_t *chunk = arg;
/* If this was a timeout (not an INIT), then do the counter
* work. We might need to just dump the association.
*/
if (!chunk) {
if (1 + asoc->counters[SctpCounterRetran] >
- asoc->maxRetrans){
+ asoc->maxRetrans) {
sctp_add_cmd(commands, SCTP_CMD_DELETE_TCB,
SCTP_NULL());
return SCTP_DISPOSITION_DELETE_TCB;
retval->counters[1] = 0;
}
- reply = sctp_make_shutdown_ack(asoc, chunk);
- if (!reply) { goto nomem; }
+ reply = sctp_make_shutdown_ack(asoc, chunk);
+ if (!reply)
+ goto nomem;
sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(reply));
-
return SCTP_DISPOSITION_CONSUME;
+
nomem:
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_disposition_t sctp_do_9_2_reshutack(...) */
+}
#endif /* 0 */
/*
* sctp_sf_do_9_2_shut
- *
+ *
* Section: 9.2
- * Upon the reception of the SHUTDOWN, the peer endpoint shall
- * - enter the SHUTDOWN-RECEIVED state,
+ * Upon the reception of the SHUTDOWN, the peer endpoint shall
+ * - enter the SHUTDOWN-RECEIVED state,
*
- * - stop accepting new data from its SCTP user
+ * - stop accepting new data from its SCTP user
*
* - verify, by checking the Cumulative TSN Ack field of the chunk,
* that all its outstanding DATA chunks have been received by the
* Once an endpoint as reached the SHUTDOWN-RECEIVED state it MUST NOT
* send a SHUTDOWN in response to a ULP request. And should discard
* subsequent SHUTDOWN chunks.
- *
+ *
* If there are still outstanding DATA chunks left, the SHUTDOWN
* receiver shall continue to follow normal data transmission
* procedures defined in Section 6 until all outstanding DATA chunks
* are acknowledged; however, the SHUTDOWN receiver MUST NOT accept
* new data from its SCTP user.
- *
+ *
* Verification Tag: 8.5 Verification Tag [Normal verification]
*
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
- * The return value is the disposition of the chunk. */
-sctp_disposition_t
-sctp_sf_do_9_2_shutdown(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- sctp_chunk_t *chunk = arg;
- sctp_shutdownhdr_t *sdh;
- sctp_disposition_t disposition;
-
- /* Convert the elaborate header. */
- sdh = (sctp_shutdownhdr_t *)chunk->skb->data;
- skb_pull(chunk->skb, sizeof(sctp_shutdownhdr_t));
- chunk->subh.shutdown_hdr = sdh;
-
- /* 8.5 When receiving an SCTP packet, the endpoint MUST ensure
- * that the value in the Verification Tag field of the
- * received SCTP packet matches its own Tag. If the received
- * Verification Tag value does not match the receiver's own
- * tag value, the receiver shall silently discard the packet...
- */
- if (ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag) {
- return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
- }
+ * The return value is the disposition of the chunk.
+ */
+sctp_disposition_t sctp_sf_do_9_2_shutdown(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ sctp_chunk_t *chunk = arg;
+ sctp_shutdownhdr_t *sdh;
+ sctp_disposition_t disposition;
- /* Upon the reception of the SHUTDOWN, the peer endpoint shall
- * - enter the SHUTDOWN-RECEIVED state,
- * - stop accepting new data from its SCTP user
- *
- * [This is implicit in the new state.]
- */
+ /* Convert the elaborate header. */
+ sdh = (sctp_shutdownhdr_t *) chunk->skb->data;
+ skb_pull(chunk->skb, sizeof(sctp_shutdownhdr_t));
+ chunk->subh.shutdown_hdr = sdh;
+
+ /* 8.5 When receiving an SCTP packet, the endpoint MUST ensure
+ * that the value in the Verification Tag field of the
+ * received SCTP packet matches its own Tag. If the received
+ * Verification Tag value does not match the receiver's own
+ * tag value, the receiver shall silently discard the packet...
+ */
+ if (ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag)
+ return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
+ /* Upon the reception of the SHUTDOWN, the peer endpoint shall
+ * - enter the SHUTDOWN-RECEIVED state,
+ * - stop accepting new data from its SCTP user
+ *
+ * [This is implicit in the new state.]
+ */
sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
- SCTP_STATE(SCTP_STATE_SHUTDOWN_RECEIVED));
+ SCTP_STATE(SCTP_STATE_SHUTDOWN_RECEIVED));
+ disposition = SCTP_DISPOSITION_CONSUME;
- disposition = SCTP_DISPOSITION_CONSUME;
+ if (sctp_outqueue_is_empty(&asoc->outqueue)) {
+ disposition =
+ sctp_sf_do_9_2_shutdown_ack(ep, asoc, type,
+ arg, commands);
+ }
- if (sctp_outqueue_is_empty(&asoc->outqueue)) {
- disposition =
- sctp_sf_do_9_2_shutdown_ack(ep, asoc, type,
- arg, commands);
- }
-
/* - verify, by checking the Cumulative TSN Ack field of the
- * chunk, that all its outstanding DATA chunks have been
- * received by the SHUTDOWN sender.
- */
+ * chunk, that all its outstanding DATA chunks have been
+ * received by the SHUTDOWN sender.
+ */
sctp_add_cmd_sf(commands, SCTP_CMD_PROCESS_CTSN,
SCTP_U32(chunk->subh.shutdown_hdr->cum_tsn_ack));
-
- return disposition;
-
-} /* sctp_sf_do_9_2_shutdown() */
-
+ return disposition;
+}
/*
* sctp_sf_do_ecn_cwr
- *
+ *
* Section: Appendix A: Explicit Congestion Notification
*
* CWR:
* Verification Tag: 8.5 Verification Tag [Normal verification]
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-
-sctp_disposition_t
-sctp_sf_do_ecn_cwr(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_do_ecn_cwr(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
- sctp_cwrhdr_t *cwr;
- sctp_chunk_t *chunk = arg;
+ sctp_cwrhdr_t *cwr;
+ sctp_chunk_t *chunk = arg;
/* 8.5 When receiving an SCTP packet, the endpoint MUST ensure
- * that the value in the Verification Tag field of the
- * received SCTP packet matches its own Tag. If the received
- * Verification Tag value does not match the receiver's own
- * tag value, the receiver shall silently discard the packet...
- */
- if (ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag) {
- return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
- }
+ * that the value in the Verification Tag field of the
+ * received SCTP packet matches its own Tag. If the received
+ * Verification Tag value does not match the receiver's own
+ * tag value, the receiver shall silently discard the packet...
+ */
+ if (ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag)
+ return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
- cwr = (sctp_cwrhdr_t *)chunk->skb->data;
+ cwr = (sctp_cwrhdr_t *) chunk->skb->data;
skb_pull(chunk->skb, sizeof(sctp_cwrhdr_t));
cwr->lowest_tsn = ntohl(cwr->lowest_tsn);
/* Does this CWR ack the last sent congestion notification? */
-
if (TSN_lte(asoc->last_ecne_tsn, cwr->lowest_tsn)) {
-
/* Stop sending ECNE. */
sctp_add_cmd_sf(commands,
SCTP_CMD_ECN_CWR,
SCTP_U32(cwr->lowest_tsn));
}
-
return SCTP_DISPOSITION_CONSUME;
-
-} /* sctp_sf_do_ecne_cwr() */
-
+}
/*
* sctp_sf_do_ecne
- *
+ *
* Section: Appendix A: Explicit Congestion Notification
*
* ECN-Echo
* Verification Tag: 8.5 Verification Tag [Normal verification]
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-
-sctp_disposition_t
-sctp_sf_do_ecne(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_do_ecne(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
- sctp_ecnehdr_t *ecne;
- sctp_chunk_t *chunk = arg;
-
+ sctp_ecnehdr_t *ecne;
+ sctp_chunk_t *chunk = arg;
+
/* 8.5 When receiving an SCTP packet, the endpoint MUST ensure
- * that the value in the Verification Tag field of the
- * received SCTP packet matches its own Tag. If the received
- * Verification Tag value does not match the receiver's own
- * tag value, the receiver shall silently discard the packet...
- */
- if (ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag) {
- return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
- }
-
- ecne = (sctp_ecnehdr_t *)chunk->skb->data;
+ * that the value in the Verification Tag field of the
+ * received SCTP packet matches its own Tag. If the received
+ * Verification Tag value does not match the receiver's own
+ * tag value, the receiver shall silently discard the packet...
+ */
+ if (ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag)
+ return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
+
+ ecne = (sctp_ecnehdr_t *) chunk->skb->data;
skb_pull(chunk->skb, sizeof(sctp_ecnehdr_t));
ecne->lowest_tsn = ntohl(ecne->lowest_tsn);
-
+
/* Casting away the const, as we are just modifying the spinlock,
* not the association itself. This should go away in the near
* future when we move to an endpoint based lock.
*/
-
- /* If this is a newer ECNE than the last CWR packet we sent out */
- if (TSN_lt(asoc->last_cwr_tsn, ecne->lowest_tsn)){
+ /* If this is a newer ECNE than the last CWR packet we sent out */
+ if (TSN_lt(asoc->last_cwr_tsn, ecne->lowest_tsn)) {
sctp_add_cmd_sf(commands, SCTP_CMD_ECN_ECNE,
SCTP_U32(ecne->lowest_tsn));
}
-
return SCTP_DISPOSITION_CONSUME;
-
-} /* sctp_sf_do_ecne() */
-
+}
/*
* Section: 6.2 Acknowledgement on Reception of DATA Chunks
- *
- * The SCTP endpoint MUST always acknowledge the reception of each valid
- * DATA chunk.
- *
+ *
+ * The SCTP endpoint MUST always acknowledge the reception of each valid
+ * DATA chunk.
+ *
* The guidelines on delayed acknowledgement algorithm specified in
* Section 4.2 of [RFC2581] SHOULD be followed. Specifically, an
* acknowledgement SHOULD be generated for at least every second packet
- * (not every second DATA chunk) received, and SHOULD be generated within
- * 200 ms of the arrival of any unacknowledged DATA chunk. In some
- * situations it may be beneficial for an SCTP transmitter to be more
- * conservative than the algorithms detailed in this document allow.
- * However, an SCTP transmitter MUST NOT be more aggressive than the
+ * (not every second DATA chunk) received, and SHOULD be generated within
+ * 200 ms of the arrival of any unacknowledged DATA chunk. In some
+ * situations it may be beneficial for an SCTP transmitter to be more
+ * conservative than the algorithms detailed in this document allow.
+ * However, an SCTP transmitter MUST NOT be more aggressive than the
* following algorithms allow.
- *
+ *
* A SCTP receiver MUST NOT generate more than one SACK for every
* incoming packet, other than to update the offered window as the
* receiving application consumes new data.
- *
+ *
* Verification Tag: 8.5 Verification Tag [Normal verification]
- *
+ *
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_eat_data_6_2(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- sctp_chunk_t *chunk = arg;
- sctp_datahdr_t *data_hdr;
+sctp_disposition_t sctp_sf_eat_data_6_2(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ sctp_chunk_t *chunk = arg;
+ sctp_datahdr_t *data_hdr;
sctp_chunk_t *err;
- size_t datalen;
+ size_t datalen;
int tmp;
- uint32_t tsn;
+ __u32 tsn;
- /* RFC 2960 8.5 Verification Tag
+ /* RFC 2960 8.5 Verification Tag
*
* When receiving an SCTP packet, the endpoint MUST ensure
- * that the value in the Verification Tag field of the
- * received SCTP packet matches its own Tag.
+ * that the value in the Verification Tag field of the
+ * received SCTP packet matches its own Tag.
*/
- if (ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag) {
-
- sctp_add_cmd_sf(commands, SCTP_CMD_REPORT_BAD_TAG,
+ if (ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag) {
+ sctp_add_cmd_sf(commands, SCTP_CMD_REPORT_BAD_TAG,
SCTP_NULL());
- return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
-
+ return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
}
- data_hdr = chunk->subh.data_hdr = (sctp_datahdr_t *)chunk->skb->data;
- skb_pull(chunk->skb, sizeof(sctp_datahdr_t));
+ data_hdr = chunk->subh.data_hdr = (sctp_datahdr_t *)chunk->skb->data;
+ skb_pull(chunk->skb, sizeof(sctp_datahdr_t));
tsn = ntohl(data_hdr->tsn);
SCTP_DEBUG_PRINTK("eat_data: TSN 0x%x.\n", tsn);
SCTP_DEBUG_PRINTK("eat_data: skb->head %p.\n", chunk->skb->head);
-
- /* ASSERT: Now skb->data is really the user data. */
+ /* ASSERT: Now skb->data is really the user data. */
- /* Process ECN based congestion.
- *
+ /* Process ECN based congestion.
+ *
* Since the chunk structure is reused for all chunks within
* a packet, we use ecn_ce_done to track if we've already
* done CE processing for this packet.
if (!chunk->ecn_ce_done) {
chunk->ecn_ce_done = 1;
- if ( INET_ECN_is_ce(chunk->skb->nh.iph->tos)
- && asoc->peer.ecn_capable) {
-
+ if (INET_ECN_is_ce(chunk->skb->nh.iph->tos) &&
+ asoc->peer.ecn_capable) {
/* Do real work as sideffect. */
- sctp_add_cmd_sf(commands, SCTP_CMD_ECN_CE,
+ sctp_add_cmd_sf(commands, SCTP_CMD_ECN_CE,
SCTP_U32(tsn));
-
- } /* if (packet experienced congestion) */
-
- } /* if (we have not done ecn_ce processing) */
+ }
+ }
tmp = sctp_tsnmap_check(&asoc->peer.tsn_map, tsn);
if (tmp < 0) {
/* This is a new TSN. */
- /* If we don't have any room in our receive window, discard.
+ /* If we don't have any room in our receive window, discard.
* Actually, allow a little bit of overflow (up to a MTU of
* of overflow).
*/
datalen = ntohs(chunk->chunk_hdr->length);
datalen -= sizeof(sctp_data_chunk_t);
- if (!asoc->rwnd || (datalen > asoc->frag_point)) {
+ if (!asoc->rwnd || (datalen > asoc->frag_point)) {
SCTP_DEBUG_PRINTK("Discarding tsn: %u datalen: %d, "
"rwnd: %d\n", tsn, datalen, asoc->rwnd);
goto discard_noforce;
- }
+ }
- /*
+ /*
* Section 3.3.10.9 No User Data (9)
*
* Cause of error
if (unlikely(0 == datalen)) {
err = sctp_make_abort_no_data(asoc, chunk, tsn);
if (err) {
- sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
+ sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
SCTP_CHUNK(err));
}
/* We are going to ABORT, so we might as well stop
* processing the rest of the chunks in the packet.
*/
sctp_add_cmd_sf(commands, SCTP_CMD_DISCARD_PACKET,SCTP_NULL());
- sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
+ sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
SCTP_STATE(SCTP_STATE_CLOSED));
sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
return SCTP_DISPOSITION_CONSUME;
sctp_add_cmd_sf(commands, SCTP_CMD_REPORT_TSN, SCTP_U32(tsn));
/* RFC 2960 6.5 Stream Identifier and Stream Sequence Number
- *
- * If an endpoint receive a DATA chunk with an invalid stream
+ *
+ * If an endpoint receive a DATA chunk with an invalid stream
* identifier, it shall acknowledge the reception of the DATA chunk
* following the normal procedure, immediately send an ERROR chunk
* with cause set to "Invalid Stream Identifier" (See Section 3.3.10)
- * and discard the DATA chunk.
+ * and discard the DATA chunk.
*/
if (ntohs(data_hdr->stream) >= asoc->c.sinit_max_instreams) {
-
- err = sctp_make_op_error(asoc, chunk, SCTP_ERROR_INV_STRM,
- &data_hdr->stream,
+ err = sctp_make_op_error(asoc, chunk, SCTP_ERROR_INV_STRM,
+ &data_hdr->stream,
sizeof(data_hdr->stream));
if (err) {
- sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
+ sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
SCTP_CHUNK(err));
}
goto discard_noforce;
-
}
- /* Send the data up to the user. Note: Schedule the
- * SCTP_CMD_CHUNK_ULP cmd before the SCTP_CMD_GEN_SACK, as the SACK
+ /* Send the data up to the user. Note: Schedule the
+ * SCTP_CMD_CHUNK_ULP cmd before the SCTP_CMD_GEN_SACK, as the SACK
* chunk needs the updated rwnd.
*/
sctp_add_cmd_sf(commands, SCTP_CMD_CHUNK_ULP, SCTP_CHUNK(chunk));
-
if (asoc->autoclose) {
sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_RESTART,
SCTP_TO(SCTP_EVENT_TIMEOUT_AUTOCLOSE));
}
- /* If this is the last chunk in a packet, we need to count it
- * toward sack generation. Note that we need to SACK every
- * OTHER packet containing data chunks, EVEN IF WE DISCARD
- * THEM. We elect to NOT generate SACK's if the chunk fails
- * the verification tag test.
+ /* If this is the last chunk in a packet, we need to count it
+ * toward sack generation. Note that we need to SACK every
+ * OTHER packet containing data chunks, EVEN IF WE DISCARD
+ * THEM. We elect to NOT generate SACK's if the chunk fails
+ * the verification tag test.
*
* RFC 2960 6.2 Acknowledgement on Reception of DATA Chunks
*
* be more conservative than the algorithms detailed in this
* document allow. However, an SCTP transmitter MUST NOT be
* more aggressive than the following algorithms allow.
- */
+ */
if (chunk->end_of_packet) {
sctp_add_cmd_sf(commands, SCTP_CMD_GEN_SACK, SCTP_NOFORCE());
-
+
/* Start the SACK timer. */
sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_RESTART,
SCTP_TO(SCTP_EVENT_TIMEOUT_SACK));
- }
+ }
return SCTP_DISPOSITION_CONSUME;
- discard_force:
+discard_force:
/* RFC 2960 6.2 Acknowledgement on Reception of DATA Chunks
*
* When a packet arrives with duplicate DATA chunk(s) and with
/* In our case, we split the MAY SACK advice up whether or not
* the last chunk is a duplicate.'
*/
- if (chunk->end_of_packet) {
+ if (chunk->end_of_packet)
sctp_add_cmd_sf(commands, SCTP_CMD_GEN_SACK, SCTP_FORCE());
- }
return SCTP_DISPOSITION_DISCARD;
discard_noforce:
SCTP_TO(SCTP_EVENT_TIMEOUT_SACK));
}
return SCTP_DISPOSITION_DISCARD;
-
-} /* sctp_sf_eat_data_6_2() */
-
+}
/*
* sctp_sf_eat_data_fast_4_4
- *
+ *
* Section: 4 (4)
* (4) In SHUTDOWN-SENT state the endpoint MUST acknowledge any received
* DATA chunks without delay.
* Verification Tag: 8.5 Verification Tag [Normal verification]
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_eat_data_fast_4_4(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- sctp_chunk_t *chunk = arg;
- sctp_datahdr_t *data_hdr;
+sctp_disposition_t sctp_sf_eat_data_fast_4_4(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ sctp_chunk_t *chunk = arg;
+ sctp_datahdr_t *data_hdr;
sctp_chunk_t *err;
- size_t datalen;
+ size_t datalen;
int tmp;
- uint32_t tsn;
+ __u32 tsn;
- /* RFC 2960 8.5 Verification Tag
+ /* RFC 2960 8.5 Verification Tag
*
* When receiving an SCTP packet, the endpoint MUST ensure
- * that the value in the Verification Tag field of the
- * received SCTP packet matches its own Tag.
+ * that the value in the Verification Tag field of the
+ * received SCTP packet matches its own Tag.
*/
-
- if (ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag) {
-
- sctp_add_cmd_sf(commands, SCTP_CMD_REPORT_BAD_TAG,
+ if (ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag) {
+ sctp_add_cmd_sf(commands, SCTP_CMD_REPORT_BAD_TAG,
SCTP_NULL());
- return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
- }
+ return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
+ }
- data_hdr = chunk->subh.data_hdr = (sctp_datahdr_t *)chunk->skb->data;
- skb_pull(chunk->skb, sizeof(sctp_datahdr_t));
+ data_hdr = chunk->subh.data_hdr = (sctp_datahdr_t *) chunk->skb->data;
+ skb_pull(chunk->skb, sizeof(sctp_datahdr_t));
tsn = ntohl(data_hdr->tsn);
SCTP_DEBUG_PRINTK("eat_data: TSN 0x%x.\n", tsn);
-
- /* ASSERT: Now skb->data is really the user data. */
- /* Process ECN based congestion.
- *
+ /* ASSERT: Now skb->data is really the user data. */
+
+ /* Process ECN based congestion.
+ *
* Since the chunk structure is reused for all chunks within
* a packet, we use ecn_ce_done to track if we've already
* done CE processing for this packet.
* We need to do ECN processing even if we plan to discard the
* chunk later.
*/
-
if (!chunk->ecn_ce_done) {
chunk->ecn_ce_done = 1;
- if ( INET_ECN_is_ce(chunk->skb->nh.iph->tos)
- && asoc->peer.ecn_capable) {
-
+ if (INET_ECN_is_ce(chunk->skb->nh.iph->tos) &&
+ asoc->peer.ecn_capable) {
/* Do real work as sideffect. */
- sctp_add_cmd_sf(commands, SCTP_CMD_ECN_CE,
+ sctp_add_cmd_sf(commands, SCTP_CMD_ECN_CE,
SCTP_U32(tsn));
-
- } /* if (packet experienced congestion) */
-
- } /* if (we have not done ecn_ce processing) */
+ }
+ }
tmp = sctp_tsnmap_check(&asoc->peer.tsn_map, tsn);
if (tmp < 0) {
/* This is a new TSN. */
- datalen = ntohs(chunk->chunk_hdr->length);
+ datalen = ntohs(chunk->chunk_hdr->length);
datalen -= sizeof(sctp_data_chunk_t);
- /*
+ /*
* Section 3.3.10.9 No User Data (9)
*
* Cause of error
if (unlikely(0 == datalen)) {
err = sctp_make_abort_no_data(asoc, chunk, tsn);
if (err) {
- sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
+ sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
SCTP_CHUNK(err));
}
/* We are going to ABORT, so we might as well stop
* processing the rest of the chunks in the packet.
*/
sctp_add_cmd_sf(commands, SCTP_CMD_DISCARD_PACKET,SCTP_NULL());
- sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
+ sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
SCTP_STATE(SCTP_STATE_CLOSED));
sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
return SCTP_DISPOSITION_CONSUME;
/* Record the fact that we have received this TSN. */
sctp_add_cmd_sf(commands, SCTP_CMD_REPORT_TSN, SCTP_U32(tsn));
-
+
/* RFC 2960 6.5 Stream Identifier and Stream Sequence Number
- *
- * If an endpoint receive a DATA chunk with an invalid stream
+ *
+ * If an endpoint receive a DATA chunk with an invalid stream
* identifier, it shall acknowledge the reception of the DATA chunk
* following the normal procedure, immediately send an ERROR chunk
* with cause set to "Invalid Stream Identifier" (See Section 3.3.10)
- * and discard the DATA chunk.
+ * and discard the DATA chunk.
*/
if (ntohs(data_hdr->stream) >= asoc->c.sinit_max_instreams) {
-
- err = sctp_make_op_error(asoc, chunk, SCTP_ERROR_INV_STRM,
- &data_hdr->stream,
+ err = sctp_make_op_error(asoc, chunk, SCTP_ERROR_INV_STRM,
+ &data_hdr->stream,
sizeof(data_hdr->stream));
if (err) {
- sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
+ sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
SCTP_CHUNK(err));
}
}
/* Go a head and force a SACK, since we are shutting down. */
gen_shutdown:
- /* Implementor's Guide.
+ /* Implementor's Guide.
*
* While in SHUTDOWN-SENT state, the SHUTDOWN sender MUST immediately
* respond to each received packet containing one or more DATA chunk(s)
*/
if (chunk->end_of_packet) {
/* We must delay the chunk creation since the cumulative
- * TSN has not been updated yet.
+ * TSN has not been updated yet.
*/
sctp_add_cmd_sf(commands, SCTP_CMD_GEN_SHUTDOWN, SCTP_NULL());
sctp_add_cmd_sf(commands, SCTP_CMD_GEN_SACK, SCTP_FORCE());
SCTP_TO(SCTP_EVENT_TIMEOUT_T2_SHUTDOWN));
}
return SCTP_DISPOSITION_CONSUME;
-
-} /* sctp_sf_eat_data_fast4_4() */
-
+}
/*
* Section: 6.2 Processing a Received SACK
* D) Any time a SACK arrives, the endpoint performs the following:
- *
+ *
* i) If Cumulative TSN Ack is less than the Cumulative TSN Ack Point,
* then drop the SACK. Since Cumulative TSN Ack is monotonically
* increasing, a SACK whose Cumulative TSN Ack is less than the
* Cumulative TSN Ack Point indicates an out-of-order SACK.
- *
+ *
* ii) Set rwnd equal to the newly received a_rwnd minus the number
* of bytes still outstanding after processing the Cumulative TSN Ack
* and the Gap Ack Blocks.
- *
+ *
* iii) If the SACK is missing a TSN that was previously
* acknowledged via a Gap Ack Block (e.g., the data receiver
* reneged on the data), then mark the corresponding DATA chunk
* timer is running for the destination address to which the DATA
* chunk was originally transmitted, then T3-rtx is started for
* that destination address.
- *
+ *
* Verification Tag: 8.5 Verification Tag [Normal verification]
*
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_eat_sack_6_2(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- sctp_chunk_t *chunk = arg;
- sctp_sackhdr_t *sackh;
- uint32_t ctsn;
-
-
- /* 8.5 When receiving an SCTP packet, the endpoint MUST ensure
- * that the value in the Verification Tag field of the
- * received SCTP packet matches its own Tag. ...
- */
- if (ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag) {
- return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
- }
+sctp_disposition_t sctp_sf_eat_sack_6_2(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ sctp_chunk_t *chunk = arg;
+ sctp_sackhdr_t *sackh;
+ __u32 ctsn;
- /* Pull the SACK chunk from the data buffer */
- sackh = sctp_sm_pull_sack(chunk);
- chunk->subh.sack_hdr = sackh;
- ctsn = ntohl(sackh->cum_tsn_ack);
-
- /* i) If Cumulative TSN Ack is less than the Cumulative TSN
- * Ack Point, then drop the SACK. Since Cumulative TSN
- * Ack is monotonically increasing, a SACK whose
- * Cumulative TSN Ack is less than the Cumulative TSN Ack
- * Point indicates an out-of-order SACK.
- */
- if (TSN_lt(ctsn, asoc->ctsn_ack_point)) {
+ /* 8.5 When receiving an SCTP packet, the endpoint MUST ensure
+ * that the value in the Verification Tag field of the
+ * received SCTP packet matches its own Tag. ...
+ */
+ if (ntohl(chunk->sctp_hdr->vtag) != asoc->c.my_vtag)
+ return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
+ /* Pull the SACK chunk from the data buffer */
+ sackh = sctp_sm_pull_sack(chunk);
+ chunk->subh.sack_hdr = sackh;
+ ctsn = ntohl(sackh->cum_tsn_ack);
+
+ /* i) If Cumulative TSN Ack is less than the Cumulative TSN
+ * Ack Point, then drop the SACK. Since Cumulative TSN
+ * Ack is monotonically increasing, a SACK whose
+ * Cumulative TSN Ack is less than the Cumulative TSN Ack
+ * Point indicates an out-of-order SACK.
+ */
+ if (TSN_lt(ctsn, asoc->ctsn_ack_point)) {
SCTP_DEBUG_PRINTK("ctsn %x\n", ctsn);
- SCTP_DEBUG_PRINTK("ctsn_ack_point %x\n",
+ SCTP_DEBUG_PRINTK("ctsn_ack_point %x\n",
asoc->ctsn_ack_point);
- return SCTP_DISPOSITION_DISCARD;
- }
+ return SCTP_DISPOSITION_DISCARD;
+ }
- /* Return this SACK for further processing. */
+ /* Return this SACK for further processing. */
sctp_add_cmd_sf(commands, SCTP_CMD_PROCESS_SACK,
SCTP_SACKH(sackh));
- /* Note: We do the rest of the work on the PROCESS_SACK
+ /* Note: We do the rest of the work on the PROCESS_SACK
* sideeffect.
*/
-
return SCTP_DISPOSITION_CONSUME;
-
-} /* sctp_sf_eat_sack_6_2() */
-
+}
/*
* Generate an ABORT in response to a packet.
- *
+ *
* Section: 8.4 Handle "Out of the blue" Packets
*
* 8) The receiver should respond to the sender of the OOTB packet
*
* Verification Tag:
*
- * The return value is the disposition of the chunk. */
-sctp_disposition_t
-sctp_sf_tabort_8_4_8(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- sctp_packet_t *packet = NULL;
- sctp_transport_t *transport = NULL;
- sctp_chunk_t *chunk = arg;
- sctp_chunk_t *abort;
- uint16_t sport;
- uint16_t dport;
- uint32_t vtag;
-
-
- /* Grub in chunk and endpoint for kewl bitz. */
+ * The return value is the disposition of the chunk.
+*/
+sctp_disposition_t sctp_sf_tabort_8_4_8(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ sctp_packet_t *packet = NULL;
+ sctp_transport_t *transport = NULL;
+ sctp_chunk_t *chunk = arg;
+ sctp_chunk_t *abort;
+ __u16 sport;
+ __u16 dport;
+ __u32 vtag;
+
+ /* Grub in chunk and endpoint for kewl bitz. */
sport = ntohs(chunk->sctp_hdr->dest);
dport = ntohs(chunk->sctp_hdr->source);
/* -- Make sure the ABORT packet's V-tag is the same as the
- * inbound packet if no association exists, otherwise use
- * the peer's vtag.
- */
- if (asoc) {
- vtag = asoc->peer.i.init_tag;
- } else {
- vtag = ntohl(chunk->sctp_hdr->vtag);
- }
-
- /* Make a transport for the bucket, Eliza... */
- transport = sctp_transport_new(sctp_source(chunk), GFP_ATOMIC);
-
- if (!transport) {
- goto nomem;
- }
+ * inbound packet if no association exists, otherwise use
+ * the peer's vtag.
+ */
+ if (asoc)
+ vtag = asoc->peer.i.init_tag;
+ else
+ vtag = ntohl(chunk->sctp_hdr->vtag);
+
+ /* Make a transport for the bucket, Eliza... */
+ transport = sctp_transport_new(sctp_source(chunk), GFP_ATOMIC);
+ if (!transport)
+ goto nomem;
- /* Make a packet for the ABORT to go into. */
- packet = t_new(sctp_packet_t, GFP_ATOMIC);
- if (!packet) {
+ /* Make a packet for the ABORT to go into. */
+ packet = t_new(sctp_packet_t, GFP_ATOMIC);
+ if (!packet)
goto nomem_packet;
- }
- packet = sctp_packet_init(packet, transport, sport, dport);
+
+ packet = sctp_packet_init(packet, transport, sport, dport);
packet = sctp_packet_config(packet, vtag, 0, NULL);
/* Make an ABORT.
- * This will set the T bit since we have no association.
- */
- abort = sctp_make_abort(NULL, chunk, 0);
- if (NULL == abort) {
- goto nomem_chunk;
- }
+ * This will set the T bit since we have no association.
+ */
+ abort = sctp_make_abort(NULL, chunk, 0);
+ if (!abort)
+ goto nomem_chunk;
- /* Set the skb to the belonging sock for accounting. */
- abort->skb->sk = ep->base.sk;
+ /* Set the skb to the belonging sock for accounting. */
+ abort->skb->sk = ep->base.sk;
- sctp_packet_append_chunk(packet, abort);
-
+ sctp_packet_append_chunk(packet, abort);
sctp_add_cmd_sf(commands, SCTP_CMD_SEND_PKT, SCTP_PACKET(packet));
-
return SCTP_DISPOSITION_DISCARD;
nomem_chunk:
sctp_packet_free(packet);
+
nomem_packet:
sctp_transport_free(transport);
+
nomem:
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_tabort_8_4_8() */
+}
/*
- * Received an ERROR chunk from peer. Generate SCTP_REMOTE_ERROR
+ * Received an ERROR chunk from peer. Generate SCTP_REMOTE_ERROR
* event as ULP notification for each cause included in the chunk.
- *
- * API 5.3.1.3 - SCTP_REMOTE_ERROR
*
- * The return value is the disposition of the chunk. */
-sctp_disposition_t
-sctp_sf_operr_notify(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
-
+ * API 5.3.1.3 - SCTP_REMOTE_ERROR
+ *
+ * The return value is the disposition of the chunk.
+*/
+sctp_disposition_t sctp_sf_operr_notify(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
sctp_chunk_t *chunk = arg;
sctp_ulpevent_t *ev;
-
while (chunk->chunk_end > chunk->skb->data) {
ev = sctp_ulpevent_make_remote_error(asoc,chunk,0, GFP_ATOMIC);
-
- if (!ev) { goto nomem; }
-
+ if (!ev)
+ goto nomem;
+
if (!sctp_add_cmd(commands, SCTP_CMD_EVENT_ULP,
SCTP_ULPEVENT(ev))) {
sctp_ulpevent_free(ev);
goto nomem;
}
}
-
return SCTP_DISPOSITION_CONSUME;
+
nomem:
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_operr_notify() */
+}
/*
* Process an inbound SHUTDOWN ACK.
- *
- * From Section 9.2:
+ *
+ * From Section 9.2:
* Upon the receipt of the SHUTDOWN ACK, the SHUTDOWN sender shall
* stop the T2-shutdown timer, send a SHUTDOWN COMPLETE chunk to its
* peer, and remove all record of the association.
*
- * The return value is the disposition. */
-sctp_disposition_t
-sctp_sf_do_9_2_final(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- sctp_chunk_t *chunk = arg;
+ * The return value is the disposition.
+ */
+sctp_disposition_t sctp_sf_do_9_2_final(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ sctp_chunk_t *chunk = arg;
sctp_chunk_t *reply;
sctp_ulpevent_t *ev;
-
- /* 10.2 H) SHUTDOWN COMPLETE notification
- *
- * When SCTP completes the shutdown procedures (section 9.2) this
- * notification is passed to the upper layer.
- */
-
+ /* 10.2 H) SHUTDOWN COMPLETE notification
+ *
+ * When SCTP completes the shutdown procedures (section 9.2) this
+ * notification is passed to the upper layer.
+ */
ev = sctp_ulpevent_make_assoc_change(asoc, 0, SCTP_SHUTDOWN_COMP,
0, 0, 0, GFP_ATOMIC);
- if (!ev) { goto nomem; }
+ if (!ev)
+ goto nomem;
sctp_add_cmd_sf(commands, SCTP_CMD_EVENT_ULP, SCTP_ULPEVENT(ev));
-
- /* Upon the receipt of the SHUTDOWN ACK, the SHUTDOWN sender shall
- * stop the T2-shutdown timer,
- */
- sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
+ /* Upon the receipt of the SHUTDOWN ACK, the SHUTDOWN sender shall
+ * stop the T2-shutdown timer,
+ */
+ sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
SCTP_TO(SCTP_EVENT_TIMEOUT_T2_SHUTDOWN));
- /* ...send a SHUTDOWN COMPLETE chunk to its peer, */
- reply = sctp_make_shutdown_complete(asoc, chunk);
- if (!reply) { goto nomem; }
-
+ /* ...send a SHUTDOWN COMPLETE chunk to its peer, */
+ reply = sctp_make_shutdown_complete(asoc, chunk);
+ if (!reply)
+ goto nomem;
+
sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
SCTP_STATE(SCTP_STATE_CLOSED));
sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(reply));
- /* ...and remove all record of the association. */
+ /* ...and remove all record of the association. */
sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
return SCTP_DISPOSITION_DELETE_TCB;
- nomem:
+nomem:
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_do_9_2_final() */
+}
/*
* RFC 2960, 8.4 - Handle "Out of the blue" Packets
* TCB was found. After sending this ABORT, the receiver of the OOTB
* packet shall discard the OOTB packet and take no further action.
*/
-sctp_disposition_t
-sctp_sf_ootb(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_ootb(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
sctp_chunk_t *chunk = arg;
struct sk_buff *skb = chunk->skb;
sctp_chunkhdr_t *ch;
- uint8_t *ch_end;
+ __u8 *ch_end;
int ootb_shut_ack = 0;
- ch = (sctp_chunkhdr_t *)chunk->chunk_hdr;
-
+ ch = (sctp_chunkhdr_t *) chunk->chunk_hdr;
do {
- ch_end = ((uint8_t *)ch) + WORD_ROUND(ntohs(ch->length));
-
- if (SCTP_CID_SHUTDOWN_ACK == ch->type) {
+ ch_end = ((__u8 *)ch) + WORD_ROUND(ntohs(ch->length));
+
+ if (SCTP_CID_SHUTDOWN_ACK == ch->type)
ootb_shut_ack = 1;
- }
- ch = (sctp_chunkhdr_t *)ch_end;
-
+ ch = (sctp_chunkhdr_t *) ch_end;
} while (ch_end < skb->tail);
-
- if (ootb_shut_ack) {
+ if (ootb_shut_ack)
sctp_sf_shut_8_4_5(ep, asoc, type, arg, commands);
- } else {
+ else
sctp_sf_tabort_8_4_8(ep, asoc, type, arg, commands);
- }
-
- return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
-} /* sctp_sf_ootb() */
+ return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
+}
/*
* Handle an "Out of the blue" SHUTDOWN ACK.
*
* Section: 8.4 5)
- * 5) If the packet contains a SHUTDOWN ACK chunk, the receiver should
- * respond to the sender of the OOTB packet with a SHUTDOWN COMPLETE.
- * When sending the SHUTDOWN COMPLETE, the receiver of the OOTB packet
- * must fill in the Verification Tag field of the outbound packet with
- * the Verification Tag received in the SHUTDOWN ACK and set the
- * T-bit in the Chunk Flags to indicate that no TCB was found.
- *
+ * 5) If the packet contains a SHUTDOWN ACK chunk, the receiver should
+ * respond to the sender of the OOTB packet with a SHUTDOWN COMPLETE.
+ * When sending the SHUTDOWN COMPLETE, the receiver of the OOTB packet
+ * must fill in the Verification Tag field of the outbound packet with
+ * the Verification Tag received in the SHUTDOWN ACK and set the
+ * T-bit in the Chunk Flags to indicate that no TCB was found.
+ *
* Verification Tag: 8.5.1 E) Rules for packet carrying a SHUTDOWN ACK
* If the receiver is in COOKIE-ECHOED or COOKIE-WAIT state the
* procedures in section 8.4 SHOULD be followed, in other words it
*
* Inputs
* (endpoint, asoc, type, arg, commands)
- *
+ *
* Outputs
* (sctp_disposition_t)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_shut_8_4_5(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_shut_8_4_5(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
- sctp_packet_t *packet = NULL;
- sctp_transport_t *transport = NULL;
- sctp_chunk_t *chunk = arg;
- sctp_chunk_t *shut;
- uint16_t sport;
- uint16_t dport;
- uint32_t vtag;
-
-
+ sctp_packet_t *packet = NULL;
+ sctp_transport_t *transport = NULL;
+ sctp_chunk_t *chunk = arg;
+ sctp_chunk_t *shut;
+ __u16 sport;
+ __u16 dport;
+ __u32 vtag;
+
/* Grub in chunk and endpoint for kewl bitz. */
sport = ntohs(chunk->sctp_hdr->dest);
dport = ntohs(chunk->sctp_hdr->source);
/* Make sure the ABORT packet's V-tag is the same as the
- * inbound packet if no association exists, otherwise use
- * the peer's vtag.
- */
- vtag = ntohl(chunk->sctp_hdr->vtag);
-
- /* Make a transport for the bucket, Eliza... */
- transport = sctp_transport_new(sctp_source(chunk), GFP_ATOMIC);
+ * inbound packet if no association exists, otherwise use
+ * the peer's vtag.
+ */
+ vtag = ntohl(chunk->sctp_hdr->vtag);
- if (!transport) {
+ /* Make a transport for the bucket, Eliza... */
+ transport = sctp_transport_new(sctp_source(chunk), GFP_ATOMIC);
+ if (!transport)
goto nomem;
- }
- /* Make a packet for the ABORT to go into. */
- packet = t_new(sctp_packet_t, GFP_ATOMIC);
- if (!packet) {
+ /* Make a packet for the ABORT to go into. */
+ packet = t_new(sctp_packet_t, GFP_ATOMIC);
+ if (!packet)
goto nomem_packet;
- }
- packet = sctp_packet_init(packet, transport, sport, dport);
+
+ packet = sctp_packet_init(packet, transport, sport, dport);
packet = sctp_packet_config(packet, vtag, 0, NULL);
/* Make an ABORT.
- * This will set the T bit since we have no association.
- */
- shut = sctp_make_shutdown_complete(NULL, chunk);
- if (!shut) {
- goto nomem_chunk;
- }
+ * This will set the T bit since we have no association.
+ */
+ shut = sctp_make_shutdown_complete(NULL, chunk);
+ if (!shut)
+ goto nomem_chunk;
/* Set the skb to the belonging sock for accounting. */
- shut->skb->sk = ep->base.sk;
-
- sctp_packet_append_chunk(packet, shut);
+ shut->skb->sk = ep->base.sk;
+ sctp_packet_append_chunk(packet, shut);
sctp_add_cmd_sf(commands, SCTP_CMD_SEND_PKT, SCTP_PACKET(packet));
return SCTP_DISPOSITION_CONSUME;
nomem_chunk:
sctp_packet_free(packet);
+
nomem_packet:
sctp_transport_free(transport);
+
nomem:
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_shut_8_4_5() */
+}
#if 0
/*
* ABORT.
*
* Log the fact and then process normally.
- *
+ *
* Section: Not specified
* Verification Tag: 8.5 Verification Tag [Normal verification]
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-lucky(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- sctp_chunk_t *chunk = arg;
-
- /* 8.5 When receiving an SCTP packet, the endpoint MUST ensure
- * that the value in the Verification Tag field of the
- * received SCTP packet matches its own Tag. ...
- */
- if (chunk->sctp_hdr->vtag != asoc->c.my_vtag) {
- return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
- }
+sctp_disposition_t lucky(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ sctp_chunk_t *chunk = arg;
+
+ /* 8.5 When receiving an SCTP packet, the endpoint MUST ensure
+ * that the value in the Verification Tag field of the
+ * received SCTP packet matches its own Tag. ...
+ */
+ if (chunk->sctp_hdr->vtag != asoc->c.my_vtag)
+ return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
return SCTP_DISPOSITION_CONSUME;
+
nomem:
return SCTP_DISPOSITION_NOMEM;
-} /* sctp_disposition_t lucky(...) */
+}
#endif /* 0 */
#if 0
/*
* The other end is doing something very stupid. We'll ignore them
* after logging their idiocy. :-)
- *
+ *
* Section: Not specified
* Verification Tag: 8.5 Verification Tag [Normal verification]
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-other_stupid(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- sctp_chunk_t *chunk = arg;
-
-
- /* 8.5 When receiving an SCTP packet, the endpoint MUST ensure
- * that the value in the Verification Tag field of the
- * received SCTP packet matches its own Tag. ...
- */
- if (chunk->sctp_hdr->vtag != asoc->c.my_vtag) {
- return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
- }
+sctp_disposition_t other_stupid(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ sctp_chunk_t *chunk = arg;
+
+ /* 8.5 When receiving an SCTP packet, the endpoint MUST ensure
+ * that the value in the Verification Tag field of the
+ * received SCTP packet matches its own Tag. ...
+ */
+ if (chunk->sctp_hdr->vtag != asoc->c.my_vtag)
+ return sctp_sf_pdiscard(ep, asoc, type, arg, commands);
return SCTP_DISPOSITION_CONSUME;
+
nomem:
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_disposition_t other_stupid(...) */
+}
#endif /* 0 */
-
/*
* The other end is violating protocol.
- *
+ *
* Section: Not specified
* Verification Tag: Not specified
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* We simply tag the chunk as a violation. The state machine will log
* the violation and continue.
*/
-sctp_disposition_t
-sctp_sf_violation(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
+sctp_disposition_t sctp_sf_violation(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
return SCTP_DISPOSITION_VIOLATION;
-
-} /* sctp_sf_violation() */
-
-
+}
/***************************************************************************
* These are the state functions for handling primitive (Section 10) events.
***************************************************************************/
/*
* sctp_sf_do_prm_asoc
- *
+ *
* Section: 10.1 ULP-to-SCTP
* B) Associate
- *
- * Format: ASSOCIATE(local SCTP instance name, destination transport addr,
- * outbound stream count)
+ *
+ * Format: ASSOCIATE(local SCTP instance name, destination transport addr,
+ * outbound stream count)
* -> association id [,destination transport addr list] [,outbound stream
- * count]
- *
+ * count]
+ *
* This primitive allows the upper layer to initiate an association to a
- * specific peer endpoint.
- *
+ * specific peer endpoint.
+ *
* The peer endpoint shall be specified by one of the transport addresses
* which defines the endpoint (see Section 1.4). If the local SCTP
* instance has not been initialized, the ASSOCIATE is considered an
- * error.
+ * error.
* [This is not relevant for the kernel implementation since we do all
* initialization at boot time. It we hadn't initialized we wouldn't
* get anywhere near this code.]
- *
+ *
* An association id, which is a local handle to the SCTP association,
* will be returned on successful establishment of the association. If
* SCTP is not able to open an SCTP association with the peer endpoint,
* an error is returned.
* [In the kernel implementation, the sctp_association_t needs to
* be created BEFORE causing this primitive to run.]
- *
+ *
* Other association parameters may be returned, including the
* complete destination transport addresses of the peer as well as the
* outbound stream count of the local endpoint. One of the transport
* sending a packet to a specific transport address. [All of this
* stuff happens when the INIT ACK arrives. This is a NON-BLOCKING
* function.]
- *
+ *
* Mandatory attributes:
- *
+ *
* o local SCTP instance name - obtained from the INITIALIZE operation.
* [This is the argument asoc.]
* o destination transport addr - specified as one of the transport
* would like to open towards this peer endpoint.
* [BUG: This is not currently implemented.]
* Optional attributes:
- *
+ *
* None.
- *
+ *
* The return value is a disposition.
*/
-sctp_disposition_t
-sctp_sf_do_prm_asoc(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_do_prm_asoc(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
sctp_chunk_t *repl;
sctp_bind_addr_t *bp;
* info from the local endpoint and the remote peer.
*/
bp = sctp_bind_addr_new(GFP_ATOMIC);
- if (NULL == bp) { goto nomem; }
+ if (!bp)
+ goto nomem;
- /* Use scoping rules to determine the subset of addresses from
- * the endpoint.
+ /* Use scoping rules to determine the subset of addresses from
+ * the endpoint.
*/
scope = sctp_scope(&asoc->peer.active_path->ipaddr);
flags = (PF_INET6 == asoc->base.sk->family) ? SCTP_ADDR6_ALLOWED : 0;
- if (asoc->peer.ipv4_address) {
+ if (asoc->peer.ipv4_address)
flags |= SCTP_ADDR4_PEERSUPP;
- }
- if (asoc->peer.ipv6_address) {
+ if (asoc->peer.ipv6_address)
flags |= SCTP_ADDR6_PEERSUPP;
- }
- error = sctp_bind_addr_copy(bp, &ep->base.bind_addr, scope,
+ error = sctp_bind_addr_copy(bp, &ep->base.bind_addr, scope,
GFP_ATOMIC, flags);
- if (0 != error) { goto nomem; }
+ if (error)
+ goto nomem;
/* FIXME: Either move address assignment out of this function
* or else move the association allocation/init into this function.
* function, so would not be a sideeffect if the allocation
* moved into this function. --jgrimm
*/
- sctp_add_cmd_sf(commands, SCTP_CMD_SET_BIND_ADDR, (sctp_arg_t)bp);
+ sctp_add_cmd_sf(commands, SCTP_CMD_SET_BIND_ADDR, (sctp_arg_t) bp);
/* RFC 2960 5.1 Normal Establishment of an Association
- *
- * A) "A" first sends an INIT chunk to "Z". In the INIT, "A"
- * must provide its Verification Tag (Tag_A) in the Initiate
- * Tag field. Tag_A SHOULD be a random number in the range of
- * 1 to 4294967295 (see 5.3.1 for Tag value selection). ...
- */
+ *
+ * A) "A" first sends an INIT chunk to "Z". In the INIT, "A"
+ * must provide its Verification Tag (Tag_A) in the Initiate
+ * Tag field. Tag_A SHOULD be a random number in the range of
+ * 1 to 4294967295 (see 5.3.1 for Tag value selection). ...
+ */
repl = sctp_make_init(asoc, bp, GFP_ATOMIC);
- if (NULL == repl) { goto nomem; }
+ if (!repl)
+ goto nomem;
/* Cast away the const modifier, as we want to just
- * rerun it through as a sideffect.
+ * rerun it through as a sideffect.
*/
- sctp_add_cmd_sf(commands, SCTP_CMD_NEW_ASOC,
- SCTP_ASOC((sctp_association_t *)asoc));
+ sctp_add_cmd_sf(commands, SCTP_CMD_NEW_ASOC,
+ SCTP_ASOC((sctp_association_t *) asoc));
- /* After sending the INIT, "A" starts the T1-init timer and
- * enters the COOKIE-WAIT state.
- */
- sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_START,
+ /* After sending the INIT, "A" starts the T1-init timer and
+ * enters the COOKIE-WAIT state.
+ */
+ sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_START,
SCTP_TO(SCTP_EVENT_TIMEOUT_T1_INIT));
-
sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(repl));
-
return SCTP_DISPOSITION_CONSUME;
+
nomem:
- if (bp) {
+ if (bp)
sctp_bind_addr_free(bp);
- }
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_do_prm_asoc() */
+}
/*
* Process the SEND primitive.
- *
+ *
* Section: 10.1 ULP-to-SCTP
* E) Send
- *
+ *
* Format: SEND(association id, buffer address, byte count [,context]
- * [,stream id] [,life time] [,destination transport address]
+ * [,stream id] [,life time] [,destination transport address]
* [,unorder flag] [,no-bundle flag] [,payload protocol-id] )
* -> result
- *
- * This is the main method to send user data via SCTP.
- *
+ *
+ * This is the main method to send user data via SCTP.
+ *
* Mandatory attributes:
- *
+ *
* o association id - local handle to the SCTP association
- *
+ *
* o buffer address - the location where the user message to be
* transmitted is stored;
- *
+ *
* o byte count - The size of the user data in number of bytes;
- *
+ *
* Optional attributes:
- *
+ *
* o context - an optional 32 bit integer that will be carried in the
* sending failure notification to the ULP if the transportation of
* this User Message fails.
- *
+ *
* o stream id - to indicate which stream to send the data on. If not
* specified, stream 0 will be used.
- *
+ *
* o life time - specifies the life time of the user data. The user data
* will not be sent by SCTP after the life time expires. This
* parameter can be used to avoid efforts to transmit stale
* user messages. SCTP notifies the ULP if the data cannot be
* initiated to transport (i.e. sent to the destination via SCTP's
* send primitive) within the life time variable. However, the
- * user data will be transmitted if SCTP has attempted to transmit a
+ * user data will be transmitted if SCTP has attempted to transmit a
* chunk before the life time expired.
- *
+ *
* o destination transport address - specified as one of the destination
* transport addresses of the peer endpoint to which this packet
* should be sent. Whenever possible, SCTP should use this destination
* transport address for sending the packets, instead of the current
- * primary path.
- *
+ * primary path.
+ *
* o unorder flag - this flag, if present, indicates that the user
- * would like the data delivered in an unordered fashion to the peer
- * (i.e., the U flag is set to 1 on all DATA chunks carrying this
+ * would like the data delivered in an unordered fashion to the peer
+ * (i.e., the U flag is set to 1 on all DATA chunks carrying this
* message).
- *
+ *
* o no-bundle flag - instructs SCTP not to bundle this user data with
- * other outbound DATA chunks. SCTP MAY still bundle even when
+ * other outbound DATA chunks. SCTP MAY still bundle even when
* this flag is present, when faced with network congestion.
- *
- * o payload protocol-id - A 32 bit unsigned integer that is to be
- * passed to the peer indicating the type of payload protocol data
+ *
+ * o payload protocol-id - A 32 bit unsigned integer that is to be
+ * passed to the peer indicating the type of payload protocol data
* being transmitted. This value is passed as opaque data by SCTP.
- *
- *
+ *
* The return value is the disposition.
*/
-sctp_disposition_t
-sctp_sf_do_prm_send(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_do_prm_send(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
- sctp_chunk_t *chunk = arg;
+ sctp_chunk_t *chunk = arg;
+
sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(chunk));
-
return SCTP_DISPOSITION_CONSUME;
-
-} /* sctp_sf_do_prm_send() */
+}
/*
* Process the SHUTDOWN primitive.
- *
+ *
* Section: 10.1:
* C) Shutdown
- *
+ *
* Format: SHUTDOWN(association id)
- * -> result
- *
+ * -> result
+ *
* Gracefully closes an association. Any locally queued user data
* will be delivered to the peer. The association will be terminated only
* after the peer acknowledges all the SCTP packets sent. A success code
* will be returned on successful termination of the association. If
* attempting to terminate the association results in a failure, an error
* code shall be returned.
- *
+ *
* Mandatory attributes:
- *
+ *
* o association id - local handle to the SCTP association
- *
+ *
* Optional attributes:
- *
+ *
* None.
*
* The return value is the disposition.
*/
-sctp_disposition_t
-sctp_sf_do_9_2_prm_shutdown(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- int disposition;
-
- /* From 9.2 Shutdown of an Association
- * Upon receipt of the SHUTDOWN primitive from its upper
- * layer, the endpoint enters SHUTDOWN-PENDING state and
- * remains there until all outstanding data has been
- * acknowledged by its peer. The endpoint accepts no new data
- * from its upper layer, but retransmits data to the far end
- * if necessary to fill gaps.
- */
-
+sctp_disposition_t sctp_sf_do_9_2_prm_shutdown(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ int disposition;
+
+ /* From 9.2 Shutdown of an Association
+ * Upon receipt of the SHUTDOWN primitive from its upper
+ * layer, the endpoint enters SHUTDOWN-PENDING state and
+ * remains there until all outstanding data has been
+ * acknowledged by its peer. The endpoint accepts no new data
+ * from its upper layer, but retransmits data to the far end
+ * if necessary to fill gaps.
+ */
sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
SCTP_STATE(SCTP_STATE_SHUTDOWN_PENDING));
- disposition = SCTP_DISPOSITION_CONSUME;
- if (sctp_outqueue_is_empty(&asoc->outqueue)) {
- disposition =
- sctp_sf_do_9_2_start_shutdown(ep, asoc, type,
+ disposition = SCTP_DISPOSITION_CONSUME;
+ if (sctp_outqueue_is_empty(&asoc->outqueue)) {
+ disposition =
+ sctp_sf_do_9_2_start_shutdown(ep, asoc, type,
arg, commands);
- }
+ }
return disposition;
-
-} /* sctp_sf_do_9_2_prm_shutdown() */
+}
/*
* Process the ABORT primitive.
*
* The return value is the disposition.
*/
-sctp_disposition_t
-sctp_sf_do_9_1_prm_abort(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_do_9_1_prm_abort(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
-
- /* From 9.1 Abort of an Association
- * Upon receipt of the ABORT primitive from its upper
- * layer, the endpoint enters CLOSED state and
- * discard all outstanding data has been
- * acknowledged by its peer. The endpoint accepts no new data
- * from its upper layer, but retransmits data to the far end
- * if necessary to fill gaps.
- */
-
- sctp_chunk_t *abort;
+ /* From 9.1 Abort of an Association
+ * Upon receipt of the ABORT primitive from its upper
+ * layer, the endpoint enters CLOSED state and
+ * discard all outstanding data has been
+ * acknowledged by its peer. The endpoint accepts no new data
+ * from its upper layer, but retransmits data to the far end
+ * if necessary to fill gaps.
+ */
+ sctp_chunk_t *abort;
sctp_disposition_t retval;
retval = SCTP_DISPOSITION_CONSUME;
- /* Generate ABORT chunk to send the peer */
+ /* Generate ABORT chunk to send the peer. */
abort = sctp_make_abort(asoc, NULL, 0);
- if (!abort) {
+ if (!abort)
retval = SCTP_DISPOSITION_NOMEM;
- } else {
- sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(abort));
- }
+ else
+ sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(abort));
- /* Even if we can't send the ABORT due to low memory delete the
+ /* Even if we can't send the ABORT due to low memory delete the
* TCB. This is a departure from our typical NOMEM handling.
*/
- /* Change to CLOSED state */
+ /* Change to CLOSED state. */
sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
SCTP_STATE(SCTP_STATE_CLOSED));
- /* Delete the established association */
+ /* Delete the established association. */
sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
-
return retval;
-
-} /* sctp_sf_do_9_1_prm_abort() */
-
+}
/* We tried an illegal operation on an association which is closed. */
-sctp_disposition_t
-sctp_sf_error_closed(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_error_closed(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
-
sctp_add_cmd_sf(commands, SCTP_CMD_REPORT_ERROR, SCTP_ERROR(-EINVAL));
return SCTP_DISPOSITION_CONSUME;
-
-} /* sctp_sf_error_closed() */
+}
/* We tried an illegal operation on an association which is shutting
* down.
*/
-sctp_disposition_t
-sctp_sf_error_shutdown(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_error_shutdown(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
-
- sctp_add_cmd_sf(commands, SCTP_CMD_REPORT_ERROR,
+ sctp_add_cmd_sf(commands, SCTP_CMD_REPORT_ERROR,
SCTP_ERROR(-ESHUTDOWN));
return SCTP_DISPOSITION_CONSUME;
-
-} /* sctp_sf_error_shutdown() */
+}
/*
* sctp_cookie_wait_prm_shutdown
- *
+ *
* Section: 4 Note: 2
* Verification Tag:
* Inputs
* (endpoint, asoc)
- *
+ *
* The RFC does not explicitly address this issue, but is the route through the
* state table when someone issues a shutdown while in COOKIE_WAIT state.
*
* Outputs
* (timers)
*/
-sctp_disposition_t
-sctp_sf_cookie_wait_prm_shutdown(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands){
-
-
- sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
+sctp_disposition_t sctp_sf_cookie_wait_prm_shutdown(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
SCTP_TO(SCTP_EVENT_TIMEOUT_T1_INIT));
sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
return SCTP_DISPOSITION_DELETE_TCB;
-
-} /* sctp_sf_cookie_wait_prm_shutdown() */
+}
/*
* sctp_cookie_echoed_prm_shutdown
- *
+ *
* Section: 4 Note: 2
* Verification Tag:
* Inputs
* (endpoint, asoc)
- *
+ *
* The RFC does not explcitly address this issue, but is the route through the
* state table when someone issues a shutdown while in COOKIE_ECHOED state.
*
* Outputs
* (timers)
*/
-sctp_disposition_t
-sctp_sf_cookie_echoed_prm_shutdown(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_cookie_echoed_prm_shutdown(
+ const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg, sctp_cmd_seq_t *commands)
{
-
/* There is a single T1 timer, so we should be able to use
- * common function with the COOKIE-WAIT state.
+ * common function with the COOKIE-WAIT state.
*/
return sctp_sf_cookie_wait_prm_shutdown(ep, asoc, type, arg, commands);
-
-} /* sctp_sf_cookie_echoed_prm_shutdown() */
+}
/*
* sctp_cookie_wait_prm_abort
* Outputs
* (timers)
*/
-sctp_disposition_t
-sctp_sf_cookie_wait_prm_abort(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands){
-
+sctp_disposition_t sctp_sf_cookie_wait_prm_abort(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
/* Stop T1-init timer */
- sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
+ sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
SCTP_TO(SCTP_EVENT_TIMEOUT_T1_INIT));
-
return sctp_sf_do_9_1_prm_abort(ep, asoc, type, arg, commands);
-
-} /* sctp_sf_cookie_wait_prm_abort() */
+}
/*
* sctp_cookie_echoed_prm_abort
* Outputs
* (timers)
*/
-sctp_disposition_t
-sctp_sf_cookie_echoed_prm_abort(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_cookie_echoed_prm_abort(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
-
/* There is a single T1 timer, so we should be able to use
* common function with the COOKIE-WAIT state.
*/
return sctp_sf_cookie_wait_prm_abort(ep, asoc, type, arg, commands);
-
-} /* sctp_sf_cookie_echoed_prm_abort() */
+}
/*
* Ignore the primitive event
*
* The return value is the disposition of the primitive.
*/
-sctp_disposition_t
-sctp_sf_ignore_primitive(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_ignore_primitive(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
SCTP_DEBUG_PRINTK("Primitive type %d is ignored.\n", type.primitive);
-
return SCTP_DISPOSITION_DISCARD;
-
-} /* sctp_sf_ignore_primitive() */
-
+}
/***************************************************************************
* These are the state functions for the OTHER events.
/*
* Start the shutdown negotiation.
- *
+ *
* From Section 9.2:
- * Once all its outstanding data has been acknowledged, the endpoint
- * shall send a SHUTDOWN chunk to its peer including in the Cumulative
- * TSN Ack field the last sequential TSN it has received from the peer.
- * It shall then start the T2-shutdown timer and enter the SHUTDOWN-SENT
- * state. If the timer expires, the endpoint must re-send the SHUTDOWN
+ * Once all its outstanding data has been acknowledged, the endpoint
+ * shall send a SHUTDOWN chunk to its peer including in the Cumulative
+ * TSN Ack field the last sequential TSN it has received from the peer.
+ * It shall then start the T2-shutdown timer and enter the SHUTDOWN-SENT
+ * state. If the timer expires, the endpoint must re-send the SHUTDOWN
* with the updated last sequential TSN received from its peer.
- *
+ *
* The return value is the disposition.
*/
-sctp_disposition_t
-sctp_sf_do_9_2_start_shutdown(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
+sctp_disposition_t sctp_sf_do_9_2_start_shutdown(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
sctp_chunk_t *reply;
-
- /* Once all its outstanding data has been acknowledged, the
- * endpoint shall send a SHUTDOWN chunk to its peer including
- * in the Cumulative TSN Ack field the last sequential TSN it
- * has received from the peer.
- */
+
+ /* Once all its outstanding data has been acknowledged, the
+ * endpoint shall send a SHUTDOWN chunk to its peer including
+ * in the Cumulative TSN Ack field the last sequential TSN it
+ * has received from the peer.
+ */
reply = sctp_make_shutdown(asoc);
- if (NULL == reply) { goto nomem; }
+ if (!reply)
+ goto nomem;
/* Set the transport for the SHUTDOWN chunk and the timeout for the
- * T2-shutdown timer.
+ * T2-shutdown timer.
*/
sctp_add_cmd_sf(commands, SCTP_CMD_SETUP_T2, SCTP_CHUNK(reply));
- /* It shall then start the T2-shutdown timer */
- sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_START,
+ /* It shall then start the T2-shutdown timer */
+ sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_START,
SCTP_TO(SCTP_EVENT_TIMEOUT_T2_SHUTDOWN));
-
- if (asoc->autoclose) {
- sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
+
+ if (asoc->autoclose)
+ sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
SCTP_TO(SCTP_EVENT_TIMEOUT_AUTOCLOSE));
- }
- /* and enter the SHUTDOWN-SENT state. */
+ /* and enter the SHUTDOWN-SENT state. */
sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
- SCTP_STATE(SCTP_STATE_SHUTDOWN_SENT));
+ SCTP_STATE(SCTP_STATE_SHUTDOWN_SENT));
sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(reply));
return SCTP_DISPOSITION_CONSUME;
+
nomem:
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_do_9_2_start_shutdown() */
-
+}
/*
* Generate a SHUTDOWN ACK now that everything is SACK'd.
- *
- * From Section 9.2:
- *
+ *
+ * From Section 9.2:
+ *
* If it has no more outstanding DATA chunks, the SHUTDOWN receiver
* shall send a SHUTDOWN ACK and start a T2-shutdown timer of its own,
* entering the SHUTDOWN-ACK-SENT state. If the timer expires, the
* endpoint must re-send the SHUTDOWN ACK.
*
- * The return value is the disposition. */
-sctp_disposition_t
-sctp_sf_do_9_2_shutdown_ack(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
+ * The return value is the disposition.
+ */
+sctp_disposition_t sctp_sf_do_9_2_shutdown_ack(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
sctp_chunk_t *chunk = (sctp_chunk_t *) arg;
sctp_chunk_t *reply;
- /* If it has no more outstanding DATA chunks, the SHUTDOWN receiver
- * shall send a SHUTDOWN ACK ...
- */
- reply = sctp_make_shutdown_ack(asoc, chunk);
- if (NULL == reply) { goto nomem; }
-
- /* Set the transport for the SHUTDOWN ACK chunk and the timeout for
- * the T2-shutdown timer.
- */
+ /* If it has no more outstanding DATA chunks, the SHUTDOWN receiver
+ * shall send a SHUTDOWN ACK ...
+ */
+ reply = sctp_make_shutdown_ack(asoc, chunk);
+ if (!reply)
+ goto nomem;
+
+ /* Set the transport for the SHUTDOWN ACK chunk and the timeout for
+ * the T2-shutdown timer.
+ */
sctp_add_cmd_sf(commands, SCTP_CMD_SETUP_T2, SCTP_CHUNK(reply));
/* and start/restart a T2-shutdown timer of its own, */
sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_RESTART,
SCTP_TO(SCTP_EVENT_TIMEOUT_T2_SHUTDOWN));
-
- if (asoc->autoclose) {
- sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
+
+ if (asoc->autoclose)
+ sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
SCTP_TO(SCTP_EVENT_TIMEOUT_AUTOCLOSE));
- }
- /* Enter the SHUTDOWN-ACK-SENT state. */
+ /* Enter the SHUTDOWN-ACK-SENT state. */
sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
- SCTP_STATE(SCTP_STATE_SHUTDOWN_ACK_SENT));
-
- sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(reply));
+ SCTP_STATE(SCTP_STATE_SHUTDOWN_ACK_SENT));
+ sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(reply));
return SCTP_DISPOSITION_CONSUME;
+
nomem:
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_do_9_2_shutdown_ack() */
+}
/*
* Ignore the event defined as other
*
* The return value is the disposition of the event.
*/
-sctp_disposition_t
-sctp_sf_ignore_other(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_ignore_other(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
- SCTP_DEBUG_PRINTK("The event other type %d is ignored\n",
+ SCTP_DEBUG_PRINTK("The event other type %d is ignored\n",
type.other);
return SCTP_DISPOSITION_DISCARD;
-
-} /* sctp_sf_ignore_primitive() */
+}
/************************************************************
* These are the state functions for handling timeout events.
/*
* RTX Timeout
- *
+ *
* Section: 6.3.3 Handle T3-rtx Expiration
- *
+ *
* Whenever the retransmission timer T3-rtx expires for a destination
* address, do the following:
* [See below]
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_do_6_3_3_rtx(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- sctp_transport_t *transport = arg;
+sctp_disposition_t sctp_sf_do_6_3_3_rtx(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ sctp_transport_t *transport = arg;
if (asoc->overall_error_count >= asoc->overall_error_threshold) {
/* CMD_ASSOC_FAILED calls CMD_DELETE_TCB. */
sctp_add_cmd_sf(commands, SCTP_CMD_ASSOC_FAILED, SCTP_NULL());
return SCTP_DISPOSITION_DELETE_TCB;
- }
-
- /* E1) For the destination address for which the timer
- * expires, adjust its ssthresh with rules defined in Section
- * 7.2.3 and set the cwnd <- MTU.
- */
-
- /* E2) For the destination address for which the timer
- * expires, set RTO <- RTO * 2 ("back off the timer"). The
- * maximum value discussed in rule C7 above (RTO.max) may be
- * used to provide an upper bound to this doubling operation.
- */
-
- /* E3) Determine how many of the earliest (i.e., lowest TSN)
- * outstanding DATA chunks for the address for which the
- * T3-rtx has expired will fit into a single packet, subject
- * to the MTU constraint for the path corresponding to the
- * destination transport address to which the retransmission
- * is being sent (this may be different from the address for
- * which the timer expires [see Section 6.4]). Call this
- * value K. Bundle and retransmit those K DATA chunks in a
- * single packet to the destination endpoint.
- *
- * Note: Any DATA chunks that were sent to the address for
- * which the T3-rtx timer expired but did not fit in one MTU
- * (rule E3 above), should be marked for retransmission and
- * sent as soon as cwnd allows (normally when a SACK arrives).
- */
-
- /* NB: Rules E4 and F1 are implicit in R1. */
+ }
+
+ /* E1) For the destination address for which the timer
+ * expires, adjust its ssthresh with rules defined in Section
+ * 7.2.3 and set the cwnd <- MTU.
+ */
+
+ /* E2) For the destination address for which the timer
+ * expires, set RTO <- RTO * 2 ("back off the timer"). The
+ * maximum value discussed in rule C7 above (RTO.max) may be
+ * used to provide an upper bound to this doubling operation.
+ */
+
+ /* E3) Determine how many of the earliest (i.e., lowest TSN)
+ * outstanding DATA chunks for the address for which the
+ * T3-rtx has expired will fit into a single packet, subject
+ * to the MTU constraint for the path corresponding to the
+ * destination transport address to which the retransmission
+ * is being sent (this may be different from the address for
+ * which the timer expires [see Section 6.4]). Call this
+ * value K. Bundle and retransmit those K DATA chunks in a
+ * single packet to the destination endpoint.
+ *
+ * Note: Any DATA chunks that were sent to the address for
+ * which the T3-rtx timer expired but did not fit in one MTU
+ * (rule E3 above), should be marked for retransmission and
+ * sent as soon as cwnd allows (normally when a SACK arrives).
+ */
+ /* NB: Rules E4 and F1 are implicit in R1. */
sctp_add_cmd_sf(commands, SCTP_CMD_RETRAN, SCTP_TRANSPORT(transport));
/* Do some failure management (Section 8.2). */
- sctp_add_cmd_sf(commands, SCTP_CMD_STRIKE, SCTP_TRANSPORT(transport));
-
- return SCTP_DISPOSITION_CONSUME;
+ sctp_add_cmd_sf(commands, SCTP_CMD_STRIKE, SCTP_TRANSPORT(transport));
-} /* sctp_sf_do_6_3_3_rtx() */
+ return SCTP_DISPOSITION_CONSUME;
+}
/*
* Generate delayed SACK on timeout
- *
+ *
* Section: 6.2 Acknowledgement on Reception of DATA Chunks
*
* The guidelines on delayed acknowledgement algorithm specified in
* allow. However, an SCTP transmitter MUST NOT be more aggressive than
* the following algorithms allow.
*/
-sctp_disposition_t
-sctp_sf_do_6_2_sack(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
-
+sctp_disposition_t sctp_sf_do_6_2_sack(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
sctp_add_cmd_sf(commands, SCTP_CMD_GEN_SACK, SCTP_FORCE());
return SCTP_DISPOSITION_CONSUME;
-
-} /* sctp_sf_do_6_2_sack() */
+}
/*
* sctp_sf_t1_timer_expire
- *
+ *
* Section: 4 Note: 2
* Verification Tag:
* Inputs
* (endpoint, asoc)
- *
- * RFC 2960 Section 4 Notes
+ *
+ * RFC 2960 Section 4 Notes
* 2) If the T1-init timer expires, the endpoint MUST retransmit INIT
* and re-start the T1-init timer without changing state. This MUST
* be repeated up to 'Max.Init.Retransmits' times. After that, the
* endpoint MUST abort the initialization process and report the
* error to SCTP user.
- *
+ *
* 3) If the T1-cookie timer expires, the endpoint MUST retransmit
* COOKIE ECHO and re-start the T1-cookie timer without changing
* state. This MUST be repeated up to 'Max.Init.Retransmits' times.
* After that, the endpoint MUST abort the initialization process and
* report the error to SCTP user.
*
- *
* Outputs
* (timers, events)
*
*/
-sctp_disposition_t
-sctp_sf_t1_timer_expire(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- sctp_chunk_t *repl;
+sctp_disposition_t sctp_sf_t1_timer_expire(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ sctp_chunk_t *repl;
sctp_bind_addr_t *bp;
- sctp_event_timeout_t timer = (sctp_event_timeout_t)arg;
+ sctp_event_timeout_t timer = (sctp_event_timeout_t) arg;
int timeout;
int attempts;
SCTP_DEBUG_PRINTK("Timer T1 expired.\n");
- if ((timeout < asoc->max_init_timeo)
- && (attempts < asoc->max_init_attempts)) {
-
+ if ((timeout < asoc->max_init_timeo) &&
+ (attempts < asoc->max_init_attempts)) {
switch (timer) {
case SCTP_EVENT_TIMEOUT_T1_INIT:
- bp = (sctp_bind_addr_t *)&asoc->base.bind_addr;
+ bp = (sctp_bind_addr_t *) &asoc->base.bind_addr;
repl = sctp_make_init(asoc, bp, GFP_ATOMIC);
break;
+
case SCTP_EVENT_TIMEOUT_T1_COOKIE:
repl = sctp_make_cookie_echo(asoc, NULL);
break;
+
default:
BUG();
break;
- }
-
- if (!repl) { goto nomem; }
+ };
+
+ if (!repl)
+ goto nomem;
sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(repl));
/* Issue a sideeffect to do the needed accounting. */
- sctp_add_cmd_sf(commands, SCTP_CMD_INIT_RESTART,
+ sctp_add_cmd_sf(commands, SCTP_CMD_INIT_RESTART,
SCTP_TO(timer));
-
- } else {
+ } else {
sctp_add_cmd_sf(commands, SCTP_CMD_INIT_FAILED, SCTP_NULL());
-
- return SCTP_DISPOSITION_DELETE_TCB;
- }
-
+ return SCTP_DISPOSITION_DELETE_TCB;
+ }
+
return SCTP_DISPOSITION_CONSUME;
+
nomem:
return SCTP_DISPOSITION_NOMEM;
-
-} /* sctp_sf_t1_timer_expire() */
+}
/* RFC2960 9.2 If the timer expires, the endpoint must re-send the SHUTDOWN
* with the updated last sequential TSN received from its peer.
* the T2-Shutdown timer, giving its peer ample opportunity to transmit
* all of its queued DATA chunks that have not yet been sent.
*/
-sctp_disposition_t
-sctp_sf_t2_timer_expire(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
+sctp_disposition_t sctp_sf_t2_timer_expire(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
sctp_chunk_t *reply = NULL;
SCTP_DEBUG_PRINTK("Timer T2 expired.\n");
-
if (asoc->overall_error_count >= asoc->overall_error_threshold) {
-
/* Note: CMD_ASSOC_FAILED calls CMD_DELETE_TCB. */
-
sctp_add_cmd_sf(commands, SCTP_CMD_ASSOC_FAILED, SCTP_NULL());
return SCTP_DISPOSITION_DELETE_TCB;
- }
+ }
- switch (asoc->state) {
+ switch (asoc->state) {
case SCTP_STATE_SHUTDOWN_SENT:
reply = sctp_make_shutdown(asoc);
break;
+
case SCTP_STATE_SHUTDOWN_ACK_SENT:
reply = sctp_make_shutdown_ack(asoc, NULL);
break;
+
default:
BUG();
break;
- }
- if (!reply) { goto nomem; }
+ };
+
+ if (!reply)
+ goto nomem;
/* Do some failure management (Section 8.2). */
- sctp_add_cmd_sf(commands, SCTP_CMD_STRIKE,
+ sctp_add_cmd_sf(commands, SCTP_CMD_STRIKE,
SCTP_TRANSPORT(asoc->shutdown_last_sent_to));
- /* Set the transport for the SHUTDOWN/ACK chunk and the timeout for
- * the T2-shutdown timer.
- */
+ /* Set the transport for the SHUTDOWN/ACK chunk and the timeout for
+ * the T2-shutdown timer.
+ */
sctp_add_cmd_sf(commands, SCTP_CMD_SETUP_T2, SCTP_CHUNK(reply));
/* Restart the T2-shutdown timer. */
sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_RESTART,
SCTP_TO(SCTP_EVENT_TIMEOUT_T2_SHUTDOWN));
-
sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(reply));
-
return SCTP_DISPOSITION_CONSUME;
+
nomem:
return SCTP_DISPOSITION_NOMEM;
+}
-} /* sctp_sf_t2_timer_expire() */
-
-/* Handle expiration of AUTOCLOSE timer. When the autoclose timer expires,
- * the association is automatically closed by starting the shutdown process.
+/* Handle expiration of AUTOCLOSE timer. When the autoclose timer expires,
+ * the association is automatically closed by starting the shutdown process.
* The work that needs to be done is same as when SHUTDOWN is initiated by
* the user. So this routine looks same as sctp_sf_do_9_2_prm_shutdown().
- */
-sctp_disposition_t
-sctp_sf_autoclose_timer_expire(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
-{
- int disposition;
-
- /* From 9.2 Shutdown of an Association
- * Upon receipt of the SHUTDOWN primitive from its upper
- * layer, the endpoint enters SHUTDOWN-PENDING state and
- * remains there until all outstanding data has been
- * acknowledged by its peer. The endpoint accepts no new data
- * from its upper layer, but retransmits data to the far end
- * if necessary to fill gaps.
- */
-
+ */
+sctp_disposition_t sctp_sf_autoclose_timer_expire(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
+{
+ int disposition;
+
+ /* From 9.2 Shutdown of an Association
+ * Upon receipt of the SHUTDOWN primitive from its upper
+ * layer, the endpoint enters SHUTDOWN-PENDING state and
+ * remains there until all outstanding data has been
+ * acknowledged by its peer. The endpoint accepts no new data
+ * from its upper layer, but retransmits data to the far end
+ * if necessary to fill gaps.
+ */
sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
- SCTP_STATE(SCTP_STATE_SHUTDOWN_PENDING));
+ SCTP_STATE(SCTP_STATE_SHUTDOWN_PENDING));
- disposition = SCTP_DISPOSITION_CONSUME;
- if (sctp_outqueue_is_empty(&asoc->outqueue)) {
- disposition =
- sctp_sf_do_9_2_start_shutdown(ep, asoc, type,
+ disposition = SCTP_DISPOSITION_CONSUME;
+ if (sctp_outqueue_is_empty(&asoc->outqueue)) {
+ disposition =
+ sctp_sf_do_9_2_start_shutdown(ep, asoc, type,
arg, commands);
- }
+ }
return disposition;
-
-} /* sctp_sf_autoclose_timer_expire() */
+}
/*****************************************************************************
* These are sa state functions which could apply to all types of events.
*
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_not_impl(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_not_impl(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
return SCTP_DISPOSITION_NOT_IMPL;
-
-} /* sctp_sf_not_impl() */
+}
/*
* This table entry represents a bug.
*
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_bug(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_bug(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
return SCTP_DISPOSITION_BUG;
-
-} /* sctp_sf_bug() */
+}
/*
* This table entry represents the firing of a timer in the wrong state.
* Since timer deletion cannot be guaranteed a timer 'may' end up firing
- * when the association is in the wrong state. This event should
- * be ignored, so as to prevent any rearming of the timer.
+ * when the association is in the wrong state. This event should
+ * be ignored, so as to prevent any rearming of the timer.
*
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_timer_ignore(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_timer_ignore(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
SCTP_DEBUG_PRINTK("Timer %d ignored.\n", type.chunk);
-
return SCTP_DISPOSITION_CONSUME;
-
-} /* sctp_sf_timer_ignore() */
-
+}
/*
* Discard the chunk.
* Verification Tag: No verification needed.
* Inputs
* (endpoint, asoc, chunk)
- *
+ *
* Outputs
* (asoc, reply_msg, msg_up, timers, counters)
*
* The return value is the disposition of the chunk.
*/
-sctp_disposition_t
-sctp_sf_discard_chunk(const sctp_endpoint_t *ep,
- const sctp_association_t *asoc,
- const sctp_subtype_t type,
- void *arg,
- sctp_cmd_seq_t *commands)
+sctp_disposition_t sctp_sf_discard_chunk(const sctp_endpoint_t *ep,
+ const sctp_association_t *asoc,
+ const sctp_subtype_t type,
+ void *arg,
+ sctp_cmd_seq_t *commands)
{
SCTP_DEBUG_PRINTK("Chunk %d is discarded\n", type.chunk);
return SCTP_DISPOSITION_DISCARD;
-
-} /* sctp_sf_discard_chunk() */
-
+}
/********************************************************************
* 2nd Level Abstractions
********************************************************************/
/* Pull the SACK chunk based on the SACK header. */
-sctp_sackhdr_t *
-sctp_sm_pull_sack(sctp_chunk_t *chunk)
+sctp_sackhdr_t *sctp_sm_pull_sack(sctp_chunk_t *chunk)
{
- sctp_sackhdr_t *sack;
- uint16_t num_blocks;
- uint16_t num_dup_tsns;
-
- sack = (sctp_sackhdr_t *)chunk->skb->data;
- skb_pull(chunk->skb, sizeof(sctp_sackhdr_t));
-
- num_blocks = ntohs(sack->num_gap_ack_blocks);
- num_dup_tsns = ntohs(sack->num_dup_tsns);
+ sctp_sackhdr_t *sack;
+ __u16 num_blocks;
+ __u16 num_dup_tsns;
- skb_pull(chunk->skb, (num_blocks + num_dup_tsns) * sizeof(uint32_t));
+ sack = (sctp_sackhdr_t *) chunk->skb->data;
+ skb_pull(chunk->skb, sizeof(sctp_sackhdr_t));
- return sack;
+ num_blocks = ntohs(sack->num_gap_ack_blocks);
+ num_dup_tsns = ntohs(sack->num_dup_tsns);
-} /* sctp_sm_pull_sack() */
+ skb_pull(chunk->skb, (num_blocks + num_dup_tsns) * sizeof(__u32));
+ return sack;
+}
#include <linux/skbuff.h>
#include <net/sctp/sctp.h>
#include <net/sctp/sctp_sm.h>
-
-sctp_sm_table_entry_t nop = {fn: sctp_sf_discard_chunk,
+
+sctp_sm_table_entry_t nop = {fn: sctp_sf_discard_chunk,
name: "sctp_sf_discard_chunk"};
sctp_sm_table_entry_t bug = {fn: sctp_sf_bug, name: "sctp_sf_bug"};
"sctp table %p possible attack:" \
" event %d exceeds max %d\n", \
_table, event_subtype._type, _max); \
- return(&bug); \
- } \
- return(&_table[event_subtype._type][(int)state]);
+ return(&bug); \
+ } \
+ return(&_table[event_subtype._type][(int)state]);
-sctp_sm_table_entry_t *
-sctp_sm_lookup_event(sctp_event_t event_type, sctp_state_t state,
- sctp_subtype_t event_subtype)
+sctp_sm_table_entry_t *sctp_sm_lookup_event(sctp_event_t event_type,
+ sctp_state_t state,
+ sctp_subtype_t event_subtype)
{
- switch (event_type) {
- case SCTP_EVENT_T_CHUNK :
- return sctp_chunk_event_lookup(event_subtype.chunk, state);
- break;
- case SCTP_EVENT_T_TIMEOUT :
- DO_LOOKUP(SCTP_EVENT_TIMEOUT_MAX, timeout,
+ switch (event_type) {
+ case SCTP_EVENT_T_CHUNK:
+ return sctp_chunk_event_lookup(event_subtype.chunk, state);
+ break;
+ case SCTP_EVENT_T_TIMEOUT:
+ DO_LOOKUP(SCTP_EVENT_TIMEOUT_MAX, timeout,
timeout_event_table);
- break;
- case SCTP_EVENT_T_OTHER :
- DO_LOOKUP(SCTP_EVENT_OTHER_MAX, other, other_event_table);
- break;
- case SCTP_EVENT_T_PRIMITIVE:
- DO_LOOKUP(SCTP_EVENT_PRIMITIVE_MAX, primitive,
- primitive_event_table);
- break;
- default:
- /* Yikes! We got an illegal event type. */
- return(&bug);
- }
+ break;
-} /* sctp_sm_lookup_event() */
+ case SCTP_EVENT_T_OTHER:
+ DO_LOOKUP(SCTP_EVENT_OTHER_MAX, other, other_event_table);
+ break;
+
+ case SCTP_EVENT_T_PRIMITIVE:
+ DO_LOOKUP(SCTP_EVENT_PRIMITIVE_MAX, primitive,
+ primitive_event_table);
+ break;
+ default:
+ /* Yikes! We got an illegal event type. */
+ return &bug;
+ };
+}
#define TYPE_SCTP_DATA { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_tabort_8_4_8, name: "sctp_sf_tabort_8_4_8"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_tabort_8_4_8, name: "sctp_sf_tabort_8_4_8"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_eat_data_6_2, name: "sctp_sf_eat_data_6_2"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_eat_data_6_2, name: "sctp_sf_eat_data_6_2"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_eat_data_fast_4_4, name: "sctp_sf_eat_data_fast_4_4"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
{fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_eat_data_6_2, name: "sctp_sf_eat_data_6_2"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_eat_data_6_2, name: "sctp_sf_eat_data_6_2"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_eat_data_fast_4_4, name: "sctp_sf_eat_data_fast_4_4"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
} /* TYPE_SCTP_DATA */
-
-
#define TYPE_SCTP_INIT { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_do_5_1B_init, name: "sctp_sf_do_5_1B_init"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_do_5_1B_init, name: "sctp_sf_do_5_1B_init"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_do_5_2_1_siminit, name: "sctp_sf_do_5_2_1_siminit"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
{fn: sctp_sf_do_5_2_1_siminit, name: "sctp_sf_do_5_2_1_siminit"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_do_5_2_1_siminit, name: "sctp_sf_do_5_2_1_siminit"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_do_5_2_2_dupinit, name: "sctp_sf_do_5_2_2_dupinit"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_do_5_2_2_dupinit, name: "sctp_sf_do_5_2_2_dupinit"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_do_5_2_2_dupinit, name: "sctp_sf_do_5_2_2_dupinit"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_do_5_2_2_dupinit, name: "sctp_sf_do_5_2_2_dupinit"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_do_5_2_2_dupinit, name: "sctp_sf_do_5_2_2_dupinit"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_do_5_2_2_dupinit, name: "sctp_sf_do_5_2_2_dupinit"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_do_5_2_2_dupinit, name: "sctp_sf_do_5_2_2_dupinit"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_do_5_2_2_dupinit, name: "sctp_sf_do_5_2_2_dupinit"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
} /* TYPE_SCTP_INIT */
#define TYPE_SCTP_INIT_ACK { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
{fn: sctp_sf_do_5_1C_ack, name: "sctp_sf_do_5_1C_ack"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
} /* TYPE_SCTP_INIT_ACK */
#define TYPE_SCTP_SACK { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_tabort_8_4_8, name: "sctp_sf_tabort_8_4_8"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_tabort_8_4_8, name: "sctp_sf_tabort_8_4_8"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_eat_sack_6_2, name: "sctp_sf_eat_sack_6_2"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_eat_sack_6_2, name: "sctp_sf_eat_sack_6_2"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_eat_sack_6_2, name: "sctp_sf_eat_sack_6_2"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_eat_sack_6_2, name: "sctp_sf_eat_sack_6_2"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
{fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_eat_sack_6_2, name: "sctp_sf_eat_sack_6_2"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_eat_sack_6_2, name: "sctp_sf_eat_sack_6_2"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_eat_sack_6_2, name: "sctp_sf_eat_sack_6_2"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_eat_sack_6_2, name: "sctp_sf_eat_sack_6_2"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
} /* TYPE_SCTP_SACK */
#define TYPE_SCTP_HEARTBEAT { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_tabort_8_4_8, name: "sctp_sf_tabort_8_4_8"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_tabort_8_4_8, name: "sctp_sf_tabort_8_4_8"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
{fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_beat_8_3, name: "sctp_sf_beat_8_3"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_beat_8_3, name: "sctp_sf_beat_8_3"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_beat_8_3, name: "sctp_sf_beat_8_3"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_beat_8_3, name: "sctp_sf_beat_8_3"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_beat_8_3, name: "sctp_sf_beat_8_3"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_beat_8_3, name: "sctp_sf_beat_8_3"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_beat_8_3, name: "sctp_sf_beat_8_3"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_beat_8_3, name: "sctp_sf_beat_8_3"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_beat_8_3, name: "sctp_sf_beat_8_3"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_beat_8_3, name: "sctp_sf_beat_8_3"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
/* This should not happen, but we are nice. */ \
- {fn: sctp_sf_beat_8_3, name: "sctp_sf_beat_8_3"}, \
-\
+ {fn: sctp_sf_beat_8_3, name: "sctp_sf_beat_8_3"}, \
} /* TYPE_SCTP_HEARTBEAT */
#define TYPE_SCTP_HEARTBEAT_ACK { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_tabort_8_4_8, name: "sctp_sf_tabort_8_4_8"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_tabort_8_4_8, name: "sctp_sf_tabort_8_4_8"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
{fn: sctp_sf_violation, name: "sctp_sf_violation"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_backbeat_8_3, name: "sctp_sf_backbeat_8_3"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_backbeat_8_3, name: "sctp_sf_backbeat_8_3"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_backbeat_8_3, name: "sctp_sf_backbeat_8_3"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_backbeat_8_3, name: "sctp_sf_backbeat_8_3"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_backbeat_8_3, name: "sctp_sf_backbeat_8_3"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_backbeat_8_3, name: "sctp_sf_backbeat_8_3"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_backbeat_8_3, name: "sctp_sf_backbeat_8_3"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_backbeat_8_3, name: "sctp_sf_backbeat_8_3"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
} /* TYPE_SCTP_HEARTBEAT_ACK */
#define TYPE_SCTP_ABORT { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_pdiscard, name: "sctp_sf_pdiscard"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_pdiscard, name: "sctp_sf_pdiscard"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
{fn: sctp_sf_cookie_wait_abort, name: "sctp_sf_cookie_wait_abort"}, \
-\
/* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_cookie_echoed_abort, \
+ {fn: sctp_sf_cookie_echoed_abort, \
name: "sctp_sf_cookie_echoed_abort"}, \
-\
/* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_do_9_1_abort, name: "sctp_sf_do_9_1_abort"}, \
-\
+ {fn: sctp_sf_do_9_1_abort, name: "sctp_sf_do_9_1_abort"}, \
/* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_do_9_1_abort, name: "sctp_sf_do_9_1_abort"}, \
-\
+ {fn: sctp_sf_do_9_1_abort, name: "sctp_sf_do_9_1_abort"}, \
/* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_do_9_1_abort, name: "sctp_sf_do_9_1_abort"}, \
-\
+ {fn: sctp_sf_do_9_1_abort, name: "sctp_sf_do_9_1_abort"}, \
/* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_do_9_1_abort, name: "sctp_sf_do_9_1_abort"}, \
-\
+ {fn: sctp_sf_do_9_1_abort, name: "sctp_sf_do_9_1_abort"}, \
/* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_do_9_1_abort, name: "sctp_sf_do_9_1_abort"}, \
-\
+ {fn: sctp_sf_do_9_1_abort, name: "sctp_sf_do_9_1_abort"}, \
} /* TYPE_SCTP_ABORT */
#define TYPE_SCTP_SHUTDOWN { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_tabort_8_4_8, name: "sctp_sf_tabort_8_4_8"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_tabort_8_4_8, name: "sctp_sf_tabort_8_4_8"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_do_9_2_shutdown, name: "sctp_sf_do_9_2_shutdown"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
{fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_do_9_2_shutdown, name: "sctp_sf_do_9_2_shutdown"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_do_9_2_shutdown_ack, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_do_9_2_shutdown_ack, \
name: "sctp_sf_do_9_2_shutdown_ack"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
} /* TYPE_SCTP_SHUTDOWN */
#define TYPE_SCTP_SHUTDOWN_ACK { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_violation, name: "sctp_sf_violation"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_violation, name: "sctp_sf_violation"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_do_9_2_final, name: "sctp_sf_do_9_2_final"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_violation, name: "sctp_sf_violation"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_do_9_2_final, name: "sctp_sf_do_9_2_final"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_violation, name: "sctp_sf_violation"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_violation, name: "sctp_sf_violation"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_do_9_2_final, name: "sctp_sf_do_9_2_final"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_violation, name: "sctp_sf_violation"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_do_9_2_final, name: "sctp_sf_do_9_2_final"}, \
} /* TYPE_SCTP_SHUTDOWN_ACK */
#define TYPE_SCTP_ERROR { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_tabort_8_4_8, name: "sctp_sf_tabort_8_4_8"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_operr_notify, name: "sctp_sf_operr_notify"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_tabort_8_4_8, name: "sctp_sf_tabort_8_4_8"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_operr_notify, name: "sctp_sf_operr_notify"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
} /* TYPE_SCTP_ERROR */
#define TYPE_SCTP_COOKIE_ECHO { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_do_5_1D_ce, name: "sctp_sf_do_5_1D_ce"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_do_5_1D_ce, name: "sctp_sf_do_5_1D_ce"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_do_5_2_4_dupcook, name: "sctp_sf_do_5_2_4_dupcook"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_do_5_2_4_dupcook, name: "sctp_sf_do_5_2_4_dupcook"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_do_5_2_4_dupcook, name: "sctp_sf_do_5_2_4_dupcook"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_do_5_2_4_dupcook, name: "sctp_sf_do_5_2_4_dupcook"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_do_5_2_4_dupcook, name: "sctp_sf_do_5_2_4_dupcook"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_do_5_2_4_dupcook, name: "sctp_sf_do_5_2_4_dupcook"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
{fn: sctp_sf_do_5_2_4_dupcook, name: "sctp_sf_do_5_2_4_dupcook"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_do_5_2_4_dupcook, name: "sctp_sf_do_5_2_4_dupcook"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_do_5_2_4_dupcook, name: "sctp_sf_do_5_2_4_dupcook"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_do_5_2_4_dupcook, name: "sctp_sf_do_5_2_4_dupcook"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_do_5_2_4_dupcook, name: "sctp_sf_do_5_2_4_dupcook"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_do_5_2_4_dupcook, name: "sctp_sf_do_5_2_4_dupcook"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_do_5_2_4_dupcook, name: "sctp_sf_do_5_2_4_dupcook"}, \
-\
} /* TYPE_SCTP_COOKIE_ECHO */
#define TYPE_SCTP_COOKIE_ACK { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_do_5_1E_ca, name: "sctp_sf_do_5_1E_ca"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
{fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_do_5_1E_ca, name: "sctp_sf_do_5_1E_ca"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
} /* TYPE_SCTP_COOKIE_ACK */
#define TYPE_SCTP_ECN_ECNE { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_do_ecne, name: "sctp_sf_do_ecne"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_do_ecne, name: "sctp_sf_do_ecne"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_do_ecne, name: "sctp_sf_do_ecne"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_do_ecne, name: "sctp_sf_do_ecne"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_do_ecne, name: "sctp_sf_do_ecne"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
{fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_do_ecne, name: "sctp_sf_do_ecne"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_do_ecne, name: "sctp_sf_do_ecne"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_do_ecne, name: "sctp_sf_do_ecne"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_do_ecne, name: "sctp_sf_do_ecne"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_do_ecne, name: "sctp_sf_do_ecne"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
} /* TYPE_SCTP_ECN_ECNE */
#define TYPE_SCTP_ECN_CWR { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_do_ecn_cwr, name: "sctp_sf_do_ecn_cwr"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_do_ecn_cwr, name: "sctp_sf_do_ecn_cwr"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_do_ecn_cwr, name: "sctp_sf_do_ecn_cwr"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
{fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_do_ecn_cwr, name: "sctp_sf_do_ecn_cwr"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_do_ecn_cwr, name: "sctp_sf_do_ecn_cwr"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_do_ecn_cwr, name: "sctp_sf_do_ecn_cwr"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
} /* TYPE_SCTP_ECN_CWR */
#define TYPE_SCTP_SHUTDOWN_COMPLETE { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_ootb, name: "sctp_sf_ootb"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
{fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_do_4_C, name: "sctp_sf_do_4_C"}, \
-\
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_do_4_C, name: "sctp_sf_do_4_C"}, \
} /* TYPE_SCTP_SHUTDOWN_COMPLETE */
/* The primary index for this table is the chunk type.
* For base protocol (RFC 2960).
*/
sctp_sm_table_entry_t chunk_event_table[SCTP_NUM_BASE_CHUNK_TYPES][SCTP_STATE_NUM_STATES] = {
- TYPE_SCTP_DATA,
- TYPE_SCTP_INIT,
- TYPE_SCTP_INIT_ACK,
- TYPE_SCTP_SACK,
- TYPE_SCTP_HEARTBEAT,
- TYPE_SCTP_HEARTBEAT_ACK,
- TYPE_SCTP_ABORT,
- TYPE_SCTP_SHUTDOWN,
- TYPE_SCTP_SHUTDOWN_ACK,
- TYPE_SCTP_ERROR,
- TYPE_SCTP_COOKIE_ECHO,
- TYPE_SCTP_COOKIE_ACK,
- TYPE_SCTP_ECN_ECNE,
- TYPE_SCTP_ECN_CWR,
- TYPE_SCTP_SHUTDOWN_COMPLETE,
+ TYPE_SCTP_DATA,
+ TYPE_SCTP_INIT,
+ TYPE_SCTP_INIT_ACK,
+ TYPE_SCTP_SACK,
+ TYPE_SCTP_HEARTBEAT,
+ TYPE_SCTP_HEARTBEAT_ACK,
+ TYPE_SCTP_ABORT,
+ TYPE_SCTP_SHUTDOWN,
+ TYPE_SCTP_SHUTDOWN_ACK,
+ TYPE_SCTP_ERROR,
+ TYPE_SCTP_COOKIE_ECHO,
+ TYPE_SCTP_COOKIE_ACK,
+ TYPE_SCTP_ECN_ECNE,
+ TYPE_SCTP_ECN_CWR,
+ TYPE_SCTP_SHUTDOWN_COMPLETE,
}; /* state_fn_t chunk_event_table[][] */
static sctp_sm_table_entry_t
chunk_event_table_asconf[SCTP_STATE_NUM_STATES] = {
/* SCTP_STATE_EMPTY */
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
/* SCTP_STATE_CLOSED */
{fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
/* SCTP_STATE_COOKIE_WAIT */
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
/* SCTP_STATE_COOKIE_ECHOED */
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
/* SCTP_STATE_ESTABLISHED */
- {fn: sctp_sf_discard_chunk,
+ {fn: sctp_sf_discard_chunk,
name: "sctp_sf_discard_chunk (will be sctp_addip_do_asconf)"},
/* SCTP_STATE_SHUTDOWN_PENDING */
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
/* SCTP_STATE_SHUTDOWN_SENT */
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
/* SCTP_STATE_SHUTDOWN_RECEIVED */
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
/* SCTP_STATE_SHUTDOWN_ACK_SENT */
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
}; /* chunk asconf */
static sctp_sm_table_entry_t
chunk_event_table_asconf_ack[SCTP_STATE_NUM_STATES] = {
/* SCTP_STATE_EMPTY */
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
/* SCTP_STATE_CLOSED */
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
/* SCTP_STATE_COOKIE_WAIT */
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
/* SCTP_STATE_COOKIE_ECHOED */
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
/* SCTP_STATE_ESTABLISHED */
- {fn: sctp_sf_discard_chunk,
+ {fn: sctp_sf_discard_chunk,
name: "sctp_sf_discard_chunk (will be sctp_addip_do_asconf_ack)"},
/* SCTP_STATE_SHUTDOWN_PENDING */
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
/* SCTP_STATE_SHUTDOWN_SENT */
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
/* SCTP_STATE_SHUTDOWN_RECEIVED */
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
/* SCTP_STATE_SHUTDOWN_ACK_SENT */
- {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
+ {fn: sctp_sf_discard_chunk, name: "sctp_sf_discard_chunk"},
}; /* chunk asconf_ack */
#define TYPE_SCTP_PRIMITIVE_INITIALIZE { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
} /* TYPE_SCTP_PRIMITIVE_INITIALIZE */
#define TYPE_SCTP_PRIMITIVE_ASSOCIATE { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_do_prm_asoc, name: "sctp_sf_do_prm_asoc"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_do_prm_asoc, name: "sctp_sf_do_prm_asoc"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
} /* TYPE_SCTP_PRIMITIVE_ASSOCIATE */
#define TYPE_SCTP_PRIMITIVE_SHUTDOWN { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_error_closed, name: "sctp_sf_error_closed"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_error_closed, name: "sctp_sf_error_closed"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
{fn: sctp_sf_cookie_wait_prm_shutdown, \
- name: "sctp_sf_cookie_wait_prm_shutdown"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_cookie_echoed_prm_shutdown, \
- name:"sctp_sf_cookie_echoed_prm_shutdown"},\
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_do_9_2_prm_shutdown, \
+ name: "sctp_sf_cookie_wait_prm_shutdown"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_cookie_echoed_prm_shutdown, \
+ name:"sctp_sf_cookie_echoed_prm_shutdown"},\
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_do_9_2_prm_shutdown, \
name: "sctp_sf_do_9_2_prm_shutdown"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_ignore_primitive, name: "sctp_sf_ignore_primitive"}, \
-\
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_ignore_primitive, name: "sctp_sf_ignore_primitive"}, \
/* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_ignore_primitive, name: "sctp_sf_ignore_primitive"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_ignore_primitive, name: "sctp_sf_ignore_primitive"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_ignore_primitive, name: "sctp_sf_ignore_primitive"}, \
-\
+ {fn: sctp_sf_ignore_primitive, name: "sctp_sf_ignore_primitive"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_ignore_primitive, name: "sctp_sf_ignore_primitive"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_ignore_primitive, name: "sctp_sf_ignore_primitive"}, \
} /* TYPE_SCTP_PRIMITIVE_SHUTDOWN */
#define TYPE_SCTP_PRIMITIVE_ABORT { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_error_closed, name: "sctp_sf_error_closed"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_error_closed, name: "sctp_sf_error_closed"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
{fn: sctp_sf_cookie_wait_prm_abort, \
- name: "sctp_sf_cookie_wait_prm_abort"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_cookie_echoed_prm_abort, \
- name: "sctp_sf_cookie_echoed_prm_abort"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_do_9_1_prm_abort, \
- name: "sctp_sf_do_9_1_prm_abort"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_do_9_1_prm_abort, \
- name: "sctp_sf_do_9_1_prm_abort"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_do_9_1_prm_abort, \
- name: "sctp_sf_do_9_1_prm_abort"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_do_9_1_prm_abort, \
- name: "sctp_sf_do_9_1_prm_abort"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_do_9_1_prm_abort, \
- name: "sctp_sf_do_9_1_prm_abort"}, \
-\
+ name: "sctp_sf_cookie_wait_prm_abort"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_cookie_echoed_prm_abort, \
+ name: "sctp_sf_cookie_echoed_prm_abort"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_do_9_1_prm_abort, \
+ name: "sctp_sf_do_9_1_prm_abort"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_do_9_1_prm_abort, \
+ name: "sctp_sf_do_9_1_prm_abort"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_do_9_1_prm_abort, \
+ name: "sctp_sf_do_9_1_prm_abort"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_do_9_1_prm_abort, \
+ name: "sctp_sf_do_9_1_prm_abort"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_do_9_1_prm_abort, \
+ name: "sctp_sf_do_9_1_prm_abort"}, \
} /* TYPE_SCTP_PRIMITIVE_ABORT */
#define TYPE_SCTP_PRIMITIVE_SEND { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_error_closed, name: "sctp_sf_error_closed"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_error_closed, name: "sctp_sf_error_closed"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_do_prm_send, name: "sctp_sf_do_prm_send"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_do_prm_send, name: "sctp_sf_do_prm_send"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
{fn: sctp_sf_do_prm_send, name: "sctp_sf_do_prm_send"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_do_prm_send, name: "sctp_sf_do_prm_send"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_do_prm_send, name: "sctp_sf_do_prm_send"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_error_shutdown, name: "sctp_sf_error_shutdown"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_error_shutdown, name: "sctp_sf_error_shutdown"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_error_shutdown, name: "sctp_sf_error_shutdown"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_error_shutdown, name: "sctp_sf_error_shutdown"}, \
-\
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_error_shutdown, name: "sctp_sf_error_shutdown"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_error_shutdown, name: "sctp_sf_error_shutdown"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_error_shutdown, name: "sctp_sf_error_shutdown"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_error_shutdown, name: "sctp_sf_error_shutdown"}, \
} /* TYPE_SCTP_PRIMITIVE_SEND */
#define TYPE_SCTP_PRIMITIVE_SETPRIMARY { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
-} /* TYPE_SCTP_PRIMITIVE_SETPRIMARY */
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+} /* TYPE_SCTP_PRIMITIVE_SETPRIMARY */
#define TYPE_SCTP_PRIMITIVE_RECEIVE { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
} /* TYPE_SCTP_PRIMITIVE_RECEIVE */
#define TYPE_SCTP_PRIMITIVE_STATUS { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
} /* TYPE_SCTP_PRIMITIVE_STATUS */
#define TYPE_SCTP_PRIMITIVE_CHANGEHEARTBEAT { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
} /* TYPE_SCTP_PRIMITIVE_CHANGEHEARTBEAT */
#define TYPE_SCTP_PRIMITIVE_REQUESTHEARTBEAT { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
} /* TYPE_SCTP_PRIMITIVE_REQUESTHEARTBEAT */
#define TYPE_SCTP_PRIMITIVE_GETSRTTREPORT { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
} /* TYPE_SCTP_PRIMITIVE_GETSRTTREPORT */
#define TYPE_SCTP_PRIMITIVE_SETFAILURETHRESHOLD { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
} /* TYPE_SCTP_PRIMITIVE_SETFAILURETHRESHOLD */
#define TYPE_SCTP_PRIMITIVE_SETPROTOPARAMETERS { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
} /* TYPE_SCTP_PRIMITIVE_SETPROTOPARAMETERS */
#define TYPE_SCTP_PRIMITIVE_RECEIVE_UNSENT { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
} /* TYPE_SCTP_PRIMITIVE_RECEIVE_UNSENT */
#define TYPE_SCTP_PRIMITIVE_RECEIVE_UNACKED { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
} /* TYPE_SCTP_PRIMITIVE_RECEIVE_UNACKED */
#define TYPE_SCTP_PRIMITIVE_DESTROY { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
} /* TYPE_SCTP_PRIMITIVE_DESTROY */
/* The primary index for this table is the primitive type.
* The secondary index for this table is the state.
*/
sctp_sm_table_entry_t primitive_event_table[SCTP_NUM_PRIMITIVE_TYPES][SCTP_STATE_NUM_STATES] = {
- TYPE_SCTP_PRIMITIVE_INITIALIZE,
- TYPE_SCTP_PRIMITIVE_ASSOCIATE,
- TYPE_SCTP_PRIMITIVE_SHUTDOWN,
- TYPE_SCTP_PRIMITIVE_ABORT,
- TYPE_SCTP_PRIMITIVE_SEND,
- TYPE_SCTP_PRIMITIVE_SETPRIMARY,
- TYPE_SCTP_PRIMITIVE_RECEIVE,
- TYPE_SCTP_PRIMITIVE_STATUS,
- TYPE_SCTP_PRIMITIVE_CHANGEHEARTBEAT,
- TYPE_SCTP_PRIMITIVE_REQUESTHEARTBEAT,
- TYPE_SCTP_PRIMITIVE_GETSRTTREPORT,
- TYPE_SCTP_PRIMITIVE_SETFAILURETHRESHOLD,
- TYPE_SCTP_PRIMITIVE_SETPROTOPARAMETERS,
- TYPE_SCTP_PRIMITIVE_RECEIVE_UNSENT,
- TYPE_SCTP_PRIMITIVE_RECEIVE_UNACKED,
- TYPE_SCTP_PRIMITIVE_DESTROY,
-}; /* primitive_event_table[][] */
+ TYPE_SCTP_PRIMITIVE_INITIALIZE,
+ TYPE_SCTP_PRIMITIVE_ASSOCIATE,
+ TYPE_SCTP_PRIMITIVE_SHUTDOWN,
+ TYPE_SCTP_PRIMITIVE_ABORT,
+ TYPE_SCTP_PRIMITIVE_SEND,
+ TYPE_SCTP_PRIMITIVE_SETPRIMARY,
+ TYPE_SCTP_PRIMITIVE_RECEIVE,
+ TYPE_SCTP_PRIMITIVE_STATUS,
+ TYPE_SCTP_PRIMITIVE_CHANGEHEARTBEAT,
+ TYPE_SCTP_PRIMITIVE_REQUESTHEARTBEAT,
+ TYPE_SCTP_PRIMITIVE_GETSRTTREPORT,
+ TYPE_SCTP_PRIMITIVE_SETFAILURETHRESHOLD,
+ TYPE_SCTP_PRIMITIVE_SETPROTOPARAMETERS,
+ TYPE_SCTP_PRIMITIVE_RECEIVE_UNSENT,
+ TYPE_SCTP_PRIMITIVE_RECEIVE_UNACKED,
+ TYPE_SCTP_PRIMITIVE_DESTROY,
+};
#define TYPE_SCTP_OTHER_NO_PENDING_TSN { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_ignore_other, name: "sctp_sf_ignore_other"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_ignore_other, name: "sctp_sf_ignore_other"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_ignore_other, name: "sctp_sf_ignore_other"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_ignore_other, name: "sctp_sf_ignore_other"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
{fn: sctp_sf_ignore_other, name: "sctp_sf_ignore_other"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_ignore_other, name: "sctp_sf_ignore_other"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_ignore_other, name: "sctp_sf_ignore_other"}, \
-\
/* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_do_9_2_start_shutdown, \
+ {fn: sctp_sf_do_9_2_start_shutdown, \
name: "sctp_do_9_2_start_shutdown"}, \
-\
/* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_ignore_other, name: "sctp_sf_ignore_other"}, \
-\
+ {fn: sctp_sf_ignore_other, name: "sctp_sf_ignore_other"}, \
/* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_do_9_2_shutdown_ack, \
+ {fn: sctp_sf_do_9_2_shutdown_ack, \
name: "sctp_sf_do_9_2_shutdown_ack"}, \
-\
/* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_ignore_other, name: "sctp_sf_ignore_other"}, \
-\
-} /* TYPE_SCTP_OTHER_NO_PENDING_TSN */
+ {fn: sctp_sf_ignore_other, name: "sctp_sf_ignore_other"}, \
+}
#define TYPE_SCTP_OTHER_ICMP_UNREACHFRAG { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
-} /* TYPE_SCTP_OTHER_ICMP_UNREACHFRAG */
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+}
sctp_sm_table_entry_t other_event_table[SCTP_NUM_OTHER_TYPES][SCTP_STATE_NUM_STATES] = {
- TYPE_SCTP_OTHER_NO_PENDING_TSN,
- TYPE_SCTP_OTHER_ICMP_UNREACHFRAG,
+ TYPE_SCTP_OTHER_NO_PENDING_TSN,
+ TYPE_SCTP_OTHER_ICMP_UNREACHFRAG,
};
#define TYPE_SCTP_EVENT_TIMEOUT_NONE { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
{fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
-} /* TYPE_SCTP_EVENT_TIMEOUT_NONE */
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+}
#define TYPE_SCTP_EVENT_TIMEOUT_T1_COOKIE { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
{fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_t1_timer_expire, name: "sctp_sf_t1_timer_expire"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
-} /* TYPE_SCTP_EVENT_TIMEOUT_T1_COOKIE */
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_t1_timer_expire, name: "sctp_sf_t1_timer_expire"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+}
#define TYPE_SCTP_EVENT_TIMEOUT_T1_INIT { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
{fn: sctp_sf_t1_timer_expire, name: "sctp_sf_t1_timer_expire"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
-} /* TYPE_SCTP_EVENT_TIMEOUT_T1_INIT */
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+}
#define TYPE_SCTP_EVENT_TIMEOUT_T2_SHUTDOWN { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
{fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_t2_timer_expire, name: "sctp_sf_t2_timer_expire"}, \
-\
- /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_t2_timer_expire, name: "sctp_sf_t2_timer_expire"}, \
-\
-} /* TYPE_SCTP_EVENT_TIMEOUT_T2_SHUTDOWN */
+ /* SCTP_STATE_COOKIE_WAIT */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_SHUTDOWN_PENDING */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_SHUTDOWN_SENT */ \
+ {fn: sctp_sf_t2_timer_expire, name: "sctp_sf_t2_timer_expire"}, \
+ /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+ {fn: sctp_sf_t2_timer_expire, name: "sctp_sf_t2_timer_expire"}, \
+}
#define TYPE_SCTP_EVENT_TIMEOUT_T3_RTX { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
- /* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_COOKIE_WAIT */ \
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
+ /* SCTP_STATE_CLOSED */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+ /* SCTP_STATE_COOKIE_WAIT */ \
{fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
- /* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_do_6_3_3_rtx, name: "sctp_sf_do_6_3_3_rtx"}, \
-\
- /* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_do_6_3_3_rtx, name: "sctp_sf_do_6_3_3_rtx"}, \
-\
+ /* SCTP_STATE_COOKIE_ECHOED */ \
+ {fn: sctp_sf_do_6_3_3_rtx, name: "sctp_sf_do_6_3_3_rtx"}, \
+ /* SCTP_STATE_ESTABLISHED */ \
+ {fn: sctp_sf_do_6_3_3_rtx, name: "sctp_sf_do_6_3_3_rtx"}, \
/* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_do_6_3_3_rtx, name: "sctp_sf_do_6_3_3_rtx"}, \
-\
+ {fn: sctp_sf_do_6_3_3_rtx, name: "sctp_sf_do_6_3_3_rtx"}, \
/* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
/* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_do_6_3_3_rtx, name: "sctp_sf_do_6_3_3_rtx"}, \
-\
+ {fn: sctp_sf_do_6_3_3_rtx, name: "sctp_sf_do_6_3_3_rtx"}, \
/* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
-} /* TYPE_SCTP_EVENT_TIMEOUT_T3_RTX */
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+}
#define TYPE_SCTP_EVENT_TIMEOUT_T4_RTO { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
/* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
/* SCTP_STATE_COOKIE_WAIT */ \
{fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
/* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
/* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
/* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
/* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
/* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
/* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
-} /* TYPE_SCTP_EVENT_TIMEOUT_T4_RTO */
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+}
#define TYPE_SCTP_EVENT_TIMEOUT_HEARTBEAT { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
/* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
/* SCTP_STATE_COOKIE_WAIT */ \
{fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
/* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
/* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_sendbeat_8_3, name: "sctp_sf_sendbeat_8_3"}, \
-\
+ {fn: sctp_sf_sendbeat_8_3, name: "sctp_sf_sendbeat_8_3"}, \
/* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_sendbeat_8_3, name: "sctp_sf_sendbeat_8_3"}, \
-\
+ {fn: sctp_sf_sendbeat_8_3, name: "sctp_sf_sendbeat_8_3"}, \
/* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
/* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
/* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
-} /* TYPE_SCTP_EVENT_TIMEOUT_HEARTBEAT */
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+}
#define TYPE_SCTP_EVENT_TIMEOUT_SACK { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
/* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
/* SCTP_STATE_COOKIE_WAIT */ \
{fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
/* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
/* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_do_6_2_sack, name: "sctp_sf_do_6_2_sack"}, \
-\
+ {fn: sctp_sf_do_6_2_sack, name: "sctp_sf_do_6_2_sack"}, \
/* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_do_6_2_sack, name: "sctp_sf_do_6_2_sack"}, \
-\
+ {fn: sctp_sf_do_6_2_sack, name: "sctp_sf_do_6_2_sack"}, \
/* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_do_6_2_sack, name: "sctp_sf_do_6_2_sack"}, \
-\
+ {fn: sctp_sf_do_6_2_sack, name: "sctp_sf_do_6_2_sack"}, \
/* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
/* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
-} /* TYPE_SCTP_EVENT_TIMEOUT_SACK */
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+}
#define TYPE_SCTP_EVENT_TIMEOUT_AUTOCLOSE { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
/* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
/* SCTP_STATE_COOKIE_WAIT */ \
{fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
/* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
/* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_autoclose_timer_expire, \
+ {fn: sctp_sf_autoclose_timer_expire, \
name: "sctp_sf_autoclose_timer_expire"}, \
-\
/* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
/* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
/* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
/* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-\
-} /* TYPE_SCTP_EVENT_TIMEOUT_AUTOCLOSE */
+ {fn: sctp_sf_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+}
#define TYPE_SCTP_EVENT_TIMEOUT_PMTU_RAISE { \
- /* SCTP_STATE_EMPTY */ \
- {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
-\
+ /* SCTP_STATE_EMPTY */ \
+ {fn: sctp_sf_bug, name: "sctp_sf_bug"}, \
/* SCTP_STATE_CLOSED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
/* SCTP_STATE_COOKIE_WAIT */ \
{fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
/* SCTP_STATE_COOKIE_ECHOED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
/* SCTP_STATE_ESTABLISHED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
/* SCTP_STATE_SHUTDOWN_PENDING */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
/* SCTP_STATE_SHUTDOWN_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
/* SCTP_STATE_SHUTDOWN_RECEIVED */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
/* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
- {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
-\
-} /* TYPE_SCTP_EVENT_TIMEOUT_PMTU_RAISE */
-
+ {fn: sctp_sf_not_impl, name: "sctp_sf_not_impl"}, \
+}
sctp_sm_table_entry_t timeout_event_table[SCTP_NUM_TIMEOUT_TYPES][SCTP_STATE_NUM_STATES] = {
- TYPE_SCTP_EVENT_TIMEOUT_NONE,
- TYPE_SCTP_EVENT_TIMEOUT_T1_COOKIE,
- TYPE_SCTP_EVENT_TIMEOUT_T1_INIT,
- TYPE_SCTP_EVENT_TIMEOUT_T2_SHUTDOWN,
- TYPE_SCTP_EVENT_TIMEOUT_T3_RTX,
+ TYPE_SCTP_EVENT_TIMEOUT_NONE,
+ TYPE_SCTP_EVENT_TIMEOUT_T1_COOKIE,
+ TYPE_SCTP_EVENT_TIMEOUT_T1_INIT,
+ TYPE_SCTP_EVENT_TIMEOUT_T2_SHUTDOWN,
+ TYPE_SCTP_EVENT_TIMEOUT_T3_RTX,
TYPE_SCTP_EVENT_TIMEOUT_T4_RTO,
- TYPE_SCTP_EVENT_TIMEOUT_HEARTBEAT,
- TYPE_SCTP_EVENT_TIMEOUT_SACK,
- TYPE_SCTP_EVENT_TIMEOUT_AUTOCLOSE,
- TYPE_SCTP_EVENT_TIMEOUT_PMTU_RAISE,
+ TYPE_SCTP_EVENT_TIMEOUT_HEARTBEAT,
+ TYPE_SCTP_EVENT_TIMEOUT_SACK,
+ TYPE_SCTP_EVENT_TIMEOUT_AUTOCLOSE,
+ TYPE_SCTP_EVENT_TIMEOUT_PMTU_RAISE,
};
-sctp_sm_table_entry_t *
-sctp_chunk_event_lookup(sctp_cid_t cid, sctp_state_t state)
+sctp_sm_table_entry_t *sctp_chunk_event_lookup(sctp_cid_t cid, sctp_state_t state)
{
- if ( state > SCTP_STATE_MAX ) { BUG(); }
- if ( cid < 0 ) { return &nop; }
-
- if ( cid <= SCTP_CID_BASE_MAX ) {
- return &chunk_event_table[cid][state];
- }
-
- switch ( cid ) {
- case SCTP_CID_ASCONF : return &chunk_event_table_asconf[state];
- case SCTP_CID_ASCONF_ACK : return &chunk_event_table_asconf_ack[state];
- default: return &nop;
- }
+ if (state > SCTP_STATE_MAX)
+ BUG();
+ if (cid < 0)
+ return &nop;
+
+ if (cid <= SCTP_CID_BASE_MAX)
+ return &chunk_event_table[cid][state];
+
+ switch (cid) {
+ case SCTP_CID_ASCONF:
+ return &chunk_event_table_asconf[state];
+
+ case SCTP_CID_ASCONF_ACK:
+ return &chunk_event_table_asconf_ack[state];
+ default:
+ return &nop;
+ };
return &nop;
}
static inline int sctp_wspace(sctp_association_t *asoc);
static inline void sctp_set_owner_w(sctp_chunk_t *chunk);
static void sctp_wfree(struct sk_buff *skb);
-static int sctp_wait_for_sndbuf(sctp_association_t *asoc, long *timeo_p,
+static int sctp_wait_for_sndbuf(sctp_association_t *asoc, long *timeo_p,
int msg_len);
static int sctp_wait_for_packet(struct sock * sk, int *err, long *timeo_p);
-struct sk_buff * sctp_skb_recv_datagram(struct sock *sk, int flags,
+struct sk_buff * sctp_skb_recv_datagram(struct sock *sk, int flags,
int noblock, int *err);
-static inline void sctp_sk_memcpy_msgname(struct sock *sk, char * msgname,
+static inline void sctp_sk_memcpy_msgname(struct sock *sk, char * msgname,
int *addr_len, struct sk_buff *skb);
-static inline void sctp_sk_addr_set(struct sock *,
+static inline void sctp_sk_addr_set(struct sock *,
const sockaddr_storage_t *newaddr,
sockaddr_storage_t *saveaddr);
-static inline void sctp_sk_addr_restore(struct sock *,
+static inline void sctp_sk_addr_restore(struct sock *,
const sockaddr_storage_t *);
static inline int sctp_sendmsg_verify_name(struct sock *, struct msghdr *);
static int sctp_bindx_add(struct sock *, struct sockaddr_storage *, int);
static int sctp_bindx_rem(struct sock *, struct sockaddr_storage *, int);
static int sctp_do_bind(struct sock *, sockaddr_storage_t *, int);
static int sctp_autobind(struct sock *sk);
-static sctp_bind_bucket_t *sctp_bucket_create(sctp_bind_hashbucket_t *head,
+static sctp_bind_bucket_t *sctp_bucket_create(sctp_bind_hashbucket_t *head,
unsigned short snum);
/* API 3.1.2 bind() - UDP Style Syntax
* The syntax of bind() is,
- *
+ *
* ret = bind(int sd, struct sockaddr *addr, int addrlen);
- *
+ *
* sd - the socket descriptor returned by socket().
* addr - the address structure (struct sockaddr_in or struct
- * sockaddr_in6 [RFC 2553]),
+ * sockaddr_in6 [RFC 2553]),
* addrlen - the size of the address structure.
- *
+ *
* The caller should use struct sockaddr_storage described in RFC 2553
* to represent addr for portability reason.
*/
-int
-sctp_bind(struct sock *sk, struct sockaddr *uaddr, int addr_len)
+int sctp_bind(struct sock *sk, struct sockaddr *uaddr, int addr_len)
{
- int retval = 0;
+ int retval = 0;
+
+ sctp_lock_sock(sk);
- sctp_lock_sock(sk);
-
SCTP_DEBUG_PRINTK("sctp_bind(sk: %p, uaddr: %p, addr_len: %d)\n",
sk, uaddr, addr_len);
/* Disallow binding twice. */
-
- if (0 == sctp_sk(sk)->ep->base.bind_addr.port) {
+ if (!sctp_sk(sk)->ep->base.bind_addr.port)
retval = sctp_do_bind(sk, (sockaddr_storage_t *)uaddr,
addr_len);
- } else {
+ else
retval = -EINVAL;
- }
-
- sctp_release_sock(sk);
- return retval;
+ sctp_release_sock(sk);
-} /* sctp_bind() */
+ return retval;
+}
-/* Bind a local address either to an endpoint or to an association. */
-static int
-sctp_do_bind(struct sock *sk, sockaddr_storage_t *newaddr, int addr_len)
+/* Bind a local address either to an endpoint or to an association. */
+static int sctp_do_bind(struct sock *sk, sockaddr_storage_t *newaddr, int addr_len)
{
sctp_opt_t *sp = sctp_sk(sk);
sctp_endpoint_t *ep = sp->ep;
sockaddr_storage_t tmpaddr, saveaddr;
unsigned short *snum;
int ret = 0;
-
- SCTP_DEBUG_PRINTK("sctp_do_bind(sk: %p, newaddr: %p, addr_len: %d)\n",
+
+ SCTP_DEBUG_PRINTK("sctp_do_bind(sk: %p, newaddr: %p, addr_len: %d)\n",
sk, newaddr, addr_len);
- /* FIXME: This function needs to handle v4-mapped-on-v6
+ /* FIXME: This function needs to handle v4-mapped-on-v6
* addresses!
*/
-
if (PF_INET == sk->family) {
- if (sa_family != AF_INET) {
+ if (sa_family != AF_INET)
return -EINVAL;
- }
}
- /* Make a local copy of the new address. */
+ /* Make a local copy of the new address. */
tmpaddr = *newaddr;
switch (sa_family) {
case AF_INET:
- if (addr_len < sizeof(struct sockaddr_in)) {
+ if (addr_len < sizeof(struct sockaddr_in))
return -EINVAL;
- }
+
ret = inet_addr_type(newaddr->v4.sin_addr.s_addr);
/* FIXME:
- * Should we allow apps to bind to non-local addresses by
- * checking the IP sysctl parameter "ip_nonlocal_bind"?
+ * Should we allow apps to bind to non-local addresses by
+ * checking the IP sysctl parameter "ip_nonlocal_bind"?
*/
- if (newaddr->v4.sin_addr.s_addr != INADDR_ANY
- && ret != RTN_LOCAL) {
+ if (newaddr->v4.sin_addr.s_addr != INADDR_ANY &&
+ ret != RTN_LOCAL)
return -EADDRNOTAVAIL;
- }
+
tmpaddr.v4.sin_port = htons(tmpaddr.v4.sin_port);
snum = &tmpaddr.v4.sin_port;
break;
+
case AF_INET6:
SCTP_V6(
/* FIXME: Hui, please verify this. Looking at
* the ipv6 code I see a SIN6_LEN_RFC2133 check.
* I'm guessing that scope_id is a newer addition.
*/
- if (addr_len < sizeof(struct sockaddr_in6)) {
+ if (addr_len < sizeof(struct sockaddr_in6))
return -EINVAL;
- }
- /* FIXME - The support for IPv6 multiple types
+
+ /* FIXME - The support for IPv6 multiple types
* of addresses need to be added later.
*/
ret = sctp_ipv6_addr_type(&newaddr->v6.sin6_addr);
- tmpaddr.v6.sin6_port = htons(tmpaddr.v6.sin6_port);
+ tmpaddr.v6.sin6_port = htons(tmpaddr.v6.sin6_port);
snum = &tmpaddr.v6.sin6_port;
break;
)
+
default:
return -EINVAL;
- } /* switch(family) */
-
+ };
SCTP_DEBUG_PRINTK("sctp_do_bind: port: %d, new port: %d\n",
bp->port, *snum);
- /* We must either be unbound, or bind to the same port. */
- if ((0 != bp->port) && (*snum != bp->port)) {
+ /* We must either be unbound, or bind to the same port. */
+ if (bp->port && (*snum != bp->port)) {
SCTP_DEBUG_PRINTK("sctp_do_bind:"
" New port %d does not match existing port "
"%d.\n", *snum, bp->port);
- return -EINVAL;
- }
-
- if (*snum && *snum < PROT_SOCK && !capable(CAP_NET_BIND_SERVICE)){
- return -EACCES;
+ return -EINVAL;
}
+ if (*snum && *snum < PROT_SOCK && !capable(CAP_NET_BIND_SERVICE))
+ return -EACCES;
- /* FIXME - Make socket understand that there might be multiple bind
- * addresses and there will be multiple source addresses involved in
- * routing and failover decisions.
- */
-
+ /* FIXME - Make socket understand that there might be multiple bind
+ * addresses and there will be multiple source addresses involved in
+ * routing and failover decisions.
+ */
sctp_sk_addr_set(sk, &tmpaddr, &saveaddr);
-
+
/* Make sure we are allowed to bind here.
* The function sctp_get_port() does duplicate address
* detection.
*/
- if (0 != (ret = sctp_get_port(sk, *snum))) {
+ if ((ret = sctp_get_port(sk, *snum))) {
sctp_sk_addr_restore(sk, &saveaddr);
- if (ret == (int)sk) {
+ if (ret == (long) sk) {
/* This endpoint has a conflicting address. */
return -EINVAL;
} else {
return -EADDRINUSE;
}
}
-
- /* Refresh ephemeral port. */
- if (0 == *snum) {
- *snum = inet_sk(sk)->num;
- }
- /* The getsockname() API depends on 'sport' being set. */
+ /* Refresh ephemeral port. */
+ if (!*snum)
+ *snum = inet_sk(sk)->num;
+
+ /* The getsockname() API depends on 'sport' being set. */
inet_sk(sk)->sport = htons(inet_sk(sk)->num);
- /* Add the address to the bind address list. */
+ /* Add the address to the bind address list. */
sctp_local_bh_disable();
sctp_write_lock(&ep->base.addr_lock);
- /* Use GFP_ATOMIC since BHs are disabled. */
- if (0 != (ret = sctp_add_bind_addr(bp, &tmpaddr, GFP_ATOMIC))) {
+ /* Use GFP_ATOMIC since BHs are disabled. */
+ if ((ret = sctp_add_bind_addr(bp, &tmpaddr, GFP_ATOMIC))) {
sctp_sk_addr_restore(sk, &saveaddr);
- } else if (0 == bp->port) {
- bp->port = *snum;
- }
+ } else if (!bp->port) {
+ bp->port = *snum;
+ }
sctp_write_unlock(&ep->base.addr_lock);
sctp_local_bh_enable();
-
- return ret;
-
-} /* sctp_do_bind */
+ return ret;
+}
/* API 8.1 sctp_bindx()
- *
+ *
* The syntax of sctp_bindx() is,
- *
+ *
* ret = sctp_bindx(int sd,
* struct sockaddr_storage *addrs,
* int addrcnt,
* int flags);
- *
+ *
* If sd is an IPv4 socket, the addresses passed must be IPv4 addresses.
* If the sd is an IPv6 socket, the addresses passed can either be IPv4
* or IPv6 addresses.
- *
+ *
* A single address may be specified as INADDR_ANY or IPV6_ADDR_ANY, see
* section 3.1.2 for this usage.
- *
+ *
* addrs is a pointer to an array of one or more socket addresses. Each
* address is contained in a struct sockaddr_storage, so each address is
* fixed length. The caller specifies the number of addresses in the
* array with addrcnt.
- *
+ *
* On success, sctp_bindx() returns 0. On failure, sctp_bindx() returns -1,
* and sets errno to the appropriate error code. [ Editor's note: need
* to fill in all error code? ]
- *
+ *
* For SCTP, the port given in each socket address must be the same, or
* sctp_bindx() will fail, setting errno to EINVAL .
- *
+ *
* The flags parameter is formed from the bitwise OR of zero or
* more of the following currently defined flags:
- *
+ *
* SCTP_BINDX_ADD_ADDR
* SCTP_BINDX_REM_ADDR
- *
+ *
* SCTP_BIND_ADD_ADDR directs SCTP to add the given addresses to the
* association, and SCTP_BIND_REM_ADDR directs SCTP to remove the given
* addresses from the association. The two flags are mutually exclusive;
* if both are given, sctp_bindx() will fail with EINVAL. A caller may not
* remove all addresses from an association; sctp_bindx() will reject such
* an attempt with EINVAL.
- *
+ *
* An application can use sctp_bindx(SCTP_BINDX_ADD_ADDR) to associate
* additional addresses with an endpoint after calling bind(). Or use
* sctp_bindx(SCTP_BINDX_REM_ADDR) to remove some addresses a listening
* socket is associated with so that no new association accepted will be
* associated with those addresses.
- *
+ *
* SCTP_BIND_ADD_ADDR is defined as 0, so that it becomes the default
* behavior for sctp_bindx() when no flags are given.
- *
+ *
* Adding and removing addresses from a connected association is optional
* functionality. Implementations that do not support this functionality
* should return EOPNOTSUPP.
* sctp_bindx() for a lock-protected call.
*/
-static int
-__sctp_bindx(struct sock *sk, struct sockaddr_storage *addrs,
- int addrcnt, int flags)
+static int __sctp_bindx(struct sock *sk, struct sockaddr_storage *addrs,
+ int addrcnt, int flags)
{
- int retval = 0;
+ int retval = 0;
SCTP_DEBUG_PRINTK("__sctp_bindx(sk: %p, addrs: %p, addrcnt: %d, "
"flags: %s)\n", sk, addrs, addrcnt,
(BINDX_ADD_ADDR == flags)?"ADD":
((BINDX_REM_ADDR == flags)?"REM":"BOGUS"));
-
- switch (flags) {
- case BINDX_ADD_ADDR:
- retval = sctp_bindx_add(sk, addrs, addrcnt);
- break;
- case BINDX_REM_ADDR:
- retval = sctp_bindx_rem(sk, addrs, addrcnt);
- break;
- default:
- retval = -EINVAL;
- break;
- }
- return retval;
-
-} /* __sctp_bindx() */
+ switch (flags) {
+ case BINDX_ADD_ADDR:
+ retval = sctp_bindx_add(sk, addrs, addrcnt);
+ break;
+
+ case BINDX_REM_ADDR:
+ retval = sctp_bindx_rem(sk, addrs, addrcnt);
+ break;
+
+ default:
+ retval = -EINVAL;
+ break;
+ };
+
+ return retval;
+}
/* BINDX with locks.
*
* NOTE: Currently unused at all ...
*/
-int
-sctp_bindx(struct sock *sk, struct sockaddr_storage *addrs, int addrcnt,
- int flags)
+int sctp_bindx(struct sock *sk, struct sockaddr_storage *addrs, int addrcnt,
+ int flags)
{
- int retval;
-
- sctp_lock_sock(sk);
- retval = __sctp_bindx(sk, addrs, addrcnt, flags);
- sctp_release_sock(sk);
+ int retval;
- return retval;
-
-} /* sctp_bindx() */
+ sctp_lock_sock(sk);
+ retval = __sctp_bindx(sk, addrs, addrcnt, flags);
+ sctp_release_sock(sk);
+ return retval;
+}
/* Add a list of addresses as bind addresses to local endpoint or
* association.
*
* Only __sctp_bindx() is supposed to call this function.
*/
-int
-sctp_bindx_add(struct sock *sk, struct sockaddr_storage *addrs, int addrcnt)
+int sctp_bindx_add(struct sock *sk, struct sockaddr_storage *addrs, int addrcnt)
{
-
int cnt;
- int retval = 0;
+ int retval = 0;
int addr_len;
SCTP_DEBUG_PRINTK("sctp_bindx_add (sk: %p, addrs: %p, addrcnt: %d)\n",
sk, addrs, addrcnt);
-
- for (cnt = 0; cnt < addrcnt; cnt++) {
+
+ for (cnt = 0; cnt < addrcnt; cnt++) {
/* The list may contain either IPv4 or IPv6 address;
* determine the address length for walking thru the list.
- */
+ */
switch (((struct sockaddr *)&addrs[cnt])->sa_family) {
- case AF_INET:
- addr_len = sizeof(struct sockaddr_in);
- break;
- case AF_INET6:
- addr_len = sizeof(struct sockaddr_in6);
- break;
- default:
- retval = -EINVAL;
- goto err_bindx_add;
- }
+ case AF_INET:
+ addr_len = sizeof(struct sockaddr_in);
+ break;
+
+ case AF_INET6:
+ addr_len = sizeof(struct sockaddr_in6);
+ break;
- retval = sctp_do_bind(sk, (sockaddr_storage_t *)&addrs[cnt],
+ default:
+ retval = -EINVAL;
+ goto err_bindx_add;
+ };
+
+ retval = sctp_do_bind(sk, (sockaddr_storage_t *)&addrs[cnt],
addr_len);
+
err_bindx_add:
- if (retval < 0) {
+ if (retval < 0) {
/* Failed. Cleanup the ones that has been added */
- if (cnt > 0) {
+ if (cnt > 0)
sctp_bindx_rem(sk, addrs, cnt);
- }
return retval;
}
-
}
/* Notify the peer(s), assuming we have (an) association(s).
* what we have now. In the former case we know the
* association already. --piggy
*
- * This code will be working on either a UDP style or a TCP style
- * socket, or say either an endpoint or an association. The socket
- * type verification code need to be added later before calling the
+ * This code will be working on either a UDP style or a TCP style
+ * socket, or say either an endpoint or an association. The socket
+ * type verification code need to be added later before calling the
* ADDIP code.
* --daisy
- */
+ */
#if CONFIG_IP_SCTP_ADDIP
/* Add these addresses to all associations on this endpoint. */
- if ( retval >= 0) {
+ if (retval >= 0) {
list_t *pos;
sctp_endpoint_t *ep;
sctp_association_t *asoc;
sctp_addip_addr_config(asoc,
SCTP_PARAM_ADD_IP,
addrs, addrcnt);
-
- } /* for (each association on the endpoint) */
- }
+ }
+ }
#endif
- return retval;
-
-} /* sctp_bindx_add() */
-
+ return retval;
+}
/* Remove a list of addresses from bind addresses list. Do not remove the
* last address.
*
* Only __sctp_bindx() is supposed to call this function.
*/
-int
-sctp_bindx_rem(struct sock *sk, struct sockaddr_storage *addrs, int addrcnt)
+int sctp_bindx_rem(struct sock *sk, struct sockaddr_storage *addrs, int addrcnt)
{
sctp_opt_t *sp = sctp_sk(sk);
sctp_endpoint_t *ep = sp->ep;
int cnt;
sctp_bind_addr_t *bp = &ep->base.bind_addr;
- int retval = 0;
- sockaddr_storage_t saveaddr;
+ int retval = 0;
+ sockaddr_storage_t saveaddr;
SCTP_DEBUG_PRINTK("sctp_bindx_rem (sk: %p, addrs: %p, addrcnt: %d)\n",
sk, addrs, addrcnt);
- for (cnt = 0; cnt < addrcnt; cnt++) {
- /* If there is only one bind address, there is nothing more
+ for (cnt = 0; cnt < addrcnt; cnt++) {
+ /* If there is only one bind address, there is nothing more
* to be removed (we need at least one address here).
- */
-
+ */
if (list_empty(&bp->address_list)) {
retval = -EBUSY;
goto err_bindx_rem;
}
- /* The list may contain either IPv4 or IPv6 address;
+ /* The list may contain either IPv4 or IPv6 address;
* determine the address length for walking thru the list.
- */
+ */
switch (((struct sockaddr *)&addrs[cnt])->sa_family) {
- case AF_INET:
- saveaddr = *((sockaddr_storage_t *)
- &addrs[cnt]);
- saveaddr.v4.sin_port =
- ntohs(saveaddr.v4.sin_port);
- /* verify the port */
- if (saveaddr.v4.sin_port != bp->port) {
- retval = -EINVAL;
- goto err_bindx_rem;
- }
- break;
- case AF_INET6:
- saveaddr = *((sockaddr_storage_t *)
- &addrs[cnt]);
- saveaddr.v6.sin6_port =
- ntohs(saveaddr.v6.sin6_port);
- /* verify the port */
- if (saveaddr.v6.sin6_port != bp->port) {
- retval = -EINVAL;
- goto err_bindx_rem;
- }
- break;
- default:
+ case AF_INET:
+ saveaddr = *((sockaddr_storage_t *)
+ &addrs[cnt]);
+ saveaddr.v4.sin_port =
+ ntohs(saveaddr.v4.sin_port);
+ /* verify the port */
+ if (saveaddr.v4.sin_port != bp->port) {
retval = -EINVAL;
goto err_bindx_rem;
- }
+ }
+ break;
- /* FIXME - There is probably a need to check if sk->saddr and
- * sk->rcv_addr are currently set to one of the addresses to
- * be removed. This is something which needs to be looked into
- * when we are fixing the outstanding issues with multi-homing
- * socket routing and failover schemes. Refer to comments in
+ case AF_INET6:
+ saveaddr = *((sockaddr_storage_t *)
+ &addrs[cnt]);
+ saveaddr.v6.sin6_port =
+ ntohs(saveaddr.v6.sin6_port);
+ /* verify the port */
+ if (saveaddr.v6.sin6_port != bp->port) {
+ retval = -EINVAL;
+ goto err_bindx_rem;
+ }
+ break;
+
+ default:
+ retval = -EINVAL;
+ goto err_bindx_rem;
+ };
+
+ /* FIXME - There is probably a need to check if sk->saddr and
+ * sk->rcv_addr are currently set to one of the addresses to
+ * be removed. This is something which needs to be looked into
+ * when we are fixing the outstanding issues with multi-homing
+ * socket routing and failover schemes. Refer to comments in
* sctp_do_bind(). -daisy
*/
sctp_local_bh_disable();
sctp_write_lock(&ep->base.addr_lock);
- retval = sctp_del_bind_addr(bp, &saveaddr);
+ retval = sctp_del_bind_addr(bp, &saveaddr);
sctp_write_unlock(&ep->base.addr_lock);
sctp_local_bh_enable();
+
err_bindx_rem:
- if (retval < 0) {
+ if (retval < 0) {
/* Failed. Add the ones that has been removed back */
- if (cnt > 0) {
+ if (cnt > 0)
sctp_bindx_add(sk, addrs, cnt);
- }
return retval;
}
-
- }
+ }
/*
- * This code will be working on either a UDP style or a TCP style
- * socket, * or say either an endpoint or an association. The socket
- * type verification code need to be added later before calling the
+ * This code will be working on either a UDP style or a TCP style
+ * socket, * or say either an endpoint or an association. The socket
+ * type verification code need to be added later before calling the
* ADDIP code.
* --daisy
*/
list_t *pos;
sctp_endpoint_t *ep;
sctp_association_t *asoc;
- ep = sctp_sk(sk)->ep;
+ ep = sctp_sk(sk)->ep;
list_for_each(pos, &ep->asocs) {
asoc = list_entry(pos, sctp_association_t, asocs);
-
sctp_addip_addr_config(asoc, SCTP_PARAM_DEL_IP,
addrs, addrcnt);
-
- } /* for (each association on the endpoint) */
- }
+ }
+ }
#endif
-
- return retval;
-
-} /* sctp_bindx_rem() */
-
+ return retval;
+}
/* Helper for tunneling sys_bindx() requests through sctp_setsockopt()
*
*
* Returns 0 if ok, <0 errno code on error.
*/
-static int
-sctp_setsockopt_bindx(struct sock* sk, struct sockaddr_storage *addrs,
- int addrssize, int op)
+static int sctp_setsockopt_bindx(struct sock* sk, struct sockaddr_storage *addrs,
+ int addrssize, int op)
{
struct sockaddr_storage *kaddrs;
int err;
size_t addrcnt;
-
+
SCTP_DEBUG_PRINTK("sctp_do_setsocktopt_bindx: sk %p addrs %p"
- " addrssize %d opt %d\n", sk, addrs,
- addrssize, op);
+ " addrssize %d opt %d\n", sk, addrs,
+ addrssize, op);
- /* Do we have an integer number of structs sockaddr_storage? */
- if (unlikely(addrssize <= 0
- || addrssize % sizeof(struct sockaddr_storage) != 0)) {
+ /* Do we have an integer number of structs sockaddr_storage? */
+ if (unlikely(addrssize <= 0 ||
+ addrssize % sizeof(struct sockaddr_storage) != 0))
return -EINVAL;
- }
- /* Check the user passed a healthy pointer. */
- if (unlikely(!access_ok(VERIFY_READ, addrs, addrssize))) {
+ /* Check the user passed a healthy pointer. */
+ if (unlikely(!access_ok(VERIFY_READ, addrs, addrssize)))
return -EFAULT;
- }
- /* Alloc space for the address array in kernel memory. */
+ /* Alloc space for the address array in kernel memory. */
kaddrs = (struct sockaddr_storage *) kmalloc(addrssize, GFP_KERNEL);
- if (unlikely(NULL == kaddrs)) {
+ if (unlikely(NULL == kaddrs))
return -ENOMEM;
+
+ if (copy_from_user(kaddrs, addrs, addrssize)) {
+ kfree(kaddrs);
+ return -EFAULT;
}
-
- __copy_from_user(kaddrs, addrs, addrssize);
+
addrcnt = addrssize / sizeof(struct sockaddr_storage);
err = __sctp_bindx(sk, kaddrs, addrcnt, op); /* Do the work. */
kfree(kaddrs);
- return err;
-
-} /* sctp_setsockopt_bindx() */
+ return err;
+}
/* API 3.1.4 close() - UDP Style Syntax
* Applications use close() to perform graceful shutdown (as described in
* Section 10.1 of [SCTP]) on ALL the associations currently represented
- * by a UDP-style socket.
- *
+ * by a UDP-style socket.
+ *
* The syntax is
- *
+ *
* ret = close(int sd);
- *
+ *
* sd - the socket descriptor of the associations to be closed.
- *
+ *
* To gracefully shutdown a specific association represented by the
* UDP-style socket, an application should use the sendmsg() call,
* passing no user data, but including the appropriate flag in the
* ancillary data (see Section xxxx).
- *
+ *
* If sd in the close() call is a branched-off socket representing only
* one association, the shutdown is performed on that association only.
*/
-
-void
-sctp_close(struct sock *sk, long timeout)
+void sctp_close(struct sock *sk, long timeout)
{
- sctp_endpoint_t *ep;
- sctp_association_t *asoc;
+ sctp_endpoint_t *ep;
+ sctp_association_t *asoc;
list_t *pos, *temp;
-
- SCTP_DEBUG_PRINTK("sctp_close(sk: 0x%x...)\n",
- (unsigned int)sk);
+ SCTP_DEBUG_PRINTK("sctp_close(sk: 0x%p...)\n", sk);
sctp_lock_sock(sk);
sk->shutdown = SHUTDOWN_MASK;
ep = sctp_sk(sk)->ep;
- /* Walk all associations on a socket, not on an endpoint. */
+ /* Walk all associations on a socket, not on an endpoint. */
list_for_each_safe(pos, temp, &ep->asocs) {
asoc = list_entry(pos, sctp_association_t, asocs);
sctp_primitive_SHUTDOWN(asoc, NULL);
- }
+ }
-
- /* Clean up any skbs sitting on the receive queue.
- */
+ /* Clean up any skbs sitting on the receive queue. */
skb_queue_purge(&sk->receive_queue);
-
- /* This will run the backlog queue. */
+
+ /* This will run the backlog queue. */
sctp_release_sock(sk);
/* Supposedly, no process has access to the socket, but
* the net layers still may.
*/
-
sctp_local_bh_disable();
sctp_bh_lock_sock(sk);
/* Hold the sock, since inet_sock_release() will put sock_put()
* and we have just a little more cleanup.
*/
- sock_hold(sk);
+ sock_hold(sk);
inet_sock_release(sk);
-
+
sctp_bh_unlock_sock(sk);
sctp_local_bh_enable();
sock_put(sk);
-
- SCTP_DBG_OBJCNT_DEC(sock);
-} /* sctp_close() */
+ SCTP_DBG_OBJCNT_DEC(sock);
+}
/* API 3.1.3 sendmsg() - UDP Style Syntax
*
* An application uses sendmsg() and recvmsg() calls to transmit data to
- * and receive data from its peer.
+ * and receive data from its peer.
*
* ssize_t sendmsg(int socket, const struct msghdr *message,
* int flags);
* user message and possibly some ancillary data.
*
* See Section 5 for complete description of the data
- * structures.
+ * structures.
*
* flags - flags sent or received with the user message, see Section
- * 5 for complete description of the flags.
+ * 5 for complete description of the flags.
*
* NB: The argument 'msg' is a user space address.
*/
/* BUG: We do not implement timeouts. */
/* BUG: We do not implement the equivalent of wait_for_tcp_memory(). */
-int
-sctp_sendmsg(struct sock *sk, struct msghdr *msg, int size)
+int sctp_sendmsg(struct sock *sk, struct msghdr *msg, int size)
{
sctp_opt_t *sp;
- sctp_endpoint_t *ep;
- sctp_association_t *asoc = NULL;
+ sctp_endpoint_t *ep;
+ sctp_association_t *asoc = NULL;
sctp_transport_t *transport;
- sctp_chunk_t *chunk = NULL;
- sockaddr_storage_t to;
+ sctp_chunk_t *chunk = NULL;
+ sockaddr_storage_t to;
struct sockaddr *msg_name = NULL;
- struct sctp_sndrcvinfo default_sinfo = {0};
+ struct sctp_sndrcvinfo default_sinfo = { 0 };
struct sctp_sndrcvinfo *sinfo;
struct sctp_initmsg *sinit;
sctp_assoc_t associd = NULL;
- sctp_cmsgs_t cmsgs = {0};
- int err;
- size_t msg_len;
+ sctp_cmsgs_t cmsgs = { 0 };
+ int err;
+ size_t msg_len;
sctp_scope_t scope;
long timeo;
- uint16_t sinfo_flags = 0;
+ __u16 sinfo_flags = 0;
- SCTP_DEBUG_PRINTK("sctp_sendmsg(sk: %p, msg: %p, "
+ SCTP_DEBUG_PRINTK("sctp_sendmsg(sk: %p, msg: %p, "
"size: %d)\n", sk, msg, size);
- err = 0;
+ err = 0;
sp = sctp_sk(sk);
ep = sp->ep;
SCTP_DEBUG_PRINTK("Using endpoint: %s.\n", ep->debug_name);
- /* Parse out the SCTP CMSGs. */
+ /* Parse out the SCTP CMSGs. */
err = sctp_msghdr_parse(msg, &cmsgs);
if (err) {
goto out_nounlock;
}
- /* Fetch the destination address for this packet. This
- * address only selects the association--it is not necessarily
- * the address we will send to.
- * For a peeled-off socket, msg_name is ignored.
- */
+ /* Fetch the destination address for this packet. This
+ * address only selects the association--it is not necessarily
+ * the address we will send to.
+ * For a peeled-off socket, msg_name is ignored.
+ */
if ((SCTP_SOCKET_UDP_HIGH_BANDWIDTH != sp->type) && msg->msg_name) {
err = sctp_sendmsg_verify_name(sk, msg);
- if (err) { return err; }
-
- memcpy(&to, msg->msg_name, msg->msg_namelen);
+ if (err)
+ return err;
+
+ memcpy(&to, msg->msg_name, msg->msg_namelen);
SCTP_DEBUG_PRINTK("Just memcpy'd. msg_name is "
"0x%x:%u.\n",
to.v4.sin_addr.s_addr, to.v4.sin_port);
-
+
to.v4.sin_port = ntohs(to.v4.sin_port);
msg_name = msg->msg_name;
- }
-
+ }
+
msg_len = get_user_iov_size(msg->msg_iov, msg->msg_iovlen);
sinfo = cmsgs.info;
sinit = cmsgs.init;
- /* Did the user specify SNDRCVINFO? */
+ /* Did the user specify SNDRCVINFO? */
if (sinfo) {
sinfo_flags = sinfo->sinfo_flags;
associd = sinfo->sinfo_assoc_id;
}
- SCTP_DEBUG_PRINTK("msg_len: %d, sinfo_flags: 0x%x\n",
+ SCTP_DEBUG_PRINTK("msg_len: %d, sinfo_flags: 0x%x\n",
msg_len, sinfo_flags);
- /* If MSG_EOF|MSG_ABORT is set, no data can be sent. Disallow
+ /* If MSG_EOF|MSG_ABORT is set, no data can be sent. Disallow
* sending 0-length messages when MSG_EOF|MSG_ABORT is not set.
*/
- if (((sinfo_flags & (MSG_EOF|MSG_ABORT)) && (msg_len > 0))
- || (!(sinfo_flags & (MSG_EOF|MSG_ABORT)) && (msg_len == 0))) {
- err = -EINVAL;
+ if (((sinfo_flags & (MSG_EOF|MSG_ABORT)) && (msg_len > 0)) ||
+ (!(sinfo_flags & (MSG_EOF|MSG_ABORT)) && (msg_len == 0))) {
+ err = -EINVAL;
goto out_nounlock;
- }
+ }
sctp_lock_sock(sk);
transport = NULL;
SCTP_DEBUG_PRINTK("About to look up association.\n");
- /* If a msg_name has been specified, assume this is to be used. */
+
+ /* If a msg_name has been specified, assume this is to be used. */
if (msg_name) {
asoc = sctp_endpoint_lookup_assoc(ep, &to, &transport);
} else {
if (SCTP_SOCKET_UDP_HIGH_BANDWIDTH == sp->type) {
if (list_empty(&ep->asocs)) {
err = -EINVAL;
- goto out_unlock;
+ goto out_unlock;
}
- asoc = list_entry(ep->asocs.next, sctp_association_t,
+ asoc = list_entry(ep->asocs.next, sctp_association_t,
asocs);
} else if (associd) {
- asoc = sctp_id2assoc(sk, associd);
+ asoc = sctp_id2assoc(sk, associd);
}
if (!asoc) {
err = -EINVAL;
goto out_unlock;
- }
- }
-
-
+ }
+ }
+
if (asoc) {
SCTP_DEBUG_PRINTK("Just looked up association: "
"%s. \n", asoc->debug_name);
SCTP_DEBUG_PRINTK("Shutting down association: %p\n",
asoc);
sctp_primitive_SHUTDOWN(asoc, NULL);
- err = 0;
+ err = 0;
goto out_unlock;
}
if (sinfo_flags & MSG_ABORT) {
goto out_unlock;
}
}
-
- /* Do we need to create the association? */
- if (NULL == asoc) {
-
+ /* Do we need to create the association? */
+ if (!asoc) {
SCTP_DEBUG_PRINTK("There is no association yet.\n");
/* Check for invalid stream against the stream counts,
- * either the default or the user specified stream counts.
+ * either the default or the user specified stream counts.
*/
if (sinfo) {
-
- if ((NULL == sinit) ||
- (sinit && (0 == sinit->sinit_num_ostreams))) {
+ if (!sinit ||
+ (sinit && !sinit->sinit_num_ostreams)) {
/* Check against the defaults. */
- if (sinfo->sinfo_stream >=
+ if (sinfo->sinfo_stream >=
sp->initmsg.sinit_num_ostreams) {
err = -EINVAL;
goto out_unlock;
}
-
} else {
- /* Check against the defaults. */
- if (sinfo->sinfo_stream >=
- sp->initmsg.sinit_num_ostreams) {
+ /* Check against the defaults. */
+ if (sinfo->sinfo_stream >=
+ sp->initmsg.sinit_num_ostreams) {
err = -EINVAL;
goto out_unlock;
}
- /* Check against the requested. */
- if (sinfo->sinfo_stream >=
+
+ /* Check against the requested. */
+ if (sinfo->sinfo_stream >=
sinit->sinit_num_ostreams) {
err = -EINVAL;
goto out_unlock;
- }
-
+ }
}
}
/*
- * API 3.1.2 bind() - UDP Style Syntax
- * If a bind() or sctp_bindx() is not called prior to a
- * sendmsg() call that initiates a new association, the
- * system picks an ephemeral port and will choose an address
- * set equivalent to binding with a wildcard address.
+ * API 3.1.2 bind() - UDP Style Syntax
+ * If a bind() or sctp_bindx() is not called prior to a
+ * sendmsg() call that initiates a new association, the
+ * system picks an ephemeral port and will choose an address
+ * set equivalent to binding with a wildcard address.
*/
- if (0 == ep->base.bind_addr.port) {
+ if (!ep->base.bind_addr.port) {
if (sctp_autobind(sk)) {
err = -EAGAIN;
goto out_unlock;
}
scope = sctp_scope(&to);
- asoc = sctp_association_new(ep, sk, scope, GFP_KERNEL);
- if (NULL == asoc) {
- err = -ENOMEM;
- goto out_unlock;
- }
+ asoc = sctp_association_new(ep, sk, scope, GFP_KERNEL);
+ if (!asoc) {
+ err = -ENOMEM;
+ goto out_unlock;
+ }
/* If the SCTP_INIT ancillary data is specified, set all
- * the association init values accordingly.
+ * the association init values accordingly.
*/
- if (NULL != sinit) {
- if (0 != sinit->sinit_num_ostreams) {
- asoc->c.sinit_num_ostreams =
+ if (sinit) {
+ if (sinit->sinit_num_ostreams) {
+ asoc->c.sinit_num_ostreams =
sinit->sinit_num_ostreams;
}
- if (0 != sinit->sinit_max_instreams) {
- if (sinit->sinit_max_instreams <=
+ if (sinit->sinit_max_instreams) {
+ if (sinit->sinit_max_instreams <=
SCTP_MAX_STREAM) {
- asoc->c.sinit_max_instreams =
- sinit->sinit_max_instreams;
+ asoc->c.sinit_max_instreams =
+ sinit->sinit_max_instreams;
} else {
- asoc->c.sinit_max_instreams =
+ asoc->c.sinit_max_instreams =
SCTP_MAX_STREAM;
}
}
- if (0 != sinit->sinit_max_attempts) {
- asoc->max_init_attempts
+ if (sinit->sinit_max_attempts) {
+ asoc->max_init_attempts
= sinit->sinit_max_attempts;
}
- if (0 != sinit->sinit_max_init_timeo) {
+ if (sinit->sinit_max_init_timeo) {
asoc->max_init_timeo
= sinit->sinit_max_init_timeo * HZ;
-
}
}
- /* Prime the peer's transport structures. */
+ /* Prime the peer's transport structures. */
transport = sctp_assoc_add_peer(asoc, &to, GFP_KERNEL);
-
- } /* if (we need an association) */
+ }
- /* ASSERT: we have a valid association at this point. */
+ /* ASSERT: we have a valid association at this point. */
SCTP_DEBUG_PRINTK("We have a valid association. \n");
- /* API 7.1.7, the sndbuf size per association bounds the
+ /* API 7.1.7, the sndbuf size per association bounds the
* maximum size of data that can be sent in a single send call.
*/
if (msg_len > sk->sndbuf) {
}
/* FIXME: In the current implementation, a single chunk is created
- * for the entire message initially, even if it has to be fragmented
- * later. As the length field in the chunkhdr is used to set
- * the chunk length, the maximum size of the chunk and hence the
- * message is limited by its type(uint16_t).
+ * for the entire message initially, even if it has to be fragmented
+ * later. As the length field in the chunkhdr is used to set
+ * the chunk length, the maximum size of the chunk and hence the
+ * message is limited by its type(__u16).
* The real fix is to fragment the message before creating the chunks.
*/
- if (msg_len > ((uint16_t)(~(uint16_t)0) -
- WORD_ROUND(sizeof(sctp_data_chunk_t)+1))) {
+ if (msg_len > ((__u16)(~(__u16)0) -
+ WORD_ROUND(sizeof(sctp_data_chunk_t)+1))) {
err = -EMSGSIZE;
goto out_free;
}
/* If fragmentation is disabled and the message length exceeds the
* association fragmentation point, return EMSGSIZE. The I-D
- * does not specify what this error is, but this looks like
- * a great fit.
- */
+ * does not specify what this error is, but this looks like
+ * a great fit.
+ */
if (sctp_sk(sk)->disable_fragments && (msg_len > asoc->frag_point)) {
err = -EMSGSIZE;
goto out_free;
- }
-
+ }
+
if (sinfo) {
/* Check for invalid stream. */
if (sinfo->sinfo_stream >= asoc->c.sinit_num_ostreams) {
- err = -EINVAL;
- goto out_free;
+ err = -EINVAL;
+ goto out_free;
}
} else {
- /* If the user didn't specify SNDRCVINFO, make up one with
+ /* If the user didn't specify SNDRCVINFO, make up one with
* some defaults.
*/
default_sinfo.sinfo_stream = asoc->defaults.stream;
}
timeo = sock_sndtimeo(sk, msg->msg_flags & MSG_DONTWAIT);
-
if (!sctp_wspace(asoc)) {
err = sctp_wait_for_sndbuf(asoc, &timeo, msg_len);
- if (err) {
+ if (err)
goto out_free;
- }
}
- /* Get enough memory for the whole message. */
- chunk = sctp_make_data_empty(asoc, sinfo, msg_len);
-
+ /* Get enough memory for the whole message. */
+ chunk = sctp_make_data_empty(asoc, sinfo, msg_len);
if (!chunk) {
err = -ENOMEM;
goto out_free;
}
-#if 0
- /* FIXME: This looks wrong so I'll comment out.
- * We should be able to use this same technique for
+#if 0
+ /* FIXME: This looks wrong so I'll comment out.
+ * We should be able to use this same technique for
* primary address override! --jgrimm
- */
-
+ */
/* If the user gave us an address, copy it in. */
- if (NULL != msg->msg_name) {
+ if (msg->msg_name) {
chunk->transport = sctp_assoc_lookup_paddr(asoc, &to);
if (!chunk->transport) {
err = -EINVAL;
}
#endif /* 0 */
- /* Copy the message from the user. */
- err = sctp_user_addto_chunk(chunk, msg_len, msg->msg_iov);
-
- if (err < 0) {
+ /* Copy the message from the user. */
+ err = sctp_user_addto_chunk(chunk, msg_len, msg->msg_iov);
+ if (err < 0)
goto out_free;
- }
SCTP_DEBUG_PRINTK("Copied message to chunk: %p.\n", chunk);
-
- /* Put the chunk->skb back into the form expected by send. */
- __skb_pull(chunk->skb, (uint8_t *)chunk->chunk_hdr
- - (uint8_t *)chunk->skb->data);
+
+ /* Put the chunk->skb back into the form expected by send. */
+ __skb_pull(chunk->skb, (__u8 *)chunk->chunk_hdr
+ - (__u8 *)chunk->skb->data);
/* Do accounting for the write space. */
sctp_set_owner_w(chunk);
- if (SCTP_STATE_CLOSED == asoc->state) {
- err = sctp_primitive_ASSOCIATE(asoc, NULL);
- if (err < 0) {
+ if (SCTP_STATE_CLOSED == asoc->state) {
+ err = sctp_primitive_ASSOCIATE(asoc, NULL);
+ if (err < 0)
goto out_free;
- }
SCTP_DEBUG_PRINTK("We associated primitively.\n");
- }
-
+ }
- /* Send it to the lower layers. */
- err = sctp_primitive_SEND(asoc, chunk);
+ /* Send it to the lower layers. */
+ err = sctp_primitive_SEND(asoc, chunk);
SCTP_DEBUG_PRINTK("We sent primitively.\n");
/* BUG: SCTP_CHECK_TIMER(sk); */
- if (0 == err) {
- err = msg_len;
+ if (!err) {
+ err = msg_len;
goto out_unlock;
- }
-
+ }
+
out_free:
- if (SCTP_STATE_CLOSED == asoc->state) {
+ if (SCTP_STATE_CLOSED == asoc->state)
sctp_association_free(asoc);
- }
- if (chunk) {
+ if (chunk)
sctp_free_chunk(chunk);
- }
+
out_unlock:
- sctp_release_sock(sk);
+ sctp_release_sock(sk);
+
out_nounlock:
return err;
-
+
#if 0
do_sock_err:
- if (msg_len) {
+ if (msg_len)
err = msg_len;
- } else {
+ else
err = sock_error(sk);
- }
goto out;
+
do_interrupted:
- if (msg_len) {
+ if (msg_len)
err = msg_len;
- }
goto out;
#endif /* 0 */
-
-} /* sctp_sendmsg() */
+}
/* API 3.1.3 recvmsg() - UDP Style Syntax
*
* user message and possibly some ancillary data.
*
* See Section 5 for complete description of the data
- * structures.
+ * structures.
*
* flags - flags sent or received with the user message, see Section
- * 5 for complete description of the flags.
- *
+ * 5 for complete description of the flags.
*/
-
-int
-sctp_recvmsg(struct sock *sk, struct msghdr *msg, int len, int noblock,
- int flags, int *addr_len)
+int sctp_recvmsg(struct sock *sk, struct msghdr *msg, int len, int noblock,
+ int flags, int *addr_len)
{
sctp_ulpevent_t *event = NULL;
struct sk_buff *skb;
int copied;
- int err = 0;
+ int err = 0;
SCTP_DEBUG_PRINTK("sctp_recvmsg("
"%s: %p, %s: %p, %s: %d, %s: %d, %s: "
"0x%x, %s: %p)\n",
- "sk", sk,
+ "sk", sk,
"msghdr", msg,
"len", len,
"knoblauch", noblock,
sctp_lock_sock(sk);
skb = sctp_skb_recv_datagram(sk, flags, noblock, &err);
- if (!skb){
+ if (!skb)
goto out;
- }
-
+
copied = skb->len;
if (skb_shinfo(skb)->frag_list) {
struct sk_buff *list;
- for (list = skb_shinfo(skb)->frag_list; list;
- list=list->next){
+ for (list = skb_shinfo(skb)->frag_list;
+ list;
+ list = list->next)
copied += list->len;
- }
}
if (copied > len) {
copied = len;
msg->msg_flags |= MSG_TRUNC;
- }
+ }
err = skb_copy_datagram_iovec(skb, 0, msg->msg_iov, copied);
- event = (sctp_ulpevent_t *)skb->cb;
+ event = (sctp_ulpevent_t *) skb->cb;
+
+ if (err)
+ goto out_free;
- if (err) { goto out_free; }
-
sock_recv_timestamp(msg, sk, skb);
-
if (sctp_ulpevent_is_notification(event)) {
msg->msg_flags |= MSG_NOTIFICATION;
} else {
- /* Copy the address. */
- if (addr_len && msg->msg_name) {
- sctp_sk_memcpy_msgname(sk, msg->msg_name,
+ /* Copy the address. */
+ if (addr_len && msg->msg_name)
+ sctp_sk_memcpy_msgname(sk, msg->msg_name,
addr_len, skb);
- }
-
}
-
+
/* Check if we allow SCTP_SNDRCVINFO. */
- if (sctp_sk(sk)->subscribe.sctp_data_io_event) {
+ if (sctp_sk(sk)->subscribe.sctp_data_io_event)
sctp_ulpevent_read_sndrcvinfo(event, msg);
- }
-
+
#if 0
- /* FIXME: we should be calling IP layer too. */
+ /* FIXME: we should be calling IP layer too. */
if (sk->protinfo.af_inet.cmsg_flags)
ip_cmsg_recv(msg, skb);
-#endif
-
+#endif
+
err = copied;
/* FIXME: We need to support MSG_EOR correctly. */
out:
sctp_release_sock(sk);
return err;
+}
-} /* sctp_recvmsg() */
-
-static inline int
-sctp_setsockopt_disable_fragments(struct sock *sk, char *optval, int optlen)
+static inline int sctp_setsockopt_disable_fragments(struct sock *sk, char *optval, int optlen)
{
int val;
- if (optlen < sizeof(int)) {
+ if (optlen < sizeof(int))
return -EINVAL;
- }
- if (get_user(val, (int *)optval)) {
+ if (get_user(val, (int *)optval))
return -EFAULT;
- }
sctp_sk(sk)->disable_fragments = (val == 0) ? 0 : 1;
return 0;
+}
-} /* sctp_setsockopt_disable_fragments() */
-
-static inline int
-sctp_setsockopt_set_events(struct sock *sk, char *optval, int optlen)
+static inline int sctp_setsockopt_set_events(struct sock *sk, char *optval, int optlen)
{
- if (optlen != sizeof(struct sctp_event_subscribe)) {
+ if (optlen != sizeof(struct sctp_event_subscribe))
return -EINVAL;
- }
-
- if (copy_from_user(&sctp_sk(sk)->subscribe, optval, optlen)) {
+ if (copy_from_user(&sctp_sk(sk)->subscribe, optval, optlen))
return -EFAULT;
- }
-
return 0;
+}
-} /* sctp_setsockopt_set_events() */
-
-static inline int
-sctp_setsockopt_autoclose(struct sock *sk, char *optval, int optlen)
+static inline int sctp_setsockopt_autoclose(struct sock *sk, char *optval, int optlen)
{
sctp_opt_t *sp = sctp_sk(sk);
- if (optlen != sizeof(int)) {
+ if (optlen != sizeof(int))
return -EINVAL;
- }
-
- if (copy_from_user(&sp->autoclose, optval, optlen)) {
+ if (copy_from_user(&sp->autoclose, optval, optlen))
return -EFAULT;
- }
-
- sp->ep->timeouts[SCTP_EVENT_TIMEOUT_AUTOCLOSE] = sp->autoclose * HZ;
+ sp->ep->timeouts[SCTP_EVENT_TIMEOUT_AUTOCLOSE] = sp->autoclose * HZ;
return 0;
-
-} /* sctp_setsockopt_autoclose() */
+}
/* API 6.2 setsockopt(), getsockopt()
- *
+ *
* Applications use setsockopt() and getsockopt() to set or retrieve
* socket options. Socket options are used to change the default
* behavior of sockets calls. They are described in Section 7.
- *
+ *
* The syntax is:
- *
+ *
* ret = getsockopt(int sd, int level, int optname, void *optval,
- * int *optlen);
+ * int *optlen);
* ret = setsockopt(int sd, int level, int optname, const void *optval,
* int optlen);
- *
+ *
* sd - the socket descript.
* level - set to IPPROTO_SCTP for all SCTP options.
* optname - the option name.
* optval - the buffer to store the value of the option.
* optlen - the size of the buffer.
- *
*/
-int
-sctp_setsockopt(struct sock *sk, int level, int optname, char *optval,
- int optlen)
+int sctp_setsockopt(struct sock *sk, int level, int optname, char *optval,
+ int optlen)
{
- int retval = 0;
- char * tmp;
- sctp_protocol_t *proto = sctp_get_protocol();
+ int retval = 0;
+ char * tmp;
+ sctp_protocol_t *proto = sctp_get_protocol();
list_t *pos;
sctp_func_t *af;
-
- SCTP_DEBUG_PRINTK("sctp_setsockopt(sk: %p... optname: %d)\n",
+ SCTP_DEBUG_PRINTK("sctp_setsockopt(sk: %p... optname: %d)\n",
sk, optname);
- /* I can hardly begin to describe how wrong this is. This is
- * so broken as to be worse than useless. The API draft
- * REALLY is NOT helpful here... I am not convinced that the
- * semantics of setsockopt() with a level OTHER THAN SOL_SCTP
- * are at all well-founded.
- */
+ /* I can hardly begin to describe how wrong this is. This is
+ * so broken as to be worse than useless. The API draft
+ * REALLY is NOT helpful here... I am not convinced that the
+ * semantics of setsockopt() with a level OTHER THAN SOL_SCTP
+ * are at all well-founded.
+ */
if (level != SOL_SCTP) {
list_for_each(pos, &proto->address_families) {
af = list_entry(pos, sctp_func_t, list);
- retval = af->setsockopt(sk, level, optname, optval,
+ retval = af->setsockopt(sk, level, optname, optval,
optlen);
- if (retval < 0) { goto out_nounlock; }
- }
- }
-
- sctp_lock_sock(sk);
+ if (retval < 0)
+ goto out_nounlock;
+ }
+ }
- switch (optname) {
- case SCTP_SOCKOPT_DEBUG_NAME:
+ sctp_lock_sock(sk);
+ switch (optname) {
+ case SCTP_SOCKOPT_DEBUG_NAME:
/* BUG! we don't ever seem to free this memory. --jgrimm */
- if (NULL == (tmp = kmalloc(optlen + 1, GFP_KERNEL))) {
- retval = -ENOMEM;
- goto out_unlock;
- }
-
- if (copy_from_user(tmp, optval, optlen)) {
- retval = -EFAULT;
- goto out_unlock;
- }
- tmp[optlen] = '\000';
- sctp_sk(sk)->ep->debug_name = tmp;
- break;
- case SCTP_SOCKOPT_BINDX_ADD:
- /* 'optlen' is the size of the addresses buffer. */
- retval = sctp_setsockopt_bindx(sk, (struct sockaddr_storage *)
+ if (NULL == (tmp = kmalloc(optlen + 1, GFP_KERNEL))) {
+ retval = -ENOMEM;
+ goto out_unlock;
+ }
+
+ if (copy_from_user(tmp, optval, optlen)) {
+ retval = -EFAULT;
+ goto out_unlock;
+ }
+ tmp[optlen] = '\000';
+ sctp_sk(sk)->ep->debug_name = tmp;
+ break;
+
+ case SCTP_SOCKOPT_BINDX_ADD:
+ /* 'optlen' is the size of the addresses buffer. */
+ retval = sctp_setsockopt_bindx(sk, (struct sockaddr_storage *)
optval, optlen, BINDX_ADD_ADDR);
- break;
- case SCTP_SOCKOPT_BINDX_REM:
- /* 'optlen' is the size of the addresses buffer. */
- retval = sctp_setsockopt_bindx(sk, (struct sockaddr_storage *)
+ break;
+
+ case SCTP_SOCKOPT_BINDX_REM:
+ /* 'optlen' is the size of the addresses buffer. */
+ retval = sctp_setsockopt_bindx(sk, (struct sockaddr_storage *)
optval, optlen, BINDX_REM_ADDR);
- break;
+ break;
+
case SCTP_DISABLE_FRAGMENTS:
retval = sctp_setsockopt_disable_fragments(sk, optval, optlen);
break;
+
case SCTP_SET_EVENTS:
retval = sctp_setsockopt_set_events(sk, optval, optlen);
break;
+
case SCTP_AUTOCLOSE:
retval = sctp_setsockopt_autoclose(sk, optval, optlen);
break;
+
default:
retval = -ENOPROTOOPT;
break;
- } /* switch(optname) */
-
- out_unlock:
- sctp_release_sock(sk);
- out_nounlock:
- return retval;
+ };
-} /* sctp_setsockopt() */
+out_unlock:
+ sctp_release_sock(sk);
+out_nounlock:
+ return retval;
+}
/* FIXME: Write comments. */
-int
-sctp_connect(struct sock *sk, struct sockaddr *uaddr, int addr_len)
+int sctp_connect(struct sock *sk, struct sockaddr *uaddr, int addr_len)
{
- return -EOPNOTSUPP; /* STUB */
-
-} /* sctp_connect() */
+ return -EOPNOTSUPP; /* STUB */
+}
/* FIXME: Write comments. */
-int
-sctp_disconnect(struct sock *sk, int flags)
+int sctp_disconnect(struct sock *sk, int flags)
{
- return -EOPNOTSUPP; /* STUB */
-
-} /* sctp_disconnect() */
-
+ return -EOPNOTSUPP; /* STUB */
+}
/* FIXME: Write comments. */
-struct sock *
-sctp_accept(struct sock *sk, int flags, int *err)
+struct sock *sctp_accept(struct sock *sk, int flags, int *err)
{
- int error = -EOPNOTSUPP;
+ int error = -EOPNOTSUPP;
*err = error;
return NULL;
-
-} /* sctp_accept() */
+}
/* FIXME: Write Comments. */
-int
-sctp_ioctl(struct sock *sk, int cmd, unsigned long arg)
+int sctp_ioctl(struct sock *sk, int cmd, unsigned long arg)
{
- return -EOPNOTSUPP; /* STUB */
-
-} /* sctp_ioctl() */
+ return -EOPNOTSUPP; /* STUB */
+}
/* This is the function which gets called during socket creation to
- * initialized the SCTP-specific portion of the sock.
+ * initialized the SCTP-specific portion of the sock.
* The sock structure should already be zero-filled memory.
*/
-int
-sctp_init_sock(struct sock *sk)
+int sctp_init_sock(struct sock *sk)
{
sctp_endpoint_t *ep;
sctp_protocol_t *proto;
proto = sctp_get_protocol();
- /* Create a per socket endpoint structure. Even if we
+ /* Create a per socket endpoint structure. Even if we
* change the data structure relationships, this may still
* be useful for storing pre-connect address information.
*/
ep = sctp_endpoint_new(proto, sk, GFP_KERNEL);
-
- if (NULL == ep) { return -ENOMEM; }
-
- sp = sctp_sk(sk);
+ if (!ep)
+ return -ENOMEM;
+
+ sp = sctp_sk(sk);
- /* Initialize the SCTP per socket area. */
+ /* Initialize the SCTP per socket area. */
- sp->ep = ep;
+ sp->ep = ep;
sp->type = SCTP_SOCKET_UDP;
/* FIXME: The next draft (04) of the SCTP Sockets Extensions
- * should include a socket option for manipulating these
- * message parameters (and a few others).
+ * should include a socket option for manipulating these
+ * message parameters (and a few others).
*/
sp->default_stream = 0;
sp->default_ppid = 0;
*/
sp->initmsg.sinit_num_ostreams = proto->max_outstreams;
sp->initmsg.sinit_max_instreams = proto->max_instreams;
- sp->initmsg.sinit_max_attempts = proto->max_retrans_init;
+ sp->initmsg.sinit_max_attempts = proto->max_retrans_init;
sp->initmsg.sinit_max_init_timeo = proto->rto_max / HZ;
/* Initialize default RTO related parameters. These parameters can
* be modified for with the SCTP_RTOINFO socket option.
- * FIXME: This are not used yet.
+ * FIXME: This are not used yet.
*/
sp->rtoinfo.srto_initial = proto->rto_initial;
sp->rtoinfo.srto_max = proto->rto_max;
sp->rtoinfo.srto_min = proto->rto_min;
- /* Initialize default event subscriptions.
+ /* Initialize default event subscriptions.
* the struct sock is initialized to zero, so only
* enable the events needed. By default, UDP-style
* sockets enable io and association change notifications.
*/
- if (SCTP_SOCKET_UDP == sp->type) {
+ if (SCTP_SOCKET_UDP == sp->type) {
sp->subscribe.sctp_data_io_event = 1;
sp->subscribe.sctp_association_event = 1;
- }
+ }
/* Default Peer Address Parameters. These defaults can
* be modified via SCTP_SET_PEER_ADDR_PARAMS
- */
+ */
sp->paddrparam.spp_hbinterval = proto->hb_interval / HZ;
- sp->paddrparam.spp_pathmaxrxt = proto->max_retrans_path;
-
+ sp->paddrparam.spp_pathmaxrxt = proto->max_retrans_path;
/* If enabled no SCTP message fragmentation will be performed.
* Configure through SCTP_DISABLE_FRAGMENTS socket option.
*/
sp->disable_fragments = 0;
-
- /* Turn on/off any Nagle-like algorithm. */
+
+ /* Turn on/off any Nagle-like algorithm. */
sp->nodelay = 0;
- /* Auto-close idle associations after the configured
+ /* Auto-close idle associations after the configured
* number of seconds. A value of 0 disables this
- * feature. Configure through the SCTP_AUTOCLOSE socket option,
- * for UDP-style sockets only.
+ * feature. Configure through the SCTP_AUTOCLOSE socket option,
+ * for UDP-style sockets only.
*/
sp->autoclose = 0;
SCTP_DBG_OBJCNT_INC(sock);
- return 0;
-
-} /* sctp_init_sock() */
+ return 0;
+}
-/* Cleanup any SCTP per socket resources. */
-int
-sctp_destroy_sock(struct sock *sk)
+/* Cleanup any SCTP per socket resources. */
+int sctp_destroy_sock(struct sock *sk)
{
-
sctp_endpoint_t *ep;
- SCTP_DEBUG_PRINTK("sctp_destroy_sock(sk: %p)\n", sk);
+ SCTP_DEBUG_PRINTK("sctp_destroy_sock(sk: %p)\n", sk);
/* Release our hold on the endpoint. */
ep = sctp_sk(sk)->ep;
- sctp_endpoint_free(ep);
-
- return 0;
-
-} /* sctp_destroy_sock() */
+ sctp_endpoint_free(ep);
+ return 0;
+}
-/* FIXME: Comments needed. */
+/* FIXME: Comments needed. */
void sctp_shutdown(struct sock *sk, int how)
{
/* UDP-style sockets do not support shutdown. */
- /* STUB */
-} /* sctp_shutdown() */
+ /* STUB */
+}
-static int
-sctp_getsockopt_sctp_status(struct sock *sk, int len, char *optval,
- int *optlen)
+static int sctp_getsockopt_sctp_status(struct sock *sk, int len, char *optval,
+ int *optlen)
{
struct sctp_status status;
sctp_endpoint_t *ep;
if (len != sizeof(status)) {
retval = -EINVAL;
- goto out_nounlock;
+ goto out_nounlock;
}
if (copy_from_user(&status, optval, sizeof(status))) {
retval = -EFAULT;
- goto out_nounlock;
+ goto out_nounlock;
}
sctp_lock_sock(sk);
assoc = sctp_id2assoc(sk, associd);
if (!assoc) {
retval = -EINVAL;
- goto out_unlock;
+ goto out_unlock;
}
- } else {
+ } else {
ep = sctp_sk(sk)->ep;
if (list_empty(&ep->asocs)) {
retval = -EINVAL;
- goto out_unlock;
+ goto out_unlock;
}
-
+
assoc = list_entry(ep->asocs.next, sctp_association_t, asocs);
}
status.sstat_outstrms = assoc->c.sinit_num_ostreams;
status.sstat_fragmentation_point = assoc->frag_point;
status.sstat_primary.spinfo_assoc_id = sctp_assoc2id(transport->asoc);
- memcpy(&status.sstat_primary.spinfo_address,
- &(transport->ipaddr), sizeof(sockaddr_storage_t));
+ memcpy(&status.sstat_primary.spinfo_address,
+ &(transport->ipaddr), sizeof(sockaddr_storage_t));
status.sstat_primary.spinfo_state = transport->state.active;
status.sstat_primary.spinfo_cwnd = transport->cwnd;
status.sstat_primary.spinfo_srtt = transport->srtt;
goto out_unlock;
}
- SCTP_DEBUG_PRINTK("sctp_getsockopt_sctp_status(%d): %d %d %p\n",
- len, status.sstat_state, status.sstat_rwnd,
+ SCTP_DEBUG_PRINTK("sctp_getsockopt_sctp_status(%d): %d %d %p\n",
+ len, status.sstat_state, status.sstat_rwnd,
status.sstat_assoc_id);
if (copy_to_user(optval, &status, len)) {
}
out_unlock:
- sctp_release_sock(sk);
+ sctp_release_sock(sk);
+
out_nounlock:
return (retval);
+}
-} /* sctp_getsockopt_sctp_status() */
-
-static inline int
-sctp_getsockopt_disable_fragments(struct sock *sk, int len,
- char *optval, int *optlen)
+static inline int sctp_getsockopt_disable_fragments(struct sock *sk, int len,
+ char *optval, int *optlen)
{
int val;
- if (len < sizeof(int)) {
+ if (len < sizeof(int))
return -EINVAL;
- }
len = sizeof(int);
-
- val = (sctp_sk(sk)->disable_fragments == 1);
-
- if (put_user(len, optlen)) {
+ val = (sctp_sk(sk)->disable_fragments == 1);
+ if (put_user(len, optlen))
return -EFAULT;
- }
-
- if (copy_to_user(optval, &val, len)) {
+ if (copy_to_user(optval, &val, len))
return -EFAULT;
- }
-
- return (0);
-
-} /* sctp_getsockopt_disable_fragments() */
+ return 0;
+}
-static inline int
-sctp_getsockopt_set_events(struct sock *sk, int len, char *optval, int *optlen)
+static inline int sctp_getsockopt_set_events(struct sock *sk, int len, char *optval, int *optlen)
{
- if (len != sizeof(struct sctp_event_subscribe)) {
+ if (len != sizeof(struct sctp_event_subscribe))
return -EINVAL;
- }
-
- if (copy_to_user(optval, &sctp_sk(sk)->subscribe, len)) {
+ if (copy_to_user(optval, &sctp_sk(sk)->subscribe, len))
return -EFAULT;
- }
-
return 0;
+}
-} /* sctp_getsockopt_set_events() */
-
-static inline int
-sctp_getsockopt_autoclose(struct sock *sk, int len, char *optval, int *optlen)
+static inline int sctp_getsockopt_autoclose(struct sock *sk, int len, char *optval, int *optlen)
{
- if (len != sizeof(int)) {
+ if (len != sizeof(int))
return -EINVAL;
- }
-
- if (copy_to_user(optval, &sctp_sk(sk)->autoclose, len)) {
+ if (copy_to_user(optval, &sctp_sk(sk)->autoclose, len))
return -EFAULT;
- }
-
return 0;
-
-} /* sctp_getsockopt_autoclose() */
-
+}
/* Helper routine to branch off an association to a new socket. */
-int
-sctp_do_peeloff(sctp_association_t *assoc, struct socket **newsock)
+int sctp_do_peeloff(sctp_association_t *assoc, struct socket **newsock)
{
struct sock *oldsk = assoc->base.sk;
struct sock *newsk;
sctp_opt_t *newsp;
int err = 0;
- /* An association cannot be branched off from an already peeled-off
- * socket.
+ /* An association cannot be branched off from an already peeled-off
+ * socket.
*/
- if (SCTP_SOCKET_UDP_HIGH_BANDWIDTH == sctp_sk(oldsk)->type) {
+ if (SCTP_SOCKET_UDP_HIGH_BANDWIDTH == sctp_sk(oldsk)->type)
return -EINVAL;
- }
- /* Create a new socket. */
+ /* Create a new socket. */
err = sock_create(PF_INET, SOCK_SEQPACKET, IPPROTO_SCTP, &tmpsock);
- if (err < 0) {
- return (err);
- }
+ if (err < 0)
+ return err;
newsk = tmpsock->sk;
newsp = sctp_sk(newsk);
newep = newsp->ep;
- /* Migrate socket buffer sizes and all the socket level options to the
- * new socket.
- */
+ /* Migrate socket buffer sizes and all the socket level options to the
+ * new socket.
+ */
newsk->sndbuf = oldsk->sndbuf;
newsk->rcvbuf = oldsk->rcvbuf;
*newsp = *oldsp;
/* Restore the ep value that was overwritten with the above structure
- * copy.
+ * copy.
*/
newsp->ep = newep;
/* Set the type of socket to indicate that it is peeled off from the
- * original socket.
+ * original socket.
*/
newsp->type = SCTP_SOCKET_UDP_HIGH_BANDWIDTH;
*newsock = tmpsock;
- return (err);
-
-} /* sctp_do_peeloff() */
+ return err;
+}
-static inline int
-sctp_getsockopt_peeloff(struct sock *sk, int len, char *optval, int *optlen)
+static inline int sctp_getsockopt_peeloff(struct sock *sk, int len, char *optval, int *optlen)
{
sctp_peeloff_arg_t peeloff;
struct socket *newsock;
int err, sd;
sctp_association_t *assoc;
- if (len != sizeof(sctp_peeloff_arg_t)) {
+ if (len != sizeof(sctp_peeloff_arg_t))
return -EINVAL;
- }
-
- if (copy_from_user(&peeloff, optval, len)) {
+ if (copy_from_user(&peeloff, optval, len))
return -EFAULT;
- }
-
assoc = sctp_id2assoc(sk, peeloff.associd);
- if (NULL == assoc) {
+ if (NULL == assoc)
return -EINVAL;
- }
SCTP_DEBUG_PRINTK("%s: sk: %p assoc: %p\n", __FUNCTION__, sk, assoc);
err = sctp_do_peeloff(assoc, &newsock);
- if (err < 0) {
- return (err);
- }
+ if (err < 0)
+ return err;
- /* Map the socket to an unused fd that can be returned to the user. */
- sd = sock_map_fd(newsock);
+ /* Map the socket to an unused fd that can be returned to the user. */
+ sd = sock_map_fd(newsock);
if (sd < 0) {
sock_release(newsock);
- return (sd);
+ return sd;
}
- SCTP_DEBUG_PRINTK("%s: sk: %p assoc: %p newsk: %p sd: %d\n",
+ SCTP_DEBUG_PRINTK("%s: sk: %p assoc: %p newsk: %p sd: %d\n",
__FUNCTION__, sk, assoc, newsock->sk, sd);
- /* Return the fd mapped to the new socket. */
+ /* Return the fd mapped to the new socket. */
peeloff.sd = sd;
- if (copy_to_user(optval, &peeloff, len)) {
+ if (copy_to_user(optval, &peeloff, len))
return -EFAULT;
- }
return 0;
+}
-} /* sctp_getsockopt_peeloff() */
-
-int
-sctp_getsockopt(struct sock *sk, int level, int optname, char *optval,
- int *optlen)
+int sctp_getsockopt(struct sock *sk, int level, int optname, char *optval,
+ int *optlen)
{
- int retval = 0;
- sctp_protocol_t *proto = sctp_get_protocol();
- sctp_func_t *af;
+ int retval = 0;
+ sctp_protocol_t *proto = sctp_get_protocol();
+ sctp_func_t *af;
list_t *pos;
int len;
-
+
SCTP_DEBUG_PRINTK("sctp_getsockopt(sk: %p, ...)\n", sk);
- /* I can hardly begin to describe how wrong this is. This is
- * so broken as to be worse than useless. The API draft
- * REALLY is NOT helpful here... I am not convinced that the
- * semantics of getsockopt() with a level OTHER THAN SOL_SCTP
- * are at all well-founded.
- */
+ /* I can hardly begin to describe how wrong this is. This is
+ * so broken as to be worse than useless. The API draft
+ * REALLY is NOT helpful here... I am not convinced that the
+ * semantics of getsockopt() with a level OTHER THAN SOL_SCTP
+ * are at all well-founded.
+ */
if (level != SOL_SCTP) {
list_for_each(pos, &proto->address_families) {
af = list_entry(pos, sctp_func_t, list);
-
- retval = af->getsockopt(sk, level, optname,
+ retval = af->getsockopt(sk, level, optname,
optval, optlen);
- if (retval < 0) { return retval; }
- }
- }
+ if (retval < 0)
+ return retval;
+ }
+ }
- if (get_user(len, optlen)) { return -EFAULT;}
+ if (get_user(len, optlen))
+ return -EFAULT;
switch (optname) {
case SCTP_STATUS:
retval = sctp_getsockopt_sctp_status(sk, len, optval, optlen);
break;
+
case SCTP_DISABLE_FRAGMENTS:
retval = sctp_getsockopt_disable_fragments(sk, len, optval,
optlen);
break;
+
case SCTP_SET_EVENTS:
retval = sctp_getsockopt_set_events(sk, len, optval, optlen);
break;
+
case SCTP_AUTOCLOSE:
retval = sctp_getsockopt_autoclose(sk, len, optval, optlen);
break;
+
case SCTP_SOCKOPT_PEELOFF:
retval = sctp_getsockopt_peeloff(sk, len, optval, optlen);
break;
+
default:
retval = -ENOPROTOOPT;
break;
- }
+ };
- return retval;
-
-} /* sctp_getsockopt() */
+ return retval;
+}
void sctp_hash(struct sock *sk)
{
- /* STUB */
-} /* void sctp_hash(struct sock *sk) */
+ /* STUB */
+}
void sctp_unhash(struct sock *sk)
{
- /* STUB */
-} /* void sctp_unhash(struct sock *sk) */
+ /* STUB */
+}
/* Check if port is acceptable. Possibly find first available port.
*
* link to the socket (struct sock) that uses it, the port number and
* a fastreuse flag (FIXME: NPI ipg).
*/
-
-int sctp_get_port(struct sock *sk, unsigned short snum)
+long sctp_get_port(struct sock *sk, unsigned short snum)
{
sctp_bind_hashbucket_t *head; /* hash list */
sctp_bind_bucket_t *pp; /* hash list port iterator */
sctp_local_bh_disable();
if (snum == 0) {
-
/* Search for an available port.
*
* 'sctp->port_rover' was the last port assigned, so
* already in the hash table; if not, we use that; if
* it is, we try next.
*/
-
int low = sysctl_local_port_range[0];
int high = sysctl_local_port_range[1];
int remaining = (high - low) + 1;
int rover;
int index;
-
+
sctp_spin_lock(&sctp->port_alloc_lock);
rover = sctp->port_rover;
do {
/* Exhausted local port range during search? */
ret = 1;
- if (remaining <= 0) {
+ if (remaining <= 0)
goto fail;
- }
/* OK, here is the one we will use. HEAD (the port
* hash table list entry) is non-NULL and we hold it's
*/
snum = rover;
pp = NULL;
-
} else {
-
/* We are given an specific port number; we verify
* that it is not being used. If it is used, we will
* exahust the search in the hash list corresponding
head = &sctp->port_hashtable[sctp_phashfn(snum)];
sctp_spin_lock(&head->lock);
for (pp = head->chain; pp; pp = pp->next) {
- if (pp->port == snum) {
+ if (pp->port == snum)
break;
- }
}
}
if (pp != NULL && pp->sk != NULL) {
-
/* We had a port hash table hit - there is an
* available port (pp != NULL) and it is being
* used by other socket (pp->sk != NULL); that other
- * socket is going to be sk2.
+ * socket is going to be sk2.
*/
-
int sk_reuse = sk->reuse;
sockaddr_storage_t tmpaddr;
struct sock *sk2 = pp->sk;
SCTP_DEBUG_PRINTK("sctp_get_port() found a "
"possible match\n");
- if (pp->fastreuse != 0 && sk->reuse != 0) {
+ if (pp->fastreuse != 0 && sk->reuse != 0)
goto success;
- }
/* FIXME - multiple addresses need to be supported
* later.
- */
-
+ */
switch (sk->family) {
case PF_INET:
tmpaddr.v4.sin_family = AF_INET;
tmpaddr.v4.sin_port = snum;
tmpaddr.v4.sin_addr.s_addr = inet_sk(sk)->rcv_saddr;
break;
+
case PF_INET6:
SCTP_V6(tmpaddr.v6.sin6_family = AF_INET6;
tmpaddr.v6.sin6_port = snum;
- tmpaddr.v6.sin6_addr
- = inet6_sk(sk)->rcv_saddr;
+ tmpaddr.v6.sin6_addr =
+ inet6_sk(sk)->rcv_saddr;
)
break;
+
default:
break;
- }
+ };
/* Run through the list of sockets bound to the port
* (pp->port) [via the pointers bind_next and
* comparing each of the addresses with the address of
* the socket sk. If we find a match, then that means
* that this port/socket (sk) combination are already
- * in an endpoint.
+ * in an endpoint.
*/
for( ; sk2 != NULL; sk2 = sk2->bind_next) {
sctp_endpoint_t *ep2;
&ep2->base.bind_addr, &tmpaddr)) {
goto found;
}
-
- } /* if (neither socket slated for reuse) */
+ }
+ }
- } /* for (every socket in this port hash bucket) */
found:
-
- /* If we found a conflict, fail. */
+ /* If we found a conflict, fail. */
if (sk2 != NULL) {
- ret = (int)sk2;
+ ret = (long) sk2;
goto fail_unlock;
}
SCTP_DEBUG_PRINTK("sctp_get_port(): Found a match\n");
- } /* if (we had a port hash-table hit) */
+ }
- /* If there was a hash table miss, create a new port. */
+ /* If there was a hash table miss, create a new port. */
ret = 1;
- if (pp == NULL && (pp = sctp_bucket_create(head, snum)) == NULL) {
+ if (pp == NULL && (pp = sctp_bucket_create(head, snum)) == NULL)
goto fail_unlock;
- }
/* In either case (hit or miss), make sure fastreuse is 1 only
* if sk->reuse is too (that is, if the caller requested
- * SO_REUSEADDR on this socket -sk-).
+ * SO_REUSEADDR on this socket -sk-).
*/
if (pp->sk == NULL) {
pp->fastreuse = sk->reuse? 1 : 0;
/* We are set, so fill up all the data in the hash table
* entry, tie the socket list information with the rest of the
- * sockets FIXME: Blurry, NPI (ipg).
+ * sockets FIXME: Blurry, NPI (ipg).
*/
success:
inet_sk(sk)->num = snum;
if (sk->prev == NULL) {
- if ((sk->bind_next = pp->sk) != NULL) {
+ if ((sk->bind_next = pp->sk) != NULL)
pp->sk->bind_pprev = &sk->bind_next;
- }
pp->sk = sk;
sk->bind_pprev = &pp->sk;
sk->prev = (struct sock *) pp;
fail_unlock:
sctp_spin_unlock(&head->lock);
+
fail:
sctp_local_bh_enable();
SCTP_DEBUG_PRINTK("sctp_get_port() ends, ret=%d\n", ret);
return ret;
-
-} /* sctp_get_port() */
-
+}
/*
* 3.1.3 listen() - UDP Style Syntax
* By default, new associations are not accepted for UDP style sockets.
* An application uses listen() to mark a socket as being able to
* accept new associations.
- *
*/
int sctp_seqpacket_listen(struct sock *sk, int backlog)
{
sctp_opt_t *sp = sctp_sk(sk);
sctp_endpoint_t *ep = sp->ep;
- /* Only UDP style sockets that are not peeled off are allowed to
- * listen().
- */
- if (SCTP_SOCKET_UDP != sp->type) {
- return (-EINVAL);
- }
+ /* Only UDP style sockets that are not peeled off are allowed to
+ * listen().
+ */
+ if (SCTP_SOCKET_UDP != sp->type)
+ return -EINVAL;
/*
* If a bind() or sctp_bindx() is not called prior to a listen()
- * call that allows new associations to be accepted, the system
- * picks an ephemeral port and will choose an address set equivalent
- * to binding with a wildcard address.
+ * call that allows new associations to be accepted, the system
+ * picks an ephemeral port and will choose an address set equivalent
+ * to binding with a wildcard address.
*
- * This is not currently spelled out in the SCTP sockets
+ * This is not currently spelled out in the SCTP sockets
* extensions draft, but follows the practice as seen in TCP
- * sockets.
+ * sockets.
*/
- if (0 == ep->base.bind_addr.port) {
- if (sctp_autobind(sk)) {
- return (-EAGAIN);
- }
+ if (!ep->base.bind_addr.port) {
+ if (sctp_autobind(sk))
+ return -EAGAIN;
}
-
-
sk->state = SCTP_SS_LISTENING;
sctp_hash_endpoint(ep);
-
return 0;
+}
-} /* sctp_seqpacket_listen() */
-
-/*
- * Move a socket to LISTENING state.
+/*
+ * Move a socket to LISTENING state.
*/
int sctp_inet_listen(struct socket *sock, int backlog)
{
sctp_lock_sock(sk);
err = -EINVAL;
- if (sock->state != SS_UNCONNECTED) {
+ if (sock->state != SS_UNCONNECTED)
goto out;
- }
-
switch (sock->type) {
case SOCK_SEQPACKET:
err = sctp_seqpacket_listen(sk, backlog);
break;
+
case SOCK_STREAM:
/* FIXME for TCP-style sockets. */
err = -EOPNOTSUPP;
+
default:
goto out;
- }
+ };
out:
sctp_release_sock(sk);
return err;
+}
-} /* sctp_inet_listen() */
-
-/*
- * This function is done by modeling the current datagram_poll() and the
- * tcp_poll(). Note that, based on these implementations, we don't
- * lock the socket in this function, even though it seems that,
- * ideally, locking or some other mechanisms can be used to ensure
- * the integrity of the counters (sndbuf and wmem_queued) used
+/*
+ * This function is done by modeling the current datagram_poll() and the
+ * tcp_poll(). Note that, based on these implementations, we don't
+ * lock the socket in this function, even though it seems that,
+ * ideally, locking or some other mechanisms can be used to ensure
+ * the integrity of the counters (sndbuf and wmem_queued) used
* in this place. We assume that we don't need locks either until proven
- * otherwise.
- *
- * Another thing to note is that we include the Async I/O support
- * here, again, by modeling the current TCP/UDP code. We don't have
- * a good way to test with it yet.
+ * otherwise.
+ *
+ * Another thing to note is that we include the Async I/O support
+ * here, again, by modeling the current TCP/UDP code. We don't have
+ * a good way to test with it yet.
*/
-unsigned int
-sctp_poll(struct file *file, struct socket *sock, poll_table *wait)
+unsigned int sctp_poll(struct file *file, struct socket *sock, poll_table *wait)
{
struct sock *sk = sock->sk;
unsigned int mask;
poll_wait(file, sk->sleep, wait);
mask = 0;
- /* Is there any exceptional events? */
- if (sk->err || !skb_queue_empty(&sk->error_queue)) {
+ /* Is there any exceptional events? */
+ if (sk->err || !skb_queue_empty(&sk->error_queue))
mask |= POLLERR;
- }
- if (sk->shutdown == SHUTDOWN_MASK) {
+ if (sk->shutdown == SHUTDOWN_MASK)
mask |= POLLHUP;
- }
/* Is it readable? Reconsider this code with TCP-style support. */
- if (!skb_queue_empty(&sk->receive_queue)
- || (sk->shutdown & RCV_SHUTDOWN)) {
+ if (!skb_queue_empty(&sk->receive_queue) ||
+ (sk->shutdown & RCV_SHUTDOWN))
mask |= POLLIN | POLLRDNORM;
- }
/*
* FIXME: We need to set SCTP_SS_DISCONNECTING for TCP-style and
* peeled off sockets. Additionally, TCP-style needs to consider
- * other establishment conditions.
+ * other establishment conditions.
*/
if (SCTP_SOCKET_UDP != sctp_sk(sk)->type) {
/* The association is going away. */
- if (SCTP_SS_DISCONNECTING == sk->state) {
+ if (SCTP_SS_DISCONNECTING == sk->state)
mask |= POLLHUP;
- }
/* The association is either gone or not ready. */
- if (SCTP_SS_CLOSED == sk->state) {
+ if (SCTP_SS_CLOSED == sk->state)
return mask;
- }
}
- /* Is it writable? */
+ /* Is it writable? */
if (sctp_writeable(sk)) {
mask |= POLLOUT | POLLWRNORM;
} else {
set_bit(SOCK_ASYNC_NOSPACE, &sk->socket->flags);
- /*
- * Since the socket is not locked, the buffer
- * might be made available after the writeable check and
- * before the bit is set. This could cause a lost I/O
+ /*
+ * Since the socket is not locked, the buffer
+ * might be made available after the writeable check and
+ * before the bit is set. This could cause a lost I/O
* signal. tcp_poll() has a race breaker for this race
- * condition. Based on their implementation, we put
+ * condition. Based on their implementation, we put
* in the following code to cover it as well.
*/
- if (sctp_writeable(sk)) {
+ if (sctp_writeable(sk))
mask |= POLLOUT | POLLWRNORM;
- }
}
-
-
return mask;
}
* 2nd Level Abstractions
********************************************************************/
-
-static sctp_bind_bucket_t *
-sctp_bucket_create(sctp_bind_hashbucket_t *head, unsigned short snum)
+static sctp_bind_bucket_t *sctp_bucket_create(sctp_bind_hashbucket_t *head, unsigned short snum)
{
sctp_bind_bucket_t *pp;
SCTP_DEBUG_PRINTK( "sctp_bucket_create() begins, snum=%d\n", snum);
pp = kmalloc(sizeof(sctp_bind_bucket_t), GFP_ATOMIC);
- if (pp != NULL) {
+ if (pp) {
pp->port = snum;
pp->fastreuse = 0;
pp->sk = NULL;
pp->next->pprev = &pp->next;
head->chain= pp;
pp->pprev = &head->chain;
- }
+ }
SCTP_DEBUG_PRINTK("sctp_bucket_create() ends, pp=%p\n", pp);
return pp;
-
-} /* sctp_bucket_create() */
+}
/* FIXME: Commments! */
__inline__ void __sctp_put_port(struct sock *sk)
{
sctp_protocol_t *sctp_proto = sctp_get_protocol();
- sctp_bind_hashbucket_t *head =
+ sctp_bind_hashbucket_t *head =
&sctp_proto->port_hashtable[sctp_phashfn(inet_sk(sk)->num)];
sctp_bind_bucket_t *pp;
*(sk->bind_pprev) = sk->bind_next;
sk->prev = NULL;
inet_sk(sk)->num = 0;
- if (pp->sk == NULL) {
+ if (pp->sk) {
if (pp->next)
pp->next->pprev = pp->pprev;
*(pp->pprev) = pp->next;
kfree(pp);
}
sctp_spin_unlock(&head->lock);
+}
-} /* __sctp_put_port() */
-
-void
-sctp_put_port(struct sock *sk)
+void sctp_put_port(struct sock *sk)
{
sctp_local_bh_disable();
__sctp_put_port(sk);
sctp_local_bh_enable();
-
-} /* sctp_put_port() */
+}
/*
- * The system picks an ephemeral port and choose an address set equivalent
- * to binding with a wildcard address.
- * One of those addresses will be the primary address for the association.
+ * The system picks an ephemeral port and choose an address set equivalent
+ * to binding with a wildcard address.
+ * One of those addresses will be the primary address for the association.
* This automatically enables the multihoming capability of SCTP.
*/
-int
-sctp_autobind(struct sock *sk)
+int sctp_autobind(struct sock *sk)
{
sockaddr_storage_t autoaddr;
- int addr_len=0;
+ int addr_len = 0;
memset(&autoaddr, 0, sizeof(sockaddr_storage_t));
-
- switch(sk->family){
+
+ switch (sk->family) {
case PF_INET:
- autoaddr.v4.sin_family = AF_INET;
- autoaddr.v4.sin_addr.s_addr = INADDR_ANY;
- autoaddr.v4.sin_port = htons(inet_sk(sk)->num);
+ autoaddr.v4.sin_family = AF_INET;
+ autoaddr.v4.sin_addr.s_addr = INADDR_ANY;
+ autoaddr.v4.sin_port = htons(inet_sk(sk)->num);
addr_len = sizeof(struct sockaddr_in);
break;
+
case PF_INET6:
SCTP_V6(
- /* FIXME: Write me for v6! */
- BUG();
+ /* FIXME: Write me for v6! */
+ BUG();
autoaddr.v6.sin6_family = AF_INET6;
autoaddr.v6.sin6_port = htons(inet_sk(sk)->num);
addr_len = sizeof(struct sockaddr_in6);
);
break;
- default: /* This should not happen. */
+
+ default: /* This should not happen. */
break;
- } /* switch(family) */
+ };
return sctp_do_bind(sk, &autoaddr, addr_len);
-
-} /* sctp_autobind() */
-
+}
/* Parse out IPPROTO_SCTP CMSG headers. Perform only minimal validation.
- *
+ *
* From RFC 2292
* 4.2 The cmsghdr Structure *
*
- * When ancillary data is sent or received, any number of ancillary data
- * objects can be specified by the msg_control and msg_controllen members of
+ * When ancillary data is sent or received, any number of ancillary data
+ * objects can be specified by the msg_control and msg_controllen members of
* the msghdr structure, because each object is preceded by
- * a cmsghdr structure defining the object's length (the cmsg_len member).
- * Historically Berkeley-derived implementations have passed only one object
+ * a cmsghdr structure defining the object's length (the cmsg_len member).
+ * Historically Berkeley-derived implementations have passed only one object
* at a time, but this API allows multiple objects to be
- * passed in a single call to sendmsg() or recvmsg(). The following example
- * shows two ancillary data objects in a control buffer.
+ * passed in a single call to sendmsg() or recvmsg(). The following example
+ * shows two ancillary data objects in a control buffer.
*
* |<--------------------------- msg_controllen -------------------------->|
* | |
*
* msg_control
* points here
- *
*/
-int
-sctp_msghdr_parse(const struct msghdr *msg,
- sctp_cmsgs_t *cmsgs)
+int sctp_msghdr_parse(const struct msghdr *msg, sctp_cmsgs_t *cmsgs)
{
struct cmsghdr *cmsg;
- for (cmsg = CMSG_FIRSTHDR(msg);
- cmsg != NULL;
+ for (cmsg = CMSG_FIRSTHDR(msg);
+ cmsg != NULL;
cmsg = CMSG_NXTHDR((struct msghdr*)msg, cmsg)) {
-
- /* Check for minimum length. The SCM code has this check. */
+ /* Check for minimum length. The SCM code has this check. */
if (cmsg->cmsg_len < sizeof(struct cmsghdr) ||
(unsigned long)(((char*)cmsg - (char*)msg->msg_control)
+ cmsg->cmsg_len) > msg->msg_controllen) {
return -EINVAL;
}
- /* Should we parse this header or ignore? */
- if (cmsg->cmsg_level != IPPROTO_SCTP) {
+ /* Should we parse this header or ignore? */
+ if (cmsg->cmsg_level != IPPROTO_SCTP)
continue;
- }
- /* Strictly check lengths following example in SCM code. */
+ /* Strictly check lengths following example in SCM code. */
switch (cmsg->cmsg_type) {
case SCTP_INIT:
- /* SCTP Socket API Extension (draft 1)
+ /* SCTP Socket API Extension (draft 1)
* 5.2.1 SCTP Initiation Structure (SCTP_INIT)
*
- * This cmsghdr structure provides information for
- * initializing new SCTP associations with sendmsg().
- * The SCTP_INITMSG socket option uses this same data
+ * This cmsghdr structure provides information for
+ * initializing new SCTP associations with sendmsg().
+ * The SCTP_INITMSG socket option uses this same data
* structure. This structure is not used for
* recvmsg().
*
* ------------ ------------ ----------------------
* IPPROTO_SCTP SCTP_INIT struct sctp_initmsg
*/
-
- if (cmsg->cmsg_len
- != CMSG_LEN(sizeof(struct sctp_initmsg)))
+ if (cmsg->cmsg_len !=
+ CMSG_LEN(sizeof(struct sctp_initmsg)))
return -EINVAL;
cmsgs->init = (struct sctp_initmsg *)CMSG_DATA(cmsg);
-
break;
case SCTP_SNDRCV:
- /* SCTP Socket API Extension (draft 1)
+ /* SCTP Socket API Extension (draft 1)
* 5.2.2 SCTP Header Information Structure(SCTP_SNDRCV)
*
- * This cmsghdr structure specifies SCTP options for
- * sendmsg() and describes SCTP header information
+ * This cmsghdr structure specifies SCTP options for
+ * sendmsg() and describes SCTP header information
* about a received message through recvmsg().
*
* cmsg_level cmsg_type cmsg_data[]
* ------------ ------------ ----------------------
* IPPROTO_SCTP SCTP_SNDRCV struct sctp_sndrcvinfo
- */
- if (cmsg->cmsg_len
- != CMSG_LEN(sizeof(struct sctp_sndrcvinfo))) {
+ */
+ if (cmsg->cmsg_len !=
+ CMSG_LEN(sizeof(struct sctp_sndrcvinfo)))
return -EINVAL;
- }
- cmsgs->info = (struct sctp_sndrcvinfo*)CMSG_DATA(cmsg);
+ cmsgs->info = (struct sctp_sndrcvinfo *)CMSG_DATA(cmsg);
/* Minimally, validate the sinfo_flags. */
- if (cmsgs->info->sinfo_flags
- & ~(MSG_UNORDERED|MSG_ADDR_OVER
- |MSG_ABORT|MSG_EOF)){
-
+ if (cmsgs->info->sinfo_flags &
+ ~(MSG_UNORDERED | MSG_ADDR_OVER |
+ MSG_ABORT | MSG_EOF))
return -EINVAL;
- }
-
break;
+
default:
return -EINVAL;
-
- } /* switch(cmsg_type) */
+ };
}
return 0;
+}
-} /* sctp_msghdr_parse()*/
-
-
-/* Setup sk->rcv_saddr before calling get_port(). */
-static inline void
-sctp_sk_addr_set(struct sock *sk,
- const sockaddr_storage_t *newaddr,
- sockaddr_storage_t *saveaddr)
+/* Setup sk->rcv_saddr before calling get_port(). */
+static inline void sctp_sk_addr_set(struct sock *sk,
+ const sockaddr_storage_t *newaddr,
+ sockaddr_storage_t *saveaddr)
{
struct inet_opt *inet = inet_sk(sk);
saveaddr->sa.sa_family = newaddr->sa.sa_family;
- switch(newaddr->sa.sa_family){
+ switch (newaddr->sa.sa_family) {
case AF_INET:
saveaddr->v4.sin_addr.s_addr = inet->rcv_saddr;
- inet->rcv_saddr = inet->saddr = newaddr->v4.sin_addr.s_addr;
- break;
+ inet->rcv_saddr = inet->saddr = newaddr->v4.sin_addr.s_addr;
+ break;
+
case AF_INET6:
SCTP_V6({
struct ipv6_pinfo *np = inet6_sk(sk);
np->rcv_saddr = np->saddr = newaddr->v6.sin6_addr;
break;
})
+
default:
break;
- }
-
-} /* sctp_sk_addr_store() */
-
+ };
+}
-/* Restore sk->rcv_saddr after failing get_port(). */
-static inline void
-sctp_sk_addr_restore(struct sock *sk, const sockaddr_storage_t *addr)
+/* Restore sk->rcv_saddr after failing get_port(). */
+static inline void sctp_sk_addr_restore(struct sock *sk, const sockaddr_storage_t *addr)
{
struct inet_opt *inet = inet_sk(sk);
- switch(addr->sa.sa_family){
+ switch (addr->sa.sa_family) {
case AF_INET:
- inet->rcv_saddr = inet->saddr = addr->v4.sin_addr.s_addr;
- break;
+ inet->rcv_saddr = inet->saddr = addr->v4.sin_addr.s_addr;
+ break;
+
case AF_INET6:
SCTP_V6({
struct ipv6_pinfo *np = inet6_sk(sk);
np->rcv_saddr = np->saddr = addr->v6.sin6_addr;
break;
})
+
default:
break;
- }
-
-} /* sctp_sk_addr_restore() */
+ };
+}
/*
* Wait for a packet..
* Note: This function is the same function as in core/datagram.c
* with a few modifications to make lksctp work.
*/
-static int
-sctp_wait_for_packet(struct sock * sk, int *err, long *timeo_p)
+static int sctp_wait_for_packet(struct sock * sk, int *err, long *timeo_p)
{
int error;
-
DECLARE_WAITQUEUE(wait, current);
__set_current_state(TASK_INTERRUPTIBLE);
/* Socket errors? */
error = sock_error(sk);
- if (error) {
+ if (error)
goto out;
- }
- if (!skb_queue_empty(&sk->receive_queue)) {
+ if (!skb_queue_empty(&sk->receive_queue))
goto ready;
- }
- /* Socket shut down? */
- if (sk->shutdown & RCV_SHUTDOWN) {
+ /* Socket shut down? */
+ if (sk->shutdown & RCV_SHUTDOWN)
goto out;
- }
- /* Sequenced packets can come disconnected. If so we report the
+ /* Sequenced packets can come disconnected. If so we report the
* problem.
*/
error = -ENOTCONN;
- /* Is there a good reason to think that we may receive some data? */
- if ((list_empty(&sctp_sk(sk)->ep->asocs))
- && (sk->state != SCTP_SS_LISTENING)) {
+ /* Is there a good reason to think that we may receive some data? */
+ if ((list_empty(&sctp_sk(sk)->ep->asocs)) &&
+ (sk->state != SCTP_SS_LISTENING))
goto out;
- }
/* Handle signals. */
- if (signal_pending(current)) {
+ if (signal_pending(current))
goto interrupted;
- }
-
- /* Let another process have a go. Since we are going to sleep
+
+ /* Let another process have a go. Since we are going to sleep
* anyway. Note: This may cause odd behaviors if the message
* does not fit in the user's buffer, but this seems to be the
- * only way to honor MSG_DONTWAIT realistically.
+ * only way to honor MSG_DONTWAIT realistically.
*/
sctp_release_sock(sk);
*timeo_p = schedule_timeout(*timeo_p);
interrupted:
error = sock_intr_errno(*timeo_p);
+
out:
remove_wait_queue(sk->sleep, &wait);
__set_current_state(TASK_RUNNING);
*err = error;
return error;
-
-} /* sctp_wait_for_packet() */
-
+}
/* Receive a datagram.
* Note: This is pretty much the same routine as in core/datagram.c
* with a few changes to make lksctp work.
*/
-struct sk_buff *
-sctp_skb_recv_datagram(struct sock *sk, int flags, int noblock, int *err)
+struct sk_buff *sctp_skb_recv_datagram(struct sock *sk, int flags, int noblock, int *err)
{
int error;
struct sk_buff *skb;
long timeo;
- /* Caller is allowed not to check sk->err before skb_recv_datagram() */
+ /* Caller is allowed not to check sk->err before skb_recv_datagram() */
error = sock_error(sk);
- if (error) {
+ if (error)
goto no_packet;
- }
timeo = sock_rcvtimeo(sk, noblock);
timeo, MAX_SCHEDULE_TIMEOUT);
do {
- /* Again only user level code calls this function,
+ /* Again only user level code calls this function,
* so nothing interrupt level
* will suddenly eat the receive_queue.
- *
+ *
* Look at current nfs client by the way...
* However, this function was corrent in any case. 8)
*/
if (flags & MSG_PEEK) {
unsigned long cpu_flags;
- sctp_spin_lock_irqsave(&sk->receive_queue.lock,
+ sctp_spin_lock_irqsave(&sk->receive_queue.lock,
cpu_flags);
skb = skb_peek(&sk->receive_queue);
- if (skb!=NULL)
+ if (skb)
atomic_inc(&skb->users);
- sctp_spin_unlock_irqrestore(&sk->receive_queue.lock,
+ sctp_spin_unlock_irqrestore(&sk->receive_queue.lock,
cpu_flags);
} else {
skb = skb_dequeue(&sk->receive_queue);
}
- if (skb) { return skb; }
+ if (skb)
+ return skb;
- /* User doesn't want to wait */
+ /* User doesn't want to wait. */
error = -EAGAIN;
if (!timeo)
goto no_packet;
-
} while (sctp_wait_for_packet(sk, err, &timeo) == 0);
return NULL;
no_packet:
*err = error;
return NULL;
+}
-} /* sctp_skb_recv_datagram() */
-
-
-/* Copy an approriately formatted address for msg_name. */
-static inline void
-sctp_sk_memcpy_msgname(struct sock *sk, char * msgname, int *addr_len,
- struct sk_buff *skb)
+/* Copy an approriately formatted address for msg_name. */
+static inline void sctp_sk_memcpy_msgname(struct sock *sk, char * msgname,
+ int *addr_len, struct sk_buff *skb)
{
struct sockaddr_in *sin;
struct sockaddr_in6 *sin6 __attribute__ ((unused));
struct sctphdr *sh;
- /* The sockets layer handles copying this out to user space. */
-
+ /* The sockets layer handles copying this out to user space. */
switch (sk->family) {
case PF_INET:
-
sin = (struct sockaddr_in *)msgname;
- if (addr_len) {
+ if (addr_len)
*addr_len = sizeof(struct sockaddr_in);
- }
sin->sin_family = AF_INET;
-
- sh = (struct sctphdr *)skb->h.raw;
+ sh = (struct sctphdr *) skb->h.raw;
sin->sin_port = sh->source;
sin->sin_addr.s_addr = skb->nh.iph->saddr;
memset(sin->sin_zero, 0, sizeof(sin->sin_zero));
break;
+
case PF_INET6:
SCTP_V6(
-
/* FIXME: Need v6 code here. We should convert
- * V4 addresses to PF_INET6 format. See ipv6/udp.c
+ * V4 addresses to PF_INET6 format. See ipv6/udp.c
* for an example. --jgrimm
*/
);
break;
+
default: /* Should not get here. */
break;
- }
-
-} /* sctp_sk_memcpy_msgname() */
-
+ };
+}
-static inline int
-sctp_sendmsg_verify_name(struct sock *sk, struct msghdr *msg)
+static inline int sctp_sendmsg_verify_name(struct sock *sk, struct msghdr *msg)
{
sockaddr_storage_t *sa;
- if (msg->msg_namelen < sizeof (struct sockaddr) ) {
+ if (msg->msg_namelen < sizeof (struct sockaddr))
return -EINVAL;
- }
- sa = (sockaddr_storage_t *)(msg->msg_name);
+
+ sa = (sockaddr_storage_t *) msg->msg_name;
switch (sa->sa.sa_family) {
case AF_INET:
- if (msg->msg_namelen < sizeof(struct sockaddr_in)) {
+ if (msg->msg_namelen < sizeof(struct sockaddr_in))
return -EINVAL;
- }
break;
+
case AF_INET6:
- if (PF_INET == sk->family) {
+ if (PF_INET == sk->family)
return -EINVAL;
- }
- SCTP_V6(
- if (msg->msg_namelen < sizeof(struct sockaddr_in6)) {
+ SCTP_V6(
+ if (msg->msg_namelen < sizeof(struct sockaddr_in6))
return -EINVAL;
- }
break;
);
+
default:
return -EINVAL;
- }
+ };
/* Disallow any illegal addresses to be used as destinations. */
- if (!sctp_addr_is_valid(sa)) {
+ if (!sctp_addr_is_valid(sa))
return -EINVAL;
- }
return 0;
-
-} /* sctp_sendmsg_verify_name() */
-
+}
/* Get the sndbuf space available at the time on the association. */
-static inline int
-sctp_wspace(sctp_association_t *asoc)
+static inline int sctp_wspace(sctp_association_t *asoc)
{
struct sock *sk = asoc->base.sk;
int amt = 0;
amt = sk->sndbuf - asoc->sndbuf_used;
- if (amt < 0) {
+ if (amt < 0)
amt = 0;
- }
return amt;
-
-} /* sctp_wspace() */
-
+}
/* Increment the used sndbuf space count of the corresponding association by
* the size of the outgoing data chunk.
- * Also, set the skb destructor for sndbuf accounting later.
+ * Also, set the skb destructor for sndbuf accounting later.
*
- * Since it is always 1-1 between chunk and skb, and also a new skb is always
- * allocated for chunk bundling in sctp_packet_transmit(), we can use the
+ * Since it is always 1-1 between chunk and skb, and also a new skb is always
+ * allocated for chunk bundling in sctp_packet_transmit(), we can use the
* destructor in the data chunk skb for the purpose of the sndbuf space
- * tracking.
+ * tracking.
*/
-static inline void
-sctp_set_owner_w(sctp_chunk_t *chunk)
+static inline void sctp_set_owner_w(sctp_chunk_t *chunk)
{
sctp_association_t *asoc = chunk->asoc;
struct sock *sk = asoc->base.sk;
-
- /* The sndbuf space is tracked per association. */
+ /* The sndbuf space is tracked per association. */
sctp_association_hold(asoc);
chunk->skb->destructor = sctp_wfree;
/* Save the chunk pointer in skb for sctp_wfree to use later. */
- *((sctp_chunk_t **)(chunk->skb->cb)) = chunk;
+ *((sctp_chunk_t **)(chunk->skb->cb)) = chunk;
asoc->sndbuf_used += SCTP_DATA_SNDSIZE(chunk);
sk->wmem_queued += SCTP_DATA_SNDSIZE(chunk);
-
-} /* sctp_set_owner_w() */
+}
/* Do accounting for the sndbuf space.
* Decrement the used sndbuf space of the corresponding association by the
* data size which was just transmitted(freed).
*/
-static void
-sctp_wfree(struct sk_buff *skb)
+static void sctp_wfree(struct sk_buff *skb)
{
sctp_association_t *asoc;
sctp_chunk_t *chunk;
struct sock *sk;
-
/* Get the saved chunk pointer. */
chunk = *((sctp_chunk_t **)(skb->cb));
asoc = chunk->asoc;
__sctp_write_space(asoc);
sctp_association_put(asoc);
+}
-} /* sctp_wfree() */
-
-
-/* Helper function to wait for space in the sndbuf. */
-static int
-sctp_wait_for_sndbuf(sctp_association_t *asoc, long *timeo_p, int msg_len)
+/* Helper function to wait for space in the sndbuf. */
+static int sctp_wait_for_sndbuf(sctp_association_t *asoc, long *timeo_p, int msg_len)
{
struct sock *sk = asoc->base.sk;
int err = 0;
long current_timeo = *timeo_p;
DECLARE_WAITQUEUE(wait, current);
- SCTP_DEBUG_PRINTK("wait_for_sndbuf: asoc=%p, timeo=%ld, msg_len=%d\n",
+ SCTP_DEBUG_PRINTK("wait_for_sndbuf: asoc=%p, timeo=%ld, msg_len=%d\n",
asoc, (long)(*timeo_p), msg_len);
-
+
/* Wait on the association specific sndbuf space. */
add_wait_queue_exclusive(&asoc->wait, &wait);
/* Increment the association's refcnt. */
sctp_association_hold(asoc);
-
for (;;) {
-
set_current_state(TASK_INTERRUPTIBLE);
-
- if (!*timeo_p) {
+ if (!*timeo_p)
goto do_nonblock;
- }
-
if (sk->err || asoc->state >= SCTP_STATE_SHUTDOWN_PENDING ||
- asoc->base.dead) {
+ asoc->base.dead)
goto do_error;
- }
-
- if (signal_pending(current)) {
+ if (signal_pending(current))
goto do_interrupted;
- }
-
- if (msg_len <= sctp_wspace(asoc)) {
+ if (msg_len <= sctp_wspace(asoc))
break;
- }
-
- /* Let another process have a go. Since we are going
- * to sleep anyway.
- */
+ /* Let another process have a go. Since we are going
+ * to sleep anyway.
+ */
sctp_release_sock(sk);
-
current_timeo = schedule_timeout(current_timeo);
-
sctp_lock_sock(sk);
-
+
*timeo_p = current_timeo;
}
out:
-
remove_wait_queue(&asoc->wait, &wait);
/* Release the association's refcnt. */
do_nonblock:
err = -EAGAIN;
goto out;
+}
-} /* sctp_wait_for_sndbuf() */
-
-
-/* If sndbuf has changed, wake up per association sndbuf waiters. */
-static void
-__sctp_write_space(sctp_association_t *asoc)
+/* If sndbuf has changed, wake up per association sndbuf waiters. */
+static void __sctp_write_space(sctp_association_t *asoc)
{
struct sock *sk = asoc->base.sk;
struct socket *sock = sk->socket;
-
+
if ((sctp_wspace(asoc) > 0) && sock) {
- if (waitqueue_active(&asoc->wait)) {
+ if (waitqueue_active(&asoc->wait))
wake_up_interruptible(&asoc->wait);
- }
+
if (sctp_writeable(sk)) {
- if (sk->sleep && waitqueue_active(sk->sleep)) {
+ if (sk->sleep && waitqueue_active(sk->sleep))
wake_up_interruptible(sk->sleep);
- }
- /* Note that we try to include the Async I/O support
- * here by modeling from the current TCP/UDP code.
- * We have not tested with it yet.
+
+ /* Note that we try to include the Async I/O support
+ * here by modeling from the current TCP/UDP code.
+ * We have not tested with it yet.
*/
- if (sock->fasync_list
- && !(sk->shutdown & SEND_SHUTDOWN)) {
+ if (sock->fasync_list &&
+ !(sk->shutdown & SEND_SHUTDOWN))
sock_wake_async(sock, 2, POLL_OUT);
- }
}
}
+}
-} /* __sctp_write_space() */
-
-
-/* If socket sndbuf has changed, wake up all per association waiters. */
-void
-sctp_write_space(struct sock *sk)
+/* If socket sndbuf has changed, wake up all per association waiters. */
+void sctp_write_space(struct sock *sk)
{
sctp_association_t *asoc;
list_t *pos;
asoc = list_entry(pos, sctp_association_t, asocs);
__sctp_write_space(asoc);
}
-
-} /* sctp_write_space() */
+}
/* Is there any sndbuf space available on the socket?
- *
- * Note that wmem_queued is the sum of the send buffers on all of the
- * associations on the same socket. For a UDP-style socket with
- * multiple associations, it is possible for it to be "unwriteable"
+ *
+ * Note that wmem_queued is the sum of the send buffers on all of the
+ * associations on the same socket. For a UDP-style socket with
+ * multiple associations, it is possible for it to be "unwriteable"
* prematurely. I assume that this is acceptable because
- * a premature "unwriteable" is better than an accidental "writeable" which
- * would cause an unwanted block under certain circumstances. For the 1-1
+ * a premature "unwriteable" is better than an accidental "writeable" which
+ * would cause an unwanted block under certain circumstances. For the 1-1
* UDP-style sockets or TCP-style sockets, this code should work.
* - Daisy
*/
-static int
-sctp_writeable(struct sock *sk)
+static int sctp_writeable(struct sock *sk)
{
-
int amt = 0;
amt = sk->sndbuf - sk->wmem_queued;
- if (amt < 0) {
+ if (amt < 0)
amt = 0;
- }
return amt;
-
-} /* sctp_writeable() */
+}
/* This proto struct describes the ULP interface for SCTP. */
struct proto sctp_prot = {
- .name = "SCTP",
- .close = sctp_close,
- .connect = sctp_connect,
- .disconnect = sctp_disconnect,
- .accept = sctp_accept,
- .ioctl = sctp_ioctl,
- .init = sctp_init_sock,
- .destroy = sctp_destroy_sock,
- .shutdown = sctp_shutdown,
- .setsockopt = sctp_setsockopt,
- .getsockopt = sctp_getsockopt,
- .sendmsg = sctp_sendmsg,
- .recvmsg = sctp_recvmsg,
- .bind = sctp_bind,
- .backlog_rcv = sctp_backlog_rcv,
- .hash = sctp_hash,
- .unhash = sctp_unhash,
- .get_port = sctp_get_port,
+ .name = "SCTP",
+ .close = sctp_close,
+ .connect = sctp_connect,
+ .disconnect = sctp_disconnect,
+ .accept = sctp_accept,
+ .ioctl = sctp_ioctl,
+ .init = sctp_init_sock,
+ .destroy = sctp_destroy_sock,
+ .shutdown = sctp_shutdown,
+ .setsockopt = sctp_setsockopt,
+ .getsockopt = sctp_getsockopt,
+ .sendmsg = sctp_sendmsg,
+ .recvmsg = sctp_recvmsg,
+ .bind = sctp_bind,
+ .backlog_rcv = sctp_backlog_rcv,
+ .hash = sctp_hash,
+ .unhash = sctp_unhash,
+ .get_port = sctp_get_port,
};
extern sctp_protocol_t sctp_proto;
static ctl_table sctp_table[] = {
- {NET_SCTP_RTO_INITIAL, "rto_initial",
- &sctp_proto.rto_initial, sizeof(int), 0644, NULL,
- &proc_dointvec_jiffies, &sysctl_jiffies},
- {NET_SCTP_RTO_MIN, "rto_min",
- &sctp_proto.rto_min, sizeof(int), 0644, NULL,
- &proc_dointvec_jiffies, &sysctl_jiffies},
- {NET_SCTP_RTO_MAX, "rto_max",
- &sctp_proto.rto_max, sizeof(int), 0644, NULL,
- &proc_dointvec_jiffies, &sysctl_jiffies},
- {NET_SCTP_VALID_COOKIE_LIFE, "valid_cookie_life",
- &sctp_proto.valid_cookie_life, sizeof(int), 0644, NULL,
- &proc_dointvec_jiffies, &sysctl_jiffies},
- {NET_SCTP_MAX_BURST, "max_burst",
- &sctp_proto.max_burst, sizeof(int), 0644, NULL,
- &proc_dointvec},
- {NET_SCTP_ASSOCIATION_MAX_RETRANS, "association_max_retrans",
- &sctp_proto.max_retrans_association, sizeof(int), 0644, NULL,
- &proc_dointvec},
- {NET_SCTP_PATH_MAX_RETRANS, "path_max_retrans",
- &sctp_proto.max_retrans_path, sizeof(int), 0644, NULL,
- &proc_dointvec},
- {NET_SCTP_MAX_INIT_RETRANSMITS, "max_init_retransmits",
- &sctp_proto.max_retrans_init, sizeof(int), 0644, NULL,
- &proc_dointvec},
- {NET_SCTP_HB_INTERVAL, "hb_interval",
- &sctp_proto.hb_interval, sizeof(int), 0644, NULL,
- &proc_dointvec_jiffies, &sysctl_jiffies},
- {NET_SCTP_RTO_ALPHA, "rto_alpha_exp_divisor",
- &sctp_proto.rto_alpha, sizeof(int), 0644, NULL,
- &proc_dointvec},
- {NET_SCTP_RTO_BETA, "rto_beta_exp_divisor",
- &sctp_proto.rto_beta, sizeof(int), 0644, NULL,
- &proc_dointvec},
- {0}
+ { NET_SCTP_RTO_INITIAL, "rto_initial",
+ &sctp_proto.rto_initial, sizeof(int), 0644, NULL,
+ &proc_dointvec_jiffies, &sysctl_jiffies },
+ { NET_SCTP_RTO_MIN, "rto_min",
+ &sctp_proto.rto_min, sizeof(int), 0644, NULL,
+ &proc_dointvec_jiffies, &sysctl_jiffies },
+ { NET_SCTP_RTO_MAX, "rto_max",
+ &sctp_proto.rto_max, sizeof(int), 0644, NULL,
+ &proc_dointvec_jiffies, &sysctl_jiffies },
+ { NET_SCTP_VALID_COOKIE_LIFE, "valid_cookie_life",
+ &sctp_proto.valid_cookie_life, sizeof(int), 0644, NULL,
+ &proc_dointvec_jiffies, &sysctl_jiffies },
+ { NET_SCTP_MAX_BURST, "max_burst",
+ &sctp_proto.max_burst, sizeof(int), 0644, NULL,
+ &proc_dointvec },
+ { NET_SCTP_ASSOCIATION_MAX_RETRANS, "association_max_retrans",
+ &sctp_proto.max_retrans_association, sizeof(int), 0644, NULL,
+ &proc_dointvec },
+ { NET_SCTP_PATH_MAX_RETRANS, "path_max_retrans",
+ &sctp_proto.max_retrans_path, sizeof(int), 0644, NULL,
+ &proc_dointvec },
+ { NET_SCTP_MAX_INIT_RETRANSMITS, "max_init_retransmits",
+ &sctp_proto.max_retrans_init, sizeof(int), 0644, NULL,
+ &proc_dointvec },
+ { NET_SCTP_HB_INTERVAL, "hb_interval",
+ &sctp_proto.hb_interval, sizeof(int), 0644, NULL,
+ &proc_dointvec_jiffies, &sysctl_jiffies },
+ { NET_SCTP_RTO_ALPHA, "rto_alpha_exp_divisor",
+ &sctp_proto.rto_alpha, sizeof(int), 0644, NULL,
+ &proc_dointvec },
+ { NET_SCTP_RTO_BETA, "rto_beta_exp_divisor",
+ &sctp_proto.rto_beta, sizeof(int), 0644, NULL,
+ &proc_dointvec },
+ { 0 }
};
static ctl_table sctp_net_table[] = {
- {NET_SCTP, "sctp", NULL, 0, 0555, sctp_table},
- {0}
+ { NET_SCTP, "sctp", NULL, 0, 0555, sctp_table },
+ { 0 }
};
static ctl_table sctp_root_table[] = {
- {CTL_NET, "net", NULL, 0, 0555, sctp_net_table},
- {0}
+ { CTL_NET, "net", NULL, 0, 0555, sctp_net_table },
+ { 0 }
};
static struct ctl_table_header * sctp_sysctl_header;
-/* Sysctl registration. */
+/* Sysctl registration. */
void sctp_sysctl_register(void)
{
sctp_sysctl_header = register_sysctl_table(sctp_root_table, 0);
+}
-} /* sctp_sysctl_register() */
-
-/* Sysctl deregistration. */
+/* Sysctl deregistration. */
void sctp_sysctl_unregister(void)
{
unregister_sysctl_table(sctp_sysctl_header);
-
-} /* sctp_sysctl_unregister() */
-
+}
*/
static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_transport.c,v 1.11 2002/06/20 05:57:01 samudrala Exp $";
-#include <linux/config.h>
#include <linux/types.h>
#include <net/sctp/sctp.h>
-/* 1st Level Abstractions. */
+/* 1st Level Abstractions. */
-/* Allocate and initialize a new transport. */
-sctp_transport_t *
-sctp_transport_new(const sockaddr_storage_t *addr, int priority)
+/* Allocate and initialize a new transport. */
+sctp_transport_t *sctp_transport_new(const sockaddr_storage_t *addr, int priority)
{
sctp_transport_t *transport;
transport = t_new(sctp_transport_t, priority);
- if (NULL == transport) {
+ if (!transport)
goto fail;
- }
- if (NULL == sctp_transport_init(transport, addr, priority)) {
+ if (!sctp_transport_init(transport, addr, priority))
goto fail_init;
- }
transport->malloced = 1;
SCTP_DBG_OBJCNT_INC(transport);
return transport;
-
+
fail_init:
kfree(transport);
+
fail:
return NULL;
+}
-} /* sctp_transport_new() */
-
-
-/* Intialize a new transport from provided memory. */
-sctp_transport_t *
-sctp_transport_init(sctp_transport_t *peer,
- const sockaddr_storage_t *addr,
- int priority)
+/* Intialize a new transport from provided memory. */
+sctp_transport_t *sctp_transport_init(sctp_transport_t *peer,
+ const sockaddr_storage_t *addr,
+ int priority)
{
- sctp_protocol_t *proto = sctp_get_protocol();
+ sctp_protocol_t *proto = sctp_get_protocol();
- /* Copy in the address. */
+ /* Copy in the address. */
peer->ipaddr = *addr;
- peer->af_specific = sctp_get_af_specific(addr);
- peer->asoc = NULL;
- peer->pmtu = peer->af_specific->get_dst_mtu(addr);
+ peer->af_specific = sctp_get_af_specific(addr);
+ peer->asoc = NULL;
+ peer->pmtu = peer->af_specific->get_dst_mtu(addr);
- /* From 6.3.1 RTO Calculation:
- *
+ /* From 6.3.1 RTO Calculation:
+ *
* C1) Until an RTT measurement has been made for a packet sent to the
* given destination transport address, set RTO to the protocol
* parameter 'RTO.Initial'.
- */
+ */
peer->rtt = 0;
peer->rto = proto->rto_initial;
peer->rttvar = 0;
peer->last_time_heard = jiffies;
peer->last_time_used = jiffies;
peer->last_time_ecne_reduced = jiffies;
-
+
peer->state.active = 1;
- peer->state.hb_allowed = 0;
-
- /* Initialize the default path max_retrans. */
+ peer->state.hb_allowed = 0;
+
+ /* Initialize the default path max_retrans. */
peer->max_retrans = proto->max_retrans_path;
peer->error_threshold = 0;
- peer->error_count = 0;
+ peer->error_count = 0;
+
+ peer->debug_name = "unnamedtransport";
- peer->debug_name = "unnamedtransport";
-
INIT_LIST_HEAD(&peer->transmitted);
INIT_LIST_HEAD(&peer->send_ready);
INIT_LIST_HEAD(&peer->transports);
peer->T3_rtx_timer.function = sctp_generate_t3_rtx_event;
peer->T3_rtx_timer.data = (unsigned long)peer;
- /* Set up the heartbeat timer. */
- init_timer(&peer->hb_timer);
- peer->hb_interval = SCTP_DEFAULT_TIMEOUT_HEARTBEAT;
- peer->hb_timer.function = sctp_generate_heartbeat_event;
- peer->hb_timer.data = (unsigned long)peer;
+ /* Set up the heartbeat timer. */
+ init_timer(&peer->hb_timer);
+ peer->hb_interval = SCTP_DEFAULT_TIMEOUT_HEARTBEAT;
+ peer->hb_timer.function = sctp_generate_heartbeat_event;
+ peer->hb_timer.data = (unsigned long)peer;
- atomic_set(&peer->refcnt, 1);
+ atomic_set(&peer->refcnt, 1);
peer->dead = 0;
peer->malloced = 0;
- return peer;
-
-} /* sctp_transport_init() */
+ return peer;
+}
/* This transport is no longer needed. Free up if possible, or
- * delay until it last reference count.
+ * delay until it last reference count.
*/
-void
-sctp_transport_free(sctp_transport_t *transport)
+void sctp_transport_free(sctp_transport_t *transport)
{
transport->dead = 1;
- /* Try to delete the heartbeat timer. */
- if (del_timer(&transport->hb_timer)) {
+ /* Try to delete the heartbeat timer. */
+ if (del_timer(&transport->hb_timer))
sctp_transport_put(transport);
- }
-
- sctp_transport_put(transport);
-
-} /* sctp_transport_free() */
+ sctp_transport_put(transport);
+}
-/* Destroy the transport data structure.
- * Assumes there are no more users of this structure.
-*/
-void
-sctp_transport_destroy(sctp_transport_t *transport)
+/* Destroy the transport data structure.
+ * Assumes there are no more users of this structure.
+ */
+void sctp_transport_destroy(sctp_transport_t *transport)
{
SCTP_ASSERT(transport->dead, "Transport is not dead", return);
- if (transport->asoc) {
+ if (transport->asoc)
sctp_association_put(transport->asoc);
- }
kfree(transport);
SCTP_DBG_OBJCNT_DEC(transport);
+}
-} /* sctp_transport_destroy() */
-
-
-/* Start T3_rtx timer if it is not already running and update the heartbeat
- * timer. This routine is called everytime a DATA chunk is sent.
- */
-void
-sctp_transport_reset_timers(sctp_transport_t *transport)
+/* Start T3_rtx timer if it is not already running and update the heartbeat
+ * timer. This routine is called everytime a DATA chunk is sent.
+ */
+void sctp_transport_reset_timers(sctp_transport_t *transport)
{
/* RFC 2960 6.3.2 Retransmission Timer Rules
*
- * R1) Every time a DATA chunk is sent to any address(including a
- * retransmission), if the T3-rtx timer of that address is not running
+ * R1) Every time a DATA chunk is sent to any address(including a
+ * retransmission), if the T3-rtx timer of that address is not running
* start it running so that it will expire after the RTO of that
* address.
*/
if (!timer_pending(&transport->T3_rtx_timer)) {
if (!mod_timer(&transport->T3_rtx_timer,
- jiffies + transport->rto)) {
+ jiffies + transport->rto))
sctp_transport_hold(transport);
- }
}
- /* When a data chunk is sent, reset the heartbeat interval. */
- if (!mod_timer(&transport->hb_timer,
- transport->hb_interval + transport->rto + jiffies)) {
+ /* When a data chunk is sent, reset the heartbeat interval. */
+ if (!mod_timer(&transport->hb_timer,
+ transport->hb_interval + transport->rto + jiffies))
sctp_transport_hold(transport);
- }
+}
-} /* sctp_transport_reset_timers() */
-
-/* This transport has been assigned to an association.
- * Initialize fields from the association or from the sock itself.
- * Register the reference count in the association.
+/* This transport has been assigned to an association.
+ * Initialize fields from the association or from the sock itself.
+ * Register the reference count in the association.
*/
-void sctp_transport_set_owner(sctp_transport_t *transport,
+void sctp_transport_set_owner(sctp_transport_t *transport,
sctp_association_t *asoc)
{
transport->asoc = asoc;
sctp_association_hold(asoc);
+}
-} /* sctp_transport_set_owner() */
-
-/* Hold a reference to a transport. */
+/* Hold a reference to a transport. */
void sctp_transport_hold(sctp_transport_t *transport)
{
atomic_inc(&transport->refcnt);
-
-} /* sctp_transport_hold() */
+}
/* Release a reference to a transport and clean up
- * if there are no more references.
+ * if there are no more references.
*/
void sctp_transport_put(sctp_transport_t *transport)
{
- if (atomic_dec_and_test(&transport->refcnt)) {
+ if (atomic_dec_and_test(&transport->refcnt))
sctp_transport_destroy(transport);
- }
-
-} /* sctp_transport_put() */
+}
/* Update transport's RTO based on the newly calculated RTT. */
-void
-sctp_transport_update_rto(sctp_transport_t *tp, uint32_t rtt)
+void sctp_transport_update_rto(sctp_transport_t *tp, __u32 rtt)
{
sctp_protocol_t *proto = sctp_get_protocol();
- /* Check for valid transport. */
+ /* Check for valid transport. */
SCTP_ASSERT(tp, "NULL transport", return);
- /* We should not be doing any RTO updates unless rto_pending is set. */
+ /* We should not be doing any RTO updates unless rto_pending is set. */
SCTP_ASSERT(tp->rto_pending, "rto_pending not set", return);
if (tp->rttvar || tp->srtt) {
/* 6.3.1 C3) When a new RTT measurement R' is made, set
- * RTTVAR <- (1 - RTO.Beta) * RTTVAR + RTO.Beta * |SRTT - R'|
+ * RTTVAR <- (1 - RTO.Beta) * RTTVAR + RTO.Beta * |SRTT - R'|
* SRTT <- (1 - RTO.Alpha) * SRTT + RTO.Alpha * R'
*/
- /* Note: The above algorithm has been rewritten to
+ /* Note: The above algorithm has been rewritten to
* express rto_beta and rto_alpha as inverse powers
- * of two.
+ * of two.
* For example, assuming the default value of RTO.Alpha of
- * 1/8, rto_alpha would be expressed as 3.
+ * 1/8, rto_alpha would be expressed as 3.
*/
- tp->rttvar = tp->rttvar - (tp->rttvar >> proto->rto_beta)
+ tp->rttvar = tp->rttvar - (tp->rttvar >> proto->rto_beta)
+ ((abs(tp->srtt - rtt)) >> proto->rto_beta);
- tp->srtt = tp->srtt - (tp->srtt >> proto->rto_alpha)
+ tp->srtt = tp->srtt - (tp->srtt >> proto->rto_alpha)
+ (rtt >> proto->rto_alpha);
-
} else {
- /* 6.3.1 C2) When the first RTT measurement R is made, set
+ /* 6.3.1 C2) When the first RTT measurement R is made, set
* SRTT <- R, RTTVAR <- R/2.
*/
tp->srtt = rtt;
tp->rttvar = rtt >> 1;
}
- /* 6.3.1 G1) Whenever RTTVAR is computed, if RTTVAR = 0, then
+ /* 6.3.1 G1) Whenever RTTVAR is computed, if RTTVAR = 0, then
* adjust RTTVAR <- G, where G is the CLOCK GRANULARITY.
*/
- if (tp->rttvar == 0) {
+ if (tp->rttvar == 0)
tp->rttvar = SCTP_CLOCK_GRANULARITY;
- }
- /* 6.3.1 C3) After the computation, update RTO <- SRTT + 4 * RTTVAR. */
+ /* 6.3.1 C3) After the computation, update RTO <- SRTT + 4 * RTTVAR. */
tp->rto = tp->srtt + (tp->rttvar << 2);
- /* 6.3.1 C6) Whenever RTO is computed, if it is less than RTO.Min
- * seconds then it is rounded up to RTO.Min seconds.
+ /* 6.3.1 C6) Whenever RTO is computed, if it is less than RTO.Min
+ * seconds then it is rounded up to RTO.Min seconds.
*/
- if (tp->rto < tp->asoc->rto_min) {
+ if (tp->rto < tp->asoc->rto_min)
tp->rto = tp->asoc->rto_min;
- }
- /* 6.3.1 C7) A maximum value may be placed on RTO provided it is
+ /* 6.3.1 C7) A maximum value may be placed on RTO provided it is
* at least RTO.max seconds.
*/
- if (tp->rto > tp->asoc->rto_max) {
+ if (tp->rto > tp->asoc->rto_max)
tp->rto = tp->asoc->rto_max;
- }
tp->rtt = rtt;
- /* Reset rto_pending so that a new RTT measurement is started when a
+ /* Reset rto_pending so that a new RTT measurement is started when a
* new data chunk is sent.
*/
tp->rto_pending = 0;
SCTP_DEBUG_PRINTK(__FUNCTION__ ": transport: %p, rtt: %d, srtt: %d "
- "rttvar: %d, rto: %d\n",
+ "rttvar: %d, rto: %d\n",
tp, rtt, tp->srtt, tp->rttvar, tp->rto);
+}
-} /* sctp_transport_update_rto() */
-
-/* This routine updates the transport's cwnd and partial_bytes_acked
+/* This routine updates the transport's cwnd and partial_bytes_acked
* parameters based on the bytes acked in the received SACK.
- */
-void
-sctp_transport_raise_cwnd(sctp_transport_t *transport, uint32_t sack_ctsn,
- uint32_t bytes_acked)
+ */
+void sctp_transport_raise_cwnd(sctp_transport_t *transport, __u32 sack_ctsn,
+ __u32 bytes_acked)
{
- uint32_t cwnd, ssthresh, flight_size, pba, pmtu;
+ __u32 cwnd, ssthresh, flight_size, pba, pmtu;
cwnd = transport->cwnd;
flight_size = transport->flight_size;
- /* The appropriate cwnd increase algorithm is performed if, and only
- * if the cumulative TSN has advanced and the congestion window is
- * being fully utilized.
+ /* The appropriate cwnd increase algorithm is performed if, and only
+ * if the cumulative TSN has advanced and the congestion window is
+ * being fully utilized.
*/
- if ((transport->asoc->ctsn_ack_point >= sack_ctsn)
- || (flight_size < cwnd)) {
+ if ((transport->asoc->ctsn_ack_point >= sack_ctsn) ||
+ (flight_size < cwnd))
return;
- }
ssthresh = transport->ssthresh;
pba = transport->partial_bytes_acked;
pmtu = transport->asoc->pmtu;
if (cwnd <= ssthresh) {
- /* RFC 2960 7.2.1, sctpimpguide-05 2.14.2 When cwnd is less
- * than or equal to ssthresh an SCTP endpoint MUST use the
- * slow start algorithm to increase cwnd only if the current
- * congestion window is being fully utilized and an incoming
- * SACK advances the Cumulative TSN Ack Point. Only when these
+ /* RFC 2960 7.2.1, sctpimpguide-05 2.14.2 When cwnd is less
+ * than or equal to ssthresh an SCTP endpoint MUST use the
+ * slow start algorithm to increase cwnd only if the current
+ * congestion window is being fully utilized and an incoming
+ * SACK advances the Cumulative TSN Ack Point. Only when these
* two conditions are met can the cwnd be increased otherwise
- * the cwnd MUST not be increased. If these conditions are met
- * then cwnd MUST be increased by at most the lesser of
- * 1) the total size of the previously outstanding DATA chunk(s)
- * acknowledged, and 2) the destination's path MTU.
+ * the cwnd MUST not be increased. If these conditions are met
+ * then cwnd MUST be increased by at most the lesser of
+ * 1) the total size of the previously outstanding DATA chunk(s)
+ * acknowledged, and 2) the destination's path MTU.
*/
- if (bytes_acked > pmtu) {
+ if (bytes_acked > pmtu)
cwnd += pmtu;
- } else {
+ else
cwnd += bytes_acked;
- }
SCTP_DEBUG_PRINTK(__FUNCTION__ ": SLOW START: transport: %p, "
- "bytes_acked: %d, cwnd: %d, ssthresh: %d, "
- "flight_size: %d, pba: %d\n",
- transport, bytes_acked, cwnd,
+ "bytes_acked: %d, cwnd: %d, ssthresh: %d, "
+ "flight_size: %d, pba: %d\n",
+ transport, bytes_acked, cwnd,
ssthresh, flight_size, pba);
} else {
- /* RFC 2960 7.2.2 Whenever cwnd is greater than ssthresh, upon
- * each SACK arrival that advances the Cumulative TSN Ack Point,
- * increase partial_bytes_acked by the total number of bytes of
- * all new chunks acknowledged in that SACK including chunks
- * acknowledged by the new Cumulative TSN Ack and by Gap Ack
+ /* RFC 2960 7.2.2 Whenever cwnd is greater than ssthresh, upon
+ * each SACK arrival that advances the Cumulative TSN Ack Point,
+ * increase partial_bytes_acked by the total number of bytes of
+ * all new chunks acknowledged in that SACK including chunks
+ * acknowledged by the new Cumulative TSN Ack and by Gap Ack
* Blocks.
- *
+ *
* When partial_bytes_acked is equal to or greater than cwnd and
- * before the arrival of the SACK the sender had cwnd or more
- * bytes of data outstanding (i.e., before arrival of the SACK,
- * flightsize was greater than or equal to cwnd), increase cwnd
- * by MTU, and reset partial_bytes_acked to
+ * before the arrival of the SACK the sender had cwnd or more
+ * bytes of data outstanding (i.e., before arrival of the SACK,
+ * flightsize was greater than or equal to cwnd), increase cwnd
+ * by MTU, and reset partial_bytes_acked to
* (partial_bytes_acked - cwnd).
*/
pba += bytes_acked;
}
SCTP_DEBUG_PRINTK(__FUNCTION__ ": CONGESTION AVOIDANCE: "
"transport: %p, bytes_acked: %d, cwnd: %d, "
- "ssthresh: %d, flight_size: %d, pba: %d\n",
- transport, bytes_acked, cwnd,
+ "ssthresh: %d, flight_size: %d, pba: %d\n",
+ transport, bytes_acked, cwnd,
ssthresh, flight_size, pba);
}
transport->cwnd = cwnd;
transport->partial_bytes_acked = pba;
+}
-} /* sctp_transport_raise_cwnd() */
-
-/* This routine is used to lower the transport's cwnd when congestion is
- * detected.
+/* This routine is used to lower the transport's cwnd when congestion is
+ * detected.
*/
-void
-sctp_transport_lower_cwnd(sctp_transport_t *transport,
- sctp_lower_cwnd_t reason)
-{
- switch(reason) {
+void sctp_transport_lower_cwnd(sctp_transport_t *transport,
+ sctp_lower_cwnd_t reason)
+{
+ switch (reason) {
case SCTP_LOWER_CWND_T3_RTX:
- /* RFC 2960 Section 7.2.3, sctpimpguide-05 Section 2.9.2
- * When the T3-rtx timer expires on an address, SCTP should
+ /* RFC 2960 Section 7.2.3, sctpimpguide-05 Section 2.9.2
+ * When the T3-rtx timer expires on an address, SCTP should
* perform slow start by:
- * ssthresh = max(cwnd/2, 2*MTU)
- * cwnd = 1*MTU
- * partial_bytes_acked = 0
- */
- transport->ssthresh = max(transport->cwnd/2,
+ * ssthresh = max(cwnd/2, 2*MTU)
+ * cwnd = 1*MTU
+ * partial_bytes_acked = 0
+ */
+ transport->ssthresh = max(transport->cwnd/2,
2*transport->asoc->pmtu);
transport->cwnd = transport->asoc->pmtu;
break;
+
case SCTP_LOWER_CWND_FAST_RTX:
/* RFC 2960 7.2.4 Adjust the ssthresh and cwnd of the
* destination address(es) to which the missing DATA chunks
* RFC 2960 7.2.3, sctpimpguide-05 2.9.2 Upon detection of
* packet losses from SACK (see Section 7.2.4), An endpoint
* should do the following:
- * ssthresh = max(cwnd/2, 2*MTU)
- * cwnd = ssthresh
+ * ssthresh = max(cwnd/2, 2*MTU)
+ * cwnd = ssthresh
* partial_bytes_acked = 0
- */
- transport->ssthresh = max(transport->cwnd/2,
+ */
+ transport->ssthresh = max(transport->cwnd/2,
2*transport->asoc->pmtu);
transport->cwnd = transport->ssthresh;
break;
+
case SCTP_LOWER_CWND_ECNE:
- /* RFC 2481 Section 6.1.2.
+ /* RFC 2481 Section 6.1.2.
* If the sender receives an ECN-Echo ACK packet
* then the sender knows that congestion was encountered in the
* network on the path from the sender to the receiver. The
- * indication of congestion should be treated just as a
- * congestion loss in non-ECN Capable TCP. That is, the TCP
+ * indication of congestion should be treated just as a
+ * congestion loss in non-ECN Capable TCP. That is, the TCP
* source halves the congestion window "cwnd" and reduces the
* slow start threshold "ssthresh".
- * A critical condition is that TCP does not react to
- * congestion indications more than once every window of
+ * A critical condition is that TCP does not react to
+ * congestion indications more than once every window of
* data (or more loosely more than once every round-trip time).
*/
- if ((jiffies - transport->last_time_ecne_reduced)
- > transport->rtt) {
- transport->ssthresh = max(transport->cwnd/2,
+ if ((jiffies - transport->last_time_ecne_reduced) >
+ transport->rtt) {
+ transport->ssthresh = max(transport->cwnd/2,
2*transport->asoc->pmtu);
transport->cwnd = transport->ssthresh;
transport->last_time_ecne_reduced = jiffies;
}
break;
+
case SCTP_LOWER_CWND_INACTIVE:
- /* RFC 2960 Section 7.2.1, sctpimpguide-05 Section 2.14.2
- * When the association does not transmit data on a given
- * transport address within an RTO, the cwnd of the transport
- * address should be adjusted to 2*MTU.
- * NOTE: Although the draft recommends that this check needs
- * to be done every RTO interval, we do it every hearbeat
- * interval.
+ /* RFC 2960 Section 7.2.1, sctpimpguide-05 Section 2.14.2
+ * When the association does not transmit data on a given
+ * transport address within an RTO, the cwnd of the transport
+ * address should be adjusted to 2*MTU.
+ * NOTE: Although the draft recommends that this check needs
+ * to be done every RTO interval, we do it every hearbeat
+ * interval.
*/
- if ((jiffies - transport->last_time_used) > transport->rto) {
+ if ((jiffies - transport->last_time_used) > transport->rto)
transport->cwnd = 2*transport->asoc->pmtu;
- }
break;
- }
+ };
transport->partial_bytes_acked = 0;
-
SCTP_DEBUG_PRINTK(__FUNCTION__ ": transport: %p reason: %d cwnd: "
- "%d ssthresh: %d\n", transport, reason,
+ "%d ssthresh: %d\n", transport, reason,
transport->cwnd, transport->ssthresh);
-
-} /* sctp_transport_lower_cwnd() */
+}
*/
static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_tsnmap.c,v 1.8 2002/07/26 22:52:32 jgrimm Exp $";
-#include <linux/config.h>
#include <linux/types.h>
#include <net/sctp/sctp.h>
#include <net/sctp/sctp_sm.h>
-static inline void _sctp_tsnmap_update(sctp_tsnmap_t *map);
-static inline void _sctp_tsnmap_update_pending_data(sctp_tsnmap_t *map);
-static inline void _sctp_tsnmap_find_gap_ack(uint8_t *map, uint16_t off,
- uint16_t len, uint16_t base,
- int *started, uint16_t *start,
- int *ended, uint16_t *end);
-
+static void _sctp_tsnmap_update(sctp_tsnmap_t *map);
+static void _sctp_tsnmap_update_pending_data(sctp_tsnmap_t *map);
+static void _sctp_tsnmap_find_gap_ack(__u8 *map, __u16 off,
+ __u16 len, __u16 base,
+ int *started, __u16 *start,
+ int *ended, __u16 *end);
/* Create a new sctp_tsnmap.
* Allocate room to store at least 'len' contiguous TSNs.
*/
-sctp_tsnmap_t *
-sctp_tsnmap_new(uint16_t len, uint32_t initial_tsn, int priority)
+sctp_tsnmap_t *sctp_tsnmap_new(__u16 len, __u32 initial_tsn, int priority)
{
sctp_tsnmap_t *retval;
-
- retval = kmalloc(sizeof(sctp_tsnmap_t)
- + sctp_tsnmap_storage_size(len),
+
+ retval = kmalloc(sizeof(sctp_tsnmap_t) +
+ sctp_tsnmap_storage_size(len),
priority);
- if (NULL == retval) {
+ if (!retval)
goto fail;
- }
- if (NULL == sctp_tsnmap_init(retval, len, initial_tsn)) {
+ if (!sctp_tsnmap_init(retval, len, initial_tsn))
goto fail_map;
- }
retval->malloced = 1;
- return(retval);
+ return retval;
+
+fail_map:
+ kfree retval;
- fail_map:
- kfree(retval);
- fail:
+fail:
return NULL;
-
-} /* sctp_tsnmap_new() */
+}
/* Initialize a block of memory as a tsnmap. */
-sctp_tsnmap_t *
-sctp_tsnmap_init(sctp_tsnmap_t *map, uint16_t len, uint32_t initial_tsn)
+sctp_tsnmap_t *sctp_tsnmap_init(sctp_tsnmap_t *map, __u16 len, __u32 initial_tsn)
{
map->tsn_map = map->raw_map;
map->overflow_map = map->tsn_map + len;
-
map->len = len;
/* Clear out a TSN ack status. */
memset(map->tsn_map, 0x00, map->len + map->len);
-
+
/* Keep track of TSNs represented by tsn_map. */
map->base_tsn = initial_tsn;
map->overflow_tsn = initial_tsn + map->len;
-
map->cumulative_tsn_ack_point = initial_tsn - 1;
map->max_tsn_seen = map->cumulative_tsn_ack_point;
-
map->malloced = 0;
-
map->pending_data = 0;
- return map;
-
-} /* sctp_tsnmap_init() */
-
+ return map;
+}
/* Test the tracking state of this TSN.
* Returns:
* >0 if the TSN has been seen (duplicate)
* <0 if the TSN is invalid (too large to track)
*/
-int
-sctp_tsnmap_check(const sctp_tsnmap_t *map, uint32_t tsn)
+int sctp_tsnmap_check(const sctp_tsnmap_t *map, __u32 tsn)
{
- int32_t gap;
+ __s32 gap;
int dup;
- /* Calculate the index into the mapping arrays. */
- gap = tsn - map->base_tsn;
+
+ /* Calculate the index into the mapping arrays. */
+ gap = tsn - map->base_tsn;
/* Verify that we can hold this TSN. */
if (gap >= (/* base */ map->len + /* overflow */ map->len)) {
}
/* Honk if we've already seen this TSN.
- * We have three cases:
+ * 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])) {
+ 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;
- }
-
- out:
- return(dup);
-
-} /* sctp_tsnmap_check() */
+out:
+ return dup;
+}
-/* Is there a gap in the TSN map? */
-int
-sctp_tsnmap_has_gap(const sctp_tsnmap_t *map)
+/* Is there a gap in the TSN map? */
+int sctp_tsnmap_has_gap(const sctp_tsnmap_t *map)
{
int has_gap;
has_gap = (map->cumulative_tsn_ack_point != map->max_tsn_seen);
return has_gap;
-
-} /* sctp_tsnmap_has_gap() */
-
+}
/* Mark this TSN as seen. */
-void
-sctp_tsnmap_mark(sctp_tsnmap_t *map, uint32_t tsn)
+void sctp_tsnmap_mark(sctp_tsnmap_t *map, __u32 tsn)
{
- int32_t gap;
-
+ __s32 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;
- }
-
- if (!TSN_lt(tsn, map->base_tsn + map->len + map->len)) {
+ if (!TSN_lt(tsn, map->base_tsn + map->len + map->len))
return;
- }
/* Bump the max. */
- if (TSN_lt(map->max_tsn_seen, tsn)) {
+ if (TSN_lt(map->max_tsn_seen, tsn))
map->max_tsn_seen = tsn;
- }
- /* Assert: TSN is in range. */
+ /* Assert: TSN is in range. */
gap = tsn - map->base_tsn;
- /* Mark the TSN as received. */
- if (gap < map->len) {
- map->tsn_map[gap]++;
- } else {
- map->overflow_map[gap - map->len]++;
- }
-
- /* Go fixup any internal TSN mapping variables including
+ /* Mark the TSN as received. */
+ if (gap < map->len)
+ map->tsn_map[gap]++;
+ else
+ map->overflow_map[gap - map->len]++;
+
+ /* Go fixup any internal TSN mapping variables including
* cumulative_tsn_ack_point.
- */
+ */
_sctp_tsnmap_update(map);
-
-} /* sctp_tsnmap_mark() */
+}
/* Retrieve the Cumulative TSN Ack Point. */
-uint32_t
-sctp_tsnmap_get_ctsn(const sctp_tsnmap_t *map)
+__u32 sctp_tsnmap_get_ctsn(const sctp_tsnmap_t *map)
{
- return(map->cumulative_tsn_ack_point);
-
-} /* sctp_tsnmap_get_ctsn() */
+ return map->cumulative_tsn_ack_point;
+}
/* Retrieve the highest TSN we've seen. */
-uint32_t
-sctp_tsnmap_get_max_tsn_seen(const sctp_tsnmap_t *map)
+__u32 sctp_tsnmap_get_max_tsn_seen(const sctp_tsnmap_t *map)
{
- return(map->max_tsn_seen);
-
-} /* sctp_tsnmap_get_max_tsn_seen() */
-
+ return map->max_tsn_seen;
+}
/* Dispose of a tsnmap. */
-void
-sctp_tsnmap_free(sctp_tsnmap_t *map)
+void sctp_tsnmap_free(sctp_tsnmap_t *map)
{
- if (map->malloced) {
+ if (map->malloced)
kfree(map);
- }
-} /* sctp_tsnmap_free() */
-
+}
/* Initialize a Gap Ack Block iterator from memory being provided. */
-void
-sctp_tsnmap_iter_init(const sctp_tsnmap_t *map, sctp_tsnmap_iter_t *iter)
+void sctp_tsnmap_iter_init(const sctp_tsnmap_t *map, sctp_tsnmap_iter_t *iter)
{
/* Only start looking one past the Cumulative TSN Ack Point. */
iter->start = map->cumulative_tsn_ack_point + 1;
+}
-} /* sctp_tsnmap_iter_init() */
-
-
-
-
-
-/* Get the next Gap Ack Blocks. Returns 0 if there was not
+/* Get the next Gap Ack Blocks. Returns 0 if there was not
* another block to get.
*/
-int
-sctp_tsnmap_next_gap_ack(const sctp_tsnmap_t *map, sctp_tsnmap_iter_t *iter,
- uint16_t *start, uint16_t *end)
+int sctp_tsnmap_next_gap_ack(const sctp_tsnmap_t *map, sctp_tsnmap_iter_t *iter,
+ __u16 *start, __u16 *end)
{
int started, ended;
- uint16_t _start, _end, offset;
-
+ __u16 _start, _end, offset;
- /* We haven't found a gap yet. */
+ /* We haven't found a gap yet. */
started = ended = 0;
-
/* Search the first mapping array. */
if (iter->start - map->base_tsn < map->len) {
offset = iter->start - map->base_tsn;
map->len, 0,
&started, &_start,
&ended, &_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) {
+ if (iter->start - map->base_tsn < map->len)
offset = 0;
- } else {
+ else
offset = iter->start - map->base_tsn - map->len;
- }
-
/* Search the overflow map. */
_sctp_tsnmap_find_gap_ack(map->overflow_map,
map->len,
map->len,
&started, &_start,
- &ended, &_end);
+ &ended, &_end);
}
/* The Gap Ack Block happens to end at the end of the
* bump the iterator forward.
*/
if (ended) {
- /* 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.
*/
- int gap = map->cumulative_tsn_ack_point
- - map->base_tsn;
-
+ int gap = map->cumulative_tsn_ack_point -
+ map->base_tsn;
+
*start = _start - gap;
*end = _end - gap;
-
+
/* Move the iterator forward. */
iter->start = map->cumulative_tsn_ack_point + *end + 1;
}
- return(ended);
-
-} /* sctp_tsnmap_next_gap_ack() */
+ return ended;
+}
/********************************************************************
* 2nd Level Abstractions
/* This private helper function updates the tsnmap buffers and
* the Cumulative TSN Ack Point.
*/
-static inline void
-_sctp_tsnmap_update(sctp_tsnmap_t *map)
+static void _sctp_tsnmap_update(sctp_tsnmap_t *map)
{
- uint32_t ctsn;
+ __u32 ctsn;
ctsn = map->cumulative_tsn_ack_point;
do {
ctsn++;
- if ( ctsn == map->overflow_tsn ) {
-
+ if (ctsn == map->overflow_tsn) {
/* Now tsn_map must have been all '1's,
* so we swap the map and check the overflow table
*/
-
- uint8_t *tmp = map->tsn_map;
- memset(tmp, 0, map->len);
- map->tsn_map = map->overflow_map;
- map->overflow_map = tmp;
+ __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;
+ map->base_tsn += map->len;
+ map->overflow_tsn += map->len;
}
- } while ( map->tsn_map[ctsn - map->base_tsn] );
+ } while (map->tsn_map[ctsn - map->base_tsn]);
map->cumulative_tsn_ack_point = ctsn - 1; /* Back up one. */
-
_sctp_tsnmap_update_pending_data(map);
+}
-} /* _sctp_tsnmap_update() */
-
-static inline void
-_sctp_tsnmap_update_pending_data(sctp_tsnmap_t *map)
+static void _sctp_tsnmap_update_pending_data(sctp_tsnmap_t *map)
{
- uint32_t cum_tsn = map->cumulative_tsn_ack_point;
- uint32_t max_tsn = map->max_tsn_seen;
- uint32_t base_tsn = map->base_tsn;
- uint16_t pending_data;
- int32_t gap, start, end, i;
+ __u32 cum_tsn = map->cumulative_tsn_ack_point;
+ __u32 max_tsn = map->max_tsn_seen;
+ __u32 base_tsn = map->base_tsn;
+ __u16 pending_data;
+ __s32 gap, start, end, i;
pending_data = max_tsn - cum_tsn;
gap = max_tsn - base_tsn;
- if (gap <= 0 || gap >= (map->len + map->len)) {
+ if (gap <= 0 || gap >= (map->len + map->len))
goto out;
- }
start = ((cum_tsn >= base_tsn) ? (cum_tsn - base_tsn + 1) : 0);
- end = ((gap > map->len ) ? map->len : gap + 1);
+ end = ((gap > map->len ) ? map->len : gap + 1);
- for (i = start; i < end; i++) {
- if (map->tsn_map[i]) { pending_data--; }
+ for (i = start; i < end; i++) {
+ if (map->tsn_map[i])
+ pending_data--;
}
- if (gap >= map->len) {
+ if (gap >= map->len) {
start = 0;
end = gap - map->len + 1;
- for (i = start; i < end; i++) {
- if (map->overflow_map[i]) { pending_data--; }
- }
+ for (i = start; i < end; i++) {
+ if (map->overflow_map[i])
+ pending_data--;
+ }
}
+
out:
map->pending_data = pending_data;
}
*
* The flags "started" and "ended" tell is if we found the beginning
* or (respectively) the end of a Gap Ack Block.
- *
*/
-
-static inline void
-_sctp_tsnmap_find_gap_ack(uint8_t *map, uint16_t off,
- uint16_t len, uint16_t base,
- int *started, uint16_t *start,
- int *ended, uint16_t *end)
+static void _sctp_tsnmap_find_gap_ack(__u8 *map, __u16 off,
+ __u16 len, __u16 base,
+ int *started, __u16 *start,
+ int *ended, __u16 *end)
{
int i = off;
-
+
/* Let's look through the entire array, but break out
* early if we have found the end of the Gap Ack Block.
*/
}
}
}
+
/* Look for the end. */
if (*started) {
/* We have found the start, let's find the
* end. If we find the end, break out.
*/
for (; i < len; i++) {
-
- if (!map[i]){
+ if (!map[i]) {
(*ended)++;
*end = base + i - 1;
break;
}
}
}
-
-} /* _sctp_tsnmap_next_gap_ack() */
+}
*/
static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_ulpevent.c,v 1.16 2002/08/21 18:34:04 jgrimm Exp $";
-#include <linux/config.h>
#include <linux/types.h>
#include <linux/skbuff.h>
#include <net/sctp/sctp_structs.h>
#include <net/sctp/sctp.h>
#include <net/sctp/sctp_sm.h>
+static void sctp_rcvmsg_rfree(struct sk_buff *skb);
+static void sctp_ulpevent_set_owner_r(struct sk_buff *skb,
+ sctp_association_t *asoc);
-static void
-sctp_rcvmsg_rfree(struct sk_buff *skb);
-static inline void
-sctp_ulpevent_set_owner_r(struct sk_buff *skb, sctp_association_t *asoc);
-
-
-/* Create a new sctp_ulpevent.
- */
-sctp_ulpevent_t *
-sctp_ulpevent_new(int size, int msg_flags, int priority)
+/* Create a new sctp_ulpevent. */
+sctp_ulpevent_t *sctp_ulpevent_new(int size, int msg_flags, int priority)
{
sctp_ulpevent_t *event;
struct sk_buff *skb;
-
- skb = alloc_skb(size, priority);
- if (NULL == skb) {
+ skb = alloc_skb(size, priority);
+ if (!skb)
goto fail;
- }
- event = (sctp_ulpevent_t *)skb->cb;
+ event = (sctp_ulpevent_t *) skb->cb;
event = sctp_ulpevent_init(event, skb, msg_flags);
-
- if (NULL == event) {
+ if (!event)
goto fail_init;
- }
event->malloced = 1;
return event;
- fail_init:
+fail_init:
kfree_skb(event->parent);
- fail:
+
+fail:
return NULL;
-
-} /* sctp_ulpevent_new() */
-
-/* Initialize an ULP event from an given skb. */
-sctp_ulpevent_t *
-sctp_ulpevent_init(sctp_ulpevent_t *event,
- struct sk_buff *parent,
- int msg_flags)
+}
+
+/* Initialize an ULP event from an given skb. */
+sctp_ulpevent_t *sctp_ulpevent_init(sctp_ulpevent_t *event,
+ struct sk_buff *parent,
+ int msg_flags)
{
memset(event, sizeof(sctp_ulpevent_t), 0x00);
-
event->msg_flags = msg_flags;
event->parent = parent;
event->malloced = 0;
return event;
+}
-} /* sctp_ulpevent_init() */
-
-
-
-/* Dispose of an event. */
-void
-sctp_ulpevent_free(sctp_ulpevent_t *event)
+/* Dispose of an event. */
+void sctp_ulpevent_free(sctp_ulpevent_t *event)
{
- if (event->malloced) {
+ if (event->malloced)
kfree_skb(event->parent);
- }
+}
-} /* sctp_ulpevent_free() */
-
-/* Is this a MSG_NOTIFICATION? */
-int
-sctp_ulpevent_is_notification(const sctp_ulpevent_t *event)
+/* Is this a MSG_NOTIFICATION? */
+int sctp_ulpevent_is_notification(const sctp_ulpevent_t *event)
{
return event->msg_flags & MSG_NOTIFICATION;
+}
-} /* sctp_ulpevent_is_notification() */
-
-
-/* Create and initialize an SCTP_ASSOC_CHANGE event.
- *
+/* Create and initialize an SCTP_ASSOC_CHANGE event.
+ *
* 5.3.1.1 SCTP_ASSOC_CHANGE
- *
+ *
* Communication notifications inform the ULP that an SCTP association
* has either begun or ended. The identifier for a new association is
* provided by this notification.
- *
+ *
* Note: There is no field checking here. If a field is unused it will be
* zero'd out.
*/
-sctp_ulpevent_t *
-sctp_ulpevent_make_assoc_change(const sctp_association_t *asoc,
- uint16_t flags,
- uint16_t state,
- uint16_t error,
- uint16_t outbound,
- uint16_t inbound,
- int priority)
+sctp_ulpevent_t *sctp_ulpevent_make_assoc_change(const sctp_association_t *asoc,
+ __u16 flags,
+ __u16 state,
+ __u16 error,
+ __u16 outbound,
+ __u16 inbound,
+ int priority)
{
-
sctp_ulpevent_t *event;
- struct sctp_assoc_change *sac;
+ struct sctp_assoc_change *sac;
event = sctp_ulpevent_new(sizeof(struct sctp_assoc_change),
MSG_NOTIFICATION,
priority);
-
- if (NULL == event) {
+ if (!event)
goto fail;
- }
sac = (struct sctp_assoc_change *)
- skb_put(event->parent,
+ skb_put(event->parent,
sizeof(struct sctp_assoc_change));
- /* Socket Extensions for SCTP
+ /* Socket Extensions for SCTP
* 5.3.1.1 SCTP_ASSOC_CHANGE
- *
+ *
* sac_type:
* It should be SCTP_ASSOC_CHANGE.
*/
-
sac->sac_type = SCTP_ASSOC_CHANGE;
-
- /* Socket Extensions for SCTP
+ /* Socket Extensions for SCTP
* 5.3.1.1 SCTP_ASSOC_CHANGE
- *
+ *
* sac_state: 32 bits (signed integer)
* This field holds one of a number of values that communicate the
- * event that happened to the association.
+ * event that happened to the association.
*/
-
sac->sac_state = state;
- /* Socket Extensions for SCTP
+ /* Socket Extensions for SCTP
* 5.3.1.1 SCTP_ASSOC_CHANGE
- *
+ *
* sac_flags: 16 bits (unsigned integer)
* Currently unused.
- *
*/
sac->sac_flags = 0;
- /* Socket Extensions for SCTP
+ /* Socket Extensions for SCTP
* 5.3.1.1 SCTP_ASSOC_CHANGE
- *
- * sac_length: sizeof (uint32_t)
+ *
+ * sac_length: sizeof (__u32)
* This field is the total length of the notification data, including
* the notification header.
*/
-
sac->sac_length = sizeof(struct sctp_assoc_change);
-
- /* Socket Extensions for SCTP
+ /* Socket Extensions for SCTP
* 5.3.1.1 SCTP_ASSOC_CHANGE
- *
+ *
* sac_error: 32 bits (signed integer)
- *
+ *
* If the state was reached due to a error condition (e.g.
* COMMUNICATION_LOST) any relevant error information is available in
* this field. This corresponds to the protocol error codes defined in
* [SCTP].
*/
-
sac->sac_error = error;
- /* Socket Extensions for SCTP
+ /* Socket Extensions for SCTP
* 5.3.1.1 SCTP_ASSOC_CHANGE
- *
+ *
* sac_outbound_streams: 16 bits (unsigned integer)
* sac_inbound_streams: 16 bits (unsigned integer)
- *
+ *
* The maximum number of streams allowed in each direction are
* available in sac_outbound_streams and sac_inbound streams.
*/
sac->sac_outbound_streams = outbound;
sac->sac_inbound_streams = inbound;
-
-
- /* Socket Extensions for SCTP
+
+ /* Socket Extensions for SCTP
* 5.3.1.1 SCTP_ASSOC_CHANGE
*
* sac_assoc_id: sizeof (sctp_assoc_t)
- *
+ *
* The association id field, holds the identifier for the association.
* All notifications for a given association have the same association
* identifier. For TCP style socket, this field is ignored.
sac->sac_assoc_id = sctp_assoc2id(asoc);
return event;
-fail:
- return NULL;
-
-} /* sctp_ulpevent_make_assoc_change() */
+fail:
+ return NULL;
+}
/* Create and initialize an SCTP_PEER_ADDR_CHANGE event.
*
* 5.3.1.2 SCTP_PEER_ADDR_CHANGE
*
* When a destination address on a multi-homed peer encounters a change
- * an interface details event is sent.
- *
+ * an interface details event is sent.
*/
-sctp_ulpevent_t *
-sctp_ulpevent_make_peer_addr_change(const sctp_association_t *asoc,
- const struct sockaddr_storage *aaddr,
- int flags,
- int state,
- int error,
- int priority)
+sctp_ulpevent_t *sctp_ulpevent_make_peer_addr_change(
+ const sctp_association_t *asoc,
+ const struct sockaddr_storage *aaddr,
+ int flags,
+ int state,
+ int error,
+ int priority)
{
sctp_ulpevent_t *event;
struct sctp_paddr_change *spc;
-
+
event = sctp_ulpevent_new(sizeof(struct sctp_paddr_change),
MSG_NOTIFICATION,
priority);
-
- if (NULL == event) {
+ if (!event)
goto fail;
- }
-
+
spc = (struct sctp_paddr_change *)
- skb_put(event->parent,
- sizeof(struct sctp_paddr_change));
-
- /* Sockets API Extensions for SCTP
+ skb_put(event->parent,
+ sizeof(struct sctp_paddr_change));
+
+ /* Sockets API Extensions for SCTP
* Section 5.3.1.2 SCTP_PEER_ADDR_CHANGE
- *
+ *
* spc_type:
- *
+ *
* It should be SCTP_PEER_ADDR_CHANGE.
*/
spc->spc_type = SCTP_PEER_ADDR_CHANGE;
- /* Sockets API Extensions for SCTP
+ /* Sockets API Extensions for SCTP
* Section 5.3.1.2 SCTP_PEER_ADDR_CHANGE
*
- * spc_length: sizeof (uint32_t)
- *
+ * spc_length: sizeof (__u32)
+ *
* This field is the total length of the notification data, including
* the notification header.
*/
- spc->spc_length = sizeof(struct sctp_paddr_change);
+ spc->spc_length = sizeof(struct sctp_paddr_change);
- /* Sockets API Extensions for SCTP
+ /* Sockets API Extensions for SCTP
* Section 5.3.1.2 SCTP_PEER_ADDR_CHANGE
*
* spc_flags: 16 bits (unsigned integer)
*/
spc->spc_flags = 0;
- /* Sockets API Extensions for SCTP
+ /* Sockets API Extensions for SCTP
* Section 5.3.1.2 SCTP_PEER_ADDR_CHANGE
*
* spc_state: 32 bits (signed integer)
*/
spc->spc_state = state;
- /* Sockets API Extensions for SCTP
+ /* Sockets API Extensions for SCTP
* Section 5.3.1.2 SCTP_PEER_ADDR_CHANGE
*
* spc_error: 32 bits (signed integer)
*/
spc->spc_error = error;
- /* Socket Extensions for SCTP
+ /* Socket Extensions for SCTP
* 5.3.1.1 SCTP_ASSOC_CHANGE
*
* sac_assoc_id: sizeof (sctp_assoc_t)
- *
+ *
* The association id field, holds the identifier for the association.
* All notifications for a given association have the same association
* identifier. For TCP style socket, this field is ignored.
*/
spc->spc_assoc_id = sctp_assoc2id(asoc);
- /* Sockets API Extensions for SCTP
+ /* Sockets API Extensions for SCTP
* Section 5.3.1.2 SCTP_PEER_ADDR_CHANGE
*
* spc_aaddr: sizeof (struct sockaddr_storage)
*/
memcpy(&spc->spc_aaddr, aaddr, sizeof(struct sockaddr_storage));
-
return event;
+
fail:
return NULL;
-
-} /* sctp_ulpevent_make_peer_addr_change() */
+}
/* Create and initialize an SCTP_REMOTE_ERROR notification.
- *
+ *
* Note: This assumes that the chunk->skb->data already points to the
- * operation error payload.
+ * operation error payload.
*
* Socket Extensions for SCTP - draft-01
- * 5.3.1.3 SCTP_REMOTE_ERROR
+ * 5.3.1.3 SCTP_REMOTE_ERROR
*
* A remote peer may send an Operational Error message to its peer.
* This message indicates a variety of error conditions on an
* included in a SCTP_REMOTE_ERROR event. Please refer to the SCTP
* specification [SCTP] and any extensions for a list of possible
* error formats.
- *
*/
-sctp_ulpevent_t *
-sctp_ulpevent_make_remote_error(const sctp_association_t *asoc,
- sctp_chunk_t *chunk,
- uint16_t flags,
- int priority)
+sctp_ulpevent_t *sctp_ulpevent_make_remote_error(const sctp_association_t *asoc,
+ sctp_chunk_t *chunk,
+ __u16 flags,
+ int priority)
{
sctp_ulpevent_t *event;
- struct sctp_remote_error *sre;
+ struct sctp_remote_error *sre;
struct sk_buff *skb;
sctp_errhdr_t *ch;
- uint16_t cause;
+ __u16 cause;
int elen;
-
+
ch = (sctp_errhdr_t *)(chunk->skb->data);
cause = ch->cause;
elen = ntohs(ch->length) - sizeof(sctp_errhdr_t);
- /* Pull off the ERROR header. */
+ /* Pull off the ERROR header. */
skb_pull(chunk->skb, sizeof(sctp_errhdr_t));
-
- /* Copy the skb to a new skb with room for us to prepend
- * notification with. */
- skb = skb_copy_expand(chunk->skb,
+
+ /* Copy the skb to a new skb with room for us to prepend
+ * notification with.
+ */
+ skb = skb_copy_expand(chunk->skb,
sizeof(struct sctp_remote_error), /* headroom */
0, /* tailroom */
priority);
- /* Pull off the rest of the cause TLV from the chunk. */
+ /* Pull off the rest of the cause TLV from the chunk. */
skb_pull(chunk->skb, elen);
-
- if (NULL == skb) {
+ if (!skb)
goto fail;
- }
- /* Embed the event fields inside the cloned skb. */
- event = (sctp_ulpevent_t *)skb->cb;
- event = sctp_ulpevent_init(event,
+ /* Embed the event fields inside the cloned skb. */
+ event = (sctp_ulpevent_t *) skb->cb;
+ event = sctp_ulpevent_init(event,
skb,
MSG_NOTIFICATION);
- if (NULL == event) {
+ if (!event)
goto fail;
- }
-
+
event->malloced = 1;
-
sre = (struct sctp_remote_error *)
skb_push(skb, sizeof(struct sctp_remote_error));
- /* Trim the buffer to the right length. */
+ /* Trim the buffer to the right length. */
skb_trim(skb, sizeof(struct sctp_remote_error) + elen);
- /* Socket Extensions for SCTP
- * 5.3.1.3 SCTP_REMOTE_ERROR
- *
+ /* Socket Extensions for SCTP
+ * 5.3.1.3 SCTP_REMOTE_ERROR
+ *
* sre_type:
* It should be SCTP_REMOTE_ERROR.
*/
sre->sre_type = SCTP_REMOTE_ERROR;
- /*
- * Socket Extensions for SCTP
- * 5.3.1.3 SCTP_REMOTE_ERROR
- *
+ /*
+ * Socket Extensions for SCTP
+ * 5.3.1.3 SCTP_REMOTE_ERROR
+ *
* sre_flags: 16 bits (unsigned integer)
* Currently unused.
*/
sre->sre_flags = 0;
- /* Socket Extensions for SCTP
- * 5.3.1.3 SCTP_REMOTE_ERROR
- *
- * sre_length: sizeof (uint32_t)
- *
- * This field is the total length of the notification data,
+ /* Socket Extensions for SCTP
+ * 5.3.1.3 SCTP_REMOTE_ERROR
+ *
+ * sre_length: sizeof (__u32)
+ *
+ * This field is the total length of the notification data,
* including the notification header.
*/
sre->sre_length = skb->len;
- /*
- * Socket Extensions for SCTP
- * 5.3.1.3 SCTP_REMOTE_ERROR
- *
+ /* Socket Extensions for SCTP
+ * 5.3.1.3 SCTP_REMOTE_ERROR
+ *
* sre_error: 16 bits (unsigned integer)
* This value represents one of the Operational Error causes defined in
* the SCTP specification, in network byte order.
*/
- sre->sre_error = cause;
+ sre->sre_error = cause;
-
- /*
- * Socket Extensions for SCTP
- * 5.3.1.3 SCTP_REMOTE_ERROR
- *
- * sre_assoc_id: sizeof (sctp_assoc_t)
+ /* Socket Extensions for SCTP
+ * 5.3.1.3 SCTP_REMOTE_ERROR
*
+ * sre_assoc_id: sizeof (sctp_assoc_t)
*
* The association id field, holds the identifier for the association.
* All notifications for a given association have the same association
* identifier. For TCP style socket, this field is ignored.
*/
-
- sre->sre_assoc_id = sctp_assoc2id(asoc);
+ sre->sre_assoc_id = sctp_assoc2id(asoc);
+ return event;
- return event;
fail:
- return NULL;
-
-} /* sctp_ulpevent_make_remote_error () */
+ return NULL;
+}
/* Create and initialize a SCTP_SEND_FAILED notification.
- *
+ *
* Socket Extensions for SCTP - draft-01
* 5.3.1.4 SCTP_SEND_FAILED
*/
-sctp_ulpevent_t *
-sctp_ulpevent_make_send_failed(const sctp_association_t *asoc,
- sctp_chunk_t *chunk,
- uint16_t flags,
- uint32_t error,
- int priority)
+sctp_ulpevent_t *sctp_ulpevent_make_send_failed(const sctp_association_t *asoc,
+ sctp_chunk_t *chunk,
+ __u16 flags,
+ __u32 error,
+ int priority)
{
sctp_ulpevent_t *event;
- struct sctp_send_failed *ssf;
+ struct sctp_send_failed *ssf;
struct sk_buff *skb;
- /* Make skb with more room so we can prepend notification. */
- skb = skb_copy_expand(chunk->skb,
+ /* Make skb with more room so we can prepend notification. */
+ skb = skb_copy_expand(chunk->skb,
sizeof(struct sctp_send_failed), /* headroom */
0, /* tailroom */
priority);
-
- if (NULL == skb) {
+ if (!skb)
goto fail;
- }
- /* Pull off the common chunk header and DATA header. */
+ /* Pull off the common chunk header and DATA header. */
skb_pull(skb, sizeof(sctp_data_chunk_t));
- /* Embed the event fields inside the cloned skb. */
- event = (sctp_ulpevent_t *)skb->cb;
+ /* Embed the event fields inside the cloned skb. */
+ event = (sctp_ulpevent_t *) skb->cb;
event = sctp_ulpevent_init(event, skb, MSG_NOTIFICATION);
- if (NULL == event) { goto fail; }
+ if (!event)
+ goto fail;
- /* Mark as malloced, even though the constructor was not
- * called.
+ /* Mark as malloced, even though the constructor was not
+ * called.
*/
event->malloced = 1;
-
+
ssf = (struct sctp_send_failed *)
skb_push(skb, sizeof(struct sctp_send_failed));
- /* Socket Extensions for SCTP
+ /* Socket Extensions for SCTP
* 5.3.1.4 SCTP_SEND_FAILED
*
* ssf_type:
- * It should be SCTP_SEND_FAILED.
+ * It should be SCTP_SEND_FAILED.
*/
ssf->ssf_type = SCTP_SEND_FAILED;
- /* Socket Extensions for SCTP
+ /* Socket Extensions for SCTP
* 5.3.1.4 SCTP_SEND_FAILED
- *
+ *
* ssf_flags: 16 bits (unsigned integer)
* The flag value will take one of the following values
*
* SCTP_DATA_UNSENT - Indicates that the data was never put on
- * the wire.
- *
+ * the wire.
+ *
* SCTP_DATA_SENT - Indicates that the data was put on the wire.
- * Note that this does not necessarily mean that the
- * data was (or was not) successfully delivered.
+ * Note that this does not necessarily mean that the
+ * data was (or was not) successfully delivered.
*/
ssf->ssf_flags = flags;
- /* Socket Extensions for SCTP
+ /* Socket Extensions for SCTP
* 5.3.1.4 SCTP_SEND_FAILED
*
- * ssf_length: sizeof (uint32_t)
+ * ssf_length: sizeof (__u32)
* This field is the total length of the notification data, including
* the notification header.
*/
ssf->ssf_length = skb->len;
-
- /* Socket Extensions for SCTP
+
+ /* Socket Extensions for SCTP
* 5.3.1.4 SCTP_SEND_FAILED
*
* ssf_error: 16 bits (unsigned integer)
*/
ssf->ssf_error = error;
- /* Socket Extensions for SCTP
+ /* Socket Extensions for SCTP
* 5.3.1.4 SCTP_SEND_FAILED
*
* ssf_info: sizeof (struct sctp_sndrcvinfo)
* The original send information associated with the undelivered
- * message.
+ * message.
*/
- memcpy(&ssf->ssf_info,
- &chunk->sinfo,
+ memcpy(&ssf->ssf_info,
+ &chunk->sinfo,
sizeof(struct sctp_sndrcvinfo));
-
- /* Socket Extensions for SCTP
+
+ /* Socket Extensions for SCTP
* 5.3.1.4 SCTP_SEND_FAILED
*
* ssf_assoc_id: sizeof (sctp_assoc_t)
* ignored.
*/
ssf->ssf_assoc_id = sctp_assoc2id(asoc);
-
-
+
return event;
-fail:
+
+fail:
return NULL;
+}
-} /* sctp_ulpevent_make_send_failed () */
-
/* Create and initialize a SCTP_SHUTDOWN_EVENT notification.
- *
+ *
* Socket Extensions for SCTP - draft-01
* 5.3.1.5 SCTP_SHUTDOWN_EVENT
*/
-sctp_ulpevent_t *
-sctp_ulpevent_make_shutdown_event(const sctp_association_t *asoc,
- uint16_t flags,
- int priority)
+sctp_ulpevent_t *sctp_ulpevent_make_shutdown_event(const sctp_association_t *asoc,
+ __u16 flags,
+ int priority)
{
sctp_ulpevent_t *event;
- struct sctp_shutdown_event *sse;
+ struct sctp_shutdown_event *sse;
event = sctp_ulpevent_new(sizeof(struct sctp_assoc_change),
MSG_NOTIFICATION,
priority);
-
- if (NULL == event) {
+ if (!event)
goto fail;
- }
sse = (struct sctp_shutdown_event *)
- skb_put(event->parent,
+ skb_put(event->parent,
sizeof(struct sctp_shutdown_event));
- /* Socket Extensions for SCTP
+ /* Socket Extensions for SCTP
* 5.3.1.5 SCTP_SHUTDOWN_EVENT
- *
+ *
* sse_type
* It should be SCTP_SHUTDOWN_EVENT
*/
sse->sse_type = SCTP_SHUTDOWN_EVENT;
- /* Socket Extensions for SCTP
+ /* Socket Extensions for SCTP
* 5.3.1.5 SCTP_SHUTDOWN_EVENT
- *
+ *
* sse_flags: 16 bits (unsigned integer)
* Currently unused.
*/
sse->sse_flags = 0;
- /* Socket Extensions for SCTP
+ /* Socket Extensions for SCTP
* 5.3.1.5 SCTP_SHUTDOWN_EVENT
*
- * sse_length: sizeof (uint32_t)
+ * sse_length: sizeof (__u32)
* This field is the total length of the notification data, including
* the notification header.
*/
sse->sse_length = sizeof(struct sctp_shutdown_event);
- /* Socket Extensions for SCTP
+ /* Socket Extensions for SCTP
* 5.3.1.5 SCTP_SHUTDOWN_EVENT
*
* sse_assoc_id: sizeof (sctp_assoc_t)
sse->sse_assoc_id = sctp_assoc2id(asoc);
return event;
-fail:
- return NULL;
-
-} /* sctp_ulpevent_make_shutdown_event () */
+fail:
+ return NULL;
+}
/* A message has been received. Package this message as a notification
* to pass it to the upper layers. Go ahead and calculate the sndrcvinfo
* even if filtered out later.
- *
+ *
* Socket Extensions for SCTP - draft-01
* 5.2.2 SCTP Header Information Structure (SCTP_SNDRCV)
- *
- */
-sctp_ulpevent_t *
-sctp_ulpevent_make_rcvmsg(sctp_association_t *asoc,
- sctp_chunk_t *chunk,
- int priority)
+ */
+sctp_ulpevent_t *sctp_ulpevent_make_rcvmsg(sctp_association_t *asoc,
+ sctp_chunk_t *chunk,
+ int priority)
{
-
sctp_ulpevent_t *event;
struct sctp_sndrcvinfo *info;
struct sk_buff *skb;
- size_t padding, len;
+ size_t padding, len;
- /* Clone the original skb, sharing the data. */
+ /* Clone the original skb, sharing the data. */
skb = skb_clone(chunk->skb, priority);
- if (NULL == skb) {
+ if (!skb)
goto fail;
- }
-
- /* 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.
*
- * RFC 2960 - Section 3.2 Chunk Field Descriptions
+ * RFC 2960 - Section 3.2 Chunk Field Descriptions
*
* The total length of a chunk(including Type, Length and Value fields)
* MUST be a multiple of 4 bytes. If the length of the chunk is not a
* bytes and this padding is not included in the chunk length field.
* The sender should never pad with more than 3 bytes. The receiver
* MUST ignore the padding bytes.
- *
- */
+ */
len = ntohs(chunk->chunk_hdr->length);
padding = WORD_ROUND(len) - len;
/* Fixup cloned skb with just this chunks data. */
skb_trim(skb, chunk->chunk_end - padding - skb->data);
- /* Set up a destructor to do rwnd accounting. */
+ /* Set up a destructor to do rwnd accounting. */
sctp_ulpevent_set_owner_r(skb, asoc);
- /* Embed the event fields inside the cloned skb. */
- event = (sctp_ulpevent_t *)skb->cb;
- /* Initialize event with flags 0. */
- event = sctp_ulpevent_init(event, skb, 0);
+ /* Embed the event fields inside the cloned skb. */
+ event = (sctp_ulpevent_t *) skb->cb;
- if (NULL == event) {
+ /* Initialize event with flags 0. */
+ event = sctp_ulpevent_init(event, skb, 0);
+ if (!event)
goto fail_init;
- }
event->malloced = 1;
- info = (struct sctp_sndrcvinfo *)&event->sndrcvinfo;
-
- /*
- * Sockets API Extensions for SCTP
+ info = (struct sctp_sndrcvinfo *) &event->sndrcvinfo;
+
+ /* Sockets API Extensions for SCTP
* Section 5.2.2 SCTP Header Information Structure (SCTP_SNDRCV)
*
* sinfo_stream: 16 bits (unsigned integer)
* For recvmsg() the SCTP stack places the message's stream number in
* this value.
*/
-
- info->sinfo_stream = ntohs(chunk->subh.data_hdr->stream);
+ info->sinfo_stream = ntohs(chunk->subh.data_hdr->stream);
- /* Sockets API Extensions for SCTP
+ /* Sockets API Extensions for SCTP
* Section 5.2.2 SCTP Header Information Structure (SCTP_SNDRCV)
*
* sinfo_ssn: 16 bits (unsigned integer)
* messages this is the same number for all deliveries of the message
* (if more than one recvmsg() is needed to read the message).
*/
+ info->sinfo_ssn = ntohs(chunk->subh.data_hdr->ssn);
- info->sinfo_ssn = ntohs(chunk->subh.data_hdr->ssn);
-
- /* Sockets API Extensions for SCTP
+ /* Sockets API Extensions for SCTP
* Section 5.2.2 SCTP Header Information Structure (SCTP_SNDRCV)
*
- * sinfo_ppid:32 bits (unsigned integer)
+ * sinfo_ppid: 32 bits (unsigned integer)
*
* In recvmsg() this value is
* the same information that was passed by the upper layer in the peer
*/
info->sinfo_ppid = ntohl(chunk->subh.data_hdr->ppid);
- /* Sockets API Extensions for SCTP
+ /* Sockets API Extensions for SCTP
* Section 5.2.2 SCTP Header Information Structure (SCTP_SNDRCV)
*
- * sinfo_flags: 16 bits (unsigned integer)
- *
+ * sinfo_flags: 16 bits (unsigned integer)
+ *
* This field may contain any of the following flags and is composed of
* a bitwise OR of these values.
- *
+ *
* recvmsg() flags:
- *
+ *
* MSG_UNORDERED - This flag is present when the message was sent
- * non-ordered.
+ * non-ordered.
*/
-
- if (chunk->chunk_hdr->flags & SCTP_DATA_UNORDERED) {
+ if (chunk->chunk_hdr->flags & SCTP_DATA_UNORDERED)
info->sinfo_flags |= MSG_UNORDERED;
- }
/* FIXME: For reassembly, we need to have the fragmentation bits.
* This really does not belong in the event structure, but
*/
event->chunk_flags = chunk->chunk_hdr->flags;
- /* With -04 draft, tsn moves into sndrcvinfo. */
+ /* With -04 draft, tsn moves into sndrcvinfo. */
info->sinfo_tsn = ntohl(chunk->subh.data_hdr->tsn);
-
+
/* Context is not used on receive. */
- info->sinfo_context = 0;
+ info->sinfo_context = 0;
- /* Sockets API Extensions for SCTP
+ /* Sockets API Extensions for SCTP
* Section 5.2.2 SCTP Header Information Structure (SCTP_SNDRCV)
- *
+ *
* sinfo_assoc_id: sizeof (sctp_assoc_t)
- *
+ *
* The association handle field, sinfo_assoc_id, holds the identifier
* for the association announced in the COMMUNICATION_UP notification.
* All notifications for a given association have the same identifier.
* Ignored for TCP-style sockets.
*/
-
- info->sinfo_assoc_id = sctp_assoc2id(asoc);
+ info->sinfo_assoc_id = sctp_assoc2id(asoc);
return event;
fail_init:
kfree_skb(skb);
-fail:
- return NULL;
-
-} /* sctp_ulpevent_make_rcvmsg() */
+
+fail:
+ return NULL;
+}
/* Return the notification type, assuming this is a notification
* event.
*/
-uint16_t
-sctp_ulpevent_get_notification_type(const sctp_ulpevent_t *event)
+__u16 sctp_ulpevent_get_notification_type(const sctp_ulpevent_t *event)
{
union sctp_notification *notification;
- notification = (union sctp_notification *)event->parent->data;
-
+ notification = (union sctp_notification *) event->parent->data;
return notification->h.sn_type;
-
-} /* sctp_ulpevent_get_notification_type() */
+}
/* Copy out the sndrcvinfo into a msghdr. */
-void
-sctp_ulpevent_read_sndrcvinfo(const sctp_ulpevent_t *event,
- struct msghdr *msghdr)
+void sctp_ulpevent_read_sndrcvinfo(const sctp_ulpevent_t *event,
+ struct msghdr *msghdr)
{
-
if (!sctp_ulpevent_is_notification(event)) {
- put_cmsg(msghdr, IPPROTO_SCTP, SCTP_SNDRCV,
- sizeof(struct sctp_sndrcvinfo),
- (void *)&event->sndrcvinfo);
+ put_cmsg(msghdr, IPPROTO_SCTP, SCTP_SNDRCV,
+ sizeof(struct sctp_sndrcvinfo),
+ (void *) &event->sndrcvinfo);
}
+}
-} /* sctp_ulpevent_read_sndrcvinfo() */
-
-/* Do accounting for bytes just read by user. */
-static void
-sctp_rcvmsg_rfree(struct sk_buff *skb)
+/* Do accounting for bytes just read by user. */
+static void sctp_rcvmsg_rfree(struct sk_buff *skb)
{
sctp_association_t *asoc;
- sctp_ulpevent_t *event;
+ sctp_ulpevent_t *event;
- /* Current stack structures assume that the rcv buffer is
- * per socket. For UDP style sockets this is not true as
- * multiple associations may be on a single UDP-style socket.
- * Use the local private area of the skb to track the owning
+ /* Current stack structures assume that the rcv buffer is
+ * per socket. For UDP style sockets this is not true as
+ * multiple associations may be on a single UDP-style socket.
+ * Use the local private area of the skb to track the owning
* association.
*/
-
- event = (sctp_ulpevent_t *)skb->cb;
+ event = (sctp_ulpevent_t *) skb->cb;
asoc = event->asoc;
-
if (asoc->rwnd_over) {
if (asoc->rwnd_over >= skb->len) {
asoc->rwnd_over -= skb->len;
asoc->rwnd += skb->len;
}
- SCTP_DEBUG_PRINTK("rwnd increased by %d to (%u, %u)\n",
- skb->len, asoc->rwnd, asoc->rwnd_over);
+ SCTP_DEBUG_PRINTK("rwnd increased by %d to (%u, %u)\n",
+ skb->len, asoc->rwnd, asoc->rwnd_over);
sctp_association_put(asoc);
-
-} /* sctp_rcvmsg_rfree() */
-
+}
/* Charge receive window for bytes recieved. */
-static inline void
-sctp_ulpevent_set_owner_r(struct sk_buff *skb, sctp_association_t *asoc)
+static void sctp_ulpevent_set_owner_r(struct sk_buff *skb, sctp_association_t *asoc)
{
sctp_ulpevent_t *event;
- /* The current stack structures assume that the rcv buffer is
- * per socket. For UDP-style sockets this is not true as
- * multiple associations may be on a single UDP-style socket.
- * We use the local private area of the skb to track the owning
+ /* The current stack structures assume that the rcv buffer is
+ * per socket. For UDP-style sockets this is not true as
+ * multiple associations may be on a single UDP-style socket.
+ * We use the local private area of the skb to track the owning
* association.
*/
sctp_association_hold(asoc);
skb->sk = asoc->base.sk;
- event = (sctp_ulpevent_t *)skb->cb;
+ event = (sctp_ulpevent_t *) skb->cb;
event->asoc = asoc;
skb->destructor = sctp_rcvmsg_rfree;
asoc->rwnd = 0;
}
- SCTP_DEBUG_PRINTK("rwnd decreased by %d to (%u, %u)\n",
- skb->len, asoc->rwnd, asoc->rwnd_over);
-
-} /* sctp_ulpevent_set_owner_r() */
-
-
+ SCTP_DEBUG_PRINTK("rwnd decreased by %d to (%u, %u)\n",
+ skb->len, asoc->rwnd, asoc->rwnd_over);
+}
*/
static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_ulpqueue.c,v 1.14 2002/08/21 18:34:04 jgrimm Exp $";
-#include <linux/config.h>
#include <linux/types.h>
#include <linux/skbuff.h>
#include <net/sock.h>
#include <net/sctp/sctp.h>
#include <net/sctp/sctp_sm.h>
-/* Forward declarations for internal helpers. */
-static inline sctp_ulpevent_t *
-sctp_ulpqueue_reasm(sctp_ulpqueue_t *ulpq, sctp_ulpevent_t *event);
-static inline sctp_ulpevent_t *
-sctp_ulpqueue_order(sctp_ulpqueue_t *ulpq, sctp_ulpevent_t *event);
-
+/* Forward declarations for internal helpers. */
+static inline sctp_ulpevent_t * sctp_ulpqueue_reasm(sctp_ulpqueue_t *ulpq,
+ sctp_ulpevent_t *event);
+static inline sctp_ulpevent_t *sctp_ulpqueue_order(sctp_ulpqueue_t *ulpq,
+ sctp_ulpevent_t *event);
/* 1st Level Abstractions */
-/* Create a new ULP queue.
- */
-sctp_ulpqueue_t *
-sctp_ulpqueue_new(sctp_association_t *asoc, uint16_t inbound, int priority)
+/* Create a new ULP queue. */
+sctp_ulpqueue_t *sctp_ulpqueue_new(sctp_association_t *asoc,
+ __u16 inbound, int priority)
{
sctp_ulpqueue_t *ulpq;
size_t size;
-
+
/* Today, there is only a fixed size of storage needed for
- * stream support, but make the interfaces acceptable for
+ * stream support, but make the interfaces acceptable for
* the future.
- */
+ */
size = sizeof(sctp_ulpqueue_t)+sctp_ulpqueue_storage_size(inbound);
ulpq = kmalloc(size, priority);
-
- if (NULL == ulpq) {
+ if (!ulpq)
goto fail;
- }
-
- if (NULL == sctp_ulpqueue_init(ulpq, asoc, inbound)) {
+ if (!sctp_ulpqueue_init(ulpq, asoc, inbound))
goto fail_init;
- }
-
ulpq->malloced = 1;
return ulpq;
-
+
fail_init:
kfree(ulpq);
+
fail:
return NULL;
-
-} /* sctp_ulpqueue_new() */
+}
-/* Initialize a ULP queue from a block of memory. */
-sctp_ulpqueue_t *
-sctp_ulpqueue_init(sctp_ulpqueue_t *ulpq, sctp_association_t *asoc,
- uint16_t inbound)
+/* Initialize a ULP queue from a block of memory. */
+sctp_ulpqueue_t *sctp_ulpqueue_init(sctp_ulpqueue_t *ulpq,
+ sctp_association_t *asoc,
+ __u16 inbound)
{
- memset(ulpq,
- sizeof(sctp_ulpqueue_t) + sctp_ulpqueue_storage_size(inbound),
+ memset(ulpq,
+ sizeof(sctp_ulpqueue_t) + sctp_ulpqueue_storage_size(inbound),
0x00);
ulpq->asoc = asoc;
ulpq->malloced = 0;
return ulpq;
+}
-} /* sctp_ulpqueue_init() */
-
-/* Flush the reassembly and ordering queues. */
-void
-sctp_ulpqueue_flush(sctp_ulpqueue_t *ulpq)
+/* Flush the reassembly and ordering queues. */
+void sctp_ulpqueue_flush(sctp_ulpqueue_t *ulpq)
{
struct sk_buff *skb;
sctp_ulpevent_t *event;
-
- while((skb = skb_dequeue(&ulpq->lobby))) {
- event = (sctp_ulpevent_t *)skb->cb;
- sctp_ulpevent_free(event);
- }
- while((skb = skb_dequeue(&ulpq->reasm))) {
- event = (sctp_ulpevent_t *)skb->cb;
- sctp_ulpevent_free(event);
+ while ((skb = skb_dequeue(&ulpq->lobby))) {
+ event = (sctp_ulpevent_t *) skb->cb;
+ sctp_ulpevent_free(event);
}
-} /* sctp_ulpqueue_flush() */
-
+ while ((skb = skb_dequeue(&ulpq->reasm))) {
+ event = (sctp_ulpevent_t *) skb->cb;
+ sctp_ulpevent_free(event);
+ }
+}
-/* Dispose of a ulpqueue. */
-void
-sctp_ulpqueue_free(sctp_ulpqueue_t *ulpq)
+/* Dispose of a ulpqueue. */
+void sctp_ulpqueue_free(sctp_ulpqueue_t *ulpq)
{
-
sctp_ulpqueue_flush(ulpq);
- if (ulpq->malloced) {
+ if (ulpq->malloced)
kfree(ulpq);
- }
+}
-} /* sctp_ulpqueue_free() */
-
-
-/* Process an incoming DATA chunk. */
-int
-sctp_ulpqueue_tail_data(sctp_ulpqueue_t *ulpq, sctp_chunk_t *chunk,
- int priority)
+/* Process an incoming DATA chunk. */
+int sctp_ulpqueue_tail_data(sctp_ulpqueue_t *ulpq, sctp_chunk_t *chunk,
+ int priority)
{
struct sk_buff_head temp;
sctp_data_chunk_t *hdr;
sctp_ulpevent_t *event;
- hdr = (sctp_data_chunk_t *)chunk->chunk_hdr;
-
- /* FIXME: Instead of event being the skb clone, we really should
- * have a new skb based chunk structure that we can convert to
+ hdr = (sctp_data_chunk_t *) chunk->chunk_hdr;
+
+ /* FIXME: Instead of event being the skb clone, we really should
+ * have a new skb based chunk structure that we can convert to
* an event. Temporarily, I'm carrying a few chunk fields in
* the event to allow reassembly. Its too painful to change
* everything at once. --jgrimm
*/
-
event = sctp_ulpevent_make_rcvmsg(chunk->asoc, chunk, priority);
-
- if (!event) {
+ if (!event)
return -ENOMEM;
- }
- /* Do reassembly if needed. */
+ /* Do reassembly if needed. */
event = sctp_ulpqueue_reasm(ulpq, event);
- /* Do ordering if needed. */
- if (NULL != event) {
- /* Create a temporary list to collect chunks on. */
+ /* Do ordering if needed. */
+ if (event) {
+ /* Create a temporary list to collect chunks on. */
skb_queue_head_init(&temp);
skb_queue_tail(&temp, event->parent);
event = sctp_ulpqueue_order(ulpq, event);
}
- /* Send event to the ULP. */
- if (NULL != event) {
- sctp_ulpqueue_tail_event(ulpq, event);
- }
+ /* Send event to the ULP. */
+ if (event)
+ sctp_ulpqueue_tail_event(ulpq, event);
return 0;
+}
-} /* sctp_ulpqueue_tail_data() */
-
-
-/* Add a new event for propogation to the ULP. */
-int
-sctp_ulpqueue_tail_event(sctp_ulpqueue_t *ulpq, sctp_ulpevent_t *event)
+/* Add a new event for propogation to the ULP. */
+int sctp_ulpqueue_tail_event(sctp_ulpqueue_t *ulpq, sctp_ulpevent_t *event)
{
- struct sock *sk;
- sk = ulpq->asoc->base.sk;
+ struct sock *sk = ulpq->asoc->base.sk;
/* If the socket is just going to throw this away, do not
* even try to deliver it.
*/
- if (sk->dead || (sk->shutdown & RCV_SHUTDOWN)) {
+ if (sk->dead || (sk->shutdown & RCV_SHUTDOWN))
goto out_free;
- }
- /* Check if the user wishes to receive this event. */
- if (!sctp_ulpevent_is_enabled(event, &sctp_sk(sk)->subscribe)) {
+ /* Check if the user wishes to receive this event. */
+ if (!sctp_ulpevent_is_enabled(event, &sctp_sk(sk)->subscribe))
goto out_free;
- }
-
- /* If we are harvesting multiple skbs they will be
- * collected on a list.
+
+ /* If we are harvesting multiple skbs they will be
+ * collected on a list.
*/
- if (event->parent->list) {
+ if (event->parent->list)
sctp_skb_list_tail(event->parent->list, &sk->receive_queue);
- } else {
+ else
skb_queue_tail(&sk->receive_queue, event->parent);
- }
-
- wake_up_interruptible(sk->sleep);
+ wake_up_interruptible(sk->sleep);
return 1;
out_free:
- if (event->parent->list) {
+ if (event->parent->list)
skb_queue_purge(event->parent->list);
- } else {
+ else
kfree_skb(event->parent);
- }
-
return 0;
-
-} /* sctp_ulpqueue_tail_event() */
-
-
+}
/* 2nd Level Abstractions */
-/* Helper function to store chunks that need to be reassembled. */
-static inline void
-sctp_ulpqueue_store_reasm(sctp_ulpqueue_t *ulpq, sctp_ulpevent_t *event)
+/* Helper function to store chunks that need to be reassembled. */
+static inline void sctp_ulpqueue_store_reasm(sctp_ulpqueue_t *ulpq, sctp_ulpevent_t *event)
{
struct sk_buff *pos, *tmp;
sctp_ulpevent_t *cevent;
- uint32_t tsn, ctsn;
- int flags __attribute ((unused));
-
+ __u32 tsn, ctsn;
+ unsigned long flags __attribute ((unused));
+
tsn = event->sndrcvinfo.sinfo_tsn;
sctp_spin_lock_irqsave(&ulpq->reasm.lock, flags);
- /* Find the right place in this list. We store them by TSN. */
+ /* Find the right place in this list. We store them by TSN. */
sctp_skb_for_each(pos, &ulpq->reasm, tmp) {
- cevent = (sctp_ulpevent_t *)pos->cb;
+ cevent = (sctp_ulpevent_t *)pos->cb;
ctsn = cevent->sndrcvinfo.sinfo_tsn;
- if (TSN_lt(tsn, ctsn)) { break; }
+ if (TSN_lt(tsn, ctsn))
+ break;
}
-
- /* If the queue is empty, we have a different function to call. */
- if (skb_peek(&ulpq->reasm)) {
+
+ /* If the queue is empty, we have a different function to call. */
+ if (skb_peek(&ulpq->reasm))
__skb_insert(event->parent, pos->prev, pos, &ulpq->reasm);
- }
- else {
+ else
__skb_queue_tail(&ulpq->reasm, event->parent);
- }
sctp_spin_unlock_irqrestore(&ulpq->reasm.lock, flags);
+}
-} /* sctp_ulpqueue_store_reasm() */
-
-/* Helper function to return an event corresponding to the reassembled
- * datagram.
+/* Helper function to return an event corresponding to the reassembled
+ * datagram.
*/
-static inline sctp_ulpevent_t *
-sctp_make_reassembled_event(struct sk_buff *f_frag, struct sk_buff *l_frag)
+static inline sctp_ulpevent_t *sctp_make_reassembled_event(struct sk_buff *f_frag, struct sk_buff *l_frag)
{
struct sk_buff *pos;
sctp_ulpevent_t *event;
pos = f_frag->next;
- /* Set the first fragment's frag_list to point to the 2nd fragment. */
- skb_shinfo(f_frag)->frag_list = pos;
+ /* Set the first fragment's frag_list to point to the 2nd fragment. */
+ skb_shinfo(f_frag)->frag_list = pos;
- /* Remove the first fragment from the reassembly queue. */
+ /* Remove the first fragment from the reassembly queue. */
__skb_unlink(f_frag, f_frag->list);
-
- do {
+ do {
pnext = pos->next;
- /* Remove the fragment from the reassembly queue. */
+ /* Remove the fragment from the reassembly queue. */
__skb_unlink(pos, pos->list);
- /* Break if we have reached the last fragment. */
- if (pos == l_frag) { break; }
+ /* Break if we have reached the last fragment. */
+ if (pos == l_frag)
+ break;
pos->next = pnext;
pos = pnext;
} while (1);
- event = (sctp_ulpevent_t *)f_frag->cb;
-
- return (event);
+ event = (sctp_ulpevent_t *) f_frag->cb;
-} /* sctp_make_reassembled_event() */
+ return event;
+}
-/* Helper function to check if an incoming chunk has filled up the last
- * missing fragment in a SCTP datagram and return the corresponding event.
+/* Helper function to check if an incoming chunk has filled up the last
+ * missing fragment in a SCTP datagram and return the corresponding event.
*/
-static inline sctp_ulpevent_t *
-sctp_ulpqueue_retrieve_reassembled(sctp_ulpqueue_t *ulpq)
+static inline sctp_ulpevent_t *sctp_ulpqueue_retrieve_reassembled(sctp_ulpqueue_t *ulpq)
{
struct sk_buff *pos, *tmp;
sctp_ulpevent_t *cevent;
struct sk_buff *first_frag = NULL;
- uint32_t ctsn, next_tsn;
- int flags __attribute ((unused));
+ __u32 ctsn, next_tsn;
+ unsigned long flags __attribute ((unused));
sctp_ulpevent_t *retval = NULL;
/* Initialized to 0 just to avoid compiler warning message. Will
- * never be used with this value. It is referenced only after it
+ * never be used with this value. It is referenced only after it
* is set when we find the first fragment of a message.
*/
next_tsn = 0;
sctp_spin_lock_irqsave(&ulpq->reasm.lock, flags);
- /* The chunks are held in the reasm queue sorted by TSN.
+
+ /* The chunks are held in the reasm queue sorted by TSN.
* Walk through the queue sequentially and look for a sequence of
- * fragmented chunks that complete a datagram.
+ * fragmented chunks that complete a datagram.
* 'first_frag' and next_tsn are reset when we find a chunk which
* is the first fragment of a datagram. Once these 2 fields are set
* we expect to find the remaining middle fragments and the last
* fragment in order. If not, first_frag is reset to NULL and we
- * start the next pass when we find another first fragment.
+ * start the next pass when we find another first fragment.
*/
sctp_skb_for_each(pos, &ulpq->reasm, tmp) {
- cevent = (sctp_ulpevent_t *)pos->cb;
+ cevent = (sctp_ulpevent_t *) pos->cb;
ctsn = cevent->sndrcvinfo.sinfo_tsn;
switch (cevent->chunk_flags & SCTP_DATA_FRAG_MASK) {
case SCTP_DATA_FIRST_FRAG:
first_frag = pos;
- next_tsn = ctsn+1;
+ next_tsn = ctsn + 1;
break;
+
case SCTP_DATA_MIDDLE_FRAG:
- if ((first_frag) && (ctsn == next_tsn)) {
+ if ((first_frag) && (ctsn == next_tsn))
next_tsn++;
- } else {
+ else
first_frag = NULL;
- }
break;
+
case SCTP_DATA_LAST_FRAG:
- if ((first_frag) && (ctsn == next_tsn)) {
+ if ((first_frag) && (ctsn == next_tsn))
retval = sctp_make_reassembled_event(
- first_frag, pos);
- } else {
+ first_frag, pos);
+ else
first_frag = NULL;
- }
break;
- }
-
- /* We have the reassembled event. There is no need to look
- * further.
+ };
+
+ /* We have the reassembled event. There is no need to look
+ * further.
*/
- if (retval) { break; }
- }
+ if (retval)
+ break;
+ }
sctp_spin_unlock_irqrestore(&ulpq->reasm.lock, flags);
- return (retval);
-
-} /* sctp_ulpqueue_retrieve_reassembled() */
+ return retval;
+}
-/* Helper function to reassemble chunks. Hold chunks on the reasm queue that
- * need reassembling.
+/* Helper function to reassemble chunks. Hold chunks on the reasm queue that
+ * need reassembling.
*/
-static inline sctp_ulpevent_t *
-sctp_ulpqueue_reasm(sctp_ulpqueue_t *ulpq, sctp_ulpevent_t *event)
+static inline sctp_ulpevent_t *sctp_ulpqueue_reasm(sctp_ulpqueue_t *ulpq,
+ sctp_ulpevent_t *event)
{
sctp_ulpevent_t *retval = NULL;
- /* FIXME: We should be using some new chunk structure here
+ /* FIXME: We should be using some new chunk structure here
* instead of carrying chunk fields in the event structure.
- * This is temporary as it is too painful to change everything
+ * This is temporary as it is too painful to change everything
* at once.
*/
- /* Check if this is part of a fragmented message. */
- if (SCTP_DATA_NOT_FRAG == (event->chunk_flags & SCTP_DATA_FRAG_MASK)) {
+ /* Check if this is part of a fragmented message. */
+ if (SCTP_DATA_NOT_FRAG == (event->chunk_flags & SCTP_DATA_FRAG_MASK))
return event;
- }
sctp_ulpqueue_store_reasm(ulpq, event);
retval = sctp_ulpqueue_retrieve_reassembled(ulpq);
return retval;
-
-} /* sctp_ulpqueue_reasm() */
-
+}
/* Helper function to gather skbs that have possibly become
- * ordered by an an incoming chunk.
+ * ordered by an an incoming chunk.
*/
-static inline void
-sctp_ulpqueue_retrieve_ordered(sctp_ulpqueue_t *ulpq,
- sctp_ulpevent_t *event)
+static inline void sctp_ulpqueue_retrieve_ordered(sctp_ulpqueue_t *ulpq,
+ sctp_ulpevent_t *event)
{
struct sk_buff *pos, *tmp;
sctp_ulpevent_t *cevent;
- uint16_t sid, csid;
- uint16_t ssn, cssn;
- int flags __attribute ((unused));
+ __u16 sid, csid;
+ __u16 ssn, cssn;
+ unsigned long flags __attribute ((unused));
sid = event->sndrcvinfo.sinfo_stream;
ssn = event->sndrcvinfo.sinfo_ssn;
- /* We are holding the chunks by stream, by SSN. */
+ /* We are holding the chunks by stream, by SSN. */
sctp_spin_lock_irqsave(&ulpq->lobby.lock, flags);
sctp_skb_for_each(pos, &ulpq->lobby, tmp) {
- cevent = (sctp_ulpevent_t *)pos->cb;
+ cevent = (sctp_ulpevent_t *) pos->cb;
csid = cevent->sndrcvinfo.sinfo_stream;
cssn = cevent->sndrcvinfo.sinfo_ssn;
-
- /* Have we gone too far? */
- if (csid > sid) { break; }
- /* Have we not gone far enough? */
- if (csid < sid) { continue; }
+ /* Have we gone too far? */
+ if (csid > sid)
+ break;
+
+ /* Have we not gone far enough? */
+ if (csid < sid)
+ continue;
+
+ if (cssn != ulpq->ssn[sid])
+ break;
- if (cssn != ulpq->ssn[sid]) { break; }
-
ulpq->ssn[sid]++;
__skb_unlink(pos, pos->list);
-
- /* Attach all gathered skbs to the event. */
+
+ /* Attach all gathered skbs to the event. */
__skb_queue_tail(event->parent->list, pos);
-
- }
+ }
sctp_spin_unlock_irqrestore(&ulpq->lobby.lock, flags);
-
-} /* sctp_ulpqueue_retrieve_ordered() */
-
+}
/* Helper function to store chunks needing ordering. */
-static inline void
-sctp_ulpqueue_store_ordered(sctp_ulpqueue_t *ulpq,
- sctp_ulpevent_t *event)
+static inline void sctp_ulpqueue_store_ordered(sctp_ulpqueue_t *ulpq,
+ sctp_ulpevent_t *event)
{
struct sk_buff *pos, *tmp;
sctp_ulpevent_t *cevent;
- uint16_t sid, csid;
- uint16_t ssn, cssn;
- int flags __attribute ((unused));
-
-
+ __u16 sid, csid;
+ __u16 ssn, cssn;
+ unsigned long flags __attribute ((unused));
+
sid = event->sndrcvinfo.sinfo_stream;
- ssn = event->sndrcvinfo.sinfo_ssn;
+ ssn = event->sndrcvinfo.sinfo_ssn;
sctp_spin_lock_irqsave(&ulpq->lobby.lock, flags);
* stream ID and then by SSN.
*/
sctp_skb_for_each(pos, &ulpq->lobby, tmp) {
- cevent = (sctp_ulpevent_t *)pos->cb;
+ cevent = (sctp_ulpevent_t *) pos->cb;
csid = cevent->sndrcvinfo.sinfo_stream;
cssn = cevent->sndrcvinfo.sinfo_ssn;
- if (csid > sid) { break; }
- if (csid == sid && SSN_lt(ssn, cssn)) { break;}
-
+ if (csid > sid)
+ break;
+ if (csid == sid && SSN_lt(ssn, cssn))
+ break;
}
-
- /* If the queue is empty, we have a different function to call. */
- if (skb_peek(&ulpq->lobby)) {
+
+ /* If the queue is empty, we have a different function to call. */
+ if (skb_peek(&ulpq->lobby))
__skb_insert(event->parent, pos->prev, pos, &ulpq->lobby);
- }
- else {
+ else
__skb_queue_tail(&ulpq->lobby, event->parent);
- }
-
sctp_spin_unlock_irqrestore(&ulpq->lobby.lock, flags);
+}
-} /* sctp_ulpqueue_store_ordered() */
-
-static inline sctp_ulpevent_t *
-sctp_ulpqueue_order(sctp_ulpqueue_t *ulpq, sctp_ulpevent_t *event)
+static inline sctp_ulpevent_t *sctp_ulpqueue_order(sctp_ulpqueue_t *ulpq,
+ sctp_ulpevent_t *event)
{
- uint16_t sid;
- uint16_t ssn;
-
+ __u16 sid, ssn;
- /* FIXME: We should be using some new chunk structure here
+ /* FIXME: We should be using some new chunk structure here
* instead of carrying chunk fields in the event structure.
- * This is temporary as it is too painful to change everything
+ * This is temporary as it is too painful to change everything
* at once.
- */
-
-
- /* Check if this message needs ordering. */
- if (SCTP_DATA_UNORDERED & event->chunk_flags) {
- return event;
- }
-
- /* Note: The stream ID must be verified before this routine. */
- sid = event->sndrcvinfo.sinfo_stream;
- ssn = event->sndrcvinfo.sinfo_ssn;
-
- /* Is this the expected SSN for this stream ID? */
-
- if (ssn != ulpq->ssn[sid]) {
-
- /* We've received something out of order, so find where it
- * needs to be placed. We order by stream and then by SSN.
- */
-
- sctp_ulpqueue_store_ordered(ulpq, event);
-
- return NULL;
- }
-
- /* Mark that the next chunk has been found. */
- ulpq->ssn[sid]++;
+ */
- /* Go find any other chunks that were waiting for
- * ordering.
- */
-
- sctp_ulpqueue_retrieve_ordered(ulpq, event);
+ /* Check if this message needs ordering. */
+ if (SCTP_DATA_UNORDERED & event->chunk_flags)
+ return event;
- return event;
-
-} /* sctp_ulpqueue_order() */
+ /* Note: The stream ID must be verified before this routine. */
+ sid = event->sndrcvinfo.sinfo_stream;
+ ssn = event->sndrcvinfo.sinfo_ssn;
+ /* Is this the expected SSN for this stream ID? */
+ if (ssn != ulpq->ssn[sid]) {
+ /* We've received something out of order, so find where it
+ * needs to be placed. We order by stream and then by SSN.
+ */
+ sctp_ulpqueue_store_ordered(ulpq, event);
+ return NULL;
+ }
+ /* Mark that the next chunk has been found. */
+ ulpq->ssn[sid]++;
+ /* Go find any other chunks that were waiting for
+ * ordering.
+ */
+ sctp_ulpqueue_retrieve_ordered(ulpq, event);
+ return event;
+}