]> git.hungrycats.org Git - linux/commitdiff
[SCTP]: Rename sctp_foo.[ch] to foo.[ch] and kill CVS tags on authors request.
authorDavid S. Miller <davem@nuts.ninka.net>
Thu, 29 Aug 2002 09:55:45 +0000 (02:55 -0700)
committerDavid S. Miller <davem@nuts.ninka.net>
Thu, 29 Aug 2002 09:55:45 +0000 (02:55 -0700)
75 files changed:
include/linux/sctp.h
include/net/sctp/command.h [new file with mode: 0644]
include/net/sctp/constants.h [new file with mode: 0644]
include/net/sctp/sctp.h
include/net/sctp/sctp_command.h [deleted file]
include/net/sctp/sctp_constants.h [deleted file]
include/net/sctp/sctp_sla1.h [deleted file]
include/net/sctp/sctp_sm.h [deleted file]
include/net/sctp/sctp_structs.h [deleted file]
include/net/sctp/sctp_tsnmap.h [deleted file]
include/net/sctp/sctp_ulpevent.h [deleted file]
include/net/sctp/sctp_ulpqueue.h [deleted file]
include/net/sctp/sctp_user.h [deleted file]
include/net/sctp/sla1.h [new file with mode: 0644]
include/net/sctp/sm.h [new file with mode: 0644]
include/net/sctp/structs.h [new file with mode: 0644]
include/net/sctp/tsnmap.h [new file with mode: 0644]
include/net/sctp/ulpevent.h [new file with mode: 0644]
include/net/sctp/ulpqueue.h [new file with mode: 0644]
include/net/sctp/user.h [new file with mode: 0644]
net/sctp/Makefile
net/sctp/adler32.c [new file with mode: 0644]
net/sctp/associola.c [new file with mode: 0644]
net/sctp/bind_addr.c [new file with mode: 0644]
net/sctp/command.c [new file with mode: 0644]
net/sctp/crc32c.c [new file with mode: 0644]
net/sctp/debug.c [new file with mode: 0644]
net/sctp/endpointola.c [new file with mode: 0644]
net/sctp/hashdriver.c [new file with mode: 0644]
net/sctp/input.c [new file with mode: 0644]
net/sctp/inqueue.c [new file with mode: 0644]
net/sctp/ipv6.c [new file with mode: 0644]
net/sctp/objcnt.c [new file with mode: 0644]
net/sctp/output.c [new file with mode: 0644]
net/sctp/outqueue.c [new file with mode: 0644]
net/sctp/primitive.c [new file with mode: 0644]
net/sctp/protocol.c [new file with mode: 0644]
net/sctp/sctp_adler32.c [deleted file]
net/sctp/sctp_associola.c [deleted file]
net/sctp/sctp_bind_addr.c [deleted file]
net/sctp/sctp_command.c [deleted file]
net/sctp/sctp_crc32c.c [deleted file]
net/sctp/sctp_debug.c [deleted file]
net/sctp/sctp_endpointola.c [deleted file]
net/sctp/sctp_hashdriver.c [deleted file]
net/sctp/sctp_input.c [deleted file]
net/sctp/sctp_inqueue.c [deleted file]
net/sctp/sctp_ipv6.c [deleted file]
net/sctp/sctp_objcnt.c [deleted file]
net/sctp/sctp_output.c [deleted file]
net/sctp/sctp_outqueue.c [deleted file]
net/sctp/sctp_primitive.c [deleted file]
net/sctp/sctp_protocol.c [deleted file]
net/sctp/sctp_sla1.c [deleted file]
net/sctp/sctp_sm_make_chunk.c [deleted file]
net/sctp/sctp_sm_sideeffect.c [deleted file]
net/sctp/sctp_sm_statefuns.c [deleted file]
net/sctp/sctp_sm_statetable.c [deleted file]
net/sctp/sctp_socket.c [deleted file]
net/sctp/sctp_sysctl.c [deleted file]
net/sctp/sctp_transport.c [deleted file]
net/sctp/sctp_tsnmap.c [deleted file]
net/sctp/sctp_ulpevent.c [deleted file]
net/sctp/sctp_ulpqueue.c [deleted file]
net/sctp/sla1.c [new file with mode: 0644]
net/sctp/sm_make_chunk.c [new file with mode: 0644]
net/sctp/sm_sideeffect.c [new file with mode: 0644]
net/sctp/sm_statefuns.c [new file with mode: 0644]
net/sctp/sm_statetable.c [new file with mode: 0644]
net/sctp/socket.c [new file with mode: 0644]
net/sctp/sysctl.c [new file with mode: 0644]
net/sctp/transport.c [new file with mode: 0644]
net/sctp/tsnmap.c [new file with mode: 0644]
net/sctp/ulpevent.c [new file with mode: 0644]
net/sctp/ulpqueue.c [new file with mode: 0644]

index b070fc1220e532a99b88da77e9f6b93376a02264..77502f62b6607db516672e270b9aaecd4bd24544 100644 (file)
@@ -8,8 +8,6 @@
  * 
  * This file is part of the SCTP kernel reference Implementation
  * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/include/linux/sctp.h,v 1.7 2002/07/17 16:13:50 jgrimm Exp $
- * 
  * Various protocol defined structures.  
  * 
  * The SCTP reference implementation is free software; 
diff --git a/include/net/sctp/command.h b/include/net/sctp/command.h
new file mode 100644 (file)
index 0000000..d299eee
--- /dev/null
@@ -0,0 +1,211 @@
+/* SCTP kernel reference Implementation Copyright (C) 1999-2001
+ * Cisco, Motorola, and IBM
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * These are the definitions needed for the command object.
+ * 
+ * The SCTP reference implementation  is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * the SCTP reference implementation  is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to one of the
+ * following email addresses:
+ * 
+ * La Monte H.P. Yarroll <piggy@acm.org>
+ * Karl Knutson <karl@athena.chicago.il.us>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+
+#ifndef __net_sctp_command_h__
+#define __net_sctp_command_h__
+
+#include <net/sctp/constants.h>
+#include <net/sctp/structs.h>
+
+
+typedef enum {
+       SCTP_CMD_NOP = 0,       /* Do nothing. */
+       SCTP_CMD_NEW_ASOC,      /* Register a new association.  */
+       SCTP_CMD_DELETE_TCB,    /* Delete the current association. */
+       SCTP_CMD_NEW_STATE,     /* Enter a new state.  */
+       SCTP_CMD_REPORT_TSN,    /* Record the arrival of a TSN.  */
+       SCTP_CMD_GEN_SACK,      /* Send a Selective ACK (maybe).  */
+       SCTP_CMD_PROCESS_SACK,  /* Process an inbound SACK.  */
+       SCTP_CMD_GEN_INIT_ACK,  /* Generate an INIT ACK chunk.  */
+       SCTP_CMD_PEER_INIT,     /* Process a INIT from the peer.  */
+       SCTP_CMD_GEN_COOKIE_ECHO, /* Generate a COOKIE ECHO chunk. */
+       SCTP_CMD_CHUNK_ULP,     /* Send a chunk to the sockets layer.  */
+       SCTP_CMD_EVENT_ULP,     /* Send a notification to the sockets layer. */
+       SCTP_CMD_REPLY,         /* Send a chunk to our peer.  */
+       SCTP_CMD_SEND_PKT,      /* Send a full packet to our peer.  */
+       SCTP_CMD_RETRAN,        /* Mark a transport for retransmission.  */
+       SCTP_CMD_ECN_CE,        /* Do delayed CE processing.   */
+       SCTP_CMD_ECN_ECNE,      /* Do delayed ECNE processing. */
+       SCTP_CMD_ECN_CWR,       /* Do delayed CWR processing.  */
+       SCTP_CMD_TIMER_START,   /* Start a timer.  */
+       SCTP_CMD_TIMER_RESTART, /* Restart a timer. */
+       SCTP_CMD_TIMER_STOP,    /* Stop a timer. */
+       SCTP_CMD_COUNTER_RESET, /* Reset a counter. */
+       SCTP_CMD_COUNTER_INC,   /* Increment a counter. */
+       SCTP_CMD_INIT_RESTART,  /* High level, do init timer work. */
+       SCTP_CMD_INIT_FAILED,   /* High level, do init failure work. */
+       SCTP_CMD_REPORT_DUP,    /* Report a duplicate TSN.  */
+       SCTP_CMD_REPORT_BIGGAP, /* Narc on a TSN (it was too high).  */
+       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_TRANSPORT_RESET,    /* Reset the status of a transport. */
+       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_DISCARD_PACKET, /* Discard the whole packet. */
+       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;
+
+#define SCTP_CMD_MAX           (SCTP_CMD_LAST - 1)
+#define SCTP_CMD_NUM_VERBS     (SCTP_CMD_MAX + 1)
+
+/* How many commands can you put in an sctp_cmd_seq_t?
+ * This is a rather arbitrary number, ideally derived from a careful
+ * analysis of the state functions, but in reality just taken from
+ * thin air in the hopes othat we don't trigger a kernel panic.
+ */
+#define SCTP_MAX_NUM_COMMANDS 14
+
+typedef union {
+       __s32 i32;
+       __u32 u32;
+       __u16 u16;
+       __u8 u8;
+       int error;
+       sctp_state_t state;
+       sctp_event_timeout_t to;
+       sctp_counter_t counter;
+       void *ptr;
+       sctp_chunk_t *chunk;
+       sctp_association_t *asoc;
+       sctp_transport_t *transport;
+       sctp_bind_addr_t  *bp;
+       sctp_init_chunk_t *init;
+       sctp_ulpevent_t *ulpevent;
+       sctp_packet_t *packet;
+       sctp_sackhdr_t *sackh;
+} 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, __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)
+{
+       sctp_arg_t retval; retval.ptr = NULL; return retval;
+}
+static inline sctp_arg_t SCTP_NOFORCE(void)
+{
+       sctp_arg_t retval; retval.i32 = 0; return retval;
+}
+static inline sctp_arg_t SCTP_FORCE(void)
+{
+       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_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)
+SCTP_ARG_CONSTRUCTOR(TO,       sctp_event_timeout_t, to)
+SCTP_ARG_CONSTRUCTOR(PTR,      void *, ptr)
+SCTP_ARG_CONSTRUCTOR(CHUNK,    sctp_chunk_t *, chunk)
+SCTP_ARG_CONSTRUCTOR(ASOC,     sctp_association_t *, asoc)
+SCTP_ARG_CONSTRUCTOR(TRANSPORT,        sctp_transport_t *, transport)
+SCTP_ARG_CONSTRUCTOR(BA,       sctp_bind_addr_t *, bp)
+SCTP_ARG_CONSTRUCTOR(PEER_INIT,        sctp_init_chunk_t *, init)
+SCTP_ARG_CONSTRUCTOR(ULPEVENT,  sctp_ulpevent_t *, ulpevent)
+SCTP_ARG_CONSTRUCTOR(PACKET,   sctp_packet_t *, packet)
+SCTP_ARG_CONSTRUCTOR(SACKH,    sctp_sackhdr_t *, sackh)
+
+typedef struct {
+       sctp_arg_t obj;
+       sctp_verb_t verb;
+} sctp_cmd_t;
+
+typedef struct {
+       sctp_cmd_t cmds[SCTP_MAX_NUM_COMMANDS];
+       __u8 next_free_slot;
+       __u8 next_cmd;
+} sctp_cmd_seq_t;
+
+
+/* Create a new sctp_command_sequence.
+ * Return NULL if creating a new sequence fails.
+ */
+sctp_cmd_seq_t *sctp_new_cmd_seq(int priority);
+
+/* Initialize a block of memory as a command sequence.
+ * Return 0 if the initialization fails.
+ */
+int sctp_init_cmd_seq(sctp_cmd_seq_t *seq);
+
+/* Add a command to an sctp_cmd_seq_t.
+ * Return 0 if the command sequence is full.
+ *
+ * Use the SCTP_* constructors defined by SCTP_ARG_CONSTRUCTOR() above
+ * to wrap data which goes in the obj argument.
+ */
+int sctp_add_cmd(sctp_cmd_seq_t *seq, sctp_verb_t verb, sctp_arg_t obj);
+
+/* Rewind an sctp_cmd_seq_t to iterate from the start.
+ * Return 0 if the rewind fails.
+ */
+int sctp_rewind_sequence(sctp_cmd_seq_t *seq);
+
+/* Return the next command structure in an sctp_cmd_seq.
+ * Return NULL at the end of the sequence.
+ */
+sctp_cmd_t *sctp_next_cmd(sctp_cmd_seq_t *seq);
+
+/* Dispose of a command sequence.  */
+void sctp_free_cmd_seq(sctp_cmd_seq_t *seq);
+
+#endif /* __net_sctp_command_h__ */
+
diff --git a/include/net/sctp/constants.h b/include/net/sctp/constants.h
new file mode 100644 (file)
index 0000000..f187be3
--- /dev/null
@@ -0,0 +1,450 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001-2002 International Business Machines Corp.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * This file is part of the implementation of the add-IP extension,
+ * based on <draft-ietf-tsvwg-addip-sctp-02.txt> June 29, 2001,
+ * for the SCTP kernel reference Implementation.
+ * 
+ * The SCTP reference implementation  is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * the SCTP reference implementation  is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty
+ *                 ************************
+ * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to one of the following email
+ * addresses:
+ * 
+ * La Monte H.P. Yarroll <piggy@acm.org>
+ * Karl Knutson <karl@athena.chicago.il.us>
+ * Randall Stewart <randall@stewart.chicago.il.us>
+ * Ken Morneau <kmorneau@cisco.com>
+ * Qiaobing Xie <qxie1@motorola.com>
+ * Xingang Guo <xingang.guo@intel.com>
+ * Sridhar Samudrala <samudrala@us.ibm.com>
+ * Daisy Chang <daisyc@us.ibm.com>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ * 
+ * There are still LOTS of bugs in this code... I always run on the motto
+ * "it is a wonder any code ever works :)"
+ * 
+ * 
+ */
+
+#ifndef __sctp_constants_h__
+#define __sctp_constants_h__
+
+#include <linux/tcp.h>  /* For TCP states used in sctp_sock_state_t */
+#include <linux/sctp.h>
+#include <linux/ipv6.h> /* For ipv6hdr. */
+#include <net/sctp/user.h>
+
+/* 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
+ * for ipv6 headers, but this seems worth the simplicity.
+ */
+
+#define SCTP_IP_OVERHEAD ((sizeof(struct sctphdr)\
+                          + 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
+ * for ipv6 headers, but this seems worth the simplicity.
+ */
+
+#define SCTP_IP_OVERHEAD ((sizeof(struct sctphdr)\
+                          + sizeof(struct ipv6hdr)\
+                          + MAX_HEADER))
+
+/* Since CIDs are sparse, we need all four of the following
+ * symbols.  CIDs are dense through SCTP_CID_BASE_MAX.
+ */
+#define SCTP_CID_BASE_MAX              SCTP_CID_SHUTDOWN_COMPLETE
+#define SCTP_CID_MAX                   SCTP_CID_ASCONF_ACK
+
+#define SCTP_NUM_BASE_CHUNK_TYPES      (SCTP_CID_BASE_MAX + 1)
+#define SCTP_NUM_CHUNK_TYPES           (SCTP_NUM_BASE_CHUNKTYPES + 2)
+
+
+/* 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;
+
+#define SCTP_EVENT_T_MAX SCTP_EVENT_T_PRIMITIVE
+#define SCTP_EVENT_T_NUM (SCTP_EVENT_T_MAX + 1)
+
+/* As a convenience for the state machine, we append SCTP_EVENT_* and
+ * SCTP_ULP_* to the list of possible chunks.
+ */
+
+typedef enum {
+
+       SCTP_EVENT_TIMEOUT_NONE = 0,
+       SCTP_EVENT_TIMEOUT_T1_COOKIE,
+       SCTP_EVENT_TIMEOUT_T1_INIT,
+       SCTP_EVENT_TIMEOUT_T2_SHUTDOWN,
+       SCTP_EVENT_TIMEOUT_T3_RTX,
+       SCTP_EVENT_TIMEOUT_T4_RTO,
+       SCTP_EVENT_TIMEOUT_HEARTBEAT,
+       SCTP_EVENT_TIMEOUT_SACK,
+       SCTP_EVENT_TIMEOUT_AUTOCLOSE,
+       SCTP_EVENT_TIMEOUT_PMTU_RAISE,
+
+} sctp_event_timeout_t;
+
+#define SCTP_EVENT_TIMEOUT_MAX         SCTP_EVENT_TIMEOUT_PMTU_RAISE
+#define SCTP_NUM_TIMEOUT_TYPES         (SCTP_EVENT_TIMEOUT_MAX + 1)
+
+typedef enum {
+
+       SCTP_EVENT_NO_PENDING_TSN = 0,
+       SCTP_EVENT_ICMP_UNREACHFRAG,
+
+} sctp_event_other_t;
+
+#define SCTP_EVENT_OTHER_MAX           SCTP_EVENT_ICMP_UNREACHFRAG
+#define SCTP_NUM_OTHER_TYPES           (SCTP_EVENT_OTHER_MAX + 1)
+
+/* 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_event_primitive_t;
+
+#define SCTP_EVENT_PRIMITIVE_MAX       SCTP_PRIMITIVE_DESTROY
+#define SCTP_NUM_PRIMITIVE_TYPES       (SCTP_EVENT_PRIMITIVE_MAX + 1)
+
+/* We define here a utility type for manipulating subtypes.
+ * The subtype constructors all work like this:
+ *
+ *     sctp_subtype_t foo = SCTP_ST_CHUNK(SCTP_CID_INIT);
+ */
+
+typedef union {
+
+       sctp_cid_t chunk;
+       sctp_event_timeout_t timeout;
+       sctp_event_other_t other;
+       sctp_event_primitive_t primitive;
+
+} sctp_subtype_t;
+
+#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_CONSTRUCTOR(CHUNK,                sctp_cid_t,             chunk)
+SCTP_SUBTYPE_CONSTRUCTOR(TIMEOUT,      sctp_event_timeout_t,   timeout)
+SCTP_SUBTYPE_CONSTRUCTOR(OTHER,                sctp_event_other_t,     other)
+SCTP_SUBTYPE_CONSTRUCTOR(PRIMITIVE,    sctp_event_primitive_t, primitive)
+
+
+#define sctp_chunk_is_control(a) (a->chunk_hdr->type != SCTP_CID_DATA)
+#define sctp_chunk_is_data(a) (a->chunk_hdr->type == SCTP_CID_DATA)
+
+/* Calculate the actual data size in a data chunk */
+#define SCTP_DATA_SNDSIZE(c) ((int)((unsigned long)(c->chunk_end)\
+                               - (unsigned long)(c->chunk_hdr)\
+                               - sizeof(sctp_data_chunk_t)))
+
+/* This is a table of printable names of sctp_param_t's.  */
+extern const char *sctp_param_tbl[];
+
+
+#define SCTP_MAX_ERROR_CAUSE  SCTP_ERROR_NONEXIST_IP
+#define SCTP_NUM_ERROR_CAUSE  10
+
+/* 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_t;
+
+
+
+/* SCTP state defines for internal state machine */
+typedef enum {
+
+       SCTP_STATE_EMPTY                = 0,
+       SCTP_STATE_CLOSED               = 1,
+       SCTP_STATE_COOKIE_WAIT          = 2,
+       SCTP_STATE_COOKIE_ECHOED        = 3,
+       SCTP_STATE_ESTABLISHED          = 4,
+       SCTP_STATE_SHUTDOWN_PENDING     = 5,
+       SCTP_STATE_SHUTDOWN_SENT        = 6,
+       SCTP_STATE_SHUTDOWN_RECEIVED    = 7,
+       SCTP_STATE_SHUTDOWN_ACK_SENT    = 8,
+
+} sctp_state_t;
+
+#define SCTP_STATE_MAX                 SCTP_STATE_SHUTDOWN_ACK_SENT
+#define SCTP_STATE_NUM_STATES          (SCTP_STATE_MAX + 1)
+
+/* These are values for sk->state.
+ * For a UDP-style SCTP socket, the states are defined as follows
+ *   (at this point of time, may change later after more discussions: FIXME)
+ * A socket in SCTP_SS_UNCONNECTED state indicates that it is not willing
+ * to accept new associations, but it can initiate the creation of new
+ * ones.
+ * A socket in SCTP_SS_LISTENING state indicates that it is willing to
+ * accept new  associations and can initiate the creation of new ones.
+ * A socket in SCTP_SS_ESTABLISHED state indicates that it is a peeled off
+ * socket with one association.
+ */
+typedef enum {
+       SCTP_SS_CLOSED         = TCP_CLOSE,
+       SCTP_SS_LISTENING      = TCP_LISTEN,
+       SCTP_SS_ESTABLISHING   = TCP_SYN_SENT,
+       SCTP_SS_ESTABLISHED    = TCP_ESTABLISHED,
+       SCTP_SS_DISCONNECTING  = TCP_CLOSING,
+} sctp_sock_state_t;
+
+/* These functions map various type to printable names.  */
+const char *sctp_cname(const sctp_subtype_t);  /* chunk types */
+const char *sctp_oname(const sctp_subtype_t);  /* other events */
+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.  */
+extern const char *sctp_state_tbl[], *sctp_evttype_tbl[], *sctp_status_tbl[];
+
+/* SCTP reachability state for each address */
+#define SCTP_ADDR_NOHB                 4
+#define SCTP_ADDR_REACHABLE            2
+#define SCTP_ADDR_NOT_REACHABLE                1
+
+
+
+
+/* Guess at how big to make the TSN mapping array.
+ * We guarantee that we can handle at least this big a gap between the
+ * cumulative ACK and the highest TSN.  In practice, we can often
+ * handle up to twice this value.
+ *
+ * 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.
+ */
+#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.
+ */
+#define SCTP_MAX_DUP_TSNS 128
+
+typedef enum {
+       SCTP_COUNTER_INIT_ERROR,
+} sctp_counter_t;
+
+/* How many counters does an association need? */
+#define SCTP_NUMBER_COUNTERS   5
+
+
+/* Here we define the default timers.  */
+
+/* cookie timer def = ? seconds */
+#define SCTP_DEFAULT_TIMEOUT_T1_COOKIE (3 * HZ)
+
+/* init timer def = 3 seconds  */
+#define SCTP_DEFAULT_TIMEOUT_T1_INIT   (3 * HZ)
+
+/* shutdown timer def = 300 ms */
+#define SCTP_DEFAULT_TIMEOUT_T2_SHUTDOWN ((300 * HZ) / 1000)
+
+/* 0 seconds + RTO */
+#define SCTP_DEFAULT_TIMEOUT_HEARTBEAT (10 * HZ)
+
+/* recv timer def = 200ms (in usec) */
+#define SCTP_DEFAULT_TIMEOUT_SACK      ((200 * HZ) / 1000)
+#define SCTP_DEFAULT_TIMEOUT_SACK_MAX  ((500 * HZ) / 1000) /* 500 ms */
+
+/* How long do we wait before attempting to raise the PMTU?  */
+#define SCTP_DEFAULT_TIMEOUT_PMTU_RAISE (10 * 60 * HZ) /* 10 Minutes */
+#define SCTP_DEFAULT_TIMEOUT_PMTU_RAISE_MIN (10 * 60 * HZ) /* 10 Minutes */
+
+/* RTO.Initial              - 3  seconds
+ * RTO.Min                  - 1  second
+ * RTO.Max                  - 60 seconds
+ * RTO.Alpha                - 1/8
+ * RTO.Beta                 - 1/4
+ */
+#define SCTP_RTO_INITIAL       (3 * HZ)
+#define SCTP_RTO_MIN           (1 * HZ)
+#define SCTP_RTO_MAX           (60 * HZ)
+
+#define SCTP_RTO_ALPHA          3   /* 1/8 when converted to right shifts. */
+#define SCTP_RTO_BETA           2   /* 1/4 when converted to right shifts. */
+
+/* Maximum number of new data packets that can be sent in a burst.  */
+#define SCTP_MAX_BURST         4
+
+#define SCTP_CLOCK_GRANULARITY 1       /* 1 jiffy */
+
+#define SCTP_DEF_MAX_INIT 6
+#define SCTP_DEF_MAX_SEND 10
+
+#define SCTP_DEFAULT_COOKIE_LIFE_SEC   60 /* seconds */
+#define SCTP_DEFAULT_COOKIE_LIFE_USEC  0  /* microseconds */
+
+#define SCTP_DEFAULT_MINWINDOW 1500    /* default minimum rwnd size */
+#define SCTP_DEFAULT_MAXWINDOW 32768   /* default rwnd size */
+#define SCTP_DEFAULT_MAXSEGMENT 1500   /* MTU size, this is the limit
+                                         * to which we will raise the P-MTU.
+                                        */
+#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
+                                        * secret will live?
+                                        */
+#define SCTP_SECRET_SIZE 32            /* Number of octets in a 256 bits. */
+
+#define SCTP_SIGNATURE_SIZE 20         /* size of a SLA-1 signature */
+
+#define SCTP_COOKIE_MULTIPLE 64 /* Pad out our cookie to make our hash
+                                * functions simpler to write.
+                                */
+
+/* These return values describe the success or failure of a number of
+ * 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_t;
+
+/* These are the commands for manipulating transports.  */
+typedef enum {
+       SCTP_TRANSPORT_UP,
+       SCTP_TRANSPORT_DOWN,
+} 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.
+ * At this point, the IPv6 scopes will be mapped to these internal scopes
+ * as much as possible.
+ */
+typedef enum {
+       SCTP_SCOPE_GLOBAL,              /* IPv4 global addresses */
+       SCTP_SCOPE_PRIVATE,             /* IPv4 private addresses */
+       SCTP_SCOPE_LINK,                /* IPv4 link local address */
+       SCTP_SCOPE_LOOPBACK,            /* IPv4 loopback address */
+       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,
+ * 192.88.99.0/24.
+ * Also, RFC 8.4, non-unicast addresses are not considered valid SCTP
+ * addresses.
+ */
+#define IS_IPV4_UNUSABLE_ADDRESS(a) \
+       ((INADDR_BROADCAST == *a) || \
+       (MULTICAST(*a)) || \
+       (((unsigned char *)(a))[0] == 0) || \
+       ((((unsigned char *)(a))[0] == 198) && \
+       (((unsigned char *)(a))[1] == 18) && \
+       (((unsigned char *)(a))[2] == 0)) || \
+       ((((unsigned char *)(a))[0] == 192) && \
+       (((unsigned char *)(a))[1] == 88) && \
+       (((unsigned char *)(a))[2] == 99)))
+
+/* IPv4 Link-local addresses: 169.254.0.0/16.  */
+#define IS_IPV4_LINK_ADDRESS(a) \
+       ((((unsigned char *)(a))[0] == 169) && \
+       (((unsigned char *)(a))[1] == 254))
+
+/* RFC 1918 "Address Allocation for Private Internets" defines the IPv4
+ * private address space as the following:
+ *
+ * 10.0.0.0 - 10.255.255.255 (10/8 prefix)
+ * 172.16.0.0.0 - 172.31.255.255 (172.16/12 prefix)
+ * 192.168.0.0 - 192.168.255.255 (192.168/16 prefix)
+ */
+#define IS_IPV4_PRIVATE_ADDRESS(a) \
+       ((((unsigned char *)(a))[0] == 10) || \
+       ((((unsigned char *)(a))[0] == 172) && \
+       (((unsigned char *)(a))[1] >= 16) && \
+       (((unsigned char *)(a))[1] < 32)) || \
+       ((((unsigned char *)(a))[0] == 192) && \
+       (((unsigned char *)(a))[1] == 168)))
+
+/* Flags used for the bind address copy functions.  */
+#define SCTP_ADDR6_ALLOWED     0x00000001      /* IPv6 address is allowed by
+                                                  local sock family */
+#define SCTP_ADDR4_PEERSUPP    0x00000002      /* IPv4 address is supported by
+                                                  peer */
+#define SCTP_ADDR6_PEERSUPP    0x00000004      /* IPv6 address is supported by
+                                                  peer */
+
+/* Reasons to lower cwnd. */
+typedef enum {
+       SCTP_LOWER_CWND_T3_RTX,
+       SCTP_LOWER_CWND_FAST_RTX,
+       SCTP_LOWER_CWND_ECNE,
+       SCTP_LOWER_CWND_INACTIVE,
+} sctp_lower_cwnd_t;
+
+#endif /* __sctp_constants_h__ */
+
index 185e39ee064cc769e4fc6470d77e5367b9f3a2f8..2c7b3c0b6534a205b10c714ce4a82761580ac201 100644 (file)
@@ -6,8 +6,6 @@
  * 
  * This file is part of the SCTP kernel reference Implementation
  * 
- * $Id: sctp.h,v 1.40 2002/08/21 18:34:03 jgrimm Exp $
- * 
  * The base lksctp header. 
  * 
  * The SCTP reference implementation is free software; 
@@ -85,9 +83,9 @@
 #include <asm/uaccess.h>
 #include <asm/page.h>
 #include <net/sock.h>
-#include <net/sctp/sctp_structs.h>
-#include <net/sctp/sctp_constants.h>
-#include <net/sctp/sctp_sm.h>
+#include <net/sctp/structs.h>
+#include <net/sctp/constants.h>
+#include <net/sctp/sm.h>
 
 
 /* Set SCTP_DEBUG flag via config if not already set. */
diff --git a/include/net/sctp/sctp_command.h b/include/net/sctp/sctp_command.h
deleted file mode 100644 (file)
index 4fd50d5..0000000
+++ /dev/null
@@ -1,213 +0,0 @@
-/* SCTP kernel reference Implementation Copyright (C) 1999-2001
- * Cisco, Motorola, and IBM
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/include/net/sctp/sctp_command.h,v 1.19 2002/08/16 19:30:49 jgrimm Exp $
- * 
- * These are the definitions needed for the command object.
- * 
- * The SCTP reference implementation  is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * the SCTP reference implementation  is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to one of the
- * following email addresses:
- * 
- * La Monte H.P. Yarroll <piggy@acm.org>
- * Karl Knutson <karl@athena.chicago.il.us>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-
-
-#ifndef __net_sctp_command_h__
-#define __net_sctp_command_h__
-
-#include <net/sctp/sctp_constants.h>
-#include <net/sctp/sctp_structs.h>
-
-
-typedef enum {
-       SCTP_CMD_NOP = 0,       /* Do nothing. */
-       SCTP_CMD_NEW_ASOC,      /* Register a new association.  */
-       SCTP_CMD_DELETE_TCB,    /* Delete the current association. */
-       SCTP_CMD_NEW_STATE,     /* Enter a new state.  */
-       SCTP_CMD_REPORT_TSN,    /* Record the arrival of a TSN.  */
-       SCTP_CMD_GEN_SACK,      /* Send a Selective ACK (maybe).  */
-       SCTP_CMD_PROCESS_SACK,  /* Process an inbound SACK.  */
-       SCTP_CMD_GEN_INIT_ACK,  /* Generate an INIT ACK chunk.  */
-       SCTP_CMD_PEER_INIT,     /* Process a INIT from the peer.  */
-       SCTP_CMD_GEN_COOKIE_ECHO, /* Generate a COOKIE ECHO chunk. */
-       SCTP_CMD_CHUNK_ULP,     /* Send a chunk to the sockets layer.  */
-       SCTP_CMD_EVENT_ULP,     /* Send a notification to the sockets layer. */
-       SCTP_CMD_REPLY,         /* Send a chunk to our peer.  */
-       SCTP_CMD_SEND_PKT,      /* Send a full packet to our peer.  */
-       SCTP_CMD_RETRAN,        /* Mark a transport for retransmission.  */
-       SCTP_CMD_ECN_CE,        /* Do delayed CE processing.   */
-       SCTP_CMD_ECN_ECNE,      /* Do delayed ECNE processing. */
-       SCTP_CMD_ECN_CWR,       /* Do delayed CWR processing.  */
-       SCTP_CMD_TIMER_START,   /* Start a timer.  */
-       SCTP_CMD_TIMER_RESTART, /* Restart a timer. */
-       SCTP_CMD_TIMER_STOP,    /* Stop a timer. */
-       SCTP_CMD_COUNTER_RESET, /* Reset a counter. */
-       SCTP_CMD_COUNTER_INC,   /* Increment a counter. */
-       SCTP_CMD_INIT_RESTART,  /* High level, do init timer work. */
-       SCTP_CMD_INIT_FAILED,   /* High level, do init failure work. */
-       SCTP_CMD_REPORT_DUP,    /* Report a duplicate TSN.  */
-       SCTP_CMD_REPORT_BIGGAP, /* Narc on a TSN (it was too high).  */
-       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_TRANSPORT_RESET,    /* Reset the status of a transport. */
-       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_DISCARD_PACKET, /* Discard the whole packet. */
-       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;
-
-#define SCTP_CMD_MAX           (SCTP_CMD_LAST - 1)
-#define SCTP_CMD_NUM_VERBS     (SCTP_CMD_MAX + 1)
-
-/* How many commands can you put in an sctp_cmd_seq_t?
- * This is a rather arbitrary number, ideally derived from a careful
- * analysis of the state functions, but in reality just taken from
- * thin air in the hopes othat we don't trigger a kernel panic.
- */
-#define SCTP_MAX_NUM_COMMANDS 14
-
-typedef union {
-       __s32 i32;
-       __u32 u32;
-       __u16 u16;
-       __u8 u8;
-       int error;
-       sctp_state_t state;
-       sctp_event_timeout_t to;
-       sctp_counter_t counter;
-       void *ptr;
-       sctp_chunk_t *chunk;
-       sctp_association_t *asoc;
-       sctp_transport_t *transport;
-       sctp_bind_addr_t  *bp;
-       sctp_init_chunk_t *init;
-       sctp_ulpevent_t *ulpevent;
-       sctp_packet_t *packet;
-       sctp_sackhdr_t *sackh;
-} 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, __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)
-{
-       sctp_arg_t retval; retval.ptr = NULL; return retval;
-}
-static inline sctp_arg_t SCTP_NOFORCE(void)
-{
-       sctp_arg_t retval; retval.i32 = 0; return retval;
-}
-static inline sctp_arg_t SCTP_FORCE(void)
-{
-       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_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)
-SCTP_ARG_CONSTRUCTOR(TO,       sctp_event_timeout_t, to)
-SCTP_ARG_CONSTRUCTOR(PTR,      void *, ptr)
-SCTP_ARG_CONSTRUCTOR(CHUNK,    sctp_chunk_t *, chunk)
-SCTP_ARG_CONSTRUCTOR(ASOC,     sctp_association_t *, asoc)
-SCTP_ARG_CONSTRUCTOR(TRANSPORT,        sctp_transport_t *, transport)
-SCTP_ARG_CONSTRUCTOR(BA,       sctp_bind_addr_t *, bp)
-SCTP_ARG_CONSTRUCTOR(PEER_INIT,        sctp_init_chunk_t *, init)
-SCTP_ARG_CONSTRUCTOR(ULPEVENT,  sctp_ulpevent_t *, ulpevent)
-SCTP_ARG_CONSTRUCTOR(PACKET,   sctp_packet_t *, packet)
-SCTP_ARG_CONSTRUCTOR(SACKH,    sctp_sackhdr_t *, sackh)
-
-typedef struct {
-       sctp_arg_t obj;
-       sctp_verb_t verb;
-} sctp_cmd_t;
-
-typedef struct {
-       sctp_cmd_t cmds[SCTP_MAX_NUM_COMMANDS];
-       __u8 next_free_slot;
-       __u8 next_cmd;
-} sctp_cmd_seq_t;
-
-
-/* Create a new sctp_command_sequence.
- * Return NULL if creating a new sequence fails.
- */
-sctp_cmd_seq_t *sctp_new_cmd_seq(int priority);
-
-/* Initialize a block of memory as a command sequence.
- * Return 0 if the initialization fails.
- */
-int sctp_init_cmd_seq(sctp_cmd_seq_t *seq);
-
-/* Add a command to an sctp_cmd_seq_t.
- * Return 0 if the command sequence is full.
- *
- * Use the SCTP_* constructors defined by SCTP_ARG_CONSTRUCTOR() above
- * to wrap data which goes in the obj argument.
- */
-int sctp_add_cmd(sctp_cmd_seq_t *seq, sctp_verb_t verb, sctp_arg_t obj);
-
-/* Rewind an sctp_cmd_seq_t to iterate from the start.
- * Return 0 if the rewind fails.
- */
-int sctp_rewind_sequence(sctp_cmd_seq_t *seq);
-
-/* Return the next command structure in an sctp_cmd_seq.
- * Return NULL at the end of the sequence.
- */
-sctp_cmd_t *sctp_next_cmd(sctp_cmd_seq_t *seq);
-
-/* Dispose of a command sequence.  */
-void sctp_free_cmd_seq(sctp_cmd_seq_t *seq);
-
-#endif /* __net_sctp_command_h__ */
-
diff --git a/include/net/sctp/sctp_constants.h b/include/net/sctp/sctp_constants.h
deleted file mode 100644 (file)
index 58e910d..0000000
+++ /dev/null
@@ -1,453 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001-2002 International Business Machines Corp.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * This file is part of the implementation of the add-IP extension,
- * based on <draft-ietf-tsvwg-addip-sctp-02.txt> June 29, 2001,
- * for the SCTP kernel reference Implementation.
- * 
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/include/net/sctp/sctp_constants.h,v 1.11 2002/07/26 22:52:32 jgrimm Exp $
- * 
- * The SCTP reference implementation  is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * the SCTP reference implementation  is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty
- *                 ************************
- * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
- * GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to one of the following email
- * addresses:
- * 
- * La Monte H.P. Yarroll <piggy@acm.org>
- * Karl Knutson <karl@athena.chicago.il.us>
- * Randall Stewart <randall@stewart.chicago.il.us>
- * Ken Morneau <kmorneau@cisco.com>
- * Qiaobing Xie <qxie1@motorola.com>
- * Xingang Guo <xingang.guo@intel.com>
- * Sridhar Samudrala <samudrala@us.ibm.com>
- * Daisy Chang <daisyc@us.ibm.com>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- * 
- * There are still LOTS of bugs in this code... I always run on the motto
- * "it is a wonder any code ever works :)"
- * 
- * 
- */
-
-#ifndef __sctp_constants_h__
-#define __sctp_constants_h__
-
-#include <linux/tcp.h>  /* For TCP states used in sctp_sock_state_t */
-#include <linux/sctp.h>
-#include <linux/ipv6.h> /* For ipv6hdr. */
-#include <net/sctp/sctp_user.h>
-
-/* 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
- * for ipv6 headers, but this seems worth the simplicity.
- */
-
-#define SCTP_IP_OVERHEAD ((sizeof(struct sctphdr)\
-                          + 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
- * for ipv6 headers, but this seems worth the simplicity.
- */
-
-#define SCTP_IP_OVERHEAD ((sizeof(struct sctphdr)\
-                          + sizeof(struct ipv6hdr)\
-                          + MAX_HEADER))
-
-/* Since CIDs are sparse, we need all four of the following
- * symbols.  CIDs are dense through SCTP_CID_BASE_MAX.
- */
-#define SCTP_CID_BASE_MAX              SCTP_CID_SHUTDOWN_COMPLETE
-#define SCTP_CID_MAX                   SCTP_CID_ASCONF_ACK
-
-#define SCTP_NUM_BASE_CHUNK_TYPES      (SCTP_CID_BASE_MAX + 1)
-#define SCTP_NUM_CHUNK_TYPES           (SCTP_NUM_BASE_CHUNKTYPES + 2)
-
-
-/* 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;
-
-#define SCTP_EVENT_T_MAX SCTP_EVENT_T_PRIMITIVE
-#define SCTP_EVENT_T_NUM (SCTP_EVENT_T_MAX + 1)
-
-/* As a convenience for the state machine, we append SCTP_EVENT_* and
- * SCTP_ULP_* to the list of possible chunks.
- */
-
-typedef enum {
-
-       SCTP_EVENT_TIMEOUT_NONE = 0,
-       SCTP_EVENT_TIMEOUT_T1_COOKIE,
-       SCTP_EVENT_TIMEOUT_T1_INIT,
-       SCTP_EVENT_TIMEOUT_T2_SHUTDOWN,
-       SCTP_EVENT_TIMEOUT_T3_RTX,
-       SCTP_EVENT_TIMEOUT_T4_RTO,
-       SCTP_EVENT_TIMEOUT_HEARTBEAT,
-       SCTP_EVENT_TIMEOUT_SACK,
-       SCTP_EVENT_TIMEOUT_AUTOCLOSE,
-       SCTP_EVENT_TIMEOUT_PMTU_RAISE,
-
-} sctp_event_timeout_t;
-
-#define SCTP_EVENT_TIMEOUT_MAX         SCTP_EVENT_TIMEOUT_PMTU_RAISE
-#define SCTP_NUM_TIMEOUT_TYPES         (SCTP_EVENT_TIMEOUT_MAX + 1)
-
-typedef enum {
-
-       SCTP_EVENT_NO_PENDING_TSN = 0,
-       SCTP_EVENT_ICMP_UNREACHFRAG,
-
-} sctp_event_other_t;
-
-#define SCTP_EVENT_OTHER_MAX           SCTP_EVENT_ICMP_UNREACHFRAG
-#define SCTP_NUM_OTHER_TYPES           (SCTP_EVENT_OTHER_MAX + 1)
-
-/* 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_event_primitive_t;
-
-#define SCTP_EVENT_PRIMITIVE_MAX       SCTP_PRIMITIVE_DESTROY
-#define SCTP_NUM_PRIMITIVE_TYPES       (SCTP_EVENT_PRIMITIVE_MAX + 1)
-
-/* We define here a utility type for manipulating subtypes.
- * The subtype constructors all work like this:
- *
- *     sctp_subtype_t foo = SCTP_ST_CHUNK(SCTP_CID_INIT);
- */
-
-typedef union {
-
-       sctp_cid_t chunk;
-       sctp_event_timeout_t timeout;
-       sctp_event_other_t other;
-       sctp_event_primitive_t primitive;
-
-} sctp_subtype_t;
-
-#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_CONSTRUCTOR(CHUNK,                sctp_cid_t,             chunk)
-SCTP_SUBTYPE_CONSTRUCTOR(TIMEOUT,      sctp_event_timeout_t,   timeout)
-SCTP_SUBTYPE_CONSTRUCTOR(OTHER,                sctp_event_other_t,     other)
-SCTP_SUBTYPE_CONSTRUCTOR(PRIMITIVE,    sctp_event_primitive_t, primitive)
-
-
-#define sctp_chunk_is_control(a) (a->chunk_hdr->type != SCTP_CID_DATA)
-#define sctp_chunk_is_data(a) (a->chunk_hdr->type == SCTP_CID_DATA)
-
-/* Calculate the actual data size in a data chunk */
-#define SCTP_DATA_SNDSIZE(c) ((int)((unsigned long)(c->chunk_end)\
-                               - (unsigned long)(c->chunk_hdr)\
-                               - sizeof(sctp_data_chunk_t)))
-
-/* This is a table of printable names of sctp_param_t's.  */
-extern const char *sctp_param_tbl[];
-
-
-#define SCTP_MAX_ERROR_CAUSE  SCTP_ERROR_NONEXIST_IP
-#define SCTP_NUM_ERROR_CAUSE  10
-
-/* 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_t;
-
-
-
-/* SCTP state defines for internal state machine */
-typedef enum {
-
-       SCTP_STATE_EMPTY                = 0,
-       SCTP_STATE_CLOSED               = 1,
-       SCTP_STATE_COOKIE_WAIT          = 2,
-       SCTP_STATE_COOKIE_ECHOED        = 3,
-       SCTP_STATE_ESTABLISHED          = 4,
-       SCTP_STATE_SHUTDOWN_PENDING     = 5,
-       SCTP_STATE_SHUTDOWN_SENT        = 6,
-       SCTP_STATE_SHUTDOWN_RECEIVED    = 7,
-       SCTP_STATE_SHUTDOWN_ACK_SENT    = 8,
-
-} sctp_state_t;
-
-#define SCTP_STATE_MAX                 SCTP_STATE_SHUTDOWN_ACK_SENT
-#define SCTP_STATE_NUM_STATES          (SCTP_STATE_MAX + 1)
-
-/* These are values for sk->state.
- * For a UDP-style SCTP socket, the states are defined as follows
- *   (at this point of time, may change later after more discussions: FIXME)
- * A socket in SCTP_SS_UNCONNECTED state indicates that it is not willing
- * to accept new associations, but it can initiate the creation of new
- * ones.
- * A socket in SCTP_SS_LISTENING state indicates that it is willing to
- * accept new  associations and can initiate the creation of new ones.
- * A socket in SCTP_SS_ESTABLISHED state indicates that it is a peeled off
- * socket with one association.
- */
-typedef enum {
-       SCTP_SS_CLOSED         = TCP_CLOSE,
-       SCTP_SS_LISTENING      = TCP_LISTEN,
-       SCTP_SS_ESTABLISHING   = TCP_SYN_SENT,
-       SCTP_SS_ESTABLISHED    = TCP_ESTABLISHED,
-       SCTP_SS_DISCONNECTING  = TCP_CLOSING,
-} sctp_sock_state_t;
-
-/* These functions map various type to printable names.  */
-const char *sctp_cname(const sctp_subtype_t);  /* chunk types */
-const char *sctp_oname(const sctp_subtype_t);  /* other events */
-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.  */
-extern const char *sctp_state_tbl[], *sctp_evttype_tbl[], *sctp_status_tbl[];
-
-/* SCTP reachability state for each address */
-#define SCTP_ADDR_NOHB                 4
-#define SCTP_ADDR_REACHABLE            2
-#define SCTP_ADDR_NOT_REACHABLE                1
-
-
-
-
-/* Guess at how big to make the TSN mapping array.
- * We guarantee that we can handle at least this big a gap between the
- * cumulative ACK and the highest TSN.  In practice, we can often
- * handle up to twice this value.
- *
- * 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.
- */
-#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.
- */
-#define SCTP_MAX_DUP_TSNS 128
-
-typedef enum {
-       SCTP_COUNTER_INIT_ERROR,
-} sctp_counter_t;
-
-/* How many counters does an association need? */
-#define SCTP_NUMBER_COUNTERS   5
-
-
-/* Here we define the default timers.  */
-
-/* cookie timer def = ? seconds */
-#define SCTP_DEFAULT_TIMEOUT_T1_COOKIE (3 * HZ)
-
-/* init timer def = 3 seconds  */
-#define SCTP_DEFAULT_TIMEOUT_T1_INIT   (3 * HZ)
-
-/* shutdown timer def = 300 ms */
-#define SCTP_DEFAULT_TIMEOUT_T2_SHUTDOWN ((300 * HZ) / 1000)
-
-/* 0 seconds + RTO */
-#define SCTP_DEFAULT_TIMEOUT_HEARTBEAT (10 * HZ)
-
-/* recv timer def = 200ms (in usec) */
-#define SCTP_DEFAULT_TIMEOUT_SACK      ((200 * HZ) / 1000)
-#define SCTP_DEFAULT_TIMEOUT_SACK_MAX  ((500 * HZ) / 1000) /* 500 ms */
-
-/* How long do we wait before attempting to raise the PMTU?  */
-#define SCTP_DEFAULT_TIMEOUT_PMTU_RAISE (10 * 60 * HZ) /* 10 Minutes */
-#define SCTP_DEFAULT_TIMEOUT_PMTU_RAISE_MIN (10 * 60 * HZ) /* 10 Minutes */
-
-/* RTO.Initial              - 3  seconds
- * RTO.Min                  - 1  second
- * RTO.Max                  - 60 seconds
- * RTO.Alpha                - 1/8
- * RTO.Beta                 - 1/4
- */
-#define SCTP_RTO_INITIAL       (3 * HZ)
-#define SCTP_RTO_MIN           (1 * HZ)
-#define SCTP_RTO_MAX           (60 * HZ)
-
-#define SCTP_RTO_ALPHA          3   /* 1/8 when converted to right shifts. */
-#define SCTP_RTO_BETA           2   /* 1/4 when converted to right shifts. */
-
-/* Maximum number of new data packets that can be sent in a burst.  */
-#define SCTP_MAX_BURST         4
-
-#define SCTP_CLOCK_GRANULARITY 1       /* 1 jiffy */
-
-#define SCTP_DEF_MAX_INIT 6
-#define SCTP_DEF_MAX_SEND 10
-
-#define SCTP_DEFAULT_COOKIE_LIFE_SEC   60 /* seconds */
-#define SCTP_DEFAULT_COOKIE_LIFE_USEC  0  /* microseconds */
-
-#define SCTP_DEFAULT_MINWINDOW 1500    /* default minimum rwnd size */
-#define SCTP_DEFAULT_MAXWINDOW 32768   /* default rwnd size */
-#define SCTP_DEFAULT_MAXSEGMENT 1500   /* MTU size, this is the limit
-                                         * to which we will raise the P-MTU.
-                                        */
-#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
-                                        * secret will live?
-                                        */
-#define SCTP_SECRET_SIZE 32            /* Number of octets in a 256 bits. */
-
-#define SCTP_SIGNATURE_SIZE 20         /* size of a SLA-1 signature */
-
-#define SCTP_COOKIE_MULTIPLE 64 /* Pad out our cookie to make our hash
-                                * functions simpler to write.
-                                */
-
-/* These return values describe the success or failure of a number of
- * 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_t;
-
-/* These are the commands for manipulating transports.  */
-typedef enum {
-       SCTP_TRANSPORT_UP,
-       SCTP_TRANSPORT_DOWN,
-} 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.
- * At this point, the IPv6 scopes will be mapped to these internal scopes
- * as much as possible.
- */
-typedef enum {
-       SCTP_SCOPE_GLOBAL,              /* IPv4 global addresses */
-       SCTP_SCOPE_PRIVATE,             /* IPv4 private addresses */
-       SCTP_SCOPE_LINK,                /* IPv4 link local address */
-       SCTP_SCOPE_LOOPBACK,            /* IPv4 loopback address */
-       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,
- * 192.88.99.0/24.
- * Also, RFC 8.4, non-unicast addresses are not considered valid SCTP
- * addresses.
- */
-#define IS_IPV4_UNUSABLE_ADDRESS(a) \
-       ((INADDR_BROADCAST == *a) || \
-       (MULTICAST(*a)) || \
-       (((unsigned char *)(a))[0] == 0) || \
-       ((((unsigned char *)(a))[0] == 198) && \
-       (((unsigned char *)(a))[1] == 18) && \
-       (((unsigned char *)(a))[2] == 0)) || \
-       ((((unsigned char *)(a))[0] == 192) && \
-       (((unsigned char *)(a))[1] == 88) && \
-       (((unsigned char *)(a))[2] == 99)))
-
-/* IPv4 Link-local addresses: 169.254.0.0/16.  */
-#define IS_IPV4_LINK_ADDRESS(a) \
-       ((((unsigned char *)(a))[0] == 169) && \
-       (((unsigned char *)(a))[1] == 254))
-
-/* RFC 1918 "Address Allocation for Private Internets" defines the IPv4
- * private address space as the following:
- *
- * 10.0.0.0 - 10.255.255.255 (10/8 prefix)
- * 172.16.0.0.0 - 172.31.255.255 (172.16/12 prefix)
- * 192.168.0.0 - 192.168.255.255 (192.168/16 prefix)
- */
-#define IS_IPV4_PRIVATE_ADDRESS(a) \
-       ((((unsigned char *)(a))[0] == 10) || \
-       ((((unsigned char *)(a))[0] == 172) && \
-       (((unsigned char *)(a))[1] >= 16) && \
-       (((unsigned char *)(a))[1] < 32)) || \
-       ((((unsigned char *)(a))[0] == 192) && \
-       (((unsigned char *)(a))[1] == 168)))
-
-/* Flags used for the bind address copy functions.  */
-#define SCTP_ADDR6_ALLOWED     0x00000001      /* IPv6 address is allowed by
-                                                  local sock family */
-#define SCTP_ADDR4_PEERSUPP    0x00000002      /* IPv4 address is supported by
-                                                  peer */
-#define SCTP_ADDR6_PEERSUPP    0x00000004      /* IPv6 address is supported by
-                                                  peer */
-
-/* Reasons to lower cwnd. */
-typedef enum {
-       SCTP_LOWER_CWND_T3_RTX,
-       SCTP_LOWER_CWND_FAST_RTX,
-       SCTP_LOWER_CWND_ECNE,
-       SCTP_LOWER_CWND_INACTIVE,
-} sctp_lower_cwnd_t;
-
-#endif /* __sctp_constants_h__ */
-
diff --git a/include/net/sctp/sctp_sla1.h b/include/net/sctp/sctp_sla1.h
deleted file mode 100644 (file)
index 9e88f67..0000000
+++ /dev/null
@@ -1,80 +0,0 @@
-/* SCTP reference Implementation 
- * Copyright (C) 1999 Cisco, Inc. 
- * Copyright (C) 1999 Motorola, Inc.
- *
- * This file originates from Randy Stewart's SCTP reference Implementation.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    Randy Stewart <rstewar1@email.mot.com>
- *    Ken Morneau   <kmorneau@cisco.com> 
- *    Qiaobing Xie  <qxie1@email.mot.com>
- */
-
-#ifndef __SLA1_h__
-#define __SLA1_h__
-
-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;
-};
-
-
-#define F1(B,C,D) (((B & C) | ((~B) & D)))       /* 0  <= t <= 19 */
-#define F2(B,C,D) (B ^ C ^ D)                   /* 20 <= t <= 39 */
-#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 */
-#define K2 0x6ed9eba1       /* 20 <= t <= 39 */
-#define K3 0x8f1bbcdc       /* 40 <= t <= 59 */
-#define K4 0xca62c1d6       /* 60 <= t <= 79 */
-
-#define H0INIT 0x67452301
-#define H1INIT 0xefcdab89
-#define H2INIT 0x98badcfe
-#define H3INIT 0x10325476
-#define H4INIT 0xc3d2e1f0
-
-extern void SLA1_Init(struct SLA_1_Context *);
-extern void SLA1_Process(struct SLA_1_Context *, const unsigned char *, int);
-extern void SLA1_Final(struct SLA_1_Context *, unsigned char *);
-
-#endif
diff --git a/include/net/sctp/sctp_sm.h b/include/net/sctp/sctp_sm.h
deleted file mode 100644 (file)
index 2d1ac58..0000000
+++ /dev/null
@@ -1,420 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001-2002 International Business Machines Corp.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * This file is part of the implementation of the add-IP extension,
- * based on <draft-ietf-tsvwg-addip-sctp-02.txt> June 29, 2001,
- * for the SCTP kernel reference Implementation.
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/include/net/sctp/sctp_sm.h,v 1.34 2002/08/21 18:34:04 jgrimm Exp $
- * 
- * These are definitions needed by the state machine.
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email addresses:
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson <karl@athena.chicago.il.us>
- *    Xingang Guo <xingang.guo@intel.com>
- *    Jon Grimm <jgrimm@us.ibm.com>
- *    Dajiang Zhang <dajiang.zhang@nokia.com>
- *    Sridhar Samudrala <sri@us.ibm.com>
- *    Daisy Chang <daisyc@us.ibm.com>
- *
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-
-
-#include <linux/types.h>
-#include <linux/compiler.h>
-#include <linux/sched.h>
-#include <linux/slab.h>
-#include <linux/in.h>
-#include <net/sctp/sctp_command.h>
-#include <net/sctp/sctp.h>
-
-#ifndef __sctp_sm_h__
-#define __sctp_sm_h__
-
-/*
- * Possible values for the disposition are:
- */
-typedef enum {
-       SCTP_DISPOSITION_DISCARD,        /* No further processing.  */
-       SCTP_DISPOSITION_CONSUME,        /* Process return values normally.  */
-       SCTP_DISPOSITION_NOMEM,          /* We ran out of memory--recover.  */
-       SCTP_DISPOSITION_DELETE_TCB,     /* Close the association.  */
-       SCTP_DISPOSITION_ABORT,          /* Close the association NOW.  */
-       SCTP_DISPOSITION_VIOLATION,      /* The peer is misbehaving.  */
-       SCTP_DISPOSITION_NOT_IMPL,       /* This entry is not implemented.  */
-       SCTP_DISPOSITION_ERROR,          /* This is plain old user error.  */
-       SCTP_DISPOSITION_BUG,            /* This is a bug.  */
-} sctp_disposition_t;
-
-typedef struct {
-       int name;
-       int action;
-} sctp_sm_command_t;
-
-typedef sctp_disposition_t (sctp_state_fn_t) (const sctp_endpoint_t *,
-                                             const sctp_association_t *,
-                                             const sctp_subtype_t type,
-                                             void *arg,
-                                             sctp_cmd_seq_t *);
-typedef void (sctp_timer_event_t) (unsigned long);
-typedef struct {
-       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
- * currently in use.
- */
-
-/* Prototypes for generic state functions. */
-sctp_state_fn_t sctp_sf_not_impl;
-sctp_state_fn_t sctp_sf_bug;
-
-/* Prototypes for gener timer state functions. */
-sctp_state_fn_t sctp_sf_timer_ignore;
-
-/* Prototypes for chunk state functions. */
-sctp_state_fn_t sctp_sf_do_9_1_abort;
-sctp_state_fn_t sctp_sf_cookie_wait_abort;
-sctp_state_fn_t sctp_sf_cookie_echoed_abort;
-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_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_eat_sack_6_2;
-sctp_state_fn_t sctp_sf_tabort_8_4_8;
-sctp_state_fn_t sctp_sf_operr_notify;
-sctp_state_fn_t sctp_sf_t1_timer_expire;
-sctp_state_fn_t sctp_sf_t2_timer_expire;
-sctp_state_fn_t sctp_sf_sendbeat_8_3;
-sctp_state_fn_t sctp_sf_beat_8_3;
-sctp_state_fn_t sctp_sf_backbeat_8_3;
-sctp_state_fn_t sctp_sf_do_9_2_final;
-sctp_state_fn_t sctp_sf_do_9_2_shutdown;
-sctp_state_fn_t sctp_sf_do_ecn_cwr;
-sctp_state_fn_t sctp_sf_do_ecne;
-sctp_state_fn_t sctp_sf_ootb;
-sctp_state_fn_t sctp_sf_shut_8_4_5;
-sctp_state_fn_t sctp_sf_pdiscard;
-sctp_state_fn_t sctp_sf_violation;
-sctp_state_fn_t sctp_sf_discard_chunk;
-sctp_state_fn_t sctp_sf_do_5_2_1_siminit;
-sctp_state_fn_t sctp_sf_do_5_2_2_dupinit;
-sctp_state_fn_t sctp_sf_do_5_2_4_dupcook;
-
-/* Prototypes for primitive event state functions.  */
-sctp_state_fn_t sctp_sf_do_prm_asoc;
-sctp_state_fn_t sctp_sf_do_prm_send;
-sctp_state_fn_t sctp_sf_do_9_2_prm_shutdown;
-sctp_state_fn_t sctp_sf_cookie_wait_prm_shutdown;
-sctp_state_fn_t sctp_sf_cookie_echoed_prm_shutdown;
-sctp_state_fn_t sctp_sf_do_9_1_prm_abort;
-sctp_state_fn_t sctp_sf_cookie_wait_prm_abort;
-sctp_state_fn_t sctp_sf_cookie_echoed_prm_abort;
-sctp_state_fn_t sctp_sf_error_closed;
-sctp_state_fn_t sctp_sf_error_shutdown;
-sctp_state_fn_t sctp_sf_ignore_primitive;
-
-/* Prototypes for other event state functions.  */
-sctp_state_fn_t sctp_sf_do_9_2_start_shutdown;
-sctp_state_fn_t sctp_sf_do_9_2_shutdown_ack;
-sctp_state_fn_t sctp_sf_ignore_other;
-
-/* Prototypes for timeout event state functions.  */
-sctp_state_fn_t sctp_sf_do_6_3_3_rtx;
-sctp_state_fn_t sctp_sf_do_6_2_sack;
-sctp_state_fn_t sctp_sf_autoclose_timer_expire;
-
-
-/* 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_sf_do_9_2_reshutack;
-sctp_state_fn_t sctp_sf_do_9_2_reshut;
-sctp_state_fn_t sctp_sf_do_9_2_shutack;
-
-sctp_state_fn_t lucky;
-sctp_state_fn_t other_stupid;
-
-/* Prototypes for timeout event state functions.  Not in use. */
-sctp_state_fn_t sctp_do_4_2_reinit;
-sctp_state_fn_t sctp_do_4_3_reecho;
-sctp_state_fn_t sctp_do_9_2_reshut;
-sctp_state_fn_t sctp_do_9_2_reshutack;
-sctp_state_fn_t sctp_do_8_3_hb_err;
-sctp_state_fn_t sctp_heartoff;
-
-/* Prototypes for addip related state functions.  Not in use. */
-sctp_state_fn_t sctp_addip_do_asconf;
-sctp_state_fn_t sctp_addip_do_asconf_ack;
-
-/* Prototypes for utility support functions.  */
-__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);
-
-time_t timeval_sub(struct timeval *, struct timeval *);
-sctp_association_t *sctp_make_temp_asoc(const sctp_endpoint_t *,
-                                       sctp_chunk_t *,
-                                       const int priority);
-__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(__u8 *cookie, __u32 curTag, __u32 hisTag);
-
-/* Prototypes for chunk-building functions.  */
-sctp_chunk_t *sctp_make_init(const sctp_association_t *,
-                            const sctp_bind_addr_t *,
-                            int priority);
-sctp_chunk_t *sctp_make_init_ack(const sctp_association_t *,
-                                const sctp_chunk_t *,
-                                const int priority);
-sctp_chunk_t *sctp_make_cookie_echo(const sctp_association_t *,
-                                   const sctp_chunk_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 __u32 lowest_tsn,
-                                const sctp_chunk_t *);
-sctp_chunk_t *sctp_make_datafrag(sctp_association_t *,
-                                const struct sctp_sndrcvinfo *sinfo,
-                                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 __u8 flags,
-                                       __u16 ssn);
-sctp_chunk_t *sctp_make_data(sctp_association_t *,
-                            const struct sctp_sndrcvinfo *sinfo,
-                            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 __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 *, __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 *,
-                                     __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_heartbeat_ack(const sctp_association_t *,
-                                     const sctp_chunk_t *,
-                                     const void *payload,
-                                     const size_t paylen);
-sctp_chunk_t *sctp_make_op_error(const sctp_association_t *,
-                                const sctp_chunk_t *chunk,
-                                __u16 cause_code,
-                                const void *payload,
-                                size_t paylen);
-void sctp_chunk_assign_tsn(sctp_chunk_t *);
-
-
-/* Prototypes for statetable processing. */
-
-int sctp_do_sm(sctp_event_t event_type, sctp_subtype_t subtype,
-              sctp_state_t state,
-               sctp_endpoint_t *,
-               sctp_association_t *asoc,
-               void *event_arg,
-               int priority);
-
-int sctp_side_effects(sctp_event_t event_type, sctp_subtype_t subtype,
-                     sctp_state_t state,
-                      sctp_endpoint_t *,
-                      sctp_association_t *asoc,
-                      void *event_arg,
-                      sctp_disposition_t status,
-                     sctp_cmd_seq_t *commands,
-                      int priority);
-
-/* 2nd level prototypes */
-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 *retval,
-                    int priority);
-
-
-int sctp_gen_sack(sctp_association_t *, int force, sctp_cmd_seq_t *);
-void sctp_do_TSNdup(sctp_association_t *, sctp_chunk_t *, long gap);
-
-void sctp_generate_t3_rtx_event(unsigned long peer);
-void sctp_generate_heartbeat_event(unsigned long peer);
-
-sctp_sackhdr_t *sctp_sm_pull_sack(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 __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 */
-__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,
-                        __u16 port);
-int sctp_addr2sockaddr(const sctpParam_t, sockaddr_storage_t *);
-int sockaddr2sctp_addr(const sockaddr_storage_t *, sctpParam_t);
-
-/* Extern declarations for major data structures.  */
-sctp_sm_table_entry_t *sctp_chunk_event_lookup(sctp_cid_t, sctp_state_t);
-extern sctp_sm_table_entry_t
-primitive_event_table[SCTP_NUM_PRIMITIVE_TYPES][SCTP_STATE_NUM_STATES];
-extern sctp_sm_table_entry_t
-other_event_table[SCTP_NUM_OTHER_TYPES][SCTP_STATE_NUM_STATES];
-extern sctp_sm_table_entry_t
-timeout_event_table[SCTP_NUM_TIMEOUT_TYPES][SCTP_STATE_NUM_STATES];
-extern sctp_timer_event_t *sctp_timer_events[SCTP_NUM_TIMEOUT_TYPES];
-
-/* These are some handy utility macros... */
-
-
-/* Get the size of a DATA chunk payload. */
-static inline __u16 sctp_data_size(sctp_chunk_t *chunk)
-{
-       __u16 size;
-
-       size = ntohs(chunk->chunk_hdr->length);
-       size -= sizeof(sctp_data_chunk_t);
-
-       return size;
-}
-
-/* Compare two TSNs */
-
-/* RFC 1982 - Serial Number Arithmetic
- *
- * 2. Comparison
- *  Then, s1 is said to be equal to s2 if and only if i1 is equal to i2,
- *  in all other cases, s1 is not equal to s2.
- *
- * s1 is said to be less than s2 if, and only if, s1 is not equal to s2,
- * and
- *
- *      (i1 < i2 and i2 - i1 < 2^(SERIAL_BITS - 1)) or
- *      (i1 > i2 and i1 - i2 > 2^(SERIAL_BITS - 1))
- *
- * s1 is said to be greater than s2 if, and only if, s1 is not equal to
- * s2, and
- *
- *      (i1 < i2 and i2 - i1 > 2^(SERIAL_BITS - 1)) or
- *      (i1 > i2 and i1 - i2 < 2^(SERIAL_BITS - 1))
- */
-
-/*
- * RFC 2960
- *  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.
- */
-
-enum {
-       TSN_SIGN_BIT = (1<<31)
-};
-
-static inline int TSN_lt(__u32 s, __u32 t)
-{
-       return (((s) - (t)) & TSN_SIGN_BIT);
-}
-
-static inline int TSN_lte(__u32 s, __u32 t)
-{
-       return (((s) == (t)) || (((s) - (t)) & TSN_SIGN_BIT));
-}
-
-/* Compare two SSNs */
-
-/*
- * RFC 2960
- *  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
- * SERIAL_BITS = 16.
- */
-enum {
-       SSN_SIGN_BIT = (1<<15)
-};
-
-static inline int SSN_lt(__u16 s, __u16 t)
-{
-       return (((s) - (t)) & SSN_SIGN_BIT);
-}
-
-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)
-{
-       if (unlikely(!sctp_add_cmd(seq, verb, obj)))
-               BUG();
-}
-
-#endif /* __sctp_sm_h__ */
diff --git a/include/net/sctp/sctp_structs.h b/include/net/sctp/sctp_structs.h
deleted file mode 100644 (file)
index bb77157..0000000
+++ /dev/null
@@ -1,1543 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001 International Business Machines Corp.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/include/net/sctp/sctp_structs.h,v 1.21 2002/08/16 19:30:49 jgrimm Exp $
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email addresses:
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    Randall Stewart       <randall@sctp.chicago.il.us>
- *    Ken Morneau           <kmorneau@cisco.com>
- *    Qiaobing Xie          <qxie1@email.mot.com>
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson          <karl@athena.chicago.il.us>
- *    Jon Grimm             <jgrimm@us.ibm.com>
- *    Xingang Guo           <xingang.guo@intel.com>
- *    Hui Huang             <hui.huang@nokia.com>
- *    Sridhar Samudrala     <sri@us.ibm.com>
- *    Daisy Chang          <daisyc@us.ibm.com>
- *    Dajiang Zhang         <dajiang.zhang@nokia.com> 
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-
-#ifndef __sctp_structs_h__
-#define __sctp_structs_h__
-
-#include <linux/time.h>                /* We get struct timespec.    */
-#include <linux/socket.h>      /* linux/in.h needs this!!    */
-#include <linux/in.h>          /* We get struct sockaddr_in. */
-#include <linux/in6.h>          /* We get struct in6_addr     */
-#include <asm/param.h>         /* We get MAXHOSTNAMELEN.     */
-#include <asm/atomic.h>                /* This gets us atomic counters.  */
-#include <linux/skbuff.h>      /* We need sk_buff_head. */
-#include <linux/tqueue.h>      /* We need tq_struct.    */
-#include <linux/sctp.h>         /* We need sctp* header structs.  */
-
-/*
- * 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 (__s64))
-                               /* Implementation specific desired alignment */
-/*
- * Definitions used for sockaddr_storage structure paddings design.
- */
-#define _SS_PAD1SIZE   (_SS_ALIGNSIZE - sizeof (sa_family_t))
-#define _SS_PAD2SIZE   (_SS_MAXSIZE - (sizeof (sa_family_t)+ \
-                              _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 */
-       __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.
- * We should wean ourselves off this.
- */
-typedef union {
-       struct sockaddr_in v4;
-       struct sockaddr_in6 v6;
-       struct sockaddr sa;
-} sockaddr_storage_t;
-
-
-/* Forward declarations for data structures. */
-struct SCTP_protocol;
-struct SCTP_endpoint;
-struct SCTP_association;
-struct SCTP_transport;
-struct SCTP_packet;
-struct SCTP_chunk;
-struct SCTP_inqueue;
-struct SCTP_outqueue;
-struct SCTP_bind_addr;
-struct sctp_opt;
-struct sctp_endpoint_common;
-
-
-typedef struct SCTP_protocol sctp_protocol_t;
-typedef struct SCTP_endpoint sctp_endpoint_t;
-typedef struct SCTP_association sctp_association_t;
-typedef struct SCTP_transport sctp_transport_t;
-typedef struct SCTP_packet sctp_packet_t;
-typedef struct SCTP_chunk sctp_chunk_t;
-typedef struct SCTP_inqueue sctp_inqueue_t;
-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;
-
-#include <net/sctp/sctp_tsnmap.h>
-#include <net/sctp/sctp_ulpevent.h>
-#include <net/sctp/sctp_ulpqueue.h>
-
-
-/* Structures useful for managing bind/connect. */
-
-typedef struct sctp_bind_bucket {
-       unsigned short  port;
-       unsigned short  fastreuse;
-       struct sctp_bind_bucket *next;
-       struct sctp_bind_bucket **pprev;
-       struct sock             *sk;
-} sctp_bind_bucket_t;
-
-typedef struct sctp_bind_hashbucket {
-       spinlock_t      lock;
-       struct sctp_bind_bucket *chain;
-} sctp_bind_hashbucket_t;
-
-/* Used for hashing all associations.  */
-typedef struct sctp_hashbucket {
-       rwlock_t        lock;
-       sctp_endpoint_common_t  *chain;
-} sctp_hashbucket_t __attribute__((__aligned__(8)));
-
-
-/* The SCTP protocol structure. */
-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.)
-        */
-       __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.
-        */
-       int rto_alpha;
-       int rto_beta;
-
-       /* Max.Burst                - 4 */
-       int max_burst;
-
-       /* 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
-        */
-       int max_retrans_association;
-       int max_retrans_path;
-       int max_retrans_init;
-
-       /* HB.interval              - 30 seconds  */
-       int hb_interval;
-
-       /* The following variables are implementation specific.  */
-
-       /* Default initialization values to be applied to new associations. */
-       __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. */
-       int ep_hashsize;
-       sctp_hashbucket_t *ep_hashbucket;
-
-       /* This is the hash of all associations. */
-       int assoc_hashsize;
-       sctp_hashbucket_t *assoc_hashbucket;
-
-       /* This is the sctp port control hash.  */
-       int port_hashsize;
-       int port_rover;
-       spinlock_t port_alloc_lock;  /* Protects port_rover. */
-       sctp_bind_hashbucket_t *port_hashtable;
-
-       /* This is the global local address list.
-        * We actively maintain this complete list of interfaces on
-        * the system by catching routing events.
-        *
-        * It is a list of struct sockaddr_storage_list.
-        */
-       list_t local_addr_list;
-       spinlock_t local_addr_lock;
-};
-
-
-/*
- * Pointers to address related SCTP functions.
- * (i.e. things that depend on the address family.)
- */
-typedef struct sctp_func {
-       int             (*queue_xmit)   (struct sk_buff *skb);
-       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 *optlen);
-       int             (*get_dst_mtu)  (const sockaddr_storage_t *address);
-       __u16           net_header_len; 
-       int             sockaddr_len;
-       sa_family_t     sa_family;
-       list_t          list;
-} sctp_func_t;
-
-sctp_func_t *sctp_get_af_specific(const sockaddr_storage_t *address);
-
-/* SCTP Socket type: UDP or TCP style. */
-typedef enum {
-       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 is our base endpointer? */
-       sctp_endpoint_t *ep;
-
-       /* Various Socket Options.  */
-       __u16 default_stream;
-       __u32 default_ppid;
-       struct sctp_initmsg initmsg;
-       struct sctp_rtoinfo rtoinfo;
-       struct sctp_paddrparams paddrparam;
-       struct sctp_event_subscribe subscribe;
-       __u32 autoclose;
-       __u8 nodelay;
-       __u8 disable_fragments;
-};
-
-
-
-/* This is our APPLICATION-SPECIFIC state cookie.
- * THIS IS NOT DICTATED BY THE SPECIFICATION.
- */
-/* 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         :
-         */
-        __u32 my_vtag;
-
-        /* Peer's      : Tag expected in every outbound packet except
-         * Verification: in the INIT chunk.
-         * Tag         :
-         */
-        __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. */
-       __u32 my_ttag;
-
-       /* Peer's Tie Tag: Assist in discovering a restarting association. */
-       __u32 peer_ttag;
-
-        /* When does this cookie expire? */
-        struct timeval expiration;
-
-       /* Number of inbound/outbound streams which are set
-        * and negotiated during the INIT process. */
-       __u16 sinit_num_ostreams;
-       __u16 sinit_max_instreams;
-
-        /* This is the first sequence number I used.  */
-       __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
-        * raw_addr_list_len field, which will be used at the time when
-        * the association TCB is re-constructed from the cookie.
-        */
-       __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 {
-       __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 {
-       __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
- * internally.
- */
-typedef union {
-       sctp_ipv4addr_param_t v4;
-       sctp_ipv6addr_param_t v6;
-} sctpIpAddress_t;
-
-/* RFC 2960.  Section 3.3.5 Heartbeat.
- *    Heartbeat Information: variable length
- *    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).
- */
-typedef struct sctp_sender_hb_info {
-       sctp_paramhdr_t param_hdr;
-       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.
- *
- * As a matter of convenience, we remember the SCTP common header for
- * 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;
-
-       /* 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
-        * fragments. For the remaining fragments, this is the link to the
-        * frag_list maintained in the first fragment.
-        */
-       struct list_head frag_list;
-
-       /* This points to the sk_buff containing the actual data.  */
-       struct sk_buff *skb;
-
-       /* 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 {
-               __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_signed_cookie_t *cookie_hdr;
-               sctp_ecnehdr_t *ecne_hdr;
-               sctp_cwrhdr_t *ecn_cwr_hdr;
-               sctp_errhdr_t *err_hdr;
-       } subh;
-
-       __u8 *chunk_end;
-
-       sctp_chunkhdr_t *chunk_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;
-
-       /* What endpoint received this chunk? */
-       sctp_endpoint_common_t *rcvr;
-
-       /* 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;
-
-       /* For an inbound chunk, this tells us where it came from.
-        * For an outbound chunk, it tells us where we'd like it to
-        * go.  It is NULL if we have no preference.
-        */
-       sctp_transport_t *transport;
-};
-
-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);
-int sctp_user_addto_chunk(sctp_chunk_t *chunk, int len, struct iovec *data);
-sctp_chunk_t *sctp_chunkify(struct sk_buff *, const sctp_association_t *,
-                           struct sock *);
-void sctp_init_source(sctp_chunk_t *chunk);
-const sockaddr_storage_t *sctp_source(const sctp_chunk_t *chunk);
-
-/* This is a structure for holding either an IPv6 or an IPv4 address.  */
-/* sin_family -- AF_INET or AF_INET6
- * sin_port -- ordinary port number
- * sin_addr -- cast to either (struct in_addr) or (struct in6_addr)
- */
-struct sockaddr_storage_list {
-       list_t list;
-       sockaddr_storage_t a;
-};
-
-typedef sctp_chunk_t *(sctp_packet_phandler_t)(sctp_association_t *);
-
-/* This structure holds lists of chunks as we are assembling for
- * transmission.
- */
-struct SCTP_packet {
-       /* These are the SCTP header values (host order) for the packet. */
-       __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).
-        */
-       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;
-};
-
-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 *,
-         __u16 sport,
-         __u16 dport);
-typedef sctp_packet_t *(sctp_outqueue_ohandler_config_t)
-        (sctp_packet_t *,
-        __u32 vtag,
-        int ecn_capable,
-        sctp_packet_phandler_t *get_prepend_chunk);
-typedef sctp_xmit_t (sctp_outqueue_ohandler_t)(sctp_packet_t *,
-                                               sctp_chunk_t *);
-typedef int (sctp_outqueue_ohandler_force_t)(sctp_packet_t *);
-
-sctp_outqueue_ohandler_init_t    sctp_packet_init;
-sctp_outqueue_ohandler_config_t  sctp_packet_config;
-sctp_outqueue_ohandler_t         sctp_packet_append_chunk;
-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
- *      transport address is defined by the combination of an IP address
- *      and an SCTP port number (where SCTP is the Transport protocol).
- *
- * RFC2960 Section 7.1 SCTP Differences from TCP Congestion control
- *
- *   o  The sender keeps a separate congestion control parameter set for
- *      each of the destination addresses it can send to (not each
- *      source-destination pair but for each destination).  The parameters
- *      should decay if the address is not used for a long enough time
- *      period.
- *
- */
-struct SCTP_transport {
-       /* A list of transports. */
-       list_t transports;
-
-       /* Reference counting. */
-       atomic_t refcnt;
-       int      dead;
-
-       /* This is the peer's IP address and port. */
-       sockaddr_storage_t ipaddr;
-
-       /* 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;
-
-       /* 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:
-        */
-       __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,
-        *              the next DATA chunk sent to this destination
-        *              should be used to compute a RTT and this flag
-        *              should be set. Every time the RTT
-        *              calculation completes (i.e. the DATA chunk
-        *              is SACK'd) clear this flag.
-         */
-       int rto_pending;
-
-
-       /*
-        * 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)
-        */
-       __u32 partial_bytes_acked;
-
-       /* Data that has been sent, but not acknowledged. */
-       __u32 flight_size;
-
-       /* PMTU       : The current known path MTU.  */
-       __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;
-
-       /* 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
-        * indication based on ECNE chunk.
-        */
-       unsigned long last_time_ecne_reduced;
-
-       /* state       : The current state of this destination,
-        *             :  i.e. DOWN, UP, ALLOW-HB, NO-HEARTBEAT, etc.
-        */
-       struct {
-               int active;
-               int hb_allowed;
-       } state;
-
-       /* These are the error stats 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
-        * 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;
-
-       /* 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;
-
-       /* We build bundle-able packets for this transport here.  */
-       sctp_packet_t packet;
-
-       /* This is the list of transports that have chunks to send.  */
-       struct list_head send_ready;
-
-       int malloced; /* Is this structure kfree()able? */
-};
-
-extern sctp_transport_t *sctp_transport_new(const sockaddr_storage_t *, int);
-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_destroy(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 *, __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
- * SCTP.  We write packets to it and read chunks from it.  It handles
- * 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 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;
-
-       int malloced;        /* Is this structure kfree()able?  */
-};
-
-sctp_inqueue_t *sctp_inqueue_new(void);
-void sctp_inqueue_init(sctp_inqueue_t *);
-void sctp_inqueue_free(sctp_inqueue_t *);
-void sctp_push_inqueue(sctp_inqueue_t *, sctp_chunk_t *packet);
-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
- * packets (it calls (*output_handler)()).
- *
- * 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.
- *
- * If you feed it SACKs, it will eat them.
- *
- * If you give it big chunks, it will fragment them.
- *
- * It assigns TSN's to data chunks.  This happens at the last possible
- * instant before transmission.
- *
- * When free()'d, it empties itself out via output_handler().
- */
-struct SCTP_outqueue {
-       sctp_association_t *asoc;
-
-       /* 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;
-
-       /* These are control chunks we want to send.  */
-       struct sk_buff_head control;
-
-       /* These are chunks that have been sacked but are above the
-        * CTSN, or cumulative tsn ack point.
-        */
-       struct list_head sacked;
-
-       /* Put chunks on this list to schedule them for
-        * retransmission.
-        */
-       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;
-
-       /* How many unackd bytes do we have in-flight?  */
-       __u32 outstanding_bytes;
-
-       /* Is this structure empty?  */
-       int empty;
-
-       /* Are we kfree()able? */
-       int malloced;
-};
-
-sctp_outqueue_t *sctp_outqueue_new(sctp_association_t *);
-void sctp_outqueue_init(sctp_association_t *, sctp_outqueue_t *);
-void sctp_outqueue_teardown(sctp_outqueue_t *);
-void sctp_outqueue_free(sctp_outqueue_t*);
-void sctp_force_outqueue(sctp_outqueue_t *);
-int sctp_push_outqueue(sctp_outqueue_t *, sctp_chunk_t *chunk);
-int sctp_flush_outqueue(sctp_outqueue_t *, int);
-int sctp_sack_outqueue(sctp_outqueue_t *, sctp_sackhdr_t *);
-int sctp_outqueue_is_empty(const sctp_outqueue_t *);
-int sctp_outqueue_set_output_handlers(sctp_outqueue_t *,
-                                      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);
-void sctp_outqueue_restart(sctp_outqueue_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.
-        */
-       __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.
-        */
-       list_t address_list;
-
-       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 *, __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,
-                       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,
-                                  int *addrs_len,
-                                  int priority);
-int sctp_raw_to_bind_addrs(sctp_bind_addr_t *bp,
-                          __u8 *raw_addr_list,
-                          int addrs_len,
-                          unsigned short port,
-                          int priority);
-
-sctp_scope_t sctp_scope(const sockaddr_storage_t *);
-int sctp_in_scope(const sockaddr_storage_t *addr, const sctp_scope_t scope);
-int sctp_is_any(const sockaddr_storage_t *addr);
-int sctp_addr_is_valid(const sockaddr_storage_t *addr);
-
-
-/* What type of sctp_endpoint_common?  */
-typedef enum {
-       SCTP_EP_TYPE_SOCKET,
-       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.
- * This common structure is useful for several purposes:
- *   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.
- *   4) Disentangle association lookup from endpoint lookup, where we
- *      do not have to find our endpoint to find our association.
- *
- */
-
-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;
-       int hashent;
-
-       /* Runtime type information.  What kind of endpoint is this? */
-       sctp_endpoint_type_t type;
-
-       /* 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?
-        */
-       atomic_t    refcnt;
-       char        dead;
-       char        malloced;
-
-       /* What socket does this endpoint belong to?  */
-       struct sock *sk;
-
-       /* This is where we receive inbound chunks.  */
-       sctp_inqueue_t   inqueue;
-
-       /* 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;
-};
-
-
-/* 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
- *   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
- *   to an SCTP endpoint.
- *
- * From an implementation perspective, each socket has one of these.
- * A TCP-style socket will have exactly one association on one of
- * these.  An UDP-style socket will have multiple associations hanging
- * off one of these.
- */
-
-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;
-
-       /* 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.
-        */
-       /* 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.
-        */
-       __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];
-
-       /* Various thresholds.  */
-
-       /* Name for debugging output... */
-       char *debug_name;
-};
-
-/* Recover the outter endpoint structure. */
-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.
-        */
-       ep = list_entry(base, sctp_endpoint_t, base);
-       return ep;
-}
-
-/* These are function signatures for manipulating endpoints.  */
-sctp_endpoint_t *sctp_endpoint_new(sctp_protocol_t *, struct sock *, int);
-sctp_endpoint_t *sctp_endpoint_init(sctp_endpoint_t *, sctp_protocol_t *,
-                                   struct sock *, int priority);
-void sctp_endpoint_free(sctp_endpoint_t *);
-void sctp_endpoint_put(sctp_endpoint_t *);
-void sctp_endpoint_hold(sctp_endpoint_t *);
-void sctp_endpoint_add_asoc(sctp_endpoint_t *, sctp_association_t *asoc);
-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 *);
-
-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,
-                      const sockaddr_storage_t *peer_addr,
-                      sctp_cid_t cid, int priority);
-__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.
- */
-
-
-/* Here we have information about each individual association. */
-struct SCTP_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.
-        */
-       sctp_endpoint_common_t base;
-
-       /* Associations on the same socket. */
-       list_t asocs;
-
-       /* This is a signature that lets us know that this is a
-        * sctp_association_t data structure.  Used for mapping an
-        * association id to an association.
-        */
-       __u32 eyecatcher;
-
-       /* This is our parent endpoint.  */
-       sctp_endpoint_t *ep;
-
-       /* These are those association elements needed in the cookie.  */
-       sctp_cookie_t c;
-
-       /* This is all information about our peer.  */
-       struct {
-               /* rwnd
-                *
-                * Peer Rwnd   : Current calculated value of the peer's rwnd.
-                */
-               __u32 rwnd;
-
-               /* transport_addr_list
-                *
-                * Peer        : A list of SCTP transport addresses that the
-                * Transport   : peer is bound to. This information is derived
-                * Address     : from the INIT or INIT ACK and is used to
-                * List        : associate an inbound packet with a given
-                *             : association. Normally this information is
-                *             : hashed or keyed for quick lookup and access
-                *             : of the TCB.
-                *
-                * It is a list of SCTP_transport's.
-                */
-               list_t transport_addr_list;
-
-               /* port
-                *   The transport layer port number.
-                */
-               __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.
-                *
-                * All of these paths live on transport_addr_list.
-                *
-                * At the bakeoffs, we discovered that the intent of
-                * primaryPath is that it only changes when the ULP
-                * asks to have it changed.  We add the activePath to
-                * designate the connection we are currently using to
-                * transmit new data and most control chunks.
-                */
-               sctp_transport_t *primary_path;
-
-               /* active_path
-                *   The path that we are currently using to
-                *   transmit new data and most control chunks.
-                */
-               sctp_transport_t *active_path;
-
-               /* retran_path
-                *
-                * RFC2960 6.4 Multi-homed SCTP Endpoints
-                * ...
-                * Furthermore, when its peer is multi-homed, an
-                * endpoint SHOULD try to retransmit a chunk to an
-                * active destination transport address that is
-                * different from the last destination address to
-                * which the DATA chunk was sent.
-                */
-               sctp_transport_t *retran_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;
-
-               /*
-                * Mapping  An array of bits or bytes indicating which out of
-                * Array    order TSN's have been received (relative to the
-                *          Last Rcvd TSN). If no gaps exist, i.e. no out of
-                *          order packets have been received, this array
-                *          will be set to all zero. This structure may be
-                *          in the form of a circular buffer or bit array.
-                *
-                * 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.
-                *
-                * 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().
-                */
-               sctp_tsnmap_t tsn_map;
-               __u8 _map[sctp_tsnmap_storage_size(SCTP_TSN_MAP_SIZE)];
-
-               /* We record duplicate TSNs here.  We clear this after
-                * every SACK.
-                * FIXME: We should move this into the tsnmap? --jgrimm
-                */
-               sctp_dup_tsn_t dup_tsns[SCTP_MAX_DUP_TSNS];
-               int next_dup_tsn;
-
-               /* Do we need to sack the peer? */
-               int sack_needed;
-
-               /* 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) */
-               __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;
-
-       /* When did we enter this state?  */
-       int state_timestamp;
-
-       /* 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.
-        */
-       __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  */
-       __u16 max_init_attempts;
-
-       /* How many times have we resent an INIT? */
-       __u16 init_retries;
-
-       /* The largest timeout or RTO value to use in attempting an INIT */
-       __u16 max_init_timeo;
-
-
-       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).
-        */
-       __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
-        *             : subtracting one from it.
-        *
-        * Most of RFC 2960 refers to this as the Cumulative TSN Ack Point. 
-        */
-
-       __u32 ctsn_ack_point;
-
-       /* The number of unacknowledged data chunks.  Reported through
-        * the SCTP_STATUS sockopt.
-        */
-       __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.
-        */
-       __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.
-        */
-       __u32 rwnd_over;
-
-       /* 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;
-
-       /* This is the wait queue head for send requests waiting on
-        * the association sndbuf space.
-        */
-       wait_queue_head_t       wait;
-
-       /* 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. */
-       __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.
-        */
-       sctp_ulpqueue_t ulpq;
-       __u8 _ssnmap[sctp_ulpqueue_storage_size(SCTP_MAX_STREAM)];
-
-       /* Need to send an ECNE Chunk? */
-       int need_ecne;
-
-       /* Last TSN that caused an ECNE Chunk to be sent.  */
-       __u32 last_ecne_tsn;
-
-       /* Last TSN that caused a CWR Chunk to be sent.  */
-       __u32 last_cwr_tsn;
-
-       /* How many duplicated TSNs have we seen?  */
-       int numduptsns;
-
-       /* Number of seconds of idle time before an association is closed.  */
-       __u32 autoclose;
-
-       /* Name for debugging output... */
-       char *debug_name;
-
-       /* These are to support
-        * "SCTP Extensions for Dynamic Reconfiguration of IP Addresses
-        *  and Enforcement of Flow and Message Limits"
-        * <draft-ietf-tsvwg-addip-sctp-02.txt>
-        * or "ADDIP" for short.
-        */
-
-       /* Is the ADDIP extension enabled for this association? */
-       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
-        * ASCONF Chunk, it MUST wait until the ASCONF-ACK Chunk
-        * returns from the previous ASCONF Chunk before sending a
-        * 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
-        * same serial number) it may just re-transmit the
-        * ASCONF-ACK. It may wish to use the arrival of a new serial
-        * number to discard the previously saved ASCONF-ACK or any
-        * other means it may choose to expire the saved ASCONF-ACK.
-        *
-        * [This is our saved ASCONF-ACK.  We invalidate it when a new
-        * ASCONF serial number arrives.]
-        */
-       sctp_chunk_t *addip_last_asconf_ack;
-
-       /* These ASCONF chunks are waiting to be sent.
-        *
-        * These chunaks can't be pushed to outqueue until receiving
-        * ASCONF_ACK for the previous ASCONF indicated by
-        * addip_last_asconf, so as to guarantee that only one ASCONF
-        * is in flight at any time.
-        *
-        * ADDIP Section 4.1.1 Congestion Control of ASCONF Chunks
-        *
-        * In defining the ASCONF Chunk transfer procedures, it is
-        * 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
-        * ASCONF Chunk, it MUST wait until the ASCONF-ACK Chunk
-        * returns from the previous ASCONF Chunk before sending a
-        * 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]
-        */
-       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.
-        */
-       __u32 addip_serial;
-};
-
-
-/* An eyecatcher for determining if we are really looking at an
- * association data structure.
- */
-enum {
-       SCTP_ASSOC_EYECATCHER = 0xa550c123,
-};
-
-/* Recover the outter association structure. */
-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.
-        */
-       asoc = list_entry(base, sctp_association_t, base);
-       return asoc;
-}
-
-/* These are function signatures for manipulating associations.  */
-
-
-sctp_association_t *
-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,
-                     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 *,
-                                         const sockaddr_storage_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 *, __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);
-
-__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);
-int sctp_cmp_addr_exact(const sockaddr_storage_t *ss1,
-                       const sockaddr_storage_t *ss2);
-sctp_chunk_t *sctp_get_ecne_prepend(sctp_association_t *asoc);
-sctp_chunk_t *sctp_get_no_prepend(sctp_association_t *asoc);
-
-
-/* A convenience structure to parse out SCTP specific CMSGs. */
-typedef struct sctp_cmsgs {
-       struct sctp_initmsg *init;
-       struct sctp_sndrcvinfo *info;
-} sctp_cmsgs_t;
-
-/* Structure for tracking memory objects */
-typedef struct {
-       char *label;
-       atomic_t *counter;
-} sctp_dbg_objcnt_entry_t;
-
-#endif /* __sctp_structs_h__ */
diff --git a/include/net/sctp/sctp_tsnmap.h b/include/net/sctp/sctp_tsnmap.h
deleted file mode 100644 (file)
index 2d75c3f..0000000
+++ /dev/null
@@ -1,161 +0,0 @@
-/* SCTP kernel reference Implementation Copyright (C) 1999-2001
- * Cisco, Motorola, Intel, and International Business Machines Corp.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/include/net/sctp/sctp_tsnmap.h,v 1.8 2002/07/16 14:51:58 jgrimm Exp $
- * 
- * These are the definitions needed for the tsnmap type.  The tsnmap is used
- * to track out of order TSNs received.
- * 
- * The SCTP reference implementation  is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * the SCTP reference implementation  is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to one of the
- * following email addresses:
- * 
- * Jon Grimm <jgrimm@us.ibm.com>
- * La Monte H.P. Yarroll <piggy@acm.org>
- * Karl Knutson <karl@athena.chicago.il.us>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-#include <net/sctp/sctp_constants.h>
-
-#ifndef __sctp_tsnmap_h__
-#define __sctp_tsnmap_h__
-
-
-
-/* RFC 2960 12.2 Parameters necessary per association (i.e. the TCB)
- * Mapping  An array of bits or bytes indicating which out of
- * Array    order TSN's have been received (relative to the
- *          Last Rcvd TSN). If no gaps exist, i.e. no out of
- *          order packets have been received, this array
- *          will be set to all zero. This structure may be
- *          in the form of a circular buffer or bit array.
- */
-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.
-        */
-       __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).
-        */
-       __u32 overflow_tsn;
-
-       /* This is the overflow array for tsn_map.
-        * It points at one of the other ping-pong buffers.
-        */
-       __u8 *overflow_map;
-
-       /* This is the TSN at tsn_map[0].  */
-       __u32 base_tsn;
-
-       /* 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.
-        *
-        * 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.
-        */
-       __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.
-        */
-       __u16 len;
-
-       /* This is the highest TSN we've marked.  */
-       __u32 max_tsn_seen;
-
-       /* No. of data chunks pending receipt. used by SCTP_STATUS sockopt */
-       __u16 pending_data;
-
-       int malloced;
-
-       __u8 raw_map[0];
-} sctp_tsnmap_t;
-
-typedef struct sctp_tsnmap_iter {
-       __u32 start;
-} sctp_tsnmap_iter_t;
-
-
-/* Create a new tsnmap.  */
-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
- * internal buffers.  We double allocate so the overflow map works.
- */
-#define sctp_tsnmap_storage_size(count) (sizeof(__u8) * (count) * 2)
-
-/* Initialize a block of memory as a tsnmap.  */
-sctp_tsnmap_t *sctp_tsnmap_init(sctp_tsnmap_t *map, __u16 len, __u32 initial_tsn);
-
-
-
-/* Test the tracking state of this TSN.
- * Returns:
- *   0 if the TSN has not yet been seen
- *  >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, __u32 tsn);
-
-/* Mark this TSN as seen.  */
-void sctp_tsnmap_mark(sctp_tsnmap_t *map, __u32 tsn);
-
-/* Retrieve the Cumulative TSN ACK Point.  */
-__u32 sctp_tsnmap_get_ctsn(const sctp_tsnmap_t *map);
-
-/* Retrieve the highest TSN we've seen.  */
-__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);
-
-/* Initialize a gap ack block interator from user-provided memory.  */
-void sctp_tsnmap_iter_init(const sctp_tsnmap_t *map, sctp_tsnmap_iter_t *iter);
-
-/* 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,
-                            __u16 *start, __u16 *end);
-
-
-#endif /* __sctp_tsnmap_h__ */
-
-
-
diff --git a/include/net/sctp/sctp_ulpevent.h b/include/net/sctp/sctp_ulpevent.h
deleted file mode 100644 (file)
index 75e8ba9..0000000
+++ /dev/null
@@ -1,137 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 International Business Machines, Corp.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001 Nokia, Inc.
- * Copyright (c) 2001 La Monte H.P. Yarroll
- *
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/include/net/sctp/sctp_ulpevent.h,v 1.5 2002/07/12 14:50:25 jgrimm Exp $
- * 
- * These are the definitions needed for the sctp_ulpevent type.  The 
- * sctp_ulpevent type is used to carry information from the state machine
- * upwards to the ULP. 
- * 
- * The SCTP reference implementation  is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * the SCTP reference implementation  is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to one of the
- * following email addresses:
- * 
- * Jon Grimm <jgrimm@us.ibm.com>
- * La Monte H.P. Yarroll <piggy@acm.org>
- * Karl Knutson <karl@athena.chicago.il.us>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-
-
-#ifndef __sctp_ulpevent_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.
- */
-typedef struct sctp_ulpevent {
-       int malloced;
-       sctp_association_t *asoc;
-       struct sk_buff *parent;
-       struct sctp_sndrcvinfo sndrcvinfo;
-       int chunk_flags; /* Temp. until we get a new chunk_t */
-       int msg_flags;
-} sctp_ulpevent_t;
-
-
-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);
-
-void sctp_ulpevent_free(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,
-                               __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_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,
-                               __u16 flags,
-                               __u32 error,
-                               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);
-
-void sctp_ulpevent_read_sndrcvinfo(const sctp_ulpevent_t *event,
-                                  struct msghdr *msghdr);
-
-__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)
-{
-       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;
-}
-
-
-#endif /* __sctp_ulpevent_h__ */
-
-
-
-
-
-
-
diff --git a/include/net/sctp/sctp_ulpqueue.h b/include/net/sctp/sctp_ulpqueue.h
deleted file mode 100644 (file)
index 83c1b97..0000000
+++ /dev/null
@@ -1,95 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 International Business Machines, Corp.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001 Nokia, Inc.
- * Copyright (c) 2001 La Monte H.P. Yarroll
- *
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/include/net/sctp/sctp_ulpqueue.h,v 1.2 2002/07/12 14:50:25 jgrimm Exp $
- * 
- * These are the definitions needed for the sctp_ulpqueue type.  The 
- * sctp_ulpqueue is the interface between the Upper Layer Protocol, or ULP,
- * and the core SCTP state machine.  This is the component which handles
- * reassembly and ordering.  
- * 
- * The SCTP reference implementation  is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * the SCTP reference implementation  is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to one of the
- * following email addresses:
- * 
- * Jon Grimm <jgrimm@us.ibm.com>
- * La Monte H.P. Yarroll <piggy@acm.org>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-
-#ifndef __sctp_ulpqueue_h__
-#define __sctp_ulpqueue_h__
-
-/* A structure to carry information to the ULP (e.g. Sockets API) */
-typedef struct sctp_ulpqueue {
-       int malloced;
-       spinlock_t lock;
-       sctp_association_t *asoc;
-       struct sk_buff_head reasm;
-       struct sk_buff_head lobby;
-       __u16 ssn[0];
-} sctp_ulpqueue_t;
-
-/* This macro assists in creation of external storage for variable length
- * internal buffers.
- */
-#define sctp_ulpqueue_storage_size(inbound) (sizeof(__u16) * (inbound))
-
-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,
-                                   __u16 inbound);
-
-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);
-
-
-/* Add a new event for propogation to the ULP. */
-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_data_empty(sctp_ulpqueue_t *);
-
-#endif /* __sctp_ulpqueue_h__ */
-
-
-
-
-
-
diff --git a/include/net/sctp/sctp_user.h b/include/net/sctp/sctp_user.h
deleted file mode 100644 (file)
index 54eaadd..0000000
+++ /dev/null
@@ -1,609 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 International Business Machines, Corp.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/include/net/sctp/sctp_user.h,v 1.17 2002/07/29 21:04:23 jgrimm Exp $
- * 
- * This header represents the structures and constants needed to support
- * the SCTP Extension to the Sockets API. 
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll    <piggy@acm.org>
- *    R. Stewart               <randall@sctp.chicago.il.us>
- *    K. Morneau               <kmorneau@cisco.com>
- *    Q. Xie                   <qxie1@email.mot.com>
- *    Karl Knutson             <karl@athena.chicago.il.us>
- *    Jon Grimm                <jgrimm@us.ibm.com>
- *    Daisy Chang              <daisyc@us.ibm.com>
- * 
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-#include <linux/types.h>
-#include <linux/socket.h>
-
-#ifndef __net_sctp_user_h__
-#define __net_sctp_user_h__
-
-
-typedef void * sctp_assoc_t;
-
-/* The following symbols come from the Sockets API Extensions for
- * SCTP <draft-ietf-tsvwg-sctpsocket-04.txt>.
- */
-enum sctp_optname {
-       SCTP_RTOINFO,
-#define SCTP_RTOINFO SCTP_RTOINFO
-       SCTP_ASSOCRTXINFO,
-#define SCTP_ASSOCRTXINFO SCTP_ASSOCRTXINFO
-       SCTP_INITMSG,
-#define SCTP_INITMSG SCTP_INITMSG
-       SCTP_AUTO_CLOSE,
-#define SCTP_AUTO_CLOSE SCTP_AUTO_CLOSE
-       SCTP_SET_PRIMARY_ADDR,
-#define SCTP_SET_PRIMARY_ADDR SCTP_SET_PRIMARY_ADDR
-       SCTP_SET_PEER_PRIMARY_ADDR, 
-#define SCTP_SET_PEER_PRIMARY_ADDR SCTP_SET_PEER_PRIMARY_ADDR
-       SCTP_SET_ADAPTATION_LAYER,      
-#define SCTP_SET_ADAPTATION_LAYER SCTP_SET_ADAPTATION_LAYER
-       SCTP_SET_STREAM_TIMEOUTS,
-#define SCTP_SET_STREAM_TIMEOUTS SCTP_SET_STREAM_TIMEOUTS
-       SCTP_DISABLE_FRAGMENTS,
-#define SCTP_DISABLE_FRAGMENTS SCTP_DISABLE_FRAGMENTS
-       SCTP_SET_PEER_ADDR_PARAMS,
-#define SCTP_SET_PEER_ADDR_PARAMS SCTP_SET_PEER_ADDR_PARAMS
-       SCTP_GET_PEER_ADDR_PARAMS,
-#define SCTP_GET_PEER_ADDR_PARAMS SCTP_GET_PEER_ADDR_PARAMS
-       SCTP_STATUS,
-#define SCTP_STATUS SCTP_STATUS
-       SCTP_GET_PEER_ADDR_INFO,
-#define SCTP_GET_PEER_ADDR_INFO SCTP_GET_PEER_ADDR_INFO
-       SCTP_SET_EVENTS,
-#define SCTP_SET_EVENTS SCTP_SET_EVENTS
-       SCTP_AUTOCLOSE,
-#define SCTP_AUTOCLOSE SCTP_AUTOCLOSE
-       SCTP_SET_DEFAULT_SEND_PARAM,
-#define SCTP_SET_DEFAULT_SEND_PARAM SCTP_SET_DEFAULT_SEND_PARAM
-
-       SCTP_SOCKOPT_DEBUG_NAME = 42, /* FIXME */
-#define SCTP_SOCKOPT_DEBUG_NAME        SCTP_SOCKOPT_DEBUG_NAME
-
-       SCTP_SOCKOPT_BINDX_ADD, /* BINDX requests for adding addresses. */
-#define SCTP_SOCKOPT_BINDX_ADD SCTP_SOCKOPT_BINDX_ADD
-       SCTP_SOCKOPT_BINDX_REM, /* BINDX requests for removing addresses. */
-#define SCTP_SOCKOPT_BINDX_REM SCTP_SOCKOPT_BINDX_REM
-       SCTP_SOCKOPT_PEELOFF,   /* peel off association. */
-#define SCTP_SOCKOPT_PEELOFF   SCTP_SOCKOPT_PEELOFF
-};
-
-
-/*
- * 5.2 SCTP msg_control Structures
- *
- * A key element of all SCTP-specific socket extensions is the use of
- * ancillary data to specify and access SCTP-specific data via the
- * struct msghdr's msg_control member used in sendmsg() and recvmsg().
- * Fine-grained control over initialization and sending parameters are
- * handled with ancillary data.
- *
- * Each ancillary data item is preceeded by a struct cmsghdr (see
- * Section 5.1), which defines the function and purpose of the data
- * 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
- *   recvmsg().
- *
- *   cmsg_level    cmsg_type      cmsg_data[]
- *   ------------  ------------   ----------------------
- *   IPPROTO_SCTP  SCTP_INIT      struct sctp_initmsg
- *
- */
-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().
- *
- *   cmsg_level    cmsg_type      cmsg_data[]
- *   ------------  ------------   ----------------------
- *   IPPROTO_SCTP  SCTP_SNDRCV    struct sctp_sndrcvinfo
- *
- */
-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)
- *
- *   This field may contain any of the following flags and is composed of
- *   a bitwise OR of these values.
- */
-
-enum sctp_sinfo_flags {
-       MSG_UNORDERED = 1,  /* Send/recieve message unordered. */
-       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 */
-};
-
-
-typedef union {
-       __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_cmsg_t;
-
-
-/*
- * 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 notificaion. The notification information has the
- *   following format:
- *
- */
-
-struct sctp_assoc_change {
-       __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;
-};
-
-/*
- *   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.
- */
-enum sctp_sac_state {
-       SCTP_COMM_UP,
-       SCTP_COMM_LOST,
-       SCTP_RESTART,
-       SCTP_SHUTDOWN_COMP,
-       SCTP_CANT_STR_ASSOC,
-};
-
-/*
- * 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 {
-       __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;
-};
-
-/*
- *    spc_state:  32 bits (signed integer)
- *
- *   This field holds one of a number of values that communicate the
- *   event that happened to the address.  They include:
- */
-enum sctp_spc_state {
-       ADDRESS_AVAILABLE,
-       ADDRESS_UNREACHABLE,
-       ADDRESS_REMOVED,
-       ADDRESS_ADDED,
-       ADDRESS_MADE_PRIM,
-};
-
-
-/*
- * 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
- *   association. The entire error TLV as it appears on the wire is
- *   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 error TLVs have the format:
- */
-struct sctp_remote_error {
-       __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 {
-       __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];
-};
-
-/*
- *   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.
- *
- *   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.
- */
-
-enum sctp_ssf_flags {
-       SCTP_DATA_UNSENT,
-       SCTP_DATA_SENT,
-};
-
-/*
- * 5.3.1.5 SCTP_SHUTDOWN_EVENT
- *
- *   When a peer sends a SHUTDOWN, SCTP delivers this notification to
- *   inform the application that it should cease sending data.
- */
-
-struct sctp_shutdown_event {
-       __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
- *   that of the peers requested adaption layer.
- */
-struct sctp_adaption_event {
-       __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.
- */
-
-struct sctp_rcv_pdapi_event {
-       __u16 pdapi_type;
-       __u16 pdapi_flags;
-       __u32 pdapi_length;
-       __u32 pdapi_indication;
-       sctp_assoc_t pdapi_assoc_id;
-};
-
-
-/*
- * Described in Section 7.3
- *   Ancillary Data and Notification Interest Options
- */
-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
- *   notification types.
- *
- */
-union sctp_notification {
-       struct {
-               __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_send_failed sn_send_failed;
-       struct sctp_shutdown_event sn_shutdown_event;
-       struct sctp_adaption_event sn_adaption_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.
- * Values from 2^15 and down are reserved.
- */
-
-enum sctp_sn_type {
-       SCTP_SN_TYPE_BASE     = (1<<15),
-       SCTP_ASSOC_CHANGE,
-       SCTP_PEER_ADDR_CHANGE,
-       SCTP_REMOTE_ERROR,
-       SCTP_SEND_FAILED,
-       SCTP_SHUTDOWN_EVENT,
-       SCTP_PARTIAL_DELIVERY_EVENT,
-       SCTP_ADAPTION_INDICATION,
-};
-
-/* 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
- * sockets API specification.
- */
-typedef enum sctp_sn_error {
-       SCTP_FAILED_THRESHOLD,
-       SCTP_RECEIVED_SACK,
-       SCTP_HEARTBEAT_SUCCESS,
-       SCTP_RESPONSE_TO_USER_REQ,
-       SCTP_INTERNAL_ERROR,
-       SCTP_SHUTDOWN_GUARD_EXPIRES,
-       SCTP_PEER_FAULTY,
-} sctp_sn_error_t;
-
-/*
- *
- * 7.1.14 Peer Address Parameters
- *
- *   Applications can enable or disable heartbeats for any peer address
- *   of an association, modify an address's heartbeat interval, force a
- *   heartbeat to be sent immediately, and adjust the address's maximum
- *   number of retransmissions sent before an address is considered
- *   unreachable. The following structure is used to access and modify an
- *   address's parameters:
- */
-
-struct sctp_paddrparams {
-       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
- *   of an association, including its reachability state, congestion
- *   window, and retransmission timer values.  This information is
- *   read-only. The following structure is used to access this
- *   information:
- */
-
-struct sctp_paddrinfo {
-       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
- *   timeout (RTO) are tunable.  See [SCTP] for more information on how
- *   these parameters are used in RTO calculation.  The peer address
- *   parameter is ignored for TCP style socket.
- */
-
-struct sctp_rtoinfo {
-       __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
- *   sent before an association is considered unreachable.
- *   See [SCTP] for more information on how this parameter is used.  The
- *   peer address parameter is ignored for TCP style socket.
- */
-
-struct sctp_assocparams {
-       __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
- *  primary. The enclosed address must be one of the association's
- *  locally bound addresses. The following structure is used to make a
- *   set primary request:
- */
-
-struct sctp_setprim {
-       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
- *  the association primary. The enclosed address must be one of the
- *  association peer's addresses. The following structure is used to
- *  make a set peer primary request:
- */
-
-struct sctp_setpeerprim {
-       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
- *   association, including association state, peer receiver window size,
- *   number of unacked data chunks, and number of data chunks pending
- *   receipt.  This information is read-only.  The following structure is
- *   used to access this information:
- */
-struct sctp_status {
-       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
- * Indication parameter for all future
- *  INIT and INIT-ACK exchanges.
- */
-
-struct sctp_setadaption {
-       __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;
-       __u32           ssto_timeout;
-       __u16           ssto_streamid_start;
-       __u16           ssto_streamid_end;
-};
-
-
-/* These are bit fields for msghdr->msg_flags.  See section 5.1.  */
-/* On user space Linux, these live in <bits/socket.h> as an enum.  */
-enum sctp_msg_flags {
-       MSG_NOTIFICATION = 0x8000,
-#define MSG_NOTIFICATION MSG_NOTIFICATION
-};
-
-/*
- * 8.1 sctp_bindx()
- *
- * 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
- * getsockopt(SCTP_SOCKOPT_PEELOFF).
- */
-typedef struct {
-       sctp_assoc_t associd;
-       int sd;
-} sctp_peeloff_arg_t;
-
-#endif /* __net_sctp_user_h__ */
-
-
-
diff --git a/include/net/sctp/sla1.h b/include/net/sctp/sla1.h
new file mode 100644 (file)
index 0000000..9e88f67
--- /dev/null
@@ -0,0 +1,80 @@
+/* SCTP reference Implementation 
+ * Copyright (C) 1999 Cisco, Inc. 
+ * Copyright (C) 1999 Motorola, Inc.
+ *
+ * This file originates from Randy Stewart's SCTP reference Implementation.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    Randy Stewart <rstewar1@email.mot.com>
+ *    Ken Morneau   <kmorneau@cisco.com> 
+ *    Qiaobing Xie  <qxie1@email.mot.com>
+ */
+
+#ifndef __SLA1_h__
+#define __SLA1_h__
+
+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;
+};
+
+
+#define F1(B,C,D) (((B & C) | ((~B) & D)))       /* 0  <= t <= 19 */
+#define F2(B,C,D) (B ^ C ^ D)                   /* 20 <= t <= 39 */
+#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 */
+#define K2 0x6ed9eba1       /* 20 <= t <= 39 */
+#define K3 0x8f1bbcdc       /* 40 <= t <= 59 */
+#define K4 0xca62c1d6       /* 60 <= t <= 79 */
+
+#define H0INIT 0x67452301
+#define H1INIT 0xefcdab89
+#define H2INIT 0x98badcfe
+#define H3INIT 0x10325476
+#define H4INIT 0xc3d2e1f0
+
+extern void SLA1_Init(struct SLA_1_Context *);
+extern void SLA1_Process(struct SLA_1_Context *, const unsigned char *, int);
+extern void SLA1_Final(struct SLA_1_Context *, unsigned char *);
+
+#endif
diff --git a/include/net/sctp/sm.h b/include/net/sctp/sm.h
new file mode 100644 (file)
index 0000000..27c055e
--- /dev/null
@@ -0,0 +1,418 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001-2002 International Business Machines Corp.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * This file is part of the implementation of the add-IP extension,
+ * based on <draft-ietf-tsvwg-addip-sctp-02.txt> June 29, 2001,
+ * for the SCTP kernel reference Implementation.
+ * 
+ * These are definitions needed by the state machine.
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email addresses:
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson <karl@athena.chicago.il.us>
+ *    Xingang Guo <xingang.guo@intel.com>
+ *    Jon Grimm <jgrimm@us.ibm.com>
+ *    Dajiang Zhang <dajiang.zhang@nokia.com>
+ *    Sridhar Samudrala <sri@us.ibm.com>
+ *    Daisy Chang <daisyc@us.ibm.com>
+ *
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+
+#include <linux/types.h>
+#include <linux/compiler.h>
+#include <linux/sched.h>
+#include <linux/slab.h>
+#include <linux/in.h>
+#include <net/sctp/command.h>
+#include <net/sctp/sctp.h>
+
+#ifndef __sctp_sm_h__
+#define __sctp_sm_h__
+
+/*
+ * Possible values for the disposition are:
+ */
+typedef enum {
+       SCTP_DISPOSITION_DISCARD,        /* No further processing.  */
+       SCTP_DISPOSITION_CONSUME,        /* Process return values normally.  */
+       SCTP_DISPOSITION_NOMEM,          /* We ran out of memory--recover.  */
+       SCTP_DISPOSITION_DELETE_TCB,     /* Close the association.  */
+       SCTP_DISPOSITION_ABORT,          /* Close the association NOW.  */
+       SCTP_DISPOSITION_VIOLATION,      /* The peer is misbehaving.  */
+       SCTP_DISPOSITION_NOT_IMPL,       /* This entry is not implemented.  */
+       SCTP_DISPOSITION_ERROR,          /* This is plain old user error.  */
+       SCTP_DISPOSITION_BUG,            /* This is a bug.  */
+} sctp_disposition_t;
+
+typedef struct {
+       int name;
+       int action;
+} sctp_sm_command_t;
+
+typedef sctp_disposition_t (sctp_state_fn_t) (const sctp_endpoint_t *,
+                                             const sctp_association_t *,
+                                             const sctp_subtype_t type,
+                                             void *arg,
+                                             sctp_cmd_seq_t *);
+typedef void (sctp_timer_event_t) (unsigned long);
+typedef struct {
+       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
+ * currently in use.
+ */
+
+/* Prototypes for generic state functions. */
+sctp_state_fn_t sctp_sf_not_impl;
+sctp_state_fn_t sctp_sf_bug;
+
+/* Prototypes for gener timer state functions. */
+sctp_state_fn_t sctp_sf_timer_ignore;
+
+/* Prototypes for chunk state functions. */
+sctp_state_fn_t sctp_sf_do_9_1_abort;
+sctp_state_fn_t sctp_sf_cookie_wait_abort;
+sctp_state_fn_t sctp_sf_cookie_echoed_abort;
+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_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_eat_sack_6_2;
+sctp_state_fn_t sctp_sf_tabort_8_4_8;
+sctp_state_fn_t sctp_sf_operr_notify;
+sctp_state_fn_t sctp_sf_t1_timer_expire;
+sctp_state_fn_t sctp_sf_t2_timer_expire;
+sctp_state_fn_t sctp_sf_sendbeat_8_3;
+sctp_state_fn_t sctp_sf_beat_8_3;
+sctp_state_fn_t sctp_sf_backbeat_8_3;
+sctp_state_fn_t sctp_sf_do_9_2_final;
+sctp_state_fn_t sctp_sf_do_9_2_shutdown;
+sctp_state_fn_t sctp_sf_do_ecn_cwr;
+sctp_state_fn_t sctp_sf_do_ecne;
+sctp_state_fn_t sctp_sf_ootb;
+sctp_state_fn_t sctp_sf_shut_8_4_5;
+sctp_state_fn_t sctp_sf_pdiscard;
+sctp_state_fn_t sctp_sf_violation;
+sctp_state_fn_t sctp_sf_discard_chunk;
+sctp_state_fn_t sctp_sf_do_5_2_1_siminit;
+sctp_state_fn_t sctp_sf_do_5_2_2_dupinit;
+sctp_state_fn_t sctp_sf_do_5_2_4_dupcook;
+
+/* Prototypes for primitive event state functions.  */
+sctp_state_fn_t sctp_sf_do_prm_asoc;
+sctp_state_fn_t sctp_sf_do_prm_send;
+sctp_state_fn_t sctp_sf_do_9_2_prm_shutdown;
+sctp_state_fn_t sctp_sf_cookie_wait_prm_shutdown;
+sctp_state_fn_t sctp_sf_cookie_echoed_prm_shutdown;
+sctp_state_fn_t sctp_sf_do_9_1_prm_abort;
+sctp_state_fn_t sctp_sf_cookie_wait_prm_abort;
+sctp_state_fn_t sctp_sf_cookie_echoed_prm_abort;
+sctp_state_fn_t sctp_sf_error_closed;
+sctp_state_fn_t sctp_sf_error_shutdown;
+sctp_state_fn_t sctp_sf_ignore_primitive;
+
+/* Prototypes for other event state functions.  */
+sctp_state_fn_t sctp_sf_do_9_2_start_shutdown;
+sctp_state_fn_t sctp_sf_do_9_2_shutdown_ack;
+sctp_state_fn_t sctp_sf_ignore_other;
+
+/* Prototypes for timeout event state functions.  */
+sctp_state_fn_t sctp_sf_do_6_3_3_rtx;
+sctp_state_fn_t sctp_sf_do_6_2_sack;
+sctp_state_fn_t sctp_sf_autoclose_timer_expire;
+
+
+/* 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_sf_do_9_2_reshutack;
+sctp_state_fn_t sctp_sf_do_9_2_reshut;
+sctp_state_fn_t sctp_sf_do_9_2_shutack;
+
+sctp_state_fn_t lucky;
+sctp_state_fn_t other_stupid;
+
+/* Prototypes for timeout event state functions.  Not in use. */
+sctp_state_fn_t sctp_do_4_2_reinit;
+sctp_state_fn_t sctp_do_4_3_reecho;
+sctp_state_fn_t sctp_do_9_2_reshut;
+sctp_state_fn_t sctp_do_9_2_reshutack;
+sctp_state_fn_t sctp_do_8_3_hb_err;
+sctp_state_fn_t sctp_heartoff;
+
+/* Prototypes for addip related state functions.  Not in use. */
+sctp_state_fn_t sctp_addip_do_asconf;
+sctp_state_fn_t sctp_addip_do_asconf_ack;
+
+/* Prototypes for utility support functions.  */
+__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);
+
+time_t timeval_sub(struct timeval *, struct timeval *);
+sctp_association_t *sctp_make_temp_asoc(const sctp_endpoint_t *,
+                                       sctp_chunk_t *,
+                                       const int priority);
+__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(__u8 *cookie, __u32 curTag, __u32 hisTag);
+
+/* Prototypes for chunk-building functions.  */
+sctp_chunk_t *sctp_make_init(const sctp_association_t *,
+                            const sctp_bind_addr_t *,
+                            int priority);
+sctp_chunk_t *sctp_make_init_ack(const sctp_association_t *,
+                                const sctp_chunk_t *,
+                                const int priority);
+sctp_chunk_t *sctp_make_cookie_echo(const sctp_association_t *,
+                                   const sctp_chunk_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 __u32 lowest_tsn,
+                                const sctp_chunk_t *);
+sctp_chunk_t *sctp_make_datafrag(sctp_association_t *,
+                                const struct sctp_sndrcvinfo *sinfo,
+                                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 __u8 flags,
+                                       __u16 ssn);
+sctp_chunk_t *sctp_make_data(sctp_association_t *,
+                            const struct sctp_sndrcvinfo *sinfo,
+                            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 __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 *, __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 *,
+                                     __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_heartbeat_ack(const sctp_association_t *,
+                                     const sctp_chunk_t *,
+                                     const void *payload,
+                                     const size_t paylen);
+sctp_chunk_t *sctp_make_op_error(const sctp_association_t *,
+                                const sctp_chunk_t *chunk,
+                                __u16 cause_code,
+                                const void *payload,
+                                size_t paylen);
+void sctp_chunk_assign_tsn(sctp_chunk_t *);
+
+
+/* Prototypes for statetable processing. */
+
+int sctp_do_sm(sctp_event_t event_type, sctp_subtype_t subtype,
+              sctp_state_t state,
+               sctp_endpoint_t *,
+               sctp_association_t *asoc,
+               void *event_arg,
+               int priority);
+
+int sctp_side_effects(sctp_event_t event_type, sctp_subtype_t subtype,
+                     sctp_state_t state,
+                      sctp_endpoint_t *,
+                      sctp_association_t *asoc,
+                      void *event_arg,
+                      sctp_disposition_t status,
+                     sctp_cmd_seq_t *commands,
+                      int priority);
+
+/* 2nd level prototypes */
+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 *retval,
+                    int priority);
+
+
+int sctp_gen_sack(sctp_association_t *, int force, sctp_cmd_seq_t *);
+void sctp_do_TSNdup(sctp_association_t *, sctp_chunk_t *, long gap);
+
+void sctp_generate_t3_rtx_event(unsigned long peer);
+void sctp_generate_heartbeat_event(unsigned long peer);
+
+sctp_sackhdr_t *sctp_sm_pull_sack(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 __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 */
+__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,
+                        __u16 port);
+int sctp_addr2sockaddr(const sctpParam_t, sockaddr_storage_t *);
+int sockaddr2sctp_addr(const sockaddr_storage_t *, sctpParam_t);
+
+/* Extern declarations for major data structures.  */
+sctp_sm_table_entry_t *sctp_chunk_event_lookup(sctp_cid_t, sctp_state_t);
+extern sctp_sm_table_entry_t
+primitive_event_table[SCTP_NUM_PRIMITIVE_TYPES][SCTP_STATE_NUM_STATES];
+extern sctp_sm_table_entry_t
+other_event_table[SCTP_NUM_OTHER_TYPES][SCTP_STATE_NUM_STATES];
+extern sctp_sm_table_entry_t
+timeout_event_table[SCTP_NUM_TIMEOUT_TYPES][SCTP_STATE_NUM_STATES];
+extern sctp_timer_event_t *sctp_timer_events[SCTP_NUM_TIMEOUT_TYPES];
+
+/* These are some handy utility macros... */
+
+
+/* Get the size of a DATA chunk payload. */
+static inline __u16 sctp_data_size(sctp_chunk_t *chunk)
+{
+       __u16 size;
+
+       size = ntohs(chunk->chunk_hdr->length);
+       size -= sizeof(sctp_data_chunk_t);
+
+       return size;
+}
+
+/* Compare two TSNs */
+
+/* RFC 1982 - Serial Number Arithmetic
+ *
+ * 2. Comparison
+ *  Then, s1 is said to be equal to s2 if and only if i1 is equal to i2,
+ *  in all other cases, s1 is not equal to s2.
+ *
+ * s1 is said to be less than s2 if, and only if, s1 is not equal to s2,
+ * and
+ *
+ *      (i1 < i2 and i2 - i1 < 2^(SERIAL_BITS - 1)) or
+ *      (i1 > i2 and i1 - i2 > 2^(SERIAL_BITS - 1))
+ *
+ * s1 is said to be greater than s2 if, and only if, s1 is not equal to
+ * s2, and
+ *
+ *      (i1 < i2 and i2 - i1 > 2^(SERIAL_BITS - 1)) or
+ *      (i1 > i2 and i1 - i2 < 2^(SERIAL_BITS - 1))
+ */
+
+/*
+ * RFC 2960
+ *  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.
+ */
+
+enum {
+       TSN_SIGN_BIT = (1<<31)
+};
+
+static inline int TSN_lt(__u32 s, __u32 t)
+{
+       return (((s) - (t)) & TSN_SIGN_BIT);
+}
+
+static inline int TSN_lte(__u32 s, __u32 t)
+{
+       return (((s) == (t)) || (((s) - (t)) & TSN_SIGN_BIT));
+}
+
+/* Compare two SSNs */
+
+/*
+ * RFC 2960
+ *  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
+ * SERIAL_BITS = 16.
+ */
+enum {
+       SSN_SIGN_BIT = (1<<15)
+};
+
+static inline int SSN_lt(__u16 s, __u16 t)
+{
+       return (((s) - (t)) & SSN_SIGN_BIT);
+}
+
+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)
+{
+       if (unlikely(!sctp_add_cmd(seq, verb, obj)))
+               BUG();
+}
+
+#endif /* __sctp_sm_h__ */
diff --git a/include/net/sctp/structs.h b/include/net/sctp/structs.h
new file mode 100644 (file)
index 0000000..3c7189c
--- /dev/null
@@ -0,0 +1,1541 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001 International Business Machines Corp.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email addresses:
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    Randall Stewart       <randall@sctp.chicago.il.us>
+ *    Ken Morneau           <kmorneau@cisco.com>
+ *    Qiaobing Xie          <qxie1@email.mot.com>
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson          <karl@athena.chicago.il.us>
+ *    Jon Grimm             <jgrimm@us.ibm.com>
+ *    Xingang Guo           <xingang.guo@intel.com>
+ *    Hui Huang             <hui.huang@nokia.com>
+ *    Sridhar Samudrala     <sri@us.ibm.com>
+ *    Daisy Chang          <daisyc@us.ibm.com>
+ *    Dajiang Zhang         <dajiang.zhang@nokia.com> 
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#ifndef __sctp_structs_h__
+#define __sctp_structs_h__
+
+#include <linux/time.h>                /* We get struct timespec.    */
+#include <linux/socket.h>      /* linux/in.h needs this!!    */
+#include <linux/in.h>          /* We get struct sockaddr_in. */
+#include <linux/in6.h>          /* We get struct in6_addr     */
+#include <asm/param.h>         /* We get MAXHOSTNAMELEN.     */
+#include <asm/atomic.h>                /* This gets us atomic counters.  */
+#include <linux/skbuff.h>      /* We need sk_buff_head. */
+#include <linux/tqueue.h>      /* We need tq_struct.    */
+#include <linux/sctp.h>         /* We need sctp* header structs.  */
+
+/*
+ * 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 (__s64))
+                               /* Implementation specific desired alignment */
+/*
+ * Definitions used for sockaddr_storage structure paddings design.
+ */
+#define _SS_PAD1SIZE   (_SS_ALIGNSIZE - sizeof (sa_family_t))
+#define _SS_PAD2SIZE   (_SS_MAXSIZE - (sizeof (sa_family_t)+ \
+                              _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 */
+       __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.
+ * We should wean ourselves off this.
+ */
+typedef union {
+       struct sockaddr_in v4;
+       struct sockaddr_in6 v6;
+       struct sockaddr sa;
+} sockaddr_storage_t;
+
+
+/* Forward declarations for data structures. */
+struct SCTP_protocol;
+struct SCTP_endpoint;
+struct SCTP_association;
+struct SCTP_transport;
+struct SCTP_packet;
+struct SCTP_chunk;
+struct SCTP_inqueue;
+struct SCTP_outqueue;
+struct SCTP_bind_addr;
+struct sctp_opt;
+struct sctp_endpoint_common;
+
+
+typedef struct SCTP_protocol sctp_protocol_t;
+typedef struct SCTP_endpoint sctp_endpoint_t;
+typedef struct SCTP_association sctp_association_t;
+typedef struct SCTP_transport sctp_transport_t;
+typedef struct SCTP_packet sctp_packet_t;
+typedef struct SCTP_chunk sctp_chunk_t;
+typedef struct SCTP_inqueue sctp_inqueue_t;
+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;
+
+#include <net/sctp/tsnmap.h>
+#include <net/sctp/ulpevent.h>
+#include <net/sctp/ulpqueue.h>
+
+
+/* Structures useful for managing bind/connect. */
+
+typedef struct sctp_bind_bucket {
+       unsigned short  port;
+       unsigned short  fastreuse;
+       struct sctp_bind_bucket *next;
+       struct sctp_bind_bucket **pprev;
+       struct sock             *sk;
+} sctp_bind_bucket_t;
+
+typedef struct sctp_bind_hashbucket {
+       spinlock_t      lock;
+       struct sctp_bind_bucket *chain;
+} sctp_bind_hashbucket_t;
+
+/* Used for hashing all associations.  */
+typedef struct sctp_hashbucket {
+       rwlock_t        lock;
+       sctp_endpoint_common_t  *chain;
+} sctp_hashbucket_t __attribute__((__aligned__(8)));
+
+
+/* The SCTP protocol structure. */
+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.)
+        */
+       __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.
+        */
+       int rto_alpha;
+       int rto_beta;
+
+       /* Max.Burst                - 4 */
+       int max_burst;
+
+       /* 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
+        */
+       int max_retrans_association;
+       int max_retrans_path;
+       int max_retrans_init;
+
+       /* HB.interval              - 30 seconds  */
+       int hb_interval;
+
+       /* The following variables are implementation specific.  */
+
+       /* Default initialization values to be applied to new associations. */
+       __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. */
+       int ep_hashsize;
+       sctp_hashbucket_t *ep_hashbucket;
+
+       /* This is the hash of all associations. */
+       int assoc_hashsize;
+       sctp_hashbucket_t *assoc_hashbucket;
+
+       /* This is the sctp port control hash.  */
+       int port_hashsize;
+       int port_rover;
+       spinlock_t port_alloc_lock;  /* Protects port_rover. */
+       sctp_bind_hashbucket_t *port_hashtable;
+
+       /* This is the global local address list.
+        * We actively maintain this complete list of interfaces on
+        * the system by catching routing events.
+        *
+        * It is a list of struct sockaddr_storage_list.
+        */
+       list_t local_addr_list;
+       spinlock_t local_addr_lock;
+};
+
+
+/*
+ * Pointers to address related SCTP functions.
+ * (i.e. things that depend on the address family.)
+ */
+typedef struct sctp_func {
+       int             (*queue_xmit)   (struct sk_buff *skb);
+       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 *optlen);
+       int             (*get_dst_mtu)  (const sockaddr_storage_t *address);
+       __u16           net_header_len; 
+       int             sockaddr_len;
+       sa_family_t     sa_family;
+       list_t          list;
+} sctp_func_t;
+
+sctp_func_t *sctp_get_af_specific(const sockaddr_storage_t *address);
+
+/* SCTP Socket type: UDP or TCP style. */
+typedef enum {
+       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 is our base endpointer? */
+       sctp_endpoint_t *ep;
+
+       /* Various Socket Options.  */
+       __u16 default_stream;
+       __u32 default_ppid;
+       struct sctp_initmsg initmsg;
+       struct sctp_rtoinfo rtoinfo;
+       struct sctp_paddrparams paddrparam;
+       struct sctp_event_subscribe subscribe;
+       __u32 autoclose;
+       __u8 nodelay;
+       __u8 disable_fragments;
+};
+
+
+
+/* This is our APPLICATION-SPECIFIC state cookie.
+ * THIS IS NOT DICTATED BY THE SPECIFICATION.
+ */
+/* 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         :
+         */
+        __u32 my_vtag;
+
+        /* Peer's      : Tag expected in every outbound packet except
+         * Verification: in the INIT chunk.
+         * Tag         :
+         */
+        __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. */
+       __u32 my_ttag;
+
+       /* Peer's Tie Tag: Assist in discovering a restarting association. */
+       __u32 peer_ttag;
+
+        /* When does this cookie expire? */
+        struct timeval expiration;
+
+       /* Number of inbound/outbound streams which are set
+        * and negotiated during the INIT process. */
+       __u16 sinit_num_ostreams;
+       __u16 sinit_max_instreams;
+
+        /* This is the first sequence number I used.  */
+       __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
+        * raw_addr_list_len field, which will be used at the time when
+        * the association TCB is re-constructed from the cookie.
+        */
+       __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 {
+       __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 {
+       __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
+ * internally.
+ */
+typedef union {
+       sctp_ipv4addr_param_t v4;
+       sctp_ipv6addr_param_t v6;
+} sctpIpAddress_t;
+
+/* RFC 2960.  Section 3.3.5 Heartbeat.
+ *    Heartbeat Information: variable length
+ *    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).
+ */
+typedef struct sctp_sender_hb_info {
+       sctp_paramhdr_t param_hdr;
+       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.
+ *
+ * As a matter of convenience, we remember the SCTP common header for
+ * 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;
+
+       /* 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
+        * fragments. For the remaining fragments, this is the link to the
+        * frag_list maintained in the first fragment.
+        */
+       struct list_head frag_list;
+
+       /* This points to the sk_buff containing the actual data.  */
+       struct sk_buff *skb;
+
+       /* 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 {
+               __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_signed_cookie_t *cookie_hdr;
+               sctp_ecnehdr_t *ecne_hdr;
+               sctp_cwrhdr_t *ecn_cwr_hdr;
+               sctp_errhdr_t *err_hdr;
+       } subh;
+
+       __u8 *chunk_end;
+
+       sctp_chunkhdr_t *chunk_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;
+
+       /* What endpoint received this chunk? */
+       sctp_endpoint_common_t *rcvr;
+
+       /* 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;
+
+       /* For an inbound chunk, this tells us where it came from.
+        * For an outbound chunk, it tells us where we'd like it to
+        * go.  It is NULL if we have no preference.
+        */
+       sctp_transport_t *transport;
+};
+
+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);
+int sctp_user_addto_chunk(sctp_chunk_t *chunk, int len, struct iovec *data);
+sctp_chunk_t *sctp_chunkify(struct sk_buff *, const sctp_association_t *,
+                           struct sock *);
+void sctp_init_source(sctp_chunk_t *chunk);
+const sockaddr_storage_t *sctp_source(const sctp_chunk_t *chunk);
+
+/* This is a structure for holding either an IPv6 or an IPv4 address.  */
+/* sin_family -- AF_INET or AF_INET6
+ * sin_port -- ordinary port number
+ * sin_addr -- cast to either (struct in_addr) or (struct in6_addr)
+ */
+struct sockaddr_storage_list {
+       list_t list;
+       sockaddr_storage_t a;
+};
+
+typedef sctp_chunk_t *(sctp_packet_phandler_t)(sctp_association_t *);
+
+/* This structure holds lists of chunks as we are assembling for
+ * transmission.
+ */
+struct SCTP_packet {
+       /* These are the SCTP header values (host order) for the packet. */
+       __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).
+        */
+       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;
+};
+
+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 *,
+         __u16 sport,
+         __u16 dport);
+typedef sctp_packet_t *(sctp_outqueue_ohandler_config_t)
+        (sctp_packet_t *,
+        __u32 vtag,
+        int ecn_capable,
+        sctp_packet_phandler_t *get_prepend_chunk);
+typedef sctp_xmit_t (sctp_outqueue_ohandler_t)(sctp_packet_t *,
+                                               sctp_chunk_t *);
+typedef int (sctp_outqueue_ohandler_force_t)(sctp_packet_t *);
+
+sctp_outqueue_ohandler_init_t    sctp_packet_init;
+sctp_outqueue_ohandler_config_t  sctp_packet_config;
+sctp_outqueue_ohandler_t         sctp_packet_append_chunk;
+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
+ *      transport address is defined by the combination of an IP address
+ *      and an SCTP port number (where SCTP is the Transport protocol).
+ *
+ * RFC2960 Section 7.1 SCTP Differences from TCP Congestion control
+ *
+ *   o  The sender keeps a separate congestion control parameter set for
+ *      each of the destination addresses it can send to (not each
+ *      source-destination pair but for each destination).  The parameters
+ *      should decay if the address is not used for a long enough time
+ *      period.
+ *
+ */
+struct SCTP_transport {
+       /* A list of transports. */
+       list_t transports;
+
+       /* Reference counting. */
+       atomic_t refcnt;
+       int      dead;
+
+       /* This is the peer's IP address and port. */
+       sockaddr_storage_t ipaddr;
+
+       /* 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;
+
+       /* 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:
+        */
+       __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,
+        *              the next DATA chunk sent to this destination
+        *              should be used to compute a RTT and this flag
+        *              should be set. Every time the RTT
+        *              calculation completes (i.e. the DATA chunk
+        *              is SACK'd) clear this flag.
+         */
+       int rto_pending;
+
+
+       /*
+        * 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)
+        */
+       __u32 partial_bytes_acked;
+
+       /* Data that has been sent, but not acknowledged. */
+       __u32 flight_size;
+
+       /* PMTU       : The current known path MTU.  */
+       __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;
+
+       /* 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
+        * indication based on ECNE chunk.
+        */
+       unsigned long last_time_ecne_reduced;
+
+       /* state       : The current state of this destination,
+        *             :  i.e. DOWN, UP, ALLOW-HB, NO-HEARTBEAT, etc.
+        */
+       struct {
+               int active;
+               int hb_allowed;
+       } state;
+
+       /* These are the error stats 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
+        * 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;
+
+       /* 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;
+
+       /* We build bundle-able packets for this transport here.  */
+       sctp_packet_t packet;
+
+       /* This is the list of transports that have chunks to send.  */
+       struct list_head send_ready;
+
+       int malloced; /* Is this structure kfree()able? */
+};
+
+extern sctp_transport_t *sctp_transport_new(const sockaddr_storage_t *, int);
+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_destroy(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 *, __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
+ * SCTP.  We write packets to it and read chunks from it.  It handles
+ * 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 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;
+
+       int malloced;        /* Is this structure kfree()able?  */
+};
+
+sctp_inqueue_t *sctp_inqueue_new(void);
+void sctp_inqueue_init(sctp_inqueue_t *);
+void sctp_inqueue_free(sctp_inqueue_t *);
+void sctp_push_inqueue(sctp_inqueue_t *, sctp_chunk_t *packet);
+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
+ * packets (it calls (*output_handler)()).
+ *
+ * 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.
+ *
+ * If you feed it SACKs, it will eat them.
+ *
+ * If you give it big chunks, it will fragment them.
+ *
+ * It assigns TSN's to data chunks.  This happens at the last possible
+ * instant before transmission.
+ *
+ * When free()'d, it empties itself out via output_handler().
+ */
+struct SCTP_outqueue {
+       sctp_association_t *asoc;
+
+       /* 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;
+
+       /* These are control chunks we want to send.  */
+       struct sk_buff_head control;
+
+       /* These are chunks that have been sacked but are above the
+        * CTSN, or cumulative tsn ack point.
+        */
+       struct list_head sacked;
+
+       /* Put chunks on this list to schedule them for
+        * retransmission.
+        */
+       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;
+
+       /* How many unackd bytes do we have in-flight?  */
+       __u32 outstanding_bytes;
+
+       /* Is this structure empty?  */
+       int empty;
+
+       /* Are we kfree()able? */
+       int malloced;
+};
+
+sctp_outqueue_t *sctp_outqueue_new(sctp_association_t *);
+void sctp_outqueue_init(sctp_association_t *, sctp_outqueue_t *);
+void sctp_outqueue_teardown(sctp_outqueue_t *);
+void sctp_outqueue_free(sctp_outqueue_t*);
+void sctp_force_outqueue(sctp_outqueue_t *);
+int sctp_push_outqueue(sctp_outqueue_t *, sctp_chunk_t *chunk);
+int sctp_flush_outqueue(sctp_outqueue_t *, int);
+int sctp_sack_outqueue(sctp_outqueue_t *, sctp_sackhdr_t *);
+int sctp_outqueue_is_empty(const sctp_outqueue_t *);
+int sctp_outqueue_set_output_handlers(sctp_outqueue_t *,
+                                      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);
+void sctp_outqueue_restart(sctp_outqueue_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.
+        */
+       __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.
+        */
+       list_t address_list;
+
+       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 *, __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,
+                       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,
+                                  int *addrs_len,
+                                  int priority);
+int sctp_raw_to_bind_addrs(sctp_bind_addr_t *bp,
+                          __u8 *raw_addr_list,
+                          int addrs_len,
+                          unsigned short port,
+                          int priority);
+
+sctp_scope_t sctp_scope(const sockaddr_storage_t *);
+int sctp_in_scope(const sockaddr_storage_t *addr, const sctp_scope_t scope);
+int sctp_is_any(const sockaddr_storage_t *addr);
+int sctp_addr_is_valid(const sockaddr_storage_t *addr);
+
+
+/* What type of sctp_endpoint_common?  */
+typedef enum {
+       SCTP_EP_TYPE_SOCKET,
+       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.
+ * This common structure is useful for several purposes:
+ *   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.
+ *   4) Disentangle association lookup from endpoint lookup, where we
+ *      do not have to find our endpoint to find our association.
+ *
+ */
+
+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;
+       int hashent;
+
+       /* Runtime type information.  What kind of endpoint is this? */
+       sctp_endpoint_type_t type;
+
+       /* 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?
+        */
+       atomic_t    refcnt;
+       char        dead;
+       char        malloced;
+
+       /* What socket does this endpoint belong to?  */
+       struct sock *sk;
+
+       /* This is where we receive inbound chunks.  */
+       sctp_inqueue_t   inqueue;
+
+       /* 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;
+};
+
+
+/* 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
+ *   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
+ *   to an SCTP endpoint.
+ *
+ * From an implementation perspective, each socket has one of these.
+ * A TCP-style socket will have exactly one association on one of
+ * these.  An UDP-style socket will have multiple associations hanging
+ * off one of these.
+ */
+
+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;
+
+       /* 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.
+        */
+       /* 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.
+        */
+       __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];
+
+       /* Various thresholds.  */
+
+       /* Name for debugging output... */
+       char *debug_name;
+};
+
+/* Recover the outter endpoint structure. */
+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.
+        */
+       ep = list_entry(base, sctp_endpoint_t, base);
+       return ep;
+}
+
+/* These are function signatures for manipulating endpoints.  */
+sctp_endpoint_t *sctp_endpoint_new(sctp_protocol_t *, struct sock *, int);
+sctp_endpoint_t *sctp_endpoint_init(sctp_endpoint_t *, sctp_protocol_t *,
+                                   struct sock *, int priority);
+void sctp_endpoint_free(sctp_endpoint_t *);
+void sctp_endpoint_put(sctp_endpoint_t *);
+void sctp_endpoint_hold(sctp_endpoint_t *);
+void sctp_endpoint_add_asoc(sctp_endpoint_t *, sctp_association_t *asoc);
+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 *);
+
+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,
+                      const sockaddr_storage_t *peer_addr,
+                      sctp_cid_t cid, int priority);
+__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.
+ */
+
+
+/* Here we have information about each individual association. */
+struct SCTP_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.
+        */
+       sctp_endpoint_common_t base;
+
+       /* Associations on the same socket. */
+       list_t asocs;
+
+       /* This is a signature that lets us know that this is a
+        * sctp_association_t data structure.  Used for mapping an
+        * association id to an association.
+        */
+       __u32 eyecatcher;
+
+       /* This is our parent endpoint.  */
+       sctp_endpoint_t *ep;
+
+       /* These are those association elements needed in the cookie.  */
+       sctp_cookie_t c;
+
+       /* This is all information about our peer.  */
+       struct {
+               /* rwnd
+                *
+                * Peer Rwnd   : Current calculated value of the peer's rwnd.
+                */
+               __u32 rwnd;
+
+               /* transport_addr_list
+                *
+                * Peer        : A list of SCTP transport addresses that the
+                * Transport   : peer is bound to. This information is derived
+                * Address     : from the INIT or INIT ACK and is used to
+                * List        : associate an inbound packet with a given
+                *             : association. Normally this information is
+                *             : hashed or keyed for quick lookup and access
+                *             : of the TCB.
+                *
+                * It is a list of SCTP_transport's.
+                */
+               list_t transport_addr_list;
+
+               /* port
+                *   The transport layer port number.
+                */
+               __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.
+                *
+                * All of these paths live on transport_addr_list.
+                *
+                * At the bakeoffs, we discovered that the intent of
+                * primaryPath is that it only changes when the ULP
+                * asks to have it changed.  We add the activePath to
+                * designate the connection we are currently using to
+                * transmit new data and most control chunks.
+                */
+               sctp_transport_t *primary_path;
+
+               /* active_path
+                *   The path that we are currently using to
+                *   transmit new data and most control chunks.
+                */
+               sctp_transport_t *active_path;
+
+               /* retran_path
+                *
+                * RFC2960 6.4 Multi-homed SCTP Endpoints
+                * ...
+                * Furthermore, when its peer is multi-homed, an
+                * endpoint SHOULD try to retransmit a chunk to an
+                * active destination transport address that is
+                * different from the last destination address to
+                * which the DATA chunk was sent.
+                */
+               sctp_transport_t *retran_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;
+
+               /*
+                * Mapping  An array of bits or bytes indicating which out of
+                * Array    order TSN's have been received (relative to the
+                *          Last Rcvd TSN). If no gaps exist, i.e. no out of
+                *          order packets have been received, this array
+                *          will be set to all zero. This structure may be
+                *          in the form of a circular buffer or bit array.
+                *
+                * 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.
+                *
+                * 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().
+                */
+               sctp_tsnmap_t tsn_map;
+               __u8 _map[sctp_tsnmap_storage_size(SCTP_TSN_MAP_SIZE)];
+
+               /* We record duplicate TSNs here.  We clear this after
+                * every SACK.
+                * FIXME: We should move this into the tsnmap? --jgrimm
+                */
+               sctp_dup_tsn_t dup_tsns[SCTP_MAX_DUP_TSNS];
+               int next_dup_tsn;
+
+               /* Do we need to sack the peer? */
+               int sack_needed;
+
+               /* 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) */
+               __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;
+
+       /* When did we enter this state?  */
+       int state_timestamp;
+
+       /* 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.
+        */
+       __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  */
+       __u16 max_init_attempts;
+
+       /* How many times have we resent an INIT? */
+       __u16 init_retries;
+
+       /* The largest timeout or RTO value to use in attempting an INIT */
+       __u16 max_init_timeo;
+
+
+       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).
+        */
+       __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
+        *             : subtracting one from it.
+        *
+        * Most of RFC 2960 refers to this as the Cumulative TSN Ack Point. 
+        */
+
+       __u32 ctsn_ack_point;
+
+       /* The number of unacknowledged data chunks.  Reported through
+        * the SCTP_STATUS sockopt.
+        */
+       __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.
+        */
+       __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.
+        */
+       __u32 rwnd_over;
+
+       /* 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;
+
+       /* This is the wait queue head for send requests waiting on
+        * the association sndbuf space.
+        */
+       wait_queue_head_t       wait;
+
+       /* 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. */
+       __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.
+        */
+       sctp_ulpqueue_t ulpq;
+       __u8 _ssnmap[sctp_ulpqueue_storage_size(SCTP_MAX_STREAM)];
+
+       /* Need to send an ECNE Chunk? */
+       int need_ecne;
+
+       /* Last TSN that caused an ECNE Chunk to be sent.  */
+       __u32 last_ecne_tsn;
+
+       /* Last TSN that caused a CWR Chunk to be sent.  */
+       __u32 last_cwr_tsn;
+
+       /* How many duplicated TSNs have we seen?  */
+       int numduptsns;
+
+       /* Number of seconds of idle time before an association is closed.  */
+       __u32 autoclose;
+
+       /* Name for debugging output... */
+       char *debug_name;
+
+       /* These are to support
+        * "SCTP Extensions for Dynamic Reconfiguration of IP Addresses
+        *  and Enforcement of Flow and Message Limits"
+        * <draft-ietf-tsvwg-addip-sctp-02.txt>
+        * or "ADDIP" for short.
+        */
+
+       /* Is the ADDIP extension enabled for this association? */
+       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
+        * ASCONF Chunk, it MUST wait until the ASCONF-ACK Chunk
+        * returns from the previous ASCONF Chunk before sending a
+        * 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
+        * same serial number) it may just re-transmit the
+        * ASCONF-ACK. It may wish to use the arrival of a new serial
+        * number to discard the previously saved ASCONF-ACK or any
+        * other means it may choose to expire the saved ASCONF-ACK.
+        *
+        * [This is our saved ASCONF-ACK.  We invalidate it when a new
+        * ASCONF serial number arrives.]
+        */
+       sctp_chunk_t *addip_last_asconf_ack;
+
+       /* These ASCONF chunks are waiting to be sent.
+        *
+        * These chunaks can't be pushed to outqueue until receiving
+        * ASCONF_ACK for the previous ASCONF indicated by
+        * addip_last_asconf, so as to guarantee that only one ASCONF
+        * is in flight at any time.
+        *
+        * ADDIP Section 4.1.1 Congestion Control of ASCONF Chunks
+        *
+        * In defining the ASCONF Chunk transfer procedures, it is
+        * 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
+        * ASCONF Chunk, it MUST wait until the ASCONF-ACK Chunk
+        * returns from the previous ASCONF Chunk before sending a
+        * 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]
+        */
+       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.
+        */
+       __u32 addip_serial;
+};
+
+
+/* An eyecatcher for determining if we are really looking at an
+ * association data structure.
+ */
+enum {
+       SCTP_ASSOC_EYECATCHER = 0xa550c123,
+};
+
+/* Recover the outter association structure. */
+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.
+        */
+       asoc = list_entry(base, sctp_association_t, base);
+       return asoc;
+}
+
+/* These are function signatures for manipulating associations.  */
+
+
+sctp_association_t *
+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,
+                     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 *,
+                                         const sockaddr_storage_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 *, __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);
+
+__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);
+int sctp_cmp_addr_exact(const sockaddr_storage_t *ss1,
+                       const sockaddr_storage_t *ss2);
+sctp_chunk_t *sctp_get_ecne_prepend(sctp_association_t *asoc);
+sctp_chunk_t *sctp_get_no_prepend(sctp_association_t *asoc);
+
+
+/* A convenience structure to parse out SCTP specific CMSGs. */
+typedef struct sctp_cmsgs {
+       struct sctp_initmsg *init;
+       struct sctp_sndrcvinfo *info;
+} sctp_cmsgs_t;
+
+/* Structure for tracking memory objects */
+typedef struct {
+       char *label;
+       atomic_t *counter;
+} sctp_dbg_objcnt_entry_t;
+
+#endif /* __sctp_structs_h__ */
diff --git a/include/net/sctp/tsnmap.h b/include/net/sctp/tsnmap.h
new file mode 100644 (file)
index 0000000..af77526
--- /dev/null
@@ -0,0 +1,159 @@
+/* SCTP kernel reference Implementation Copyright (C) 1999-2001
+ * Cisco, Motorola, Intel, and International Business Machines Corp.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * These are the definitions needed for the tsnmap type.  The tsnmap is used
+ * to track out of order TSNs received.
+ * 
+ * The SCTP reference implementation  is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * the SCTP reference implementation  is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to one of the
+ * following email addresses:
+ * 
+ * Jon Grimm <jgrimm@us.ibm.com>
+ * La Monte H.P. Yarroll <piggy@acm.org>
+ * Karl Knutson <karl@athena.chicago.il.us>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+#include <net/sctp/constants.h>
+
+#ifndef __sctp_tsnmap_h__
+#define __sctp_tsnmap_h__
+
+
+
+/* RFC 2960 12.2 Parameters necessary per association (i.e. the TCB)
+ * Mapping  An array of bits or bytes indicating which out of
+ * Array    order TSN's have been received (relative to the
+ *          Last Rcvd TSN). If no gaps exist, i.e. no out of
+ *          order packets have been received, this array
+ *          will be set to all zero. This structure may be
+ *          in the form of a circular buffer or bit array.
+ */
+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.
+        */
+       __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).
+        */
+       __u32 overflow_tsn;
+
+       /* This is the overflow array for tsn_map.
+        * It points at one of the other ping-pong buffers.
+        */
+       __u8 *overflow_map;
+
+       /* This is the TSN at tsn_map[0].  */
+       __u32 base_tsn;
+
+       /* 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.
+        *
+        * 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.
+        */
+       __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.
+        */
+       __u16 len;
+
+       /* This is the highest TSN we've marked.  */
+       __u32 max_tsn_seen;
+
+       /* No. of data chunks pending receipt. used by SCTP_STATUS sockopt */
+       __u16 pending_data;
+
+       int malloced;
+
+       __u8 raw_map[0];
+} sctp_tsnmap_t;
+
+typedef struct sctp_tsnmap_iter {
+       __u32 start;
+} sctp_tsnmap_iter_t;
+
+
+/* Create a new tsnmap.  */
+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
+ * internal buffers.  We double allocate so the overflow map works.
+ */
+#define sctp_tsnmap_storage_size(count) (sizeof(__u8) * (count) * 2)
+
+/* Initialize a block of memory as a tsnmap.  */
+sctp_tsnmap_t *sctp_tsnmap_init(sctp_tsnmap_t *map, __u16 len, __u32 initial_tsn);
+
+
+
+/* Test the tracking state of this TSN.
+ * Returns:
+ *   0 if the TSN has not yet been seen
+ *  >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, __u32 tsn);
+
+/* Mark this TSN as seen.  */
+void sctp_tsnmap_mark(sctp_tsnmap_t *map, __u32 tsn);
+
+/* Retrieve the Cumulative TSN ACK Point.  */
+__u32 sctp_tsnmap_get_ctsn(const sctp_tsnmap_t *map);
+
+/* Retrieve the highest TSN we've seen.  */
+__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);
+
+/* Initialize a gap ack block interator from user-provided memory.  */
+void sctp_tsnmap_iter_init(const sctp_tsnmap_t *map, sctp_tsnmap_iter_t *iter);
+
+/* 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,
+                            __u16 *start, __u16 *end);
+
+
+#endif /* __sctp_tsnmap_h__ */
+
+
+
diff --git a/include/net/sctp/ulpevent.h b/include/net/sctp/ulpevent.h
new file mode 100644 (file)
index 0000000..6e90b83
--- /dev/null
@@ -0,0 +1,135 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 International Business Machines, Corp.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001 Nokia, Inc.
+ * Copyright (c) 2001 La Monte H.P. Yarroll
+ *
+ * These are the definitions needed for the sctp_ulpevent type.  The 
+ * sctp_ulpevent type is used to carry information from the state machine
+ * upwards to the ULP. 
+ * 
+ * The SCTP reference implementation  is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * the SCTP reference implementation  is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to one of the
+ * following email addresses:
+ * 
+ * Jon Grimm <jgrimm@us.ibm.com>
+ * La Monte H.P. Yarroll <piggy@acm.org>
+ * Karl Knutson <karl@athena.chicago.il.us>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+
+#ifndef __sctp_ulpevent_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.
+ */
+typedef struct sctp_ulpevent {
+       int malloced;
+       sctp_association_t *asoc;
+       struct sk_buff *parent;
+       struct sctp_sndrcvinfo sndrcvinfo;
+       int chunk_flags; /* Temp. until we get a new chunk_t */
+       int msg_flags;
+} sctp_ulpevent_t;
+
+
+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);
+
+void sctp_ulpevent_free(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,
+                               __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_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,
+                               __u16 flags,
+                               __u32 error,
+                               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);
+
+void sctp_ulpevent_read_sndrcvinfo(const sctp_ulpevent_t *event,
+                                  struct msghdr *msghdr);
+
+__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)
+{
+       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;
+}
+
+
+#endif /* __sctp_ulpevent_h__ */
+
+
+
+
+
+
+
diff --git a/include/net/sctp/ulpqueue.h b/include/net/sctp/ulpqueue.h
new file mode 100644 (file)
index 0000000..7e7d896
--- /dev/null
@@ -0,0 +1,93 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 International Business Machines, Corp.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001 Nokia, Inc.
+ * Copyright (c) 2001 La Monte H.P. Yarroll
+ * 
+ * These are the definitions needed for the sctp_ulpqueue type.  The 
+ * sctp_ulpqueue is the interface between the Upper Layer Protocol, or ULP,
+ * and the core SCTP state machine.  This is the component which handles
+ * reassembly and ordering.  
+ * 
+ * The SCTP reference implementation  is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * the SCTP reference implementation  is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to one of the
+ * following email addresses:
+ * 
+ * Jon Grimm <jgrimm@us.ibm.com>
+ * La Monte H.P. Yarroll <piggy@acm.org>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#ifndef __sctp_ulpqueue_h__
+#define __sctp_ulpqueue_h__
+
+/* A structure to carry information to the ULP (e.g. Sockets API) */
+typedef struct sctp_ulpqueue {
+       int malloced;
+       spinlock_t lock;
+       sctp_association_t *asoc;
+       struct sk_buff_head reasm;
+       struct sk_buff_head lobby;
+       __u16 ssn[0];
+} sctp_ulpqueue_t;
+
+/* This macro assists in creation of external storage for variable length
+ * internal buffers.
+ */
+#define sctp_ulpqueue_storage_size(inbound) (sizeof(__u16) * (inbound))
+
+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,
+                                   __u16 inbound);
+
+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);
+
+
+/* Add a new event for propogation to the ULP. */
+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_data_empty(sctp_ulpqueue_t *);
+
+#endif /* __sctp_ulpqueue_h__ */
+
+
+
+
+
+
diff --git a/include/net/sctp/user.h b/include/net/sctp/user.h
new file mode 100644 (file)
index 0000000..2bf6d0e
--- /dev/null
@@ -0,0 +1,607 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 International Business Machines, Corp.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * This header represents the structures and constants needed to support
+ * the SCTP Extension to the Sockets API. 
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll    <piggy@acm.org>
+ *    R. Stewart               <randall@sctp.chicago.il.us>
+ *    K. Morneau               <kmorneau@cisco.com>
+ *    Q. Xie                   <qxie1@email.mot.com>
+ *    Karl Knutson             <karl@athena.chicago.il.us>
+ *    Jon Grimm                <jgrimm@us.ibm.com>
+ *    Daisy Chang              <daisyc@us.ibm.com>
+ * 
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+#include <linux/types.h>
+#include <linux/socket.h>
+
+#ifndef __net_sctp_user_h__
+#define __net_sctp_user_h__
+
+
+typedef void * sctp_assoc_t;
+
+/* The following symbols come from the Sockets API Extensions for
+ * SCTP <draft-ietf-tsvwg-sctpsocket-04.txt>.
+ */
+enum sctp_optname {
+       SCTP_RTOINFO,
+#define SCTP_RTOINFO SCTP_RTOINFO
+       SCTP_ASSOCRTXINFO,
+#define SCTP_ASSOCRTXINFO SCTP_ASSOCRTXINFO
+       SCTP_INITMSG,
+#define SCTP_INITMSG SCTP_INITMSG
+       SCTP_AUTO_CLOSE,
+#define SCTP_AUTO_CLOSE SCTP_AUTO_CLOSE
+       SCTP_SET_PRIMARY_ADDR,
+#define SCTP_SET_PRIMARY_ADDR SCTP_SET_PRIMARY_ADDR
+       SCTP_SET_PEER_PRIMARY_ADDR, 
+#define SCTP_SET_PEER_PRIMARY_ADDR SCTP_SET_PEER_PRIMARY_ADDR
+       SCTP_SET_ADAPTATION_LAYER,      
+#define SCTP_SET_ADAPTATION_LAYER SCTP_SET_ADAPTATION_LAYER
+       SCTP_SET_STREAM_TIMEOUTS,
+#define SCTP_SET_STREAM_TIMEOUTS SCTP_SET_STREAM_TIMEOUTS
+       SCTP_DISABLE_FRAGMENTS,
+#define SCTP_DISABLE_FRAGMENTS SCTP_DISABLE_FRAGMENTS
+       SCTP_SET_PEER_ADDR_PARAMS,
+#define SCTP_SET_PEER_ADDR_PARAMS SCTP_SET_PEER_ADDR_PARAMS
+       SCTP_GET_PEER_ADDR_PARAMS,
+#define SCTP_GET_PEER_ADDR_PARAMS SCTP_GET_PEER_ADDR_PARAMS
+       SCTP_STATUS,
+#define SCTP_STATUS SCTP_STATUS
+       SCTP_GET_PEER_ADDR_INFO,
+#define SCTP_GET_PEER_ADDR_INFO SCTP_GET_PEER_ADDR_INFO
+       SCTP_SET_EVENTS,
+#define SCTP_SET_EVENTS SCTP_SET_EVENTS
+       SCTP_AUTOCLOSE,
+#define SCTP_AUTOCLOSE SCTP_AUTOCLOSE
+       SCTP_SET_DEFAULT_SEND_PARAM,
+#define SCTP_SET_DEFAULT_SEND_PARAM SCTP_SET_DEFAULT_SEND_PARAM
+
+       SCTP_SOCKOPT_DEBUG_NAME = 42, /* FIXME */
+#define SCTP_SOCKOPT_DEBUG_NAME        SCTP_SOCKOPT_DEBUG_NAME
+
+       SCTP_SOCKOPT_BINDX_ADD, /* BINDX requests for adding addresses. */
+#define SCTP_SOCKOPT_BINDX_ADD SCTP_SOCKOPT_BINDX_ADD
+       SCTP_SOCKOPT_BINDX_REM, /* BINDX requests for removing addresses. */
+#define SCTP_SOCKOPT_BINDX_REM SCTP_SOCKOPT_BINDX_REM
+       SCTP_SOCKOPT_PEELOFF,   /* peel off association. */
+#define SCTP_SOCKOPT_PEELOFF   SCTP_SOCKOPT_PEELOFF
+};
+
+
+/*
+ * 5.2 SCTP msg_control Structures
+ *
+ * A key element of all SCTP-specific socket extensions is the use of
+ * ancillary data to specify and access SCTP-specific data via the
+ * struct msghdr's msg_control member used in sendmsg() and recvmsg().
+ * Fine-grained control over initialization and sending parameters are
+ * handled with ancillary data.
+ *
+ * Each ancillary data item is preceeded by a struct cmsghdr (see
+ * Section 5.1), which defines the function and purpose of the data
+ * 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
+ *   recvmsg().
+ *
+ *   cmsg_level    cmsg_type      cmsg_data[]
+ *   ------------  ------------   ----------------------
+ *   IPPROTO_SCTP  SCTP_INIT      struct sctp_initmsg
+ *
+ */
+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().
+ *
+ *   cmsg_level    cmsg_type      cmsg_data[]
+ *   ------------  ------------   ----------------------
+ *   IPPROTO_SCTP  SCTP_SNDRCV    struct sctp_sndrcvinfo
+ *
+ */
+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)
+ *
+ *   This field may contain any of the following flags and is composed of
+ *   a bitwise OR of these values.
+ */
+
+enum sctp_sinfo_flags {
+       MSG_UNORDERED = 1,  /* Send/recieve message unordered. */
+       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 */
+};
+
+
+typedef union {
+       __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_cmsg_t;
+
+
+/*
+ * 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 notificaion. The notification information has the
+ *   following format:
+ *
+ */
+
+struct sctp_assoc_change {
+       __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;
+};
+
+/*
+ *   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.
+ */
+enum sctp_sac_state {
+       SCTP_COMM_UP,
+       SCTP_COMM_LOST,
+       SCTP_RESTART,
+       SCTP_SHUTDOWN_COMP,
+       SCTP_CANT_STR_ASSOC,
+};
+
+/*
+ * 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 {
+       __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;
+};
+
+/*
+ *    spc_state:  32 bits (signed integer)
+ *
+ *   This field holds one of a number of values that communicate the
+ *   event that happened to the address.  They include:
+ */
+enum sctp_spc_state {
+       ADDRESS_AVAILABLE,
+       ADDRESS_UNREACHABLE,
+       ADDRESS_REMOVED,
+       ADDRESS_ADDED,
+       ADDRESS_MADE_PRIM,
+};
+
+
+/*
+ * 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
+ *   association. The entire error TLV as it appears on the wire is
+ *   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 error TLVs have the format:
+ */
+struct sctp_remote_error {
+       __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 {
+       __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];
+};
+
+/*
+ *   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.
+ *
+ *   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.
+ */
+
+enum sctp_ssf_flags {
+       SCTP_DATA_UNSENT,
+       SCTP_DATA_SENT,
+};
+
+/*
+ * 5.3.1.5 SCTP_SHUTDOWN_EVENT
+ *
+ *   When a peer sends a SHUTDOWN, SCTP delivers this notification to
+ *   inform the application that it should cease sending data.
+ */
+
+struct sctp_shutdown_event {
+       __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
+ *   that of the peers requested adaption layer.
+ */
+struct sctp_adaption_event {
+       __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.
+ */
+
+struct sctp_rcv_pdapi_event {
+       __u16 pdapi_type;
+       __u16 pdapi_flags;
+       __u32 pdapi_length;
+       __u32 pdapi_indication;
+       sctp_assoc_t pdapi_assoc_id;
+};
+
+
+/*
+ * Described in Section 7.3
+ *   Ancillary Data and Notification Interest Options
+ */
+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
+ *   notification types.
+ *
+ */
+union sctp_notification {
+       struct {
+               __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_send_failed sn_send_failed;
+       struct sctp_shutdown_event sn_shutdown_event;
+       struct sctp_adaption_event sn_adaption_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.
+ * Values from 2^15 and down are reserved.
+ */
+
+enum sctp_sn_type {
+       SCTP_SN_TYPE_BASE     = (1<<15),
+       SCTP_ASSOC_CHANGE,
+       SCTP_PEER_ADDR_CHANGE,
+       SCTP_REMOTE_ERROR,
+       SCTP_SEND_FAILED,
+       SCTP_SHUTDOWN_EVENT,
+       SCTP_PARTIAL_DELIVERY_EVENT,
+       SCTP_ADAPTION_INDICATION,
+};
+
+/* 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
+ * sockets API specification.
+ */
+typedef enum sctp_sn_error {
+       SCTP_FAILED_THRESHOLD,
+       SCTP_RECEIVED_SACK,
+       SCTP_HEARTBEAT_SUCCESS,
+       SCTP_RESPONSE_TO_USER_REQ,
+       SCTP_INTERNAL_ERROR,
+       SCTP_SHUTDOWN_GUARD_EXPIRES,
+       SCTP_PEER_FAULTY,
+} sctp_sn_error_t;
+
+/*
+ *
+ * 7.1.14 Peer Address Parameters
+ *
+ *   Applications can enable or disable heartbeats for any peer address
+ *   of an association, modify an address's heartbeat interval, force a
+ *   heartbeat to be sent immediately, and adjust the address's maximum
+ *   number of retransmissions sent before an address is considered
+ *   unreachable. The following structure is used to access and modify an
+ *   address's parameters:
+ */
+
+struct sctp_paddrparams {
+       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
+ *   of an association, including its reachability state, congestion
+ *   window, and retransmission timer values.  This information is
+ *   read-only. The following structure is used to access this
+ *   information:
+ */
+
+struct sctp_paddrinfo {
+       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
+ *   timeout (RTO) are tunable.  See [SCTP] for more information on how
+ *   these parameters are used in RTO calculation.  The peer address
+ *   parameter is ignored for TCP style socket.
+ */
+
+struct sctp_rtoinfo {
+       __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
+ *   sent before an association is considered unreachable.
+ *   See [SCTP] for more information on how this parameter is used.  The
+ *   peer address parameter is ignored for TCP style socket.
+ */
+
+struct sctp_assocparams {
+       __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
+ *  primary. The enclosed address must be one of the association's
+ *  locally bound addresses. The following structure is used to make a
+ *   set primary request:
+ */
+
+struct sctp_setprim {
+       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
+ *  the association primary. The enclosed address must be one of the
+ *  association peer's addresses. The following structure is used to
+ *  make a set peer primary request:
+ */
+
+struct sctp_setpeerprim {
+       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
+ *   association, including association state, peer receiver window size,
+ *   number of unacked data chunks, and number of data chunks pending
+ *   receipt.  This information is read-only.  The following structure is
+ *   used to access this information:
+ */
+struct sctp_status {
+       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
+ * Indication parameter for all future
+ *  INIT and INIT-ACK exchanges.
+ */
+
+struct sctp_setadaption {
+       __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;
+       __u32           ssto_timeout;
+       __u16           ssto_streamid_start;
+       __u16           ssto_streamid_end;
+};
+
+
+/* These are bit fields for msghdr->msg_flags.  See section 5.1.  */
+/* On user space Linux, these live in <bits/socket.h> as an enum.  */
+enum sctp_msg_flags {
+       MSG_NOTIFICATION = 0x8000,
+#define MSG_NOTIFICATION MSG_NOTIFICATION
+};
+
+/*
+ * 8.1 sctp_bindx()
+ *
+ * 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
+ * getsockopt(SCTP_SOCKOPT_PEELOFF).
+ */
+typedef struct {
+       sctp_assoc_t associd;
+       int sd;
+} sctp_peeloff_arg_t;
+
+#endif /* __net_sctp_user_h__ */
+
+
+
index ff293fd2fac4a30b38f8dc653b5168f88f903147..3e0bb4c052876aa0637c16127ceefbc16dffd91a 100644 (file)
@@ -4,24 +4,24 @@
 
 obj-$(CONFIG_IP_SCTP) += sctp.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
+sctp-y := sm_statetable.o sm_statefuns.o sm_sideeffect.o \
+         protocol.o endpointola.o associola.o \
+         transport.o sm_make_chunk.o ulpevent.o \
+         inqueue.o outqueue.o ulpqueue.o command.o \
+         tsnmap.o bind_addr.o socket.o primitive.o \
+         output.o input.o hashdriver.o sla1.o \
+         debug.o
 
 ifeq ($(CONFIG_SCTP_ADLER32), y)
-sctp-y += sctp_adler32.o
+sctp-y += adler32.o
 else
-sctp-y += sctp_crc32c.o
+sctp-y += crc32c.o
 endif
 
-sctp-$(CONFIG_SCTP_DBG_OBJCNT) += sctp_objcnt.o
-sctp-$(CONFIG_SYSCTL) += sctp_sysctl.o
+sctp-$(CONFIG_SCTP_DBG_OBJCNT) += objcnt.o
+sctp-$(CONFIG_SYSCTL) += sysctl.o
 
-sctp-$(subst m,y,$(CONFIG_IPV6))       += sctp_ipv6.o
+sctp-$(subst m,y,$(CONFIG_IPV6))       += ipv6.o
 
 sctp-objs := $(sctp-y)
 
diff --git a/net/sctp/adler32.c b/net/sctp/adler32.c
new file mode 100644 (file)
index 0000000..1444417
--- /dev/null
@@ -0,0 +1,148 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * This file has direct heritage from the SCTP user-level reference 
+ * implementation by R. Stewart, et al.  These functions implement the
+ * Adler-32 algorithm as specified by RFC 2960. 
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    Randall Stewart <rstewar1@email.mot.com>
+ *    Ken Morneau     <kmorneau@cisco.com>
+ *    Qiaobing Xie    <qxie1@email.mot.com>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+/* 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.
+ *
+ * Well, the above is a bit of a lie, I have optimized this a small
+ * tad, but I have commented the original lines below
+ */
+
+#include <linux/types.h> 
+#include <net/sctp/sctp.h>
+
+#define BASE 65521 /* largest prime smaller than 65536 */
+
+
+/* Performance work as shown this pig to be the
+ * worst CPU wise guy. I have done what I could think
+ * of on my flight to Austrialia but I am sure some
+ * clever assembly could speed this up, but of
+ * course this would require the dreaded #ifdef's for
+ * architecture. If you can speed this up more, pass
+ * it back and we will incorporate it :-)
+ */
+
+unsigned long update_adler32(unsigned long adler,
+                            unsigned char *buf, int len)
+{
+       __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;
+}
+
+__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.
+        */
+       __u32 adler = 1L;
+       __u32 zero = 0L;
+
+       /* Calculate the CRC up to the checksum field. */
+       adler = update_adler32(adler, ptr, 
+                              sizeof(struct sctphdr) - sizeof(__u32));
+       /* Skip over the checksum field. */
+       adler = update_adler32(adler, (unsigned char *) &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;
+}
diff --git a/net/sctp/associola.c b/net/sctp/associola.c
new file mode 100644 (file)
index 0000000..f88e7fe
--- /dev/null
@@ -0,0 +1,1033 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 International Business Machines Corp.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001 La Monte H.P. Yarroll
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * This module provides the abstraction for an SCTP association.
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson          <karl@athena.chicago.il.us>
+ *    Jon Grimm             <jgrimm@us.ibm.com>
+ *    Xingang Guo           <xingang.guo@intel.com>
+ *    Hui Huang             <hui.huang@nokia.com>
+ *    Sridhar Samudrala            <sri@us.ibm.com>
+ *    Daisy Chang          <daisyc@us.ibm.com>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/types.h>
+#include <linux/fcntl.h>
+#include <linux/poll.h>
+#include <linux/init.h>
+#include <linux/sched.h>
+
+#include <linux/sched.h>
+#include <linux/slab.h>
+#include <linux/in.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 *asoc;
+
+       asoc = t_new(sctp_association_t, priority);
+       if (!asoc)
+               goto fail;
+
+       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;
+}
+
+/* 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_opt_t *sp;
+       int i;
+
+       /* Retrieve the SCTP per socket area.  */
+       sp = sctp_sk((struct sock *)sk);
+
+       /* Init all variables to a known value.  */
+       memset(asoc, 0, sizeof(sctp_association_t));
+
+       /* Discarding const is appropriate here.  */
+       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. */
+       asoc->base.type = SCTP_EP_TYPE_ASSOCIATION;
+
+       /* Initialize the object handling fields. */
+       atomic_set(&asoc->base.refcnt, 1);
+       asoc->base.dead = 0;
+       asoc->base.malloced = 0;
+
+       /* 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->pmtu = 0;
+       asoc->frag_point = 0;
+
+       /* Initialize the default association max_retrans and RTO values. */
+       asoc->max_retrans = ep->proto->max_retrans_association;
+       asoc->rto_initial = ep->proto->rto_initial;
+       asoc->rto_max = ep->proto->rto_max;
+       asoc->rto_min = ep->proto->rto_min;
+
+       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.
+        */
+       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;
+       }
+
+       /* 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->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;
+
+       /* 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)
+               asoc->rwnd = SCTP_DEFAULT_MINWINDOW;
+       else
+               asoc->rwnd = sk->rcvbuf;
+
+       asoc->rwnd_over = 0;
+
+       /* 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.peer_vtag = 0;
+       asoc->c.my_ttag   = 0;
+       asoc->c.peer_ttag = 0;
+
+       asoc->c.initial_tsn = sctp_generate_tsn(ep);
+
+       asoc->next_tsn = asoc->c.initial_tsn;
+
+       asoc->ctsn_ack_point = asoc->next_tsn - 1;
+
+       asoc->unack_data = 0;
+
+       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:
+        * ...
+        * 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.
+        */
+       asoc->addip_serial = asoc->c.initial_tsn;
+
+       /* 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
+        *
+        * 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.]
+        */
+       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,
+                                   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))
+               goto fail_init;
+
+       /* 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);
+
+       asoc->need_ecne = 0;
+
+       asoc->debug_name = "unnamedasoc";
+       asoc->eyecatcher = SCTP_ASSOC_EYECATCHER;
+
+       /* Assume that peer would support both address types unless we are
+        * told otherwise.
+        */
+       asoc->peer.ipv4_address = 1;
+       asoc->peer.ipv6_address = 1;
+       INIT_LIST_HEAD(&asoc->asocs);
+
+       asoc->autoclose = sp->autoclose;
+
+       return asoc;
+
+fail_init:
+       sctp_endpoint_put(asoc->ep);
+       sock_put(asoc->base.sk);
+       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)
+{
+       sctp_transport_t *transport;
+       sctp_endpoint_t *ep;
+       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.
+        */
+       asoc->base.dead = 1;
+
+       /* 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);
+
+       /* Dispose of any pending chunks on the inqueue. */
+       sctp_inqueue_free(&asoc->base.inqueue);
+
+       /* 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
+        * should be able to go through and make a guess based
+        * 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]))
+                       sctp_association_put(asoc);
+       }
+
+       /* 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);
+       }
+
+       asoc->eyecatcher = 0;
+
+       sctp_association_put(asoc);
+}
+
+
+/* Cleanup and free up an association. */
+static void sctp_association_destroy(sctp_association_t *asoc)
+{
+       SCTP_ASSERT(asoc->base.dead, "Assoc is not dead", return);
+
+       sctp_endpoint_put(asoc->ep);
+       sock_put(asoc->base.sk);
+
+       if (asoc->base.malloced) {
+               kfree(asoc);
+               SCTP_DBG_OBJCNT_DEC(assoc);
+       }
+}
+
+
+/* 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 *peer;
+       sctp_opt_t *sp;
+       const __u16 *port;
+
+       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;
+       };
+
+       /* Set the port if it has not been set yet.  */
+        if (0 == asoc->peer.port) {
+                asoc->peer.port = *port;
+        }
+
+       SCTP_ASSERT(*port == asoc->peer.port, ":Invalid port\n",
+                   return NULL);
+
+       /* Check to see if this is a duplicate. */
+       peer = sctp_assoc_lookup_paddr(asoc, addr);
+       if (peer)
+               return peer;
+
+       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;
+       }
+
+       SCTP_DEBUG_PRINTK("sctp_assoc_add_peer:association %p PMTU set to "
+                         "%d\n", asoc, 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
+        * initialize the packet structure anyway.
+        */
+       (asoc->outqueue.init_output)(&peer->packet,
+                                    peer,
+                                    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).
+        */
+       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
+        * later when we process the INIT.
+        */
+       peer->ssthresh = SCTP_DEFAULT_MAXWINDOW;
+
+       peer->partial_bytes_acked = 0;
+       peer->flight_size = 0;
+
+       peer->error_threshold = peer->max_retrans;
+
+       /* Update the overall error threshold value of the association
+        * taking the new peer's error threshold into account.
+        */
+       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.
+        */
+       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;
+
+       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 *t;
+       list_t *pos;
+
+       /* Cycle through all transports searching for a peer address. */
+
+       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;
+       }
+
+       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)
+{
+       sctp_transport_t *t = NULL;
+       sctp_transport_t *first;
+       sctp_transport_t *second;
+       sctp_ulpevent_t *event;
+       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();
+       };
+
+       /* Generate and send a SCTP_PEER_ADDR_CHANGE notification to the
+        * user.
+        */
+       event = sctp_ulpevent_make_peer_addr_change(asoc,
+                               (struct sockaddr_storage *) &transport->ipaddr,
+                               0, spc_state, error, GFP_ATOMIC);
+       if (event)
+               sctp_ulpqueue_tail_event(&asoc->ulpq, event);
+
+       /* Select new active and retran paths. */
+
+       /* Look for the two most recently used active transports.
+        *
+        * This code produces the wrong ordering whenever jiffies
+        * rolls over, but we still get usable transports, so we don't
+        * worry about it.
+        */
+       first = NULL; second = NULL;
+
+       list_for_each(pos, &asoc->peer.transport_addr_list) {
+               t = list_entry(pos, sctp_transport_t, transports);
+
+               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)
+                       second = t;
+       }
+
+       /* RFC 2960 6.4 Multi-Homed SCTP Endpoints
+        *
+        * By default, an endpoint should always transmit to the
+        * primary path, unless the SCTP user explicitly specifies the
+        * destination transport address (and possibly source
+        * transport address) to use.
+        *
+        * [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) {
+               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) {
+               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;
+}
+
+/* Hold a reference to an association. */
+void sctp_association_hold(sctp_association_t *asoc)
+{
+       atomic_inc(&asoc->base.refcnt);
+}
+
+/* Release a reference to an association and cleanup
+ * if there are no more references.
+ */
+void sctp_association_put(sctp_association_t *asoc)
+{
+       if (atomic_dec_and_test(&asoc->base.refcnt))
+               sctp_association_destroy(asoc);
+}
+
+/* Allocate the next TSN, Transmission Sequence Number, for the given
+ * association.
+ */
+__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.
+        */
+       __u32 retval = asoc->next_tsn;
+       asoc->next_tsn++;
+       asoc->unack_data++;
+
+        return retval;
+}
+
+/* Allocate 'num' TSNs by incrementing the association's TSN by num. */
+__u32 __sctp_association_get_tsn_block(sctp_association_t *asoc, int num)
+{
+       __u32 retval = asoc->next_tsn;
+
+       asoc->next_tsn += num;
+       asoc->unack_data += num;
+
+       return retval;
+}
+
+/* Fetch the next Stream Sequence Number for stream number 'sid'.  */
+__u16 __sctp_association_get_next_ssn(sctp_association_t *asoc, __u16 sid)
+{
+       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 len;
+       const void *base1;
+       const void *base2;
+
+       if (ss1->sa.sa_family != ss2->sa.sa_family)
+               return 0;
+       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)
+                       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))
+                                   goto match;
+
+                       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;
+}
+
+/* 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 len;
+       const void *base1;
+       const void *base2;
+
+       if (ss1->sa.sa_family != ss2->sa.sa_family)
+               return 0;
+       if (ss1->v4.sin_port != ss2->v4.sin_port)
+               return 0;
+
+       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);
+                       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));
+}
+
+/* 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 *chunk;
+       int need_ecne;
+       __u32 lowest_tsn;
+
+       /* 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;
+
+       if (need_ecne) {
+               chunk = sctp_make_ecne(asoc, lowest_tsn);
+
+               /* ECNE is not mandatory to the flow.  Being unable to
+                * alloc mem is not deadly.  We are just unable to help
+                * out the network.  If we run out of memory, just return
+                * NULL.
+                */
+       } else {
+               chunk = NULL;
+       }
+
+       return chunk;
+}
+
+/* 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)
+{
+       return NULL;
+}
+
+/*
+ * Find which transport this TSN was sent on.
+ */
+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;
+       __u32 key = htonl(tsn);
+
+       match = NULL;
+
+       /*
+        * FIXME: In general, find a more efficient data structure for
+        * 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) {
+               chunk = list_entry(entry, sctp_chunk_t, transmitted_list);
+
+               if (key == chunk->subh.data_hdr->tsn) {
+                       match = active;
+                       goto out;
+               }
+       }
+
+       /* 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;
+               list_for_each(entry, &transport->transmitted) {
+                       chunk = list_entry(entry, sctp_chunk_t,
+                                          transmitted_list);
+                       if (key == chunk->subh.data_hdr->tsn) {
+                               match = transport;
+                               goto out;
+                       }
+               }
+       }
+out:
+       return match;
+}
+
+/* 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 *transport;
+
+       sctp_read_lock(&asoc->base.addr_lock);
+
+       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))
+                       goto out;
+       }
+       transport = NULL;
+
+out:
+       sctp_read_unlock(&asoc->base.addr_lock);
+       return transport;
+}
+
+/* Do delayed input processing.  This is scheduled by sctp_rcv(). */
+static void sctp_assoc_bh_rcv(sctp_association_t *asoc)
+{
+       sctp_endpoint_t *ep;
+       sctp_chunk_t *chunk;
+       struct sock *sk;
+       sctp_inqueue_t *inqueue;
+       int state, subtype;
+       sctp_assoc_t associd = sctp_assoc2id(asoc);
+       int error = 0;
+
+       /* The association should be held so we should be safe. */
+       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;
+
+               /* Remember where the last DATA chunk came from so we
+                * know where to send the SACK.
+                */
+               if (sctp_chunk_is_data(chunk))
+                       asoc->peer.last_data_from = 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),
+                                  state, ep, asoc, chunk, GFP_ATOMIC);
+
+               /* 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;
+       }
+
+err_out:
+       /* Is this the right way to pass errors up to the ULP?  */
+       if (error)
+               sk->err = -error;
+out:
+}
+
+/* This routine moves an association from its old sk to a new sk.  */
+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.
+        */
+       list_del(&assoc->asocs);
+
+       /* Release references to the old endpoint and the sock.  */
+       sctp_endpoint_put(assoc->ep);
+       sock_put(assoc->base.sk);
+
+       /* Get a reference to the new endpoint.  */
+       assoc->ep = newsp->ep;
+       sctp_endpoint_hold(assoc->ep);
+
+       /* Get a reference to the new sock.  */
+       assoc->base.sk = newsk;
+       sock_hold(assoc->base.sk);
+
+       /* Add the association to the new endpoint's list of associations.  */
+       sctp_endpoint_add_asoc(newsp->ep, assoc);
+}
+
+/* Update an association (possibly from unexpected COOKIE-ECHO processing).  */
+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.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,
+                        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) {
+               asoc->next_tsn = new->next_tsn;
+               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++) {
+                       asoc->ssn[i]      = 0;
+                       asoc->ulpq.ssn[i] = 0;
+               }
+       } else {
+               asoc->ctsn_ack_point = asoc->next_tsn - 1;
+       }
+}
+
+/* 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.
+ */
+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)
+               return asoc->peer.active_path;
+
+       /* Otherwise, find the next transport in a round-robin fashion. */
+
+       t = asoc->shutdown_last_sent_to;
+       pos = &t->transports;
+       next = NULL;
+
+       while (1) {
+               /* Skip the head. */
+               if (pos->next == head)
+                       pos = head->next;
+               else
+                       pos = pos->next;
+
+               t = list_entry(pos, sctp_transport_t, transports);
+
+               /* Try to find an active transport. */
+
+               if (t->state.active) {
+                       break;
+               } else {
+                       /* Keep track of the next transport in case
+                        * we don't find any active transport.
+                        */
+                       if (!next)
+                               next = t;
+               }
+
+               /* We have exhausted the list, but didn't find any
+                * other active transports.  If so, use the next
+                * transport.
+                */
+               if (t == asoc->shutdown_last_sent_to) {
+                       t = next;
+                       break;
+               }
+       }
+
+       return t;
+}
diff --git a/net/sctp/bind_addr.c b/net/sctp/bind_addr.c
new file mode 100644 (file)
index 0000000..7f11044
--- /dev/null
@@ -0,0 +1,529 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) Cisco 1999,2000 
+ * Copyright (c) Motorola 1999,2000,2001 
+ * Copyright (c) International Business Machines Corp., 2001
+ * Copyright (c) La Monte H.P. Yarroll 2001
+ *
+ * This file is part of the SCTP kernel reference implementation.
+ * 
+ * A collection class to handle the storage of transport addresses. 
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson          <karl@athena.chicago.il.us>
+ *    Jon Grimm             <jgrimm@us.ibm.com>
+ *    Daisy Chang           <daisyc@us.ibm.com>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/types.h>
+#include <linux/sched.h>
+#include <linux/in.h>
+#include <net/sock.h>
+#include <net/ipv6.h>
+#include <net/if_inet6.h>
+#include <net/sctp/sctp.h>
+#include <net/sctp/sm.h>
+
+/* Forward declarations for internal helpers. */
+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 *);
+
+/* First Level Abstractions. */
+
+/* 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)
+{
+       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;
+       }
+
+out:
+       if (error)
+               sctp_bind_addr_clean(dest);
+       
+       return error;
+}
+
+/* Create a new SCTP_bind_addr from nothing.  */
+sctp_bind_addr_t *sctp_bind_addr_new(int priority)
+{
+       sctp_bind_addr_t *retval;
+
+       retval = t_new(sctp_bind_addr_t, priority);
+       if (!retval)
+               goto nomem;
+
+       sctp_bind_addr_init(retval, 0);
+       retval->malloced = 1;
+       SCTP_DBG_OBJCNT_INC(bind_addr);
+
+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, __u16 port)
+{
+       bp->malloced = 0;
+
+       INIT_LIST_HEAD(&bp->address_list);
+       bp->port = port;
+}
+
+/* Dispose of the address list. */
+static void sctp_bind_addr_clean(sctp_bind_addr_t *bp)
+{
+       struct sockaddr_storage_list *addr;
+       list_t *pos, *temp;
+
+       /* Empty the bind address list. */
+       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);
+       }
+}
+
+/* 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);
+
+       if (bp->malloced) {
+               kfree(bp);
+               SCTP_DBG_OBJCNT_DEC(bind_addr);
+       }
+}
+
+/* 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)
+{
+       struct sockaddr_storage_list *addr;
+
+       /* 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 (!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;
+}
+
+/* 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)
+{
+       list_t *pos, *temp;
+       struct sockaddr_storage_list *addr;
+
+       list_for_each_safe(pos, temp, &bp->address_list) {
+               addr = list_entry(pos, struct sockaddr_storage_list, list);
+               if (sctp_cmp_addr_exact(&addr->a, del_addr)) {
+                       /* Found the exact match. */
+                       list_del(pos);
+                       kfree(addr);
+                       SCTP_DBG_OBJCNT_DEC(addr);
+
+                       return 0;
+               }
+       }
+
+       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 rawaddr;
+       sctpParam_t addrparms;
+       sctpParam_t retval;
+       int addrparms_len;
+       sctpIpAddress_t rawaddr_space;
+       int len;
+       struct sockaddr_storage_list *addr;
+       list_t *pos;
+
+       retval.v = NULL;
+       addrparms_len = 0;
+       len = 0;
+
+       /* Allocate enough memory at once. */
+       list_for_each(pos, &bp->address_list) {
+               len += sizeof(sctp_ipv6addr_param_t);
+       }
+
+       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) {
+               addr = list_entry(pos, struct sockaddr_storage_list, list);
+               len = sockaddr2sctp_addr(&addr->a, rawaddr);
+               memcpy(addrparms.v, rawaddr.v, len);
+               addrparms.v += len;
+               addrparms_len += len;
+       }
+
+end_raw:
+       *addrs_len = addrparms_len;
+       return retval;
+}
+
+/*
+ * 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, __u8 *raw_addr_list,
+                          int addrs_len, __u16 port, int priority)
+{
+       sctpParam_t rawaddr;
+       sockaddr_storage_t addr;
+       int retval = 0;
+       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
+                   || SCTP_PARAM_IPV6_ADDRESS==rawaddr.p->type) {
+                       sctp_param2sockaddr(&addr, rawaddr, port);
+                       retval = sctp_add_bind_addr(bp, &addr, priority);
+                       if (retval) {
+                               /* Can't finish building the list, clean up. */
+                               sctp_bind_addr_clean(bp);
+                               break;
+                       }
+
+                       len = ntohs(rawaddr.p->length);
+                       addrs_len -= len;
+                       raw_addr_list += len;
+               } else {
+                       /* Corrupted raw addr list! */
+                       retval = -EINVAL;
+                       sctp_bind_addr_clean(bp);
+                       break;
+               }
+       }
+
+       return retval;
+}
+
+/********************************************************************
+ * 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)
+{
+       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;
+       }
+
+       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)
+{
+       sctp_protocol_t *proto = sctp_get_protocol();
+       int error = 0;
+
+       if (sctp_is_any(addr)) {
+               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))))
+                       error = sctp_add_bind_addr(dest, addr, priority);
+       }
+
+       return error;
+}
+
+/* Is addr one of the wildcards?  */
+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)
+                       retval = 1;
+               break;
+
+       case AF_INET6:
+               SCTP_V6(
+                       if (IPV6_ADDR_ANY ==
+                           sctp_ipv6_addr_type(&addr->v6.sin6_addr))
+                               retval = 1;
+                       );
+               break;
+
+       default:
+               break;
+       };
+
+       return retval;
+}
+
+/* Is 'addr' valid for '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: 
+                * 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
+                * SHOULD include all of its addresses with level greater
+                * 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;
+
+               /* 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.  There is no filtering for IPv4 vs. IPv6 addresses
+                * based on scoping alone.
+                */
+               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
+                * scoping is needed.  According to the IPv6 scoping draft,
+                * 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.
+                * (see the comments for IPv4 above)
+                */
+               if (addr_scope <= scope)
+                       return 1;
+               break;
+
+       default:
+               return 0;
+       };
+
+       return 0;
+}
+
+/********************************************************************
+ * 3rd Level Abstractions
+ ********************************************************************/
+
+/* What is the scope of '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.
+                * 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?
+                */
+
+               /* Check for unusable SCTP addresses. */
+               if (IS_IPV4_UNUSABLE_ADDRESS(&addr->v4.sin_addr.s_addr)) {
+                       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)) {
+                       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
+                                * fields. See IFA_* in <net/if_inet6.h>.
+                                * 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;
+}
+
+/* This function checks if the address is a valid address to be used for
+ * SCTP.
+ *
+ * Output:
+ * 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)
+{
+       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))
+                       return 0;
+               break;
+
+       case AF_INET6:
+               SCTP_V6(
+               {
+                       int ret = sctp_ipv6_addr_type(&addr->v6.sin6_addr);
+
+                       /* Is this a non-unicast address */
+                       if (!(ret & IPV6_ADDR_UNICAST))
+                               return 0;
+                       break;
+               });
+
+       default:
+               return 0;
+       };
+
+       return 1;
+}
diff --git a/net/sctp/command.c b/net/sctp/command.c
new file mode 100644 (file)
index 0000000..95d919b
--- /dev/null
@@ -0,0 +1,104 @@
+/* SCTP kernel reference Implementation Copyright (C) 1999-2001
+ * Cisco, Motorola, and IBM
+ * Copyright 2001 La Monte H.P. Yarroll
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * These functions manipulate sctp command sequences.
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson <karl@athena.chicago.il.us>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/types.h>
+#include <net/sctp/sctp.h>
+#include <net/sctp/sm.h>
+
+/* Create a new sctp_command_sequence.  */
+sctp_cmd_seq_t *sctp_new_cmd_seq(int 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;
+}
+
+/* Initialize a block of memory as a command sequence. */
+int sctp_init_cmd_seq(sctp_cmd_seq_t *seq)
+{
+       memset(seq, 0, sizeof(sctp_cmd_seq_t));
+       return 1;               /* We always succeed.  */
+}
+
+/* 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)
+{
+       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;
+}
+
+/* Rewind an sctp_cmd_seq_t to iterate from the start.  */
+int sctp_rewind_sequence(sctp_cmd_seq_t *seq)
+{
+       seq->next_cmd = 0;
+       return 1;               /* We always succeed. */
+}
+
+/* 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 *retval = NULL;
+
+       if (seq->next_cmd < seq->next_free_slot)
+               retval = &seq->cmds[seq->next_cmd++];
+
+       return retval;
+}
+
+/* Dispose of a command sequence.  */
+void sctp_free_cmd_seq(sctp_cmd_seq_t *seq)
+{
+       kfree(seq);
+}
diff --git a/net/sctp/crc32c.c b/net/sctp/crc32c.c
new file mode 100644 (file)
index 0000000..4f8e8a9
--- /dev/null
@@ -0,0 +1,187 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 International Business Machines, Corp.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * SCTP Checksum functions
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    Dinakaran Joseph 
+ *    Jon Grimm <jgrimm@us.ibm.com>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+/* The following code has been taken directly from
+ * draft-ietf-tsvwg-sctpcsum-03.txt
+ *
+ * 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,
+};
+     
+__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(__u32)); i++)
+               CRC32C(crc32, buffer[i]);
+
+       /* Skip checksum field of the header. */
+       for (i = 0; i < sizeof(__u32); i++)
+               CRC32C(crc32, 0);
+
+       /* Calculate the rest of the CRC. */
+       for (i = sizeof(struct sctphdr); i < length ; i++)
+               CRC32C(crc32, buffer[i]);
+
+       result = ~crc32;
+
+       /*  result  now holds the negated polynomial remainder;
+        *  since the table and algorithm is "reflected" [williams95].
+        *  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.
+        *  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
+        *  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;
+}
+
+
diff --git a/net/sctp/debug.c b/net/sctp/debug.c
new file mode 100644 (file)
index 0000000..7e7d1ea
--- /dev/null
@@ -0,0 +1,215 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001 International Business Machines Corp.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * This file is part of the implementation of the add-IP extension,
+ * based on <draft-ietf-tsvwg-addip-sctp-02.txt> June 29, 2001,
+ * for the SCTP kernel reference Implementation.
+ * 
+ * This file converts numerical ID value to alphabetical names for SCTP
+ * terms such as chunk type, parameter time, event type, etc.
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson          <karl@athena.chicago.il.us>
+ *    Xingang Guo           <xingang.guo@intel.com>
+ *    Jon Grimm             <jgrimm@us.ibm.com>
+ *    Daisy Chang          <daisyc@us.ibm.com>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <net/sctp/sctp.h>
+
+#if SCTP_DEBUG
+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",
+       "INIT",
+       "INIT_ACK",
+       "SACK",
+       "HEARTBEAT",
+       "HEARTBEAT_ACK",
+       "ABORT",
+       "SHUTDOWN",
+       "SHUTDOWN_ACK",
+       "ERROR",
+       "COOKIE_ECHO",
+       "COOKIE_ACK",
+       "ECN_ECNE",
+       "ECN_CWR",
+       "SHUTDOWN_COMPLETE",
+};
+
+/* Lookup "chunk type" debug name. */
+const char *sctp_cname(const sctp_subtype_t cid)
+{
+       if (cid.chunk < 0)
+               return "illegal chunk id";
+       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";
+
+       default:
+               return "unknown chunk";
+       };
+       return "unknown chunk";
+}
+
+/* These are printable form of variable-length parameters. */
+const char *sctp_param_tbl[SCTP_PARAM_ECN_CAPABLE + 1] = {
+       "",
+       "PARAM_HEATBEAT_INFO",
+       "",
+       "",
+       "",
+       "PARAM_IPV4_ADDRESS",
+       "PARAM_IPV6_ADDRESS",
+       "PARAM_STATE_COOKIE",
+       "PARAM_UNRECOGNIZED_PARAMETERS",
+       "PARAM_COOKIE_PRESERVATIVE",
+       "",
+       "PARAM_HOST_NAME_ADDRESS",
+       "PARAM_SUPPORTED_ADDRESS_TYPES",
+};
+
+/* These are printable forms of the states.  */
+const char *sctp_state_tbl[SCTP_STATE_NUM_STATES] = {
+       "STATE_EMPTY",
+       "STATE_CLOSED",
+       "STATE_COOKIE_WAIT",
+       "STATE_COOKIE_ECHOED",
+       "STATE_ESTABLISHED",
+       "STATE_SHUTDOWN_PENDING",
+       "STATE_SHUTDOWN_SENT",
+       "STATE_SHUTDOWN_RECEIVED",
+       "STATE_SHUTDOWN_ACK_SENT",
+};
+
+/* Events that could change the state of an association.  */
+const char *sctp_evttype_tbl[] = {
+       "EVENT_T_unknown",
+       "EVENT_T_CHUNK",
+       "EVENT_T_TIMEOUT",
+       "EVENT_T_OTHER",
+       "EVENT_T_PRIMITIVE"
+};
+
+/* Return value of a state function */
+const char *sctp_status_tbl[] = {
+       "DISPOSITION_DISCARD",
+       "DISPOSITION_CONSUME",
+       "DISPOSITION_NOMEM",
+       "DISPOSITION_DELETE_TCB",
+       "DISPOSITION_ABORT",
+       "DISPOSITION_VIOLATION",
+       "DISPOSITION_NOT_IMPL",
+       "DISPOSITION_ERROR",
+       "DISPOSITION_BUG"
+};
+
+/* Printable forms of primitives */
+static const char *sctp_primitive_tbl[SCTP_NUM_PRIMITIVE_TYPES] = {
+       "PRIMITIVE_INITIALIZE",
+       "PRIMITIVE_ASSOCIATE",
+       "PRIMITIVE_SHUTDOWN",
+       "PRIMITIVE_ABORT",
+       "PRIMITIVE_SEND",
+       "PRIMITIVE_SETPRIMARY",
+       "PRIMITIVE_RECEIVE",
+       "PRIMITIVE_STATUS",
+       "PRIMITIVE_CHANGEHEARTBEAT",
+       "PRIMITIVE_REQUESTHEARTBEAT",
+       "PRIMITIVE_GETSRTTREPORT",
+       "PRIMITIVE_SETFAILURETHRESHOLD",
+       "PRIMITIVE_SETPROTOPARAMETERS",
+       "PRIMITIVE_RECEIVE_UNSENT",
+       "PRIMITIVE_RECEIVE_UNACKED",
+       "PRIMITIVE_DESTROY"
+};
+
+/* Lookup primitive debug name. */
+const char *sctp_pname(const sctp_subtype_t id)
+{
+       if (id.primitive < 0)
+               return "illegal primitive";
+       if (id.primitive <= SCTP_EVENT_PRIMITIVE_MAX)
+               return sctp_primitive_tbl[id.primitive];
+       return "unknown_primitive";
+}
+
+static const char *sctp_other_tbl[] = {
+       "NO_PENDING_TSN",
+       "ICMP_UNREACHFRAG"
+};
+
+/* Lookup "other" debug name. */
+const char *sctp_oname(const sctp_subtype_t id)
+{
+       if (id.other < 0)
+               return "illegal 'other' event";
+       if (id.other < SCTP_EVENT_OTHER_MAX)
+               return sctp_other_tbl[id.other];
+       return "unknown 'other' event";
+}
+
+static const char *sctp_timer_tbl[] = {
+       "TIMEOUT_NONE",
+       "TIMEOUT_T1_COOKIE",
+       "TIMEOUT_T1_INIT",
+       "TIMEOUT_T2_SHUTDOWN",
+       "TIMEOUT_T3_RTX",
+       "TIMEOUT_T4_RTO",
+       "TIMEOUT_HEARTBEAT",
+       "TIMEOUT_SACK",
+       "TIMEOUT_AUTOCLOSE",
+       "TIMEOUT_PMTU_RAISE",
+};
+
+/* Lookup timer debug name. */
+const char *sctp_tname(const sctp_subtype_t id)
+{
+       if (id.timeout < 0)
+               return "illegal 'timer' event";
+       if (id.timeout <= SCTP_EVENT_TIMEOUT_MAX)
+               return sctp_timer_tbl[id.timeout];
+       return "unknown_timer";
+}
diff --git a/net/sctp/endpointola.c b/net/sctp/endpointola.c
new file mode 100644 (file)
index 0000000..6c8cb0d
--- /dev/null
@@ -0,0 +1,369 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 International Business Machines, Corp.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001 Nokia, Inc.
+ * Copyright (c) 2001 La Monte H.P. Yarroll
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * This abstraction represents an SCTP endpoint.   
+ *
+ * This file is part of the implementation of the add-IP extension,
+ * based on <draft-ietf-tsvwg-addip-sctp-02.txt> June 29, 2001,
+ * for the SCTP kernel reference Implementation.
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson <karl@athena.chicago.il.us>
+ *    Jon Grimm <jgrimm@austin.ibm.com>
+ *    Daisy Chang <daisyc@us.ibm.com>
+ *    Dajiang Zhang <dajiang.zhang@nokia.com>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/types.h>
+#include <linux/sched.h>
+#include <linux/slab.h>
+#include <linux/in.h>
+#include <linux/random.h>      /* get_random_bytes() */
+#include <net/sock.h>
+#include <net/ipv6.h>
+#include <net/sctp/sctp.h>
+#include <net/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.
+ * 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 *ep;
+
+       /* 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;
+
+fail_init:
+       kfree(ep);
+fail:
+       return NULL;
+}
+
+/*
+ * 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)
+{
+       memset(ep, 0, sizeof(sctp_endpoint_t));
+
+       /* Initialize the base structure. */
+       /* What type of endpoint are we?  */
+       ep->base.type = SCTP_EP_TYPE_SOCKET;
+
+        /* Initialize the basic object fields. */
+       atomic_set(&ep->base.refcnt, 1);
+       ep->base.dead     = 0;
+       ep->base.malloced = 1;
+
+       /* 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,
+                                   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;
+       sock_hold(ep->base.sk);
+
+       /* This pointer is useful to access the default protocol parameter
+        * values.
+        */
+       ep->proto = proto;
+
+       /* 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]
+               = SCTP_DEFAULT_TIMEOUT_T1_COOKIE;
+       ep->timeouts[SCTP_EVENT_TIMEOUT_T1_INIT]
+               = SCTP_DEFAULT_TIMEOUT_T1_INIT;
+       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]
+               = SCTP_DEFAULT_TIMEOUT_HEARTBEAT;
+       ep->timeouts[SCTP_EVENT_TIMEOUT_SACK]
+               = SCTP_DEFAULT_TIMEOUT_SACK;
+       ep->timeouts[SCTP_EVENT_TIMEOUT_AUTOCLOSE]
+               = sctp_sk(sk)->autoclose * HZ;
+       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?
+        */
+       sk->rcvbuf = SCTP_DEFAULT_MAXWINDOW;
+       sk->sndbuf = SCTP_DEFAULT_MAXWINDOW * 2;
+
+       /* Use SCTP specific send buffer space queues.  */
+       sk->write_space = sctp_write_space;
+       sk->use_write_queue = 1;
+
+       /* 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";
+       return ep;
+}
+
+/* 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);
+}
+
+/* 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;
+       sctp_endpoint_put(ep);
+}
+
+/* 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! */
+       sctp_unhash_endpoint(ep);
+
+       /* 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)
+               sctp_put_port(ep->base.sk);
+
+       /* Give up our hold on the sock. */
+       if (ep->base.sk)
+               sock_put(ep->base.sk);
+
+       /* Finally, free up our memory. */
+       if (ep->base.malloced) {
+               kfree(ep);
+               SCTP_DBG_OBJCNT_DEC(ep);
+       }
+}
+
+/* Hold a reference to an endpoint. */
+void sctp_endpoint_hold(sctp_endpoint_t *ep)
+{
+       atomic_inc(&ep->base.refcnt);
+}
+
+/* Release a reference to an endpoint and clean up if there are
+ * no more references.
+ */
+void sctp_endpoint_put(sctp_endpoint_t *ep)
+{
+       if (atomic_dec_and_test(&ep->base.refcnt))
+               sctp_endpoint_destroy(ep);
+}
+
+/* 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;
+
+       sctp_read_lock(&ep->base.addr_lock);
+       if (ep->base.bind_addr.port == laddr->v4.sin_port) {
+               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;
+}
+
+/* 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)
+{
+       int rport;
+       sctp_association_t *asoc;
+       list_t *pos;
+
+       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)
+                               return asoc;
+               }
+       }
+
+       *transport = NULL;
+       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)
+{
+       sctp_association_t *asoc;
+
+       sctp_local_bh_disable();
+       asoc = __sctp_endpoint_lookup_assoc(ep, paddr, transport);
+       sctp_local_bh_enable();
+
+       return asoc;
+}
+
+/* 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)
+{
+       sctp_association_t *asoc;
+       struct sock *sk;
+       sctp_transport_t *transport;
+       sctp_chunk_t *chunk;
+       sctp_inqueue_t *inqueue;
+       sctp_subtype_t subtype;
+       sctp_state_t state;
+       int error = 0;
+
+       if (ep->base.dead)
+               goto out;
+
+       asoc = NULL;
+       inqueue = &ep->base.inqueue;
+       sk      = ep->base.sk;
+
+       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.
+                */
+               if (NULL == chunk->asoc) {
+                       asoc = sctp_endpoint_lookup_assoc(ep,
+                                                         sctp_source(chunk),
+                                                         &transport);
+                       chunk->asoc = asoc;
+                       chunk->transport = transport;
+               }
+
+               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 && sctp_chunk_is_data(chunk))
+                       asoc->peer.last_data_from = chunk->transport;
+
+               if (chunk->transport)
+                       chunk->transport->last_time_heard = jiffies;
+
+               /* 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;
+
+               /* Check to see if the endpoint is freed in response to 
+                * the incoming chunk. If so, get out of the while loop.
+                */ 
+               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:
+}
+
+
+
+
diff --git a/net/sctp/hashdriver.c b/net/sctp/hashdriver.c
new file mode 100644 (file)
index 0000000..bb21267
--- /dev/null
@@ -0,0 +1,128 @@
+/* SCTP reference Implementation Copyright (C) 1999 Cisco And Motorola
+ * 
+ * This file origiantes from Randy Stewart's SCTP reference Implementation.
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    Randy Stewart  rstewar1@email.mot.com
+ *    Ken Morneau    kmorneau@cisco.com
+ *    Qiaobing Xie   qxie1@email.mot.com
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorperated into the next SCTP release.
+ * 
+ * There are still LOTS of bugs in this code... I always run on the motto
+ * "it is a wonder any code ever works :)"
+ */
+
+#include <linux/types.h>
+#include <asm/string.h>
+#include <net/sctp/sctp.h>
+#include <net/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.
+ */
+
+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;
+       struct SLA_1_Context context;
+
+       __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;
+
+       /* 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
+        */
+
+       /* 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 */
+}
+
diff --git a/net/sctp/input.c b/net/sctp/input.c
new file mode 100644 (file)
index 0000000..abfcb29
--- /dev/null
@@ -0,0 +1,662 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 International Business Machines, Corp.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001 Nokia, Inc.
+ * Copyright (c) 2001 La Monte H.P. Yarroll
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * These functions handle all input from the IP layer into SCTP. 
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson <karl@athena.chicago.il.us>
+ *    Xingang Guo <xingang.guo@intel.com>
+ *    Jon Grimm <jgrimm@us.ibm.com>
+ *    Hui Huang <hui.huang@nokia.com>
+ *    Daisy Chang <daisyc@us.ibm.com>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/types.h>
+#include <linux/list.h> /* For struct list_head */
+#include <linux/socket.h>
+#include <linux/ip.h>
+#include <linux/time.h> /* For struct timeval */
+#include <net/sock.h>
+#include <linux/ipsec.h>
+#include <net/sctp/sctp.h>
+#include <net/sctp/sm.h>
+
+/* Forward declarations for internal helpers. */
+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);
+
+/* 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)
+{
+       sockaddr_storage_t *ret = NULL;
+       void *to, *saddr, *daddr;
+       __u16 *port;
+       size_t len;
+       struct sctphdr *sh;
+
+       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;
+               break;
+
+       case 6:
+               SCTP_V6(
+                       to = &addr->v6.sin6_addr;
+                       port = &addr->v6.sin6_port;
+                       saddr = &skb->nh.ipv6h->saddr;
+                       daddr = &skb->nh.ipv6h->daddr;
+                       len = sizeof(struct in6_addr);
+                       addr->v6.sin6_family = AF_INET6;
+                       addr->v6.sin6_flowinfo = 0; /* FIXME */
+                       addr->v6.sin6_scope_id = 0; /* FIXME */
+                       break;
+                       )
+
+       default:
+               goto out;
+       };
+
+       sh = (struct sctphdr *) skb->h.raw;
+       if (is_saddr) {
+               *port  = ntohs(sh->source);
+               memcpy(to, saddr, len);
+       } else {
+               *port = ntohs(sh->dest);
+               memcpy(to, daddr, len);
+       }
+
+       ret = addr;
+out:
+       return ret;
+}
+
+/* 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)
+{
+       struct sock *sk;
+       sctp_association_t *asoc;
+       sctp_endpoint_t *ep = NULL;
+       sctp_endpoint_common_t *rcvr;
+       sctp_transport_t *transport = NULL;
+       sctp_chunk_t *chunk;
+       struct sctphdr *sh;
+       sockaddr_storage_t src;
+       sockaddr_storage_t dest;
+       int ret = 0;
+
+       if (skb->pkt_type!=PACKET_HOST)
+               goto discard_it;
+
+       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))
+               goto bad_packet;
+       if (sctp_rcv_checksum(skb) < 0)
+               goto bad_packet;
+
+       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
+        * 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))
+               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.
+        */
+       if (!asoc) {
+               ep = __sctp_rcv_lookup_endpoint(&dest);
+               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))
+               goto discard_release;
+
+       /* 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;
+
+       /* Set the source address.  */
+       sctp_init_source(chunk);
+
+       /* Remember where we came from.  */
+       chunk->transport = transport;
+
+       /* Acquire access to the sock lock. Note: We are safe from other
+        * bottom halves on this lock, but a user may be in the lock too,
+        * so check if it is busy.
+        */
+       sctp_bh_lock_sock(sk);
+
+       if (__sctp_sock_busy(sk)) {
+               sk_add_backlog(sk, (struct sk_buff *) chunk);
+       } else {
+               sctp_backlog_rcv(sk, (struct sk_buff *) chunk);
+       }
+
+       /* Release the sock and any reference counts we took in the
+        * lookup calls.
+        */
+       sctp_bh_unlock_sock(sk);
+       if (asoc) {
+               sctp_association_put(asoc);
+       } else {
+               sctp_endpoint_put(ep);
+       }
+       sock_put(sk);
+       return ret;
+
+bad_packet:
+#if 0 /* FIXME */
+       SCTP_INC_STATS(SctpInErrs);
+#endif /* FIXME*/
+
+discard_it:
+       kfree_skb(skb);
+       return ret;
+
+discard_release:
+       /* Release any structures we may be holding. */
+       if (asoc) {
+               sock_put(asoc->base.sk);
+               sctp_association_put(asoc);
+       } else {
+               sock_put(ep->base.sk);
+               sctp_endpoint_put(ep);
+       }
+
+       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.
+ */
+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;
+       inqueue = &chunk->rcvr->inqueue;
+
+       sctp_push_inqueue(inqueue, chunk);
+        return 0;
+}
+
+/*
+ * This routine is called by the ICMP module when it gets some
+ * sort of error condition.  If err < 0 then the socket should
+ * be closed and the error returned to the user.  If err > 0
+ * it's just the icmp type << 8 | icmp code.  After adjustment
+ * header points to the first 8 bytes of the sctp header.  We need
+ * to find the appropriate port.
+ *
+ * The locking strategy used here is very "optimistic". When
+ * someone else accesses the socket the ICMP is just dropped
+ * and for some paths there is no check at all.
+ * A more general error queue to queue errors for later handling
+ * is probably better.
+ *
+ */
+void sctp_v4_err(struct sk_buff *skb, u32 info)
+{
+       /* This should probably involve a call to SCTPhandleICMP().  */
+}
+
+/*
+ * RFC 2960, 8.4 - Handle "Out of the blue" Packets.
+ *
+ * This function scans all the chunks in the OOTB packet to determine if
+ * the packet should be discarded right away.  If a response might be needed
+ * for this packet, or, if further processing is possible, the packet will
+ * be queued to a proper inqueue for the next phase of handling.
+ *
+ * Output:
+ * 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)
+{
+       sctp_chunkhdr_t *ch;
+       __u8 *ch_end;
+
+       ch = (sctp_chunkhdr_t *) skb->data;
+
+       /* Scan through all the chunks in the packet.  */
+       do {
+               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
+                * further action.
+                */
+               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
+                * and take no further action.
+                */
+               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)
+                       goto discard;
+
+               if (ch->type == SCTP_CID_ERROR) {
+                       /* FIXME - Need to check the "Stale cookie" ERROR. */
+                       goto discard;
+               }
+
+               ch = (sctp_chunkhdr_t *) ch_end;
+       } while (ch_end < skb->tail);
+
+       return 0;
+
+discard:
+       return 1;
+}
+
+/* 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;
+
+       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)
+               (*epp)->pprev = &epb->next;
+       *epp = epb;
+       epb->pprev = epp;
+       sctp_write_unlock(&head->lock);
+}
+
+/* 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();
+}
+
+/* Remove endpoint from the hash table.  */
+void __sctp_unhash_endpoint(sctp_endpoint_t *ep)
+{
+       sctp_hashbucket_t *head;
+       sctp_endpoint_common_t *epb;
+
+       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)
+                       epb->next->pprev = epb->pprev;
+               *epb->pprev = epb->next;
+               epb->pprev = NULL;
+       }
+
+       sctp_write_unlock(&head->lock);
+}
+
+/* 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();
+}
+
+/* Look up an endpoint. */
+sctp_endpoint_t *__sctp_rcv_lookup_endpoint(const sockaddr_storage_t *laddr)
+{
+       sctp_hashbucket_t *head;
+       sctp_endpoint_common_t *epb;
+       sctp_endpoint_t *ep;
+       int hash;
+
+       hash = sctp_ep_hashfn(laddr->v4.sin_port);
+       head = &sctp_proto.ep_hashbucket[hash];
+       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;
+       }
+
+       ep = sctp_sk((sctp_get_ctl_sock()))->ep;
+       epb = &ep->base;
+
+hit:
+       sctp_endpoint_hold(ep);
+       sock_hold(epb->sk);
+       read_unlock(&head->lock);
+       return ep;
+}
+
+/* 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();
+}
+
+/* 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)
+               (*epp)->pprev = &epb->next;
+       *epp = epb;
+       epb->pprev = epp;
+       sctp_write_unlock(&head->lock);
+}
+
+/* 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();
+}
+
+/* Remove association from the hash table.  */
+void __sctp_unhash_established(sctp_association_t *asoc)
+{
+       sctp_hashbucket_t *head;
+       sctp_endpoint_common_t *epb;
+
+       epb = &asoc->base;
+
+       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)
+                       epb->next->pprev = epb->pprev;
+               *epb->pprev = epb->next;
+               epb->pprev = NULL;
+       }
+
+       sctp_write_unlock(&head->lock);
+}
+
+/* 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_hashbucket_t *head;
+       sctp_endpoint_common_t *epb;
+       sctp_association_t *asoc;
+       sctp_transport_t *transport;
+       int hash;
+
+       /* Optimize here for direct hit, only listening connections can
+        * 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;
+       }
+
+       read_unlock(&head->lock);
+
+       return NULL;
+
+hit:
+       *transportp = transport;
+       sctp_association_hold(asoc);
+       sock_hold(epb->sk);
+       read_unlock(&head->lock);
+       return asoc;
+}
+
+/*
+ * 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
+ *    or INIT-ACK chunk the receiver SHOULD use not only the
+ *    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
+ * to look within the INIT or INIT-ACK. Any implementation that
+ * does not do this, may not be able to establish associations
+ * in certain circumstances.
+ *
+ */
+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;
+       sctp_chunkhdr_t *ch;
+       __u8 *ch_end, *data;
+       sctpParam_t parm;
+
+       ch = (sctp_chunkhdr_t *) skb->data;
+
+       ch_end = ((__u8 *) ch) + WORD_ROUND(ntohs(ch->length));
+
+       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
+        * 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) {
+               parm.v  = data;
+
+               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))
+                       continue;
+
+               sctp_param2sockaddr(paddr, parm, ntohs(sh->source));
+
+               asoc = __sctp_rcv_lookup_association(laddr, paddr, transportp);
+               if (asoc)
+                       return asoc;
+       }
+
+       return NULL;
+}
+
+/* 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 *asoc;
+
+       asoc = __sctp_rcv_lookup_association(laddr, paddr, transportp);
+
+       /* Further lookup for INIT-ACK packet.
+        * SCTP Implementors Guide, 2.18 Handling of address
+        * parameters within the INIT or INIT-ACK.
+        */
+       if (!asoc)
+               asoc = __sctp_rcv_initack_lookup(skb, laddr, transportp);
+
+       return asoc;
+}
+
+
+
+
+
diff --git a/net/sctp/inqueue.c b/net/sctp/inqueue.c
new file mode 100644 (file)
index 0000000..1c5132a
--- /dev/null
@@ -0,0 +1,199 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2002 International Business Machines, Corp.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * These functions are the methods for accessing the SCTP inqueue.
+ *
+ * An SCTP inqueue is a queue into which you push SCTP packets
+ * (which might be bundles or fragments of chunks) and out of which you
+ * pop SCTP whole chunks.
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson <karl@athena.chicago.il.us>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <net/sctp/sctp.h>
+#include <net/sctp/sm.h>
+#include <linux/interrupt.h>
+
+/* Initialize an SCTP_inqueue.  */
+void sctp_inqueue_init(sctp_inqueue_t *queue)
+{
+       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;
+
+       queue->malloced = 0;
+}
+
+/* Create an initialized SCTP_inqueue.  */
+sctp_inqueue_t *sctp_inqueue_new(void)
+{
+       sctp_inqueue_t *retval;
+
+       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)
+{
+       sctp_chunk_t *chunk;
+
+       /* 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,
+        * free it as well.
+        */
+       if (queue->in_progress)
+               sctp_free_chunk(queue->in_progress);
+
+       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.
+ */
+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);
+       q->immediate.routine(q->immediate.data);
+}
+
+/* Extract a chunk from an SCTP inqueue.
+ *
+ * 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 *chunk;
+       sctp_chunkhdr_t *ch = NULL;
+
+       /* The assumption is that we are safe to process the chunks
+        * at this time.
+        */
+
+       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) {
+                       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;
+
+                       /* Force chunk->skb->data to chunk->chunk_end.  */
+                       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 (!chunk) {
+               /* Is the queue empty?  */
+               if (skb_queue_empty(&queue->in))
+                       return NULL;
+
+               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;
+       }
+
+        chunk->chunk_hdr = 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.  */
+
+       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;
+       }
+
+       SCTP_DEBUG_PRINTK("+++sctp_pop_inqueue+++ chunk %p[%s],"
+                         " 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;
+}
+
+/* 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
+ * inqueue and process it.
+ */
+void sctp_inqueue_set_th_handler(sctp_inqueue_t *q,
+                                void (*callback)(void *), void *arg)
+{
+       q->immediate.routine = callback;
+       q->immediate.data = arg;
+}
+
diff --git a/net/sctp/ipv6.c b/net/sctp/ipv6.c
new file mode 100644 (file)
index 0000000..7fba365
--- /dev/null
@@ -0,0 +1,263 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 2001 Nokia, Inc.
+ * Copyright (c) 2001 La Monte H.P. Yarroll
+ * Copyright (c) 2002 International Business Machines, Corp.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * SCTP over IPv6.
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    Le Yanqun             <yanqun.le@nokia.com>
+ *    Hui Huang                    <hui.huang@nokia.com>
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Sridhar Samudrala            <sri@us.ibm.com>
+ *    Jon Grimm             <jgrimm@us.ibm.com>
+ *
+ * Based on:
+ *      linux/net/ipv6/tcp_ipv6.c
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#define __NO_VERSION__
+#include <linux/module.h>
+#include <linux/errno.h>
+#include <linux/types.h>
+#include <linux/socket.h>
+#include <linux/sockios.h>
+#include <linux/net.h>
+#include <linux/sched.h>
+#include <linux/in.h>
+#include <linux/in6.h>
+#include <linux/netdevice.h>
+#include <linux/init.h>
+#include <linux/ipsec.h>
+
+#include <linux/ipv6.h>
+#include <linux/icmpv6.h>
+#include <linux/random.h>
+
+#include <net/protocol.h>
+#include <net/tcp.h>
+#include <net/ndisc.h>
+#include <net/ipv6.h>
+#include <net/transp_v6.h>
+#include <net/addrconf.h>
+#include <net/ip6_route.h>
+#include <net/inet_common.h>
+#include <net/inet_ecn.h>
+#include <net/sctp/sctp.h>
+
+#include <asm/uaccess.h>
+
+/* FIXME: Cleanup so we don't need TEST_FRAME here. */
+#ifndef TEST_FRAME
+/* FIXME: Comments. */
+static inline void sctp_v6_err(struct sk_buff *skb,
+                              struct inet6_skb_parm *opt,
+                              int type, int code, int offset, __u32 info)
+{
+       /* BUG.  WRITE ME.  */
+}
+
+/* Based on tcp_v6_xmit() in tcp_ipv6.c. */
+static inline int sctp_v6_xmit(struct sk_buff *skb)
+{
+       struct sock *sk = skb->sk;
+       struct ipv6_pinfo *np = inet6_sk(sk);
+       struct flowi fl;
+       struct dst_entry *dst;
+       struct in6_addr saddr;
+       int err = 0;
+
+       fl.proto = sk->protocol;
+       fl.fl6_dst = &np->daddr;
+       fl.fl6_src = NULL;
+
+       fl.fl6_flowlabel = np->flow_label;
+       IP6_ECN_flow_xmit(sk, fl.fl6_flowlabel);
+       fl.oif = sk->bound_dev_if;
+       fl.uli_u.ports.sport = inet_sk(sk)->sport;
+       fl.uli_u.ports.dport = inet_sk(sk)->dport;
+
+       if (np->opt && np->opt->srcrt) {
+               struct rt0_hdr *rt0 = (struct rt0_hdr *) np->opt->srcrt;
+               fl.nl_u.ip6_u.daddr = rt0->addr;
+       }
+
+       dst = __sk_dst_check(sk, np->dst_cookie);
+
+       if (dst == NULL) {
+               dst = ip6_route_output(sk, &fl);
+
+               if (dst->error) {
+                       sk->err_soft = -dst->error;
+                       dst_release(dst);
+                       return -sk->err_soft;
+               }
+               ip6_dst_store(sk, dst, NULL);
+       }
+
+       skb->dst = dst_clone(dst);
+
+       /* FIXME: This is all temporary until real source address
+        * selection is done.
+        */
+       if (ipv6_addr_any(&np->saddr)) {
+               err = ipv6_get_saddr(dst, fl.fl6_dst, &saddr);
+
+               if (err)
+                       printk(KERN_ERR "sctp_v6_xmit: no saddr available\n");
+
+               /* 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.
+                */
+               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;
+
+       return ip6_xmit(sk, skb, &fl, np->opt);
+}
+#endif /* TEST_FRAME */
+
+/* Returns the mtu for the given v6 destination 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;
+
+       fl.proto = 0;
+       fl.fl6_dst = (struct in6_addr *)&address->v6.sin6_addr;
+       fl.fl6_src = NULL;
+       fl.fl6_flowlabel = 0;
+       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;
+               SCTP_DEBUG_PRINTK("sctp_v6_get_dst_mtu: "
+                                 "ip6_route_output: dev:%s pmtu:%d\n", 
+                                 dst->dev->name, dst_mtu);
+               dst_release(dst);
+       } else {
+               SCTP_DEBUG_PRINTK("sctp_v6_get_dst_mtu: "
+                                 "ip6_route_output failed, returning "
+                                 "%d as dst_mtu\n", 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,
+       .getname    = inet6_getname,
+       .poll       = sctp_poll,
+       .ioctl      = inet6_ioctl,
+       .listen     = sctp_inet_listen,
+       .shutdown   = inet_shutdown,
+       .setsockopt = inet_setsockopt,
+       .getsockopt = inet_getsockopt,
+       .sendmsg    = inet_sendmsg,
+       .recvmsg    = inet_recvmsg,
+       .mmap       = sock_no_mmap,
+};
+
+static struct inet_protosw sctpv6_protosw = {
+       .type          = SOCK_SEQPACKET,
+       .protocol      = IPPROTO_SCTP,
+       .prot          = &sctp_prot,
+       .ops           = &inet6_seqpacket_ops,
+       .capability    = -1,
+       .no_check      = 0,
+       .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 sctp_func_t sctp_ipv6_specific = {
+       .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),
+       .sa_family       = AF_INET6,
+};
+
+/* Initialize IPv6 support and register with inet6 stack.  */
+int sctp_v6_init(void)
+{
+       /* Add SCTPv6 to inetsw6 linked list. */
+       inet6_register_protosw(&sctpv6_protosw);
+
+       /* 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;
+}
+
+/* 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);
+}
diff --git a/net/sctp/objcnt.c b/net/sctp/objcnt.c
new file mode 100644 (file)
index 0000000..e246a93
--- /dev/null
@@ -0,0 +1,133 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 2001 International Business Machines Corp.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * Support for memory object debugging.  This allows one to monitor the
+ * object allocations/deallocations for types instrumented for this 
+ * via the proc fs. 
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    Jon Grimm             <jgrimm@us.ibm.com>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <net/sctp/sctp.h>
+
+/*
+ * 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.
+ */
+
+SCTP_DBG_OBJCNT(sock);
+SCTP_DBG_OBJCNT(ep);
+SCTP_DBG_OBJCNT(transport);
+SCTP_DBG_OBJCNT(assoc);
+SCTP_DBG_OBJCNT(bind_addr);
+SCTP_DBG_OBJCNT(chunk);
+SCTP_DBG_OBJCNT(addr);
+
+/* An array to make it easy to pretty print the debug information
+ * to the proc fs.
+ */
+sctp_dbg_objcnt_entry_t sctp_dbg_objcnt[] = {
+       SCTP_DBG_OBJCNT_ENTRY(sock),
+       SCTP_DBG_OBJCNT_ENTRY(ep),
+       SCTP_DBG_OBJCNT_ENTRY(assoc),
+       SCTP_DBG_OBJCNT_ENTRY(transport),
+       SCTP_DBG_OBJCNT_ENTRY(chunk),
+       SCTP_DBG_OBJCNT_ENTRY(bind_addr),
+       SCTP_DBG_OBJCNT_ENTRY(addr),
+};
+
+/* 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)
+{
+       int len = 0;
+       off_t pos = 0;
+       int entries;
+       int i;
+       char temp[128];
+
+       /* How many entries? */
+       entries = sizeof(sctp_dbg_objcnt)/sizeof(sctp_dbg_objcnt[0]);
+
+       /* Walk the entries and print out the debug information
+        * for proc fs.
+        */
+       for (i = 0; i < entries; i++) {
+               pos += 128;
+
+               /* Skip ahead. */
+               if (pos <= offset) {
+                       len = 0;
+                       continue;
+               }
+               /* Print out each entry. */
+               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)
+                       goto done;
+       }
+
+done:
+       *start = buffer + len - (pos - offset);
+       len = pos - offset;
+       if (len > length)
+               len = length;
+
+       return len;
+}
+
+/* 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);
+}
+
+/* Cleanup the objcount entry in the proc filesystem.  */
+void sctp_dbg_objcnt_exit(void)
+{
+       remove_proc_entry("sctp_dbg_objcount", proc_net_sctp);
+}
+
+
diff --git a/net/sctp/output.c b/net/sctp/output.c
new file mode 100644 (file)
index 0000000..d6a37b6
--- /dev/null
@@ -0,0 +1,558 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 International Business Machines, Corp.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * These functions handle output processing.
+ *  
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson          <karl@athena.chicago.il.us>
+ *    Jon Grimm             <jgrimm@austin.ibm.com>
+ *    Sridhar Samudrala     <sri@us.ibm.com>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/types.h>
+#include <linux/kernel.h>
+#include <linux/wait.h>
+#include <linux/time.h>
+#include <linux/ip.h>
+#include <linux/ipv6.h>
+#include <linux/init.h>
+#include <net/inet_ecn.h> 
+#include <net/icmp.h>
+
+#ifndef TEST_FRAME
+#include <net/tcp.h>
+#endif /* TEST_FRAME (not defined) */
+
+#include <linux/socket.h> /* for sa_family_t */
+#include <linux/types.h> /* For NULL, etc... */
+#include <net/sock.h>
+
+#include <net/sctp/sctp.h>
+#include <net/sctp/sm.h>
+
+/* Forward declarations for private helpers. */
+__u32 count_crc(__u8 *ptr, __u16  count);
+static void sctp_packet_reset(sctp_packet_t *packet);
+static sctp_xmit_t sctp_packet_append_data(sctp_packet_t *packet,
+                                          sctp_chunk_t *chunk);
+
+/* Config a packet.
+ * This appears to be a followup set of initializations.)
+ */
+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->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)
+               sctp_packet_reset(packet);
+       return packet;
+}
+
+/* Initialize the packet structure. */
+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->destination_port = dport;
+       skb_queue_head_init(&packet->chunks);
+       packet->vtag = 0;
+       packet->ecn_capable = 0;
+       packet->get_prepend_chunk = NULL;
+       packet->has_cookie_echo = 0;
+       packet->malloced = 0;
+       sctp_packet_reset(packet);
+       return packet;
+}
+
+/* Free a packet.  */
+void sctp_packet_free(sctp_packet_t *packet)
+{
+       sctp_chunk_t *chunk;
+
+        while (NULL !=
+              (chunk = (sctp_chunk_t *)skb_dequeue(&packet->chunks))) {
+               sctp_free_chunk(chunk);
+       }
+
+       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
+ * is not present in the packet, it transmits the input packet.
+ * Data can be bundled with a packet containing a COOKIE_ECHO chunk as long
+ * 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 retval;
+       int error = 0;
+
+       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)
+                               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;
+       };
+
+       return retval;
+}
+
+/* Append a chunk to the offered packet reporting back any inability to do
+ * so.
+ */
+sctp_xmit_t sctp_packet_append_chunk(sctp_packet_t *packet, sctp_chunk_t *chunk)
+{
+       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) ?
+                (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);
+
+               /* Both control chunks and data chunks with TSNs are
+                * 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;
+               }
+       } 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).
+        */
+
+       /* 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)
+                       goto finish;
+       } 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,
+                      (struct sk_buff *)chunk);
+       packet->size += chunk_len;
+
+finish:
+       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)
+{
+       sctp_transport_t *transport = packet->transport;
+       sctp_association_t *asoc = transport->asoc;
+       struct sctphdr *sh;
+       __u32 crc32;
+       struct sk_buff *nskb;
+       sctp_chunk_t *chunk;
+       struct sock *sk;
+       int err = 0;
+       int padding;            /* How much padding do we need?  */
+       __u8 packet_has_data = 0;
+
+       /* 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;
+
+       /* Allocate the new skb.  */
+       nskb = dev_alloc_skb(packet->size);
+       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.
+        */
+       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
+        *
+        * 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 multiple of 4 bytes, the sender MUST
+        * 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");
+       while (NULL != (chunk = (sctp_chunk_t *)
+                       skb_dequeue(&packet->chunks))) {
+               chunk->num_times_sent++;
+               chunk->sent_at = jiffies;
+               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
+                        * 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)) {
+                               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);
+               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, "
+                                 "%s %d\n",
+                                 "*** Chunk", chunk,
+                                 sctp_cname(SCTP_ST_CHUNK(
+                                         chunk->chunk_hdr->type)),
+                                 chunk->has_tsn ? "TSN" : "No 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,
+                                 "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
+                * acknowledged or have failed.
+                */
+               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.
+        *
+        * 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((__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
+        *   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
+        *
+        * 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...
+        */
+
+       /* 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)
+               sk->dst_cache->obsolete = 1;
+
+       if (packet_has_data) {
+               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) &&
+                   (asoc->autoclose)) {
+                       timer = &asoc->timers[SCTP_EVENT_TIMEOUT_AUTOCLOSE];
+                       timeout = asoc->timeouts[SCTP_EVENT_TIMEOUT_AUTOCLOSE];
+
+                       if (!mod_timer(timer, jiffies + timeout))
+                               sctp_association_hold(asoc);
+               }
+       }
+
+       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;
+}
+
+/********************************************************************
+ * 2nd Level Abstractions
+ ********************************************************************/
+
+/*
+ * This private function resets the packet to a fresh state.
+ */
+static void sctp_packet_reset(sctp_packet_t *packet)
+{
+       sctp_chunk_t *chunk = NULL;
+
+       packet->size = SCTP_IP_OVERHEAD;
+
+       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
+        * any other chunks get appended.
+        */
+       if (chunk)
+               sctp_packet_append_chunk(packet, 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;
+       size_t datasize, rwnd, inflight;
+       sctp_transport_t *transport = packet->transport;
+       __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).
+                        */
+                       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.
+        * 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, "
+                                 "ssthresh: %d, flight_size: %d, "
+                                 "pba: %d\n",
+                                  __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.
+        */
+       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;
+
+       /* Keep track of how many bytes are in flight to the receiver. */
+       transport->asoc->outqueue.outstanding_bytes += datasize;
+
+       /* Update our view of the receiver's rwnd. */
+       if (datasize < rwnd) {
+               rwnd -= datasize;
+       } else {
+               rwnd = 0;
+       }
+
+       transport->asoc->peer.rwnd = rwnd;
+
+finish:
+       return retval;
+}
+
+
+
+
diff --git a/net/sctp/outqueue.c b/net/sctp/outqueue.c
new file mode 100644 (file)
index 0000000..5277952
--- /dev/null
@@ -0,0 +1,1442 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001-2002 International Business Machines Corp.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * These functions implement the outqueue class.   The outqueue handles
+ * bundling and queueing of outgoing SCTP chunks.  
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson          <karl@athena.chicago.il.us> 
+ *    Perry Melange         <pmelange@null.cc.uic.edu>
+ *    Xingang Guo           <xingang.guo@intel.com>
+ *    Hui Huang            <hui.huang@nokia.com>
+ *    Sridhar Samudrala     <sri@us.ibm.com>
+ *    Jon Grimm             <jgrimm@us.ibm.com>
+ *     
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/types.h>
+#include <linux/list.h> /* For struct list_head */
+#include <linux/socket.h>
+#include <linux/ip.h>
+#include <net/sock.h>          /* For skb_set_owner_w */
+
+#include <net/sctp/sctp.h>
+
+/* Declare internal functions here.  */
+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 *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)
+{
+       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;
+
+       /* Throw away unacknowledged chunks. */
+       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,
+                                          transmitted_list);
+                       sctp_free_chunk(chunk);
+               }
+       }
+
+       /* Throw away any leftover chunks. */
+       while ((chunk = (sctp_chunk_t *) skb_dequeue(&q->out)))
+               sctp_free_chunk(chunk);
+}
+
+/* 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);
+}
+
+/* Transmit any pending partial chunks.  */
+void sctp_force_outqueue(sctp_outqueue_t *q)
+{
+       /* Do we really need this? */
+       /* BUG */
+}
+
+/* Put a new chunk in an SCTP_outqueue.  */
+int sctp_push_outqueue(sctp_outqueue_t *q, sctp_chunk_t *chunk)
+{
+       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:
+                       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");
+
+                       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;
+}
+
+/* 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,
+                    __u8 fast_retransmit)
+{
+       struct list_head *lchunk;
+       sctp_chunk_t *chunk;
+       int error = 0;
+       struct list_head tlist;
+
+       if (fast_retransmit) {
+               sctp_transport_lower_cwnd(transport, SCTP_LOWER_CWND_FAST_RTX);
+       } else {
+               sctp_transport_lower_cwnd(transport, SCTP_LOWER_CWND_T3_RTX);
+       }
+
+       INIT_LIST_HEAD(&tlist);
+
+       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
+                * not yet acked should be added to the retransmit queue.
+                */  
+               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.
+                        */
+                       q->asoc->peer.rwnd += 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
+                        * 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
+                        * data chunk is sent.
+                        */
+                       if (chunk->rtt_in_progress) {
+                               chunk->rtt_in_progress = 0;
+                               transport->rto_pending = 0;
+                       }
+                       list_add_tail(lchunk, &q->retransmit);
+               }
+       }
+
+       /* Reconstruct the transmitted queue with chunks that are not
+        * eligible for retransmission.
+        */
+       while (NULL != (lchunk = sctp_list_dequeue(&tlist)))
+               list_add_tail(lchunk, &transport->transmitted);
+
+       SCTP_DEBUG_PRINTK("%s: transport: %p, fast_retransmit: %d, "
+                         "cwnd: %d, ssthresh: %d, flight_size: %d, "
+                         "pba: %d\n", __FUNCTION__,
+                         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;
+}
+
+/*
+ * 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.
+ * We assume that pkt->transport has already been set.
+ *
+ * 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)
+{
+       struct list_head *lqueue;
+       struct list_head *lchunk;
+       sctp_transport_t *transport = pkt->transport;
+       sctp_xmit_t status;
+       sctp_chunk_t *chunk;
+       sctp_association_t *asoc;
+       int error = 0;
+
+       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)
+        * 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.
+        *
+        * [Just to be painfully clear, if we are retransmitting
+        * because a timeout just happened, we should send only ONE
+        * packet of retransmitted data.]
+        */
+       lchunk = sctp_list_dequeue(lqueue);
+
+       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) {
+                       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)
+                               *start_timer = 1;
+
+                       /* If we are retransmitting, we should only
+                        * send a single packet.
+                        */
+                       if (rtx_timeout) {
+                               list_add(lchunk, lqueue);
+                               lchunk = NULL;
+                       }
+
+                       /* Bundle lchunk in the next round.  */
+                       break;
+
+               case SCTP_XMIT_RWND_FULL:
+                       /* Send this packet. */
+                       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.
+                        */
+                       list_add_tail(lchunk,
+                                     &transport->transmitted);
+                       *start_timer = 1;
+                       q->empty = 0;
+
+                       /* Retrieve a new chunk to bundle. */
+                       lchunk = sctp_list_dequeue(lqueue);
+                       break;
+               };
+       }
+
+       return error;
+}
+
+/* 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, __u32 tsn)
+{
+       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.
+        */
+       if (!skb_queue_empty(queue)) {
+               SCTP_DEBUG_PRINTK("sctp_xmit_frag: q not empty. "
+                                 "adding 0x%x to outqueue\n",
+                                  ntohl(frag->subh.data_hdr->tsn));
+               if (pos) {
+                       skb_insert(pos, (struct sk_buff *) frag);
+               } else {
+                       skb_queue_tail(queue, (struct sk_buff *) frag);
+               }
+               return;
+       }
+
+       /* Add the chunk fragment to the packet.  */
+       status = (*q->build_output)(packet, frag);
+       switch (status) {
+       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",
+                                 ntohl(frag->subh.data_hdr->tsn));
+               if (pos) {
+                       skb_insert(pos, (struct sk_buff *) frag);
+               } else {
+                       skb_queue_tail(queue, (struct sk_buff *) frag);
+               }
+               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
+                        */
+                       SCTP_DEBUG_PRINTK("sctp_xmit_frag: force output "
+                                         "failed. adding 0x%x to outqueue\n",
+                                         ntohl(frag->subh.data_hdr->tsn));
+                       if (pos) {
+                               skb_insert(pos, (struct sk_buff *) frag);
+                       } else {
+                               skb_queue_tail(queue, (struct sk_buff *) frag);
+                       }
+               } else {
+                       SCTP_DEBUG_PRINTK("sctp_xmit_frag: force output "
+                                         "success. 0x%x sent\n",
+                                         ntohl(frag->subh.data_hdr->tsn));
+                       list_add_tail(&frag->transmitted_list,
+                                     &transport->transmitted);
+
+                       sctp_transport_reset_timers(transport);
+               }
+               break;
+
+       default:
+               BUG();
+       };
+}
+
+/* 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)
+{
+       sctp_association_t *asoc = frag->asoc;
+       struct list_head *lfrag, *frag_list;
+       __u32 tsn;
+       int nfrags = 1;
+       struct sk_buff *pos;
+
+       /* Count the number of fragments. */
+       frag_list = &frag->frag_list;
+       list_for_each(lfrag, frag_list) {
+               nfrags++;
+       }
+
+       /* Get a TSN block of nfrags TSNs. */
+       tsn = __sctp_association_get_tsn_block(asoc, nfrags);
+
+       pos = skb_peek(&q->out);
+       /* Transmit the first fragment. */
+       sctp_xmit_frag(q, pos, packet, frag, tsn++);
+
+       /* Transmit the rest of fragments. */
+       frag_list = &frag->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++);
+       }
+}
+
+/* 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;
+       __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;
+
+       /* 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));
+
+       /* 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,
+                                       data_ptr, SCTP_DATA_FIRST_FRAG, ssn); 
+
+       if (!first_frag)
+               goto err;
+
+       /* All the fragments are added to the frag_list of the first chunk. */
+       frag_list = &first_frag->frag_list;
+
+       chunk_data_len -= max_frag_data_len;
+       data_ptr += max_frag_data_len;
+
+       /* 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;
+
+               /* 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;
+
+       /* Add the last fragment to the first fragment's frag_list. */
+       list_add_tail(&frag->frag_list, frag_list);
+
+       /* Free the original chunk. */
+       sctp_free_chunk(chunk);
+
+       return first_frag;
+
+err:
+       /* 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. */
+               flist = &first_frag->frag_list;
+               while (NULL != (lfrag = sctp_list_dequeue(flist))) {
+                       frag = list_entry(lfrag, sctp_chunk_t, frag_list);
+                       sctp_free_chunk(frag);
+               }
+
+               /* Free the first fragment. */
+               sctp_free_chunk(first_frag);
+       }
+
+       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.
+ */
+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;
+       __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;
+       struct sk_buff_head *queue;
+       sctp_transport_t *transport = NULL;
+       sctp_transport_t *new_transport;
+       sctp_chunk_t *chunk;
+       sctp_xmit_t status;
+       int error = 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;
+
+       /*
+        * 6.10 Bundling
+        *   ...
+        *   When bundling control chunks with DATA chunks, an
+        *   endpoint MUST place control chunks first in the outbound
+        *   SCTP packet.  The transmitter MUST transmit DATA chunks
+        *   within a SCTP packet in increasing order of TSN.
+        *   ...
+        */
+       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) {
+                       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)
+                               new_transport = asoc->peer.active_path;     
+               }
+
+               /* Are we switching transports?
+                * Take care of transport locks.
+                */
+               if (new_transport != transport) {
+                       transport = new_transport;
+                       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);
+               }
+
+               switch (chunk->chunk_hdr->type) {
+               /*
+                * 6.10 Bundling
+                *   ...
+                *   An endpoint MUST NOT bundle INIT, INIT ACK or SHUTDOWN
+                *   COMPLETE with any other chunks.  [Send them immediately.]
+                */
+               case SCTP_CID_INIT:
+               case SCTP_CID_INIT_ACK:
+               case SCTP_CID_SHUTDOWN_COMPLETE:
+                       (*q->init_output)(&singleton, transport, sport, dport);
+                       (*q->config_output)(&singleton, vtag, ecn_capable,
+                                           s_ecne_handler);
+                       (void) (*q->build_output)(&singleton, chunk);
+                       error = (*q->force_output)(&singleton);
+                       if (error < 0)
+                               return(error);
+                       break;
+
+               case SCTP_CID_ABORT:
+               case SCTP_CID_SACK:
+               case SCTP_CID_HEARTBEAT:
+               case SCTP_CID_HEARTBEAT_ACK:
+               case SCTP_CID_SHUTDOWN:
+               case SCTP_CID_SHUTDOWN_ACK:
+               case SCTP_CID_ERROR:
+               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);
+                       break;
+
+               case SCTP_CID_ASCONF:
+               case SCTP_CID_ASCONF_ACK:
+                       (void) (*q->build_output)(packet, chunk);
+                       break;
+
+               default:
+                       /* We built a chunk with an illegal type! */
+                       BUG();                  
+               };
+       }
+
+       /* 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)
+                       break;
+
+               /* fallthru */
+       case SCTP_STATE_ESTABLISHED:
+       case SCTP_STATE_SHUTDOWN_PENDING:
+       case SCTP_STATE_SHUTDOWN_RECEIVED:
+               /*
+                * RFC 2960 6.1  Transmission of DATA Chunks
+                *
+                * C) When the time comes for the sender to transmit,
+                * before sending new DATA chunks, the sender MUST
+                * first transmit any outstanding DATA chunks which
+                * are marked for retransmission (limited by the
+                * current cwnd).
+                */
+               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)) {
+                               list_add_tail(&transport->send_ready,
+                                             &transport_list);
+                       }
+
+                       packet = &transport->packet;
+                       (*q->config_output)(packet, vtag,
+                                           ecn_capable, ecne_handler);
+               retran:
+                       error = sctp_flush_retran_queue(q,
+                                                       packet,
+                                                       rtx_timeout,
+                                                       &start_timer);
+
+                       if (start_timer)
+                               sctp_transport_reset_timers(transport);
+               }
+
+               /* Finally, transmit new packets.  */
+               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.
+                        */
+                       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,
+                                               GFP_ATOMIC);
+                               if (event) {
+                                       sctp_ulpqueue_tail_event(&asoc->ulpq,
+                                                                event);
+                               }
+
+                               /* 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)
+                               new_transport = asoc->peer.active_path;
+
+                       /* Change packets if necessary.  */
+                       if (new_transport != transport) {
+                               transport = new_transport;
+
+                               /* Schedule to have this transport's
+                                * packet flushed.
+                                */
+                               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);
+                       }
+
+                       SCTP_DEBUG_PRINTK("sctp_transmit_packet(%p, %p[%s]), ",
+                                         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 ?
+                                         atomic_read(&chunk->skb->users) :
+                                         -1);
+
+                       /* Add the chunk to the packet.  */
+                       status = (*q->build_output)(packet, chunk);
+
+                       switch (status) {
+                       case SCTP_XMIT_PMTU_FULL:
+                       case SCTP_XMIT_RWND_FULL:
+                               /* We could not append this chunk, so put
+                                * the chunk back on the output queue.
+                                */
+                               SCTP_DEBUG_PRINTK("sctp_flush_outqueue: could"
+                                  "not transmit TSN: 0x%x, status: %d\n",
+                                  ntohl(chunk->subh.data_hdr->tsn), status);
+                               skb_queue_head(queue, (struct sk_buff *)chunk);
+                               goto sctp_flush_out;
+                               break;
+
+                       case SCTP_XMIT_MUST_FRAG: {
+                               sctp_chunk_t *frag;
+
+                               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.
+                                        */
+                                       sctp_free_chunk(chunk);
+                                       error = -ENOMEM;
+                                       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
+                        * conditions.
+                        *
+                        * Is this really a problem?  Won't this behave
+                        * like a lost TSN?
+                        */
+                       list_add_tail(&chunk->transmitted_list,
+                                     &transport->transmitted);
+
+                       sctp_transport_reset_timers(transport);
+
+                       q->empty = 0;
+               }
+               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 ) {
+               sctp_transport_t *t = list_entry(ltransport,
+                                                sctp_transport_t, send_ready);
+               if (t != transport)
+                       transport = t;
+
+               packet = &transport->packet;
+               if (packet->size != SCTP_IP_OVERHEAD)
+                       error = (*q->force_output)(packet);
+       }
+
+       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)
+{
+       q->init_output = init;
+       q->config_output = config;
+       q->append_output = append;
+       q->build_output = build;
+       q->force_output = force;
+       return 0;
+}
+
+/* Update unack_data based on the incoming SACK chunk */
+static void sctp_sack_update_unack_data(sctp_association_t *assoc,
+                                       sctp_sackhdr_t *sack)
+{
+       sctp_sack_variable_t *frags;
+       __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) -
+                               ntohs(frags[i].gab.start) + 1));
+       }
+
+       assoc->unack_data = unack_data;
+}
+
+/* This is where we REALLY process a SACK.
+ *
+ * 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)
+{
+       sctp_chunk_t *tchunk;
+       list_t *lchunk, *transport_list, *pos;
+       __u32 tsn;
+       __u32 sack_ctsn;
+       __u32 ctsn;
+       sctp_transport_t *transport;
+       int outstanding;
+       __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.
+        */
+       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;
+
+       /* Update unack_data field in the assoc. */
+       sctp_sack_update_unack_data(q->asoc, sack);
+
+       ctsn = q->asoc->ctsn_ack_point;
+
+       SCTP_DEBUG_PRINTK("%s: sack Cumulative TSN Ack is 0x%x.\n",
+                         __FUNCTION__,
+                         sack_ctsn);
+       SCTP_DEBUG_PRINTK("%s: Cumulative TSN Ack of association "
+                         "%p is 0x%x.\n", __FUNCTION__, 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) {
+               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;
+
+       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;
+       }
+
+       SCTP_DEBUG_PRINTK("sack queue is empty.\n");      
+finish:
+       return q->empty;
+}
+
+/* Is the outqueue empty?  */
+int sctp_outqueue_is_empty(const sctp_outqueue_t *q)
+{
+       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
+ * retransmit parameters are also updated and if needed a fast retransmit
+ * process is started.
+ *
+ * I added coherent debug information output.  --xguo
+ *
+ * Instead of printing 'sacked' or 'kept' for each TSN on the
+ * 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)
+{
+       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
+       __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).
+        * -1: We need to initialize.
+        */
+       int dbg_prt_state = -1;
+#endif /* SCTP_DEBUG */
+
+       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);
+               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 ((!tchunk->tsn_gap_acked) &&
+                                   (1 == tchunk->num_times_sent) &&
+                                   (tchunk->rtt_in_progress)) {
+                                       rtt = jiffies - tchunk->sent_at;
+                                       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;
+
+                               if (!tchunk->tsn_gap_acked) {
+                                       tchunk->tsn_gap_acked = 1;
+                                       bytes_acked += sctp_data_size(tchunk);
+                               }
+
+                               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'.
+                                */
+                               if (!tchunk->tsn_gap_acked) {
+                                       tchunk->tsn_gap_acked = 1;
+                                       bytes_acked += sctp_data_size(tchunk);
+                                       if (TSN_lt(highest_new_tsn_in_sack,
+                                                  tsn)) {
+                                               highest_new_tsn_in_sack = tsn;
+                                       }
+                               }
+                               list_add_tail(lchunk, &tlist);
+                       }
+
+#if SCTP_DEBUG
+                       switch (dbg_prt_state) {
+                       case 0: /* last TSN was ACKed */
+                               if (dbg_last_ack_tsn + 1 == tsn) {
+                                       /* This TSN belongs to the
+                                        * current ACK range.
+                                        */
+                                       break;
+                               }
+
+                               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);
+                               dbg_ack_tsn = tsn;
+                               break;
+
+                       case 1: /* The last TSN was NOT ACKed. */
+                               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);
+                               dbg_prt_state = 0;
+                               dbg_ack_tsn = tsn;
+                       };
+
+                       dbg_last_ack_tsn = tsn;
+#endif /* SCTP_DEBUG */
+
+               } else {
+                       if (tchunk->tsn_gap_acked) {
+                               SCTP_DEBUG_PRINTK("%s: Receiver reneged on data "
+                                                 "TSN: 0x%x\n",
+                                                 __FUNCTION__,
+                                                 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.
+                                */
+                               restart_timer = 1;
+                       }
+
+                       list_add_tail(lchunk, &tlist);
+
+#if SCTP_DEBUG
+                       /* See the above comments on ACK-ed TSNs. */
+                       switch (dbg_prt_state) {
+                       case 1:
+                               if (dbg_last_kept_tsn + 1 == tsn)
+                                       break;
+
+                               if (dbg_last_kept_tsn != dbg_kept_tsn)
+                                       SCTP_DEBUG_PRINTK("-%08x",
+                                                         dbg_last_kept_tsn);
+
+                               SCTP_DEBUG_PRINTK(",%08x", tsn);
+                               dbg_kept_tsn = tsn;
+                               break;
+
+                       case 0:
+                               if (dbg_last_ack_tsn != dbg_ack_tsn)
+                                       SCTP_DEBUG_PRINTK("-%08x",
+                                                         dbg_last_ack_tsn);
+                               SCTP_DEBUG_PRINTK("\n");
+
+                               /* FALL THROUGH... */
+                       default:
+                               SCTP_DEBUG_PRINTK("KEPT: %08x",tsn);
+                               dbg_prt_state = 1;
+                               dbg_kept_tsn = tsn;
+                       };
+
+                       dbg_last_kept_tsn = tsn;
+#endif /* SCTP_DEBUG */
+               }
+       }
+
+#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");
+               }
+       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");
+               }
+       };
+#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
+                        * DATA chunk was last sent.
+                        * 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
+                        * 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_transport_raise_cwnd(transport, sack_ctsn,
+                                                 bytes_acked);
+
+                       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
+                        * closed for an indefinite time.
+                        * 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)) {
+                               SCTP_DEBUG_PRINTK("%s: SACK received for zero "
+                                                 "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) {
+                       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))
+                               sctp_transport_hold(transport);
+               }
+       }
+
+       /* 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);
+
+               /* 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
+                * 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.
+                */
+               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,
+                                         tchunk->tsn_missing_report);
+               }
+               if (tchunk->tsn_missing_report >= 4) {
+                       tchunk->fast_retransmit = 1;
+                       do_fast_retransmit = 1;
+               }
+
+               list_add_tail(lchunk, transmitted_queue);
+       }
+
+       if (transport) {
+               if (do_fast_retransmit)
+                       sctp_retransmit(q, transport, do_fast_retransmit);
+
+               SCTP_DEBUG_PRINTK("%s: transport: %p, cwnd: %d, "
+                                 "ssthresh: %d, flight_size: %d, pba: %d\n",
+                                 __FUNCTION__,
+                                 transport, transport->cwnd,
+                                 transport->ssthresh, transport->flight_size,
+                                 transport->partial_bytes_acked);
+       }
+}
+
+/* Is the given TSN acked by this packet?  */
+static int sctp_acked(sctp_sackhdr_t *sack, __u32 tsn)
+{
+       int i;
+       sctp_sack_variable_t *frags;
+       __u16 gap;
+       __u32 ctsn = ntohl(sack->cum_tsn_ack);
+
+        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;
+}
diff --git a/net/sctp/primitive.c b/net/sctp/primitive.c
new file mode 100644 (file)
index 0000000..c4ca650
--- /dev/null
@@ -0,0 +1,202 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * These functions implement the SCTP primitive functions from Section 10.
+ * 
+ * Note that the descriptions from the specification are USER level
+ * functions--this file is the functions which populate the struct proto
+ * for SCTP which is the BOTTOM of the sockets interface.
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Narasimha Budihal     <narasimha@refcode.org>
+ *    Karl Knutson          <karl@athena.chicago.il.us>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/types.h>
+#include <linux/list.h> /* For struct list_head */
+#include <linux/socket.h>
+#include <linux/ip.h>
+#include <linux/time.h> /* For struct timeval */
+#include <net/sock.h>
+#include <net/sctp/sctp.h>
+#include <net/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; \
+       sctp_state_t state; \
+       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; \
+       \
+       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)
+ * -> association id [,destination transport addr list] [,outbound stream
+ *    count]
+ *
+ * This primitive allows the upper layer to initiate an association to a
+ * 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.
+ */
+
+/* This is called in the code as sctp_primitive_ASSOCIATE.  */
+
+DECLARE_PRIMITIVE(ASSOCIATE)
+
+/* 10.1 ULP-to-SCTP
+ * C) Shutdown
+ *
+ * Format: SHUTDOWN(association id)
+ * -> 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.
+ */
+
+DECLARE_PRIMITIVE(SHUTDOWN);
+
+/* 10.1 ULP-to-SCTP
+ * C) Abort
+ *
+ * Format: Abort(association id [, cause code])
+ * -> 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
+ * attempting to abort the association results in a failure, an error
+ * code shall be returned.
+ */
+
+DECLARE_PRIMITIVE(ABORT);
+
+/* 10.1 ULP-to-SCTP
+ * E) Send
+ *
+ * Format: SEND(association id, buffer address, byte count [,context]
+ *         [,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.
+ *
+ * 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
+ *    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.
+ *
+ *  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
+ *    message).
+ *
+ *  o no-bundle flag - instructs SCTP not to bundle this user data with
+ *    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
+ *    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 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);
+
+       return error;
+}
diff --git a/net/sctp/protocol.c b/net/sctp/protocol.c
new file mode 100644 (file)
index 0000000..8c087d1
--- /dev/null
@@ -0,0 +1,591 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 International Business Machines, Corp.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001 Nokia, Inc.
+ * Copyright (c) 2001 La Monte H.P. Yarroll
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * Initialization/cleanup for SCTP protocol support.  
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson <karl@athena.chicago.il.us>
+ *    Jon Grimm <jgrimm@us.ibm.com>
+ *    Sridhar Samudrala <sri@us.ibm.com>
+ *    Daisy Chang <daisyc@us.ibm.com>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/module.h>
+#include <linux/init.h>
+#include <linux/netdevice.h>
+#include <linux/inetdevice.h>
+#include <net/protocol.h>
+#include <net/ip.h>
+#include <net/ipv6.h>
+#include <net/sctp/sctp.h>
+#include <net/addrconf.h>
+#include <net/inet_common.h>
+
+/* Global data structures. */
+sctp_protocol_t sctp_proto;
+struct proc_dir_entry  *proc_net_sctp;
+
+/* This is the global socket data structure used for responding to
+ * the Out-of-the-blue (OOTB) packets.  A control sock will be created
+ * for this socket at the initialization time.
+ */
+static struct socket *sctp_ctl_socket;
+
+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;
+}
+
+/* Set up the proc fs entry for the SCTP protocol. */
+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;
+               }
+       }
+}
+
+/* Clean up the proc fs entry for the SCTP protocol. */
+void sctp_proc_exit(void)
+{
+       if (proc_net_sctp) {
+               proc_net_sctp= NULL;
+               remove_proc_entry("net/sctp", 0);
+       }
+}
+
+/* Private helper to extract ipv4 address and stash them in
+ * the protocol structure.
+ */
+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 sockaddr_storage_list *addr;
+
+       read_lock(&inetdev_lock);
+       if ((in_dev = __in_dev_get(dev)) == NULL) {
+               read_unlock(&inetdev_lock);
+               return;
+       }
+
+       read_lock(&in_dev->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);
+                       addr->a.v4.sin_family = AF_INET;
+                       addr->a.v4.sin_port = 0;
+                       addr->a.v4.sin_addr.s_addr = ifa->ifa_local;
+                       list_add_tail(&addr->list, &proto->local_addr_list);
+               }
+       }
+
+       read_unlock(&in_dev->lock);
+       read_unlock(&inetdev_lock);
+}
+
+/* Private helper to extract ipv6 address and stash them in
+ * the protocol structure.
+ * FIXME: Make this an address family function.
+ */
+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
+       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_unlock(&addrconf_lock);
+               return;
+       }
+
+       read_lock_bh(&in6_dev->lock);
+       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_port = 0;
+                       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 */
+#endif /* SCTP_V6_SUPPORT */
+}
+
+/* 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)
+{
+       struct net_device *dev;
+
+       read_lock(&dev_base_lock);
+       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);
+}
+
+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);
+}
+
+/* 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;
+
+       list_for_each_safe(pos, temp, &proto->local_addr_list) {
+               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);
+}
+
+/* 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)
+{
+       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
+                        * 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,
+                                                          priority);
+                               if (error)
+                                       goto end_copy;
+                       }
+               }
+       }
+
+end_copy:
+       sctp_spin_unlock_irqrestore(&proto->local_addr_lock, flags);
+
+       return error;
+}
+
+/* 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 = {
+               .dst   = address->v4.sin_addr.s_addr,
+               .src   = 0,
+               .iif   = 0,
+               .oif   = 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",
+                                 dst_mtu);
+       } else {
+               dst_mtu = rt->u.dst.pmtu;
+               SCTP_DEBUG_PRINTK("sctp_v4_get_dst_mtu: "
+                                 "ip_route_output_key: dev:%s pmtu:%d\n",
+                                 rt->u.dst.dev->name, dst_mtu);
+               ip_rt_put(rt);
+       }
+
+       return dst_mtu;
+}
+
+/* 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)
+{
+       long flags __attribute__ ((unused));
+
+       sctp_spin_lock_irqsave(&sctp_proto.local_addr_lock, flags);
+       __sctp_free_local_addr_list(&sctp_proto);
+       __sctp_get_local_addr_list(&sctp_proto);
+       sctp_spin_unlock_irqrestore(&sctp_proto.local_addr_lock, flags);
+
+       return NOTIFY_DONE;
+}
+
+/*
+ * 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 err = 0;
+       int family = PF_INET;
+
+       SCTP_V6(family = PF_INET6;)
+
+       err = sock_create(family, SOCK_SEQPACKET, IPPROTO_SCTP,
+                         &sctp_ctl_socket);
+       if (err < 0) {
+               printk(KERN_ERR 
+                       "SCTP: Failed to create the SCTP control socket.\n");
+               return err;
+       }
+       sctp_ctl_socket->sk->allocation = GFP_ATOMIC;
+       inet_sk(sctp_ctl_socket->sk)->ttl = MAXTTL;
+
+       return 0;
+}
+
+/* Get the table of functions for manipulating a particular address
+ * family.
+ */
+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;
+
+       retval = NULL;
+
+       /* 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;
+               }
+       }
+
+       return retval;
+}
+
+/* Registration for netdev events.  */
+struct notifier_block sctp_netdev_notifier = {
+       .notifier_call = sctp_netdev_event,
+};
+
+/* 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,
+       .socketpair  = sock_no_socketpair,
+       .accept      = inet_accept,
+       .getname     = inet_getname,      /* Semantics are different.  */
+       .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,
+       .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,
+       .protocol   = IPPROTO_SCTP,
+       .prot       = &sctp_prot,
+       .ops        = &inet_seqpacket_ops,
+       .capability = -1,
+       .no_check   = 0,
+       .flags      = SCTP_PROTOSW_FLAG
+};
+
+/* 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.  */
+sctp_func_t sctp_ipv4_specific = {
+       .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),
+       .sa_family      = AF_INET,
+};
+
+/* Initialize the universe into something sensible.  */
+int sctp_init(void)
+{
+       int i;
+       int status = 0;
+
+       /* Add SCTP to inetsw linked list.  */
+       inet_register_protosw(&sctp_protosw);
+
+       /* Add SCTP to inet_protos hash table.  */
+       inet_add_protocol(&sctp_protocol);
+
+       /* Initialize proc fs directory.  */
+       sctp_proc_init();
+
+       /* 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;
+
+       /* 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;
+
+       /* 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.
+        */
+       sctp_proto.max_instreams    = SCTP_MAX_STREAM;
+       sctp_proto.max_outstreams   = SCTP_MAX_STREAM;
+
+       /* 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");
+               status = -ENOMEM;
+               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.  */
+       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");
+               status = -ENOMEM;
+               goto err_ehash_alloc;
+       }
+
+       for (i = 0; i < sctp_proto.ep_hashsize; i++) {
+               sctp_proto.ep_hashbucket[i].lock = RW_LOCK_UNLOCKED;
+               sctp_proto.ep_hashbucket[i].chain = NULL;
+       }
+
+       /* Allocate and initialize the SCTP port hash table.  */
+       sctp_proto.port_hashsize = 4096;
+       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.");
+               status = -ENOMEM;
+               goto err_bhash_alloc;
+       }
+
+       sctp_proto.port_alloc_lock = SPIN_LOCK_UNLOCKED;
+       sctp_proto.port_rover = sysctl_local_port_range[0] - 1;
+       for (i = 0; i < sctp_proto.port_hashsize; i++) {
+               sctp_proto.port_hashtable[i].lock = SPIN_LOCK_UNLOCKED;
+               sctp_proto.port_hashtable[i].chain = NULL;
+       }
+
+       sctp_sysctl_register();
+
+       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)
+               goto err_v6_init;
+
+       /* 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;
+       }
+
+       /* Initialize the local address list. */
+       INIT_LIST_HEAD(&sctp_proto.local_addr_list);
+       sctp_proto.local_addr_lock = SPIN_LOCK_UNLOCKED;
+
+       register_inetaddr_notifier(&sctp_netdev_notifier);
+       sctp_get_local_addr_list(&sctp_proto);
+
+       return 0;
+
+err_ctl_sock_init:
+       sctp_v6_exit();
+err_v6_init:
+       sctp_sysctl_unregister();
+       list_del(&sctp_ipv4_specific.list);
+       kfree(sctp_proto.port_hashtable);
+err_bhash_alloc:
+       kfree(sctp_proto.ep_hashbucket);
+err_ehash_alloc:
+       kfree(sctp_proto.assoc_hashbucket);
+err_ahash_alloc:
+       sctp_dbg_objcnt_exit();
+       sctp_proc_exit();
+       inet_del_protocol(&sctp_protocol);
+       inet_unregister_protosw(&sctp_protosw);
+       return status;
+}
+
+/* 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.  */
+       unregister_inetaddr_notifier(&sctp_netdev_notifier);
+       sctp_free_local_addr_list(&sctp_proto);
+
+       /* Free the control endpoint.  */
+       sock_release(sctp_ctl_socket);
+
+       sctp_v6_exit();
+       sctp_sysctl_unregister();
+       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);
+}
+
+module_init(sctp_init);
+module_exit(sctp_exit);
+
+MODULE_AUTHOR("Linux Kernel SCTP developers <lksctp-developers@lists.sourceforge.net>");
+MODULE_DESCRIPTION("Support for the SCTP protocol (RFC2960)");
+MODULE_LICENSE("GPL");
+
diff --git a/net/sctp/sctp_adler32.c b/net/sctp/sctp_adler32.c
deleted file mode 100644 (file)
index ebacd8e..0000000
+++ /dev/null
@@ -1,150 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_adler32.c,v 1.5 2002/06/13 16:03:38 jgrimm Exp $
- * 
- * This file has direct heritage from the SCTP user-level reference 
- * implementation by R. Stewart, et al.  These functions implement the
- * Adler-32 algorithm as specified by RFC 2960. 
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    Randall Stewart <rstewar1@email.mot.com>
- *    Ken Morneau     <kmorneau@cisco.com>
- *    Qiaobing Xie    <qxie1@email.mot.com>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_adler32.c,v 1.5 2002/06/13 16:03:38 jgrimm Exp $";
-
-/* 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.
- *
- * Well, the above is a bit of a lie, I have optimized this a small
- * tad, but I have commented the original lines below
- */
-
-#include <linux/types.h> 
-#include <net/sctp/sctp.h>
-
-#define BASE 65521 /* largest prime smaller than 65536 */
-
-
-/* Performance work as shown this pig to be the
- * worst CPU wise guy. I have done what I could think
- * of on my flight to Austrialia but I am sure some
- * clever assembly could speed this up, but of
- * course this would require the dreaded #ifdef's for
- * architecture. If you can speed this up more, pass
- * it back and we will incorporate it :-)
- */
-
-unsigned long update_adler32(unsigned long adler,
-                            unsigned char *buf, int len)
-{
-       __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;
-}
-
-__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.
-        */
-       __u32 adler = 1L;
-       __u32 zero = 0L;
-
-       /* Calculate the CRC up to the checksum field. */
-       adler = update_adler32(adler, ptr, 
-                              sizeof(struct sctphdr) - sizeof(__u32));
-       /* Skip over the checksum field. */
-       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;
-}
diff --git a/net/sctp/sctp_associola.c b/net/sctp/sctp_associola.c
deleted file mode 100644 (file)
index 4694f76..0000000
+++ /dev/null
@@ -1,1036 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 International Business Machines Corp.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001 La Monte H.P. Yarroll
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_associola.c,v 1.48 2002/08/16 19:30:49 jgrimm Exp $
- * 
- * This module provides the abstraction for an SCTP association.
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson          <karl@athena.chicago.il.us>
- *    Jon Grimm             <jgrimm@us.ibm.com>
- *    Xingang Guo           <xingang.guo@intel.com>
- *    Hui Huang             <hui.huang@nokia.com>
- *    Sridhar Samudrala            <sri@us.ibm.com>
- *    Daisy Chang          <daisyc@us.ibm.com>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_associola.c,v 1.48 2002/08/16 19:30:49 jgrimm Exp $";
-
-#include <linux/types.h>
-#include <linux/fcntl.h>
-#include <linux/poll.h>
-#include <linux/init.h>
-#include <linux/sched.h>
-
-#include <linux/sched.h>
-#include <linux/slab.h>
-#include <linux/in.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 *asoc;
-
-       asoc = t_new(sctp_association_t, priority);
-       if (!asoc)
-               goto fail;
-
-       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;
-}
-
-/* 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_opt_t *sp;
-       int i;
-
-       /* Retrieve the SCTP per socket area.  */
-       sp = sctp_sk((struct sock *)sk);
-
-       /* Init all variables to a known value.  */
-       memset(asoc, 0, sizeof(sctp_association_t));
-
-       /* Discarding const is appropriate here.  */
-       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. */
-       asoc->base.type = SCTP_EP_TYPE_ASSOCIATION;
-
-       /* Initialize the object handling fields. */
-       atomic_set(&asoc->base.refcnt, 1);
-       asoc->base.dead = 0;
-       asoc->base.malloced = 0;
-
-       /* 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->pmtu = 0;
-       asoc->frag_point = 0;
-
-       /* Initialize the default association max_retrans and RTO values. */
-       asoc->max_retrans = ep->proto->max_retrans_association;
-       asoc->rto_initial = ep->proto->rto_initial;
-       asoc->rto_max = ep->proto->rto_max;
-       asoc->rto_min = ep->proto->rto_min;
-
-       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.
-        */
-       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;
-       }
-
-       /* 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->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;
-
-       /* 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)
-               asoc->rwnd = SCTP_DEFAULT_MINWINDOW;
-       else
-               asoc->rwnd = sk->rcvbuf;
-
-       asoc->rwnd_over = 0;
-
-       /* 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.peer_vtag = 0;
-       asoc->c.my_ttag   = 0;
-       asoc->c.peer_ttag = 0;
-
-       asoc->c.initial_tsn = sctp_generate_tsn(ep);
-
-       asoc->next_tsn = asoc->c.initial_tsn;
-
-       asoc->ctsn_ack_point = asoc->next_tsn - 1;
-
-       asoc->unack_data = 0;
-
-       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:
-        * ...
-        * 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.
-        */
-       asoc->addip_serial = asoc->c.initial_tsn;
-
-       /* 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
-        *
-        * 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.]
-        */
-       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,
-                                   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))
-               goto fail_init;
-
-       /* 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);
-
-       asoc->need_ecne = 0;
-
-       asoc->debug_name = "unnamedasoc";
-       asoc->eyecatcher = SCTP_ASSOC_EYECATCHER;
-
-       /* Assume that peer would support both address types unless we are
-        * told otherwise.
-        */
-       asoc->peer.ipv4_address = 1;
-       asoc->peer.ipv6_address = 1;
-       INIT_LIST_HEAD(&asoc->asocs);
-
-       asoc->autoclose = sp->autoclose;
-
-       return asoc;
-
-fail_init:
-       sctp_endpoint_put(asoc->ep);
-       sock_put(asoc->base.sk);
-       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)
-{
-       sctp_transport_t *transport;
-       sctp_endpoint_t *ep;
-       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.
-        */
-       asoc->base.dead = 1;
-
-       /* 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);
-
-       /* Dispose of any pending chunks on the inqueue. */
-       sctp_inqueue_free(&asoc->base.inqueue);
-
-       /* 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
-        * should be able to go through and make a guess based
-        * 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]))
-                       sctp_association_put(asoc);
-       }
-
-       /* 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);
-       }
-
-       asoc->eyecatcher = 0;
-
-       sctp_association_put(asoc);
-}
-
-
-/* Cleanup and free up an association. */
-static void sctp_association_destroy(sctp_association_t *asoc)
-{
-       SCTP_ASSERT(asoc->base.dead, "Assoc is not dead", return);
-
-       sctp_endpoint_put(asoc->ep);
-       sock_put(asoc->base.sk);
-
-       if (asoc->base.malloced) {
-               kfree(asoc);
-               SCTP_DBG_OBJCNT_DEC(assoc);
-       }
-}
-
-
-/* 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 *peer;
-       sctp_opt_t *sp;
-       const __u16 *port;
-
-       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;
-       };
-
-       /* Set the port if it has not been set yet.  */
-        if (0 == asoc->peer.port) {
-                asoc->peer.port = *port;
-        }
-
-       SCTP_ASSERT(*port == asoc->peer.port, ":Invalid port\n",
-                   return NULL);
-
-       /* Check to see if this is a duplicate. */
-       peer = sctp_assoc_lookup_paddr(asoc, addr);
-       if (peer)
-               return peer;
-
-       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;
-       }
-
-       SCTP_DEBUG_PRINTK("sctp_assoc_add_peer:association %p PMTU set to "
-                         "%d\n", asoc, 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
-        * initialize the packet structure anyway.
-        */
-       (asoc->outqueue.init_output)(&peer->packet,
-                                    peer,
-                                    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).
-        */
-       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
-        * later when we process the INIT.
-        */
-       peer->ssthresh = SCTP_DEFAULT_MAXWINDOW;
-
-       peer->partial_bytes_acked = 0;
-       peer->flight_size = 0;
-
-       peer->error_threshold = peer->max_retrans;
-
-       /* Update the overall error threshold value of the association
-        * taking the new peer's error threshold into account.
-        */
-       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.
-        */
-       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;
-
-       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 *t;
-       list_t *pos;
-
-       /* Cycle through all transports searching for a peer address. */
-
-       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;
-       }
-
-       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)
-{
-       sctp_transport_t *t = NULL;
-       sctp_transport_t *first;
-       sctp_transport_t *second;
-       sctp_ulpevent_t *event;
-       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();
-       };
-
-       /* Generate and send a SCTP_PEER_ADDR_CHANGE notification to the
-        * user.
-        */
-       event = sctp_ulpevent_make_peer_addr_change(asoc,
-                               (struct sockaddr_storage *) &transport->ipaddr,
-                               0, spc_state, error, GFP_ATOMIC);
-       if (event)
-               sctp_ulpqueue_tail_event(&asoc->ulpq, event);
-
-       /* Select new active and retran paths. */
-
-       /* Look for the two most recently used active transports.
-        *
-        * This code produces the wrong ordering whenever jiffies
-        * rolls over, but we still get usable transports, so we don't
-        * worry about it.
-        */
-       first = NULL; second = NULL;
-
-       list_for_each(pos, &asoc->peer.transport_addr_list) {
-               t = list_entry(pos, sctp_transport_t, transports);
-
-               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)
-                       second = t;
-       }
-
-       /* RFC 2960 6.4 Multi-Homed SCTP Endpoints
-        *
-        * By default, an endpoint should always transmit to the
-        * primary path, unless the SCTP user explicitly specifies the
-        * destination transport address (and possibly source
-        * transport address) to use.
-        *
-        * [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) {
-               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) {
-               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;
-}
-
-/* Hold a reference to an association. */
-void sctp_association_hold(sctp_association_t *asoc)
-{
-       atomic_inc(&asoc->base.refcnt);
-}
-
-/* Release a reference to an association and cleanup
- * if there are no more references.
- */
-void sctp_association_put(sctp_association_t *asoc)
-{
-       if (atomic_dec_and_test(&asoc->base.refcnt))
-               sctp_association_destroy(asoc);
-}
-
-/* Allocate the next TSN, Transmission Sequence Number, for the given
- * association.
- */
-__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.
-        */
-       __u32 retval = asoc->next_tsn;
-       asoc->next_tsn++;
-       asoc->unack_data++;
-
-        return retval;
-}
-
-/* Allocate 'num' TSNs by incrementing the association's TSN by num. */
-__u32 __sctp_association_get_tsn_block(sctp_association_t *asoc, int num)
-{
-       __u32 retval = asoc->next_tsn;
-
-       asoc->next_tsn += num;
-       asoc->unack_data += num;
-
-       return retval;
-}
-
-/* Fetch the next Stream Sequence Number for stream number 'sid'.  */
-__u16 __sctp_association_get_next_ssn(sctp_association_t *asoc, __u16 sid)
-{
-       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 len;
-       const void *base1;
-       const void *base2;
-
-       if (ss1->sa.sa_family != ss2->sa.sa_family)
-               return 0;
-       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)
-                       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))
-                                   goto match;
-
-                       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;
-}
-
-/* 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 len;
-       const void *base1;
-       const void *base2;
-
-       if (ss1->sa.sa_family != ss2->sa.sa_family)
-               return 0;
-       if (ss1->v4.sin_port != ss2->v4.sin_port)
-               return 0;
-
-       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);
-                       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));
-}
-
-/* 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 *chunk;
-       int need_ecne;
-       __u32 lowest_tsn;
-
-       /* 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;
-
-       if (need_ecne) {
-               chunk = sctp_make_ecne(asoc, lowest_tsn);
-
-               /* ECNE is not mandatory to the flow.  Being unable to
-                * alloc mem is not deadly.  We are just unable to help
-                * out the network.  If we run out of memory, just return
-                * NULL.
-                */
-       } else {
-               chunk = NULL;
-       }
-
-       return chunk;
-}
-
-/* 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)
-{
-       return NULL;
-}
-
-/*
- * Find which transport this TSN was sent on.
- */
-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;
-       __u32 key = htonl(tsn);
-
-       match = NULL;
-
-       /*
-        * FIXME: In general, find a more efficient data structure for
-        * 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) {
-               chunk = list_entry(entry, sctp_chunk_t, transmitted_list);
-
-               if (key == chunk->subh.data_hdr->tsn) {
-                       match = active;
-                       goto out;
-               }
-       }
-
-       /* 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;
-               list_for_each(entry, &transport->transmitted) {
-                       chunk = list_entry(entry, sctp_chunk_t,
-                                          transmitted_list);
-                       if (key == chunk->subh.data_hdr->tsn) {
-                               match = transport;
-                               goto out;
-                       }
-               }
-       }
-out:
-       return match;
-}
-
-/* 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 *transport;
-
-       sctp_read_lock(&asoc->base.addr_lock);
-
-       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))
-                       goto out;
-       }
-       transport = NULL;
-
-out:
-       sctp_read_unlock(&asoc->base.addr_lock);
-       return transport;
-}
-
-/* Do delayed input processing.  This is scheduled by sctp_rcv(). */
-static void sctp_assoc_bh_rcv(sctp_association_t *asoc)
-{
-       sctp_endpoint_t *ep;
-       sctp_chunk_t *chunk;
-       struct sock *sk;
-       sctp_inqueue_t *inqueue;
-       int state, subtype;
-       sctp_assoc_t associd = sctp_assoc2id(asoc);
-       int error = 0;
-
-       /* The association should be held so we should be safe. */
-       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;
-
-               /* Remember where the last DATA chunk came from so we
-                * know where to send the SACK.
-                */
-               if (sctp_chunk_is_data(chunk))
-                       asoc->peer.last_data_from = 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),
-                                  state, ep, asoc, chunk, GFP_ATOMIC);
-
-               /* 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;
-       }
-
-err_out:
-       /* Is this the right way to pass errors up to the ULP?  */
-       if (error)
-               sk->err = -error;
-out:
-}
-
-/* This routine moves an association from its old sk to a new sk.  */
-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.
-        */
-       list_del(&assoc->asocs);
-
-       /* Release references to the old endpoint and the sock.  */
-       sctp_endpoint_put(assoc->ep);
-       sock_put(assoc->base.sk);
-
-       /* Get a reference to the new endpoint.  */
-       assoc->ep = newsp->ep;
-       sctp_endpoint_hold(assoc->ep);
-
-       /* Get a reference to the new sock.  */
-       assoc->base.sk = newsk;
-       sock_hold(assoc->base.sk);
-
-       /* Add the association to the new endpoint's list of associations.  */
-       sctp_endpoint_add_asoc(newsp->ep, assoc);
-}
-
-/* Update an association (possibly from unexpected COOKIE-ECHO processing).  */
-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.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,
-                        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) {
-               asoc->next_tsn = new->next_tsn;
-               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++) {
-                       asoc->ssn[i]      = 0;
-                       asoc->ulpq.ssn[i] = 0;
-               }
-       } else {
-               asoc->ctsn_ack_point = asoc->next_tsn - 1;
-       }
-}
-
-/* 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.
- */
-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)
-               return asoc->peer.active_path;
-
-       /* Otherwise, find the next transport in a round-robin fashion. */
-
-       t = asoc->shutdown_last_sent_to;
-       pos = &t->transports;
-       next = NULL;
-
-       while (1) {
-               /* Skip the head. */
-               if (pos->next == head)
-                       pos = head->next;
-               else
-                       pos = pos->next;
-
-               t = list_entry(pos, sctp_transport_t, transports);
-
-               /* Try to find an active transport. */
-
-               if (t->state.active) {
-                       break;
-               } else {
-                       /* Keep track of the next transport in case
-                        * we don't find any active transport.
-                        */
-                       if (!next)
-                               next = t;
-               }
-
-               /* We have exhausted the list, but didn't find any
-                * other active transports.  If so, use the next
-                * transport.
-                */
-               if (t == asoc->shutdown_last_sent_to) {
-                       t = next;
-                       break;
-               }
-       }
-
-       return t;
-}
diff --git a/net/sctp/sctp_bind_addr.c b/net/sctp/sctp_bind_addr.c
deleted file mode 100644 (file)
index 5cfa1af..0000000
+++ /dev/null
@@ -1,533 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) Cisco 1999,2000 
- * Copyright (c) Motorola 1999,2000,2001 
- * Copyright (c) International Business Machines Corp., 2001
- * Copyright (c) La Monte H.P. Yarroll 2001
- *
- * This file is part of the SCTP kernel reference implementation.
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_bind_addr.c,v 1.16 2002/07/12 15:15:45 jgrimm Exp $
- * 
- * A collection class to handle the storage of transport addresses. 
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson          <karl@athena.chicago.il.us>
- *    Jon Grimm             <jgrimm@us.ibm.com>
- *    Daisy Chang           <daisyc@us.ibm.com>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_bind_addr.c,v 1.16 2002/07/12 15:15:45 jgrimm Exp $";
-
-#include <linux/types.h>
-#include <linux/sched.h>
-#include <linux/in.h>
-#include <net/sock.h>
-#include <net/ipv6.h>
-#include <net/if_inet6.h>
-#include <net/sctp/sctp.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 *,
-                             sctp_scope_t scope, int priority, int flags);
-static void sctp_bind_addr_clean(sctp_bind_addr_t *);
-
-/* First Level Abstractions. */
-
-/* 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)
-{
-       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;
-       }
-
-out:
-       if (error)
-               sctp_bind_addr_clean(dest);
-       
-       return error;
-}
-
-/* Create a new SCTP_bind_addr from nothing.  */
-sctp_bind_addr_t *sctp_bind_addr_new(int priority)
-{
-       sctp_bind_addr_t *retval;
-
-       retval = t_new(sctp_bind_addr_t, priority);
-       if (!retval)
-               goto nomem;
-
-       sctp_bind_addr_init(retval, 0);
-       retval->malloced = 1;
-       SCTP_DBG_OBJCNT_INC(bind_addr);
-
-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, __u16 port)
-{
-       bp->malloced = 0;
-
-       INIT_LIST_HEAD(&bp->address_list);
-       bp->port = port;
-}
-
-/* Dispose of the address list. */
-static void sctp_bind_addr_clean(sctp_bind_addr_t *bp)
-{
-       struct sockaddr_storage_list *addr;
-       list_t *pos, *temp;
-
-       /* Empty the bind address list. */
-       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);
-       }
-}
-
-/* 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);
-
-       if (bp->malloced) {
-               kfree(bp);
-               SCTP_DBG_OBJCNT_DEC(bind_addr);
-       }
-}
-
-/* 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)
-{
-       struct sockaddr_storage_list *addr;
-
-       /* 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 (!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;
-}
-
-/* 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)
-{
-       list_t *pos, *temp;
-       struct sockaddr_storage_list *addr;
-
-       list_for_each_safe(pos, temp, &bp->address_list) {
-               addr = list_entry(pos, struct sockaddr_storage_list, list);
-               if (sctp_cmp_addr_exact(&addr->a, del_addr)) {
-                       /* Found the exact match. */
-                       list_del(pos);
-                       kfree(addr);
-                       SCTP_DBG_OBJCNT_DEC(addr);
-
-                       return 0;
-               }
-       }
-
-       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 rawaddr;
-       sctpParam_t addrparms;
-       sctpParam_t retval;
-       int addrparms_len;
-       sctpIpAddress_t rawaddr_space;
-       int len;
-       struct sockaddr_storage_list *addr;
-       list_t *pos;
-
-       retval.v = NULL;
-       addrparms_len = 0;
-       len = 0;
-
-       /* Allocate enough memory at once. */
-       list_for_each(pos, &bp->address_list) {
-               len += sizeof(sctp_ipv6addr_param_t);
-       }
-
-       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) {
-               addr = list_entry(pos, struct sockaddr_storage_list, list);
-               len = sockaddr2sctp_addr(&addr->a, rawaddr);
-               memcpy(addrparms.v, rawaddr.v, len);
-               addrparms.v += len;
-               addrparms_len += len;
-       }
-
-end_raw:
-       *addrs_len = addrparms_len;
-       return retval;
-}
-
-/*
- * 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, __u8 *raw_addr_list,
-                          int addrs_len, __u16 port, int priority)
-{
-       sctpParam_t rawaddr;
-       sockaddr_storage_t addr;
-       int retval = 0;
-       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
-                   || SCTP_PARAM_IPV6_ADDRESS==rawaddr.p->type) {
-                       sctp_param2sockaddr(&addr, rawaddr, port);
-                       retval = sctp_add_bind_addr(bp, &addr, priority);
-                       if (retval) {
-                               /* Can't finish building the list, clean up. */
-                               sctp_bind_addr_clean(bp);
-                               break;
-                       }
-
-                       len = ntohs(rawaddr.p->length);
-                       addrs_len -= len;
-                       raw_addr_list += len;
-               } else {
-                       /* Corrupted raw addr list! */
-                       retval = -EINVAL;
-                       sctp_bind_addr_clean(bp);
-                       break;
-               }
-       }
-
-       return retval;
-}
-
-/********************************************************************
- * 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)
-{
-       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;
-       }
-
-       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)
-{
-       sctp_protocol_t *proto = sctp_get_protocol();
-       int error = 0;
-
-       if (sctp_is_any(addr)) {
-               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))))
-                       error = sctp_add_bind_addr(dest, addr, priority);
-       }
-
-       return error;
-}
-
-/* Is addr one of the wildcards?  */
-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)
-                       retval = 1;
-               break;
-
-       case AF_INET6:
-               SCTP_V6(
-                       if (IPV6_ADDR_ANY ==
-                           sctp_ipv6_addr_type(&addr->v6.sin6_addr))
-                               retval = 1;
-                       );
-               break;
-
-       default:
-               break;
-       };
-
-       return retval;
-}
-
-/* Is 'addr' valid for '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: 
-                * 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
-                * SHOULD include all of its addresses with level greater
-                * 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;
-
-               /* 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.  There is no filtering for IPv4 vs. IPv6 addresses
-                * based on scoping alone.
-                */
-               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
-                * scoping is needed.  According to the IPv6 scoping draft,
-                * 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.
-                * (see the comments for IPv4 above)
-                */
-               if (addr_scope <= scope)
-                       return 1;
-               break;
-
-       default:
-               return 0;
-       };
-
-       return 0;
-}
-
-/********************************************************************
- * 3rd Level Abstractions
- ********************************************************************/
-
-/* What is the scope of '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.
-                * 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?
-                */
-
-               /* Check for unusable SCTP addresses. */
-               if (IS_IPV4_UNUSABLE_ADDRESS(&addr->v4.sin_addr.s_addr)) {
-                       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)) {
-                       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
-                                * fields. See IFA_* in <net/if_inet6.h>.
-                                * 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;
-}
-
-/* This function checks if the address is a valid address to be used for
- * SCTP.
- *
- * Output:
- * 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)
-{
-       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))
-                       return 0;
-               break;
-
-       case AF_INET6:
-               SCTP_V6(
-               {
-                       int ret = sctp_ipv6_addr_type(&addr->v6.sin6_addr);
-
-                       /* Is this a non-unicast address */
-                       if (!(ret & IPV6_ADDR_UNICAST))
-                               return 0;
-                       break;
-               });
-
-       default:
-               return 0;
-       };
-
-       return 1;
-}
diff --git a/net/sctp/sctp_command.c b/net/sctp/sctp_command.c
deleted file mode 100644 (file)
index a3f28ed..0000000
+++ /dev/null
@@ -1,108 +0,0 @@
-/* SCTP kernel reference Implementation Copyright (C) 1999-2001
- * Cisco, Motorola, and IBM
- * Copyright 2001 La Monte H.P. Yarroll
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_command.c,v 1.4 2002/04/24 16:33:39 jgrimm Exp $
- * 
- * These functions manipulate sctp command sequences.
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson <karl@athena.chicago.il.us>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_command.c,v 1.4 2002/04/24 16:33:39 jgrimm Exp $";
-
-#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 *retval = t_new(sctp_cmd_seq_t, priority);
-
-       /* XXX Check for NULL? -DaveM */
-       sctp_init_cmd_seq(retval);
-
-       return retval;
-}
-
-/* Initialize a block of memory as a command sequence. */
-int sctp_init_cmd_seq(sctp_cmd_seq_t *seq)
-{
-       memset(seq, 0, sizeof(sctp_cmd_seq_t));
-       return 1;               /* We always succeed.  */
-}
-
-/* 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)
-{
-       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;
-}
-
-/* Rewind an sctp_cmd_seq_t to iterate from the start.  */
-int sctp_rewind_sequence(sctp_cmd_seq_t *seq)
-{
-       seq->next_cmd = 0;
-       return 1;               /* We always succeed. */
-}
-
-/* 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 *retval = NULL;
-
-       if (seq->next_cmd < seq->next_free_slot)
-               retval = &seq->cmds[seq->next_cmd++];
-
-       return retval;
-}
-
-/* Dispose of a command sequence.  */
-void sctp_free_cmd_seq(sctp_cmd_seq_t *seq)
-{
-       kfree(seq);
-}
diff --git a/net/sctp/sctp_crc32c.c b/net/sctp/sctp_crc32c.c
deleted file mode 100644 (file)
index 300de0c..0000000
+++ /dev/null
@@ -1,190 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 International Business Machines, Corp.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_crc32c.c,v 1.9 2002/07/12 14:50:25 jgrimm Exp $
- * 
- * SCTP Checksum functions
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    Dinakaran Joseph 
- *    Jon Grimm <jgrimm@us.ibm.com>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_crc32c.c,v 1.9 2002/07/12 14:50:25 jgrimm Exp $";
-
-/* The following code has been taken directly from
- * draft-ietf-tsvwg-sctpcsum-03.txt
- *
- * 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,
-};
-     
-__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(__u32)); i++)
-               CRC32C(crc32, buffer[i]);
-
-       /* Skip checksum field of the header. */
-       for (i = 0; i < sizeof(__u32); i++)
-               CRC32C(crc32, 0);
-
-       /* Calculate the rest of the CRC. */
-       for (i = sizeof(struct sctphdr); i < length ; i++)
-               CRC32C(crc32, buffer[i]);
-
-       result = ~crc32;
-
-       /*  result  now holds the negated polynomial remainder;
-        *  since the table and algorithm is "reflected" [williams95].
-        *  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.
-        *  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
-        *  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;
-}
-
-
diff --git a/net/sctp/sctp_debug.c b/net/sctp/sctp_debug.c
deleted file mode 100644 (file)
index b90e031..0000000
+++ /dev/null
@@ -1,218 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001 International Business Machines Corp.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * This file is part of the implementation of the add-IP extension,
- * based on <draft-ietf-tsvwg-addip-sctp-02.txt> June 29, 2001,
- * for the SCTP kernel reference Implementation.
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_debug.c,v 1.10 2002/07/12 14:50:25 jgrimm Exp $
- * 
- * This file converts numerical ID value to alphabetical names for SCTP
- * terms such as chunk type, parameter time, event type, etc.
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson          <karl@athena.chicago.il.us>
- *    Xingang Guo           <xingang.guo@intel.com>
- *    Jon Grimm             <jgrimm@us.ibm.com>
- *    Daisy Chang          <daisyc@us.ibm.com>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_debug.c,v 1.10 2002/07/12 14:50:25 jgrimm Exp $";
-
-#include <net/sctp/sctp.h>
-
-#if SCTP_DEBUG
-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",
-       "INIT",
-       "INIT_ACK",
-       "SACK",
-       "HEARTBEAT",
-       "HEARTBEAT_ACK",
-       "ABORT",
-       "SHUTDOWN",
-       "SHUTDOWN_ACK",
-       "ERROR",
-       "COOKIE_ECHO",
-       "COOKIE_ACK",
-       "ECN_ECNE",
-       "ECN_CWR",
-       "SHUTDOWN_COMPLETE",
-};
-
-/* Lookup "chunk type" debug name. */
-const char *sctp_cname(const sctp_subtype_t cid)
-{
-       if (cid.chunk < 0)
-               return "illegal chunk id";
-       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";
-
-       default:
-               return "unknown chunk";
-       };
-       return "unknown chunk";
-}
-
-/* These are printable form of variable-length parameters. */
-const char *sctp_param_tbl[SCTP_PARAM_ECN_CAPABLE + 1] = {
-       "",
-       "PARAM_HEATBEAT_INFO",
-       "",
-       "",
-       "",
-       "PARAM_IPV4_ADDRESS",
-       "PARAM_IPV6_ADDRESS",
-       "PARAM_STATE_COOKIE",
-       "PARAM_UNRECOGNIZED_PARAMETERS",
-       "PARAM_COOKIE_PRESERVATIVE",
-       "",
-       "PARAM_HOST_NAME_ADDRESS",
-       "PARAM_SUPPORTED_ADDRESS_TYPES",
-};
-
-/* These are printable forms of the states.  */
-const char *sctp_state_tbl[SCTP_STATE_NUM_STATES] = {
-       "STATE_EMPTY",
-       "STATE_CLOSED",
-       "STATE_COOKIE_WAIT",
-       "STATE_COOKIE_ECHOED",
-       "STATE_ESTABLISHED",
-       "STATE_SHUTDOWN_PENDING",
-       "STATE_SHUTDOWN_SENT",
-       "STATE_SHUTDOWN_RECEIVED",
-       "STATE_SHUTDOWN_ACK_SENT",
-};
-
-/* Events that could change the state of an association.  */
-const char *sctp_evttype_tbl[] = {
-       "EVENT_T_unknown",
-       "EVENT_T_CHUNK",
-       "EVENT_T_TIMEOUT",
-       "EVENT_T_OTHER",
-       "EVENT_T_PRIMITIVE"
-};
-
-/* Return value of a state function */
-const char *sctp_status_tbl[] = {
-       "DISPOSITION_DISCARD",
-       "DISPOSITION_CONSUME",
-       "DISPOSITION_NOMEM",
-       "DISPOSITION_DELETE_TCB",
-       "DISPOSITION_ABORT",
-       "DISPOSITION_VIOLATION",
-       "DISPOSITION_NOT_IMPL",
-       "DISPOSITION_ERROR",
-       "DISPOSITION_BUG"
-};
-
-/* Printable forms of primitives */
-static const char *sctp_primitive_tbl[SCTP_NUM_PRIMITIVE_TYPES] = {
-       "PRIMITIVE_INITIALIZE",
-       "PRIMITIVE_ASSOCIATE",
-       "PRIMITIVE_SHUTDOWN",
-       "PRIMITIVE_ABORT",
-       "PRIMITIVE_SEND",
-       "PRIMITIVE_SETPRIMARY",
-       "PRIMITIVE_RECEIVE",
-       "PRIMITIVE_STATUS",
-       "PRIMITIVE_CHANGEHEARTBEAT",
-       "PRIMITIVE_REQUESTHEARTBEAT",
-       "PRIMITIVE_GETSRTTREPORT",
-       "PRIMITIVE_SETFAILURETHRESHOLD",
-       "PRIMITIVE_SETPROTOPARAMETERS",
-       "PRIMITIVE_RECEIVE_UNSENT",
-       "PRIMITIVE_RECEIVE_UNACKED",
-       "PRIMITIVE_DESTROY"
-};
-
-/* Lookup primitive debug name. */
-const char *sctp_pname(const sctp_subtype_t id)
-{
-       if (id.primitive < 0)
-               return "illegal primitive";
-       if (id.primitive <= SCTP_EVENT_PRIMITIVE_MAX)
-               return sctp_primitive_tbl[id.primitive];
-       return "unknown_primitive";
-}
-
-static const char *sctp_other_tbl[] = {
-       "NO_PENDING_TSN",
-       "ICMP_UNREACHFRAG"
-};
-
-/* Lookup "other" debug name. */
-const char *sctp_oname(const sctp_subtype_t id)
-{
-       if (id.other < 0)
-               return "illegal 'other' event";
-       if (id.other < SCTP_EVENT_OTHER_MAX)
-               return sctp_other_tbl[id.other];
-       return "unknown 'other' event";
-}
-
-static const char *sctp_timer_tbl[] = {
-       "TIMEOUT_NONE",
-       "TIMEOUT_T1_COOKIE",
-       "TIMEOUT_T1_INIT",
-       "TIMEOUT_T2_SHUTDOWN",
-       "TIMEOUT_T3_RTX",
-       "TIMEOUT_T4_RTO",
-       "TIMEOUT_HEARTBEAT",
-       "TIMEOUT_SACK",
-       "TIMEOUT_AUTOCLOSE",
-       "TIMEOUT_PMTU_RAISE",
-};
-
-/* Lookup timer debug name. */
-const char *sctp_tname(const sctp_subtype_t id)
-{
-       if (id.timeout < 0)
-               return "illegal 'timer' event";
-       if (id.timeout <= SCTP_EVENT_TIMEOUT_MAX)
-               return sctp_timer_tbl[id.timeout];
-       return "unknown_timer";
-}
diff --git a/net/sctp/sctp_endpointola.c b/net/sctp/sctp_endpointola.c
deleted file mode 100644 (file)
index 1c49ef9..0000000
+++ /dev/null
@@ -1,372 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 International Business Machines, Corp.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001 Nokia, Inc.
- * Copyright (c) 2001 La Monte H.P. Yarroll
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_endpointola.c,v 1.26 2002/08/16 19:30:49 jgrimm Exp $
- * 
- * This abstraction represents an SCTP endpoint.   
- *
- * This file is part of the implementation of the add-IP extension,
- * based on <draft-ietf-tsvwg-addip-sctp-02.txt> June 29, 2001,
- * for the SCTP kernel reference Implementation.
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson <karl@athena.chicago.il.us>
- *    Jon Grimm <jgrimm@austin.ibm.com>
- *    Daisy Chang <daisyc@us.ibm.com>
- *    Dajiang Zhang <dajiang.zhang@nokia.com>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_endpointola.c,v 1.26 2002/08/16 19:30:49 jgrimm Exp $";
-
-#include <linux/types.h>
-#include <linux/sched.h>
-#include <linux/slab.h>
-#include <linux/in.h>
-#include <linux/random.h>      /* get_random_bytes() */
-#include <net/sock.h>
-#include <net/ipv6.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.
- * 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 *ep;
-
-       /* 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;
-
-fail_init:
-       kfree(ep);
-fail:
-       return NULL;
-}
-
-/*
- * 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)
-{
-       memset(ep, 0, sizeof(sctp_endpoint_t));
-
-       /* Initialize the base structure. */
-       /* What type of endpoint are we?  */
-       ep->base.type = SCTP_EP_TYPE_SOCKET;
-
-        /* Initialize the basic object fields. */
-       atomic_set(&ep->base.refcnt, 1);
-       ep->base.dead     = 0;
-       ep->base.malloced = 1;
-
-       /* 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,
-                                   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;
-       sock_hold(ep->base.sk);
-
-       /* This pointer is useful to access the default protocol parameter
-        * values.
-        */
-       ep->proto = proto;
-
-       /* 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]
-               = SCTP_DEFAULT_TIMEOUT_T1_COOKIE;
-       ep->timeouts[SCTP_EVENT_TIMEOUT_T1_INIT]
-               = SCTP_DEFAULT_TIMEOUT_T1_INIT;
-       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]
-               = SCTP_DEFAULT_TIMEOUT_HEARTBEAT;
-       ep->timeouts[SCTP_EVENT_TIMEOUT_SACK]
-               = SCTP_DEFAULT_TIMEOUT_SACK;
-       ep->timeouts[SCTP_EVENT_TIMEOUT_AUTOCLOSE]
-               = sctp_sk(sk)->autoclose * HZ;
-       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?
-        */
-       sk->rcvbuf = SCTP_DEFAULT_MAXWINDOW;
-       sk->sndbuf = SCTP_DEFAULT_MAXWINDOW * 2;
-
-       /* Use SCTP specific send buffer space queues.  */
-       sk->write_space = sctp_write_space;
-       sk->use_write_queue = 1;
-
-       /* 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";
-       return ep;
-}
-
-/* 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);
-}
-
-/* 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;
-       sctp_endpoint_put(ep);
-}
-
-/* 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! */
-       sctp_unhash_endpoint(ep);
-
-       /* 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)
-               sctp_put_port(ep->base.sk);
-
-       /* Give up our hold on the sock. */
-       if (ep->base.sk)
-               sock_put(ep->base.sk);
-
-       /* Finally, free up our memory. */
-       if (ep->base.malloced) {
-               kfree(ep);
-               SCTP_DBG_OBJCNT_DEC(ep);
-       }
-}
-
-/* Hold a reference to an endpoint. */
-void sctp_endpoint_hold(sctp_endpoint_t *ep)
-{
-       atomic_inc(&ep->base.refcnt);
-}
-
-/* Release a reference to an endpoint and clean up if there are
- * no more references.
- */
-void sctp_endpoint_put(sctp_endpoint_t *ep)
-{
-       if (atomic_dec_and_test(&ep->base.refcnt))
-               sctp_endpoint_destroy(ep);
-}
-
-/* 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;
-
-       sctp_read_lock(&ep->base.addr_lock);
-       if (ep->base.bind_addr.port == laddr->v4.sin_port) {
-               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;
-}
-
-/* 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)
-{
-       int rport;
-       sctp_association_t *asoc;
-       list_t *pos;
-
-       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)
-                               return asoc;
-               }
-       }
-
-       *transport = NULL;
-       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)
-{
-       sctp_association_t *asoc;
-
-       sctp_local_bh_disable();
-       asoc = __sctp_endpoint_lookup_assoc(ep, paddr, transport);
-       sctp_local_bh_enable();
-
-       return asoc;
-}
-
-/* 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)
-{
-       sctp_association_t *asoc;
-       struct sock *sk;
-       sctp_transport_t *transport;
-       sctp_chunk_t *chunk;
-       sctp_inqueue_t *inqueue;
-       sctp_subtype_t subtype;
-       sctp_state_t state;
-       int error = 0;
-
-       if (ep->base.dead)
-               goto out;
-
-       asoc = NULL;
-       inqueue = &ep->base.inqueue;
-       sk      = ep->base.sk;
-
-       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.
-                */
-               if (NULL == chunk->asoc) {
-                       asoc = sctp_endpoint_lookup_assoc(ep,
-                                                         sctp_source(chunk),
-                                                         &transport);
-                       chunk->asoc = asoc;
-                       chunk->transport = transport;
-               }
-
-               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 && sctp_chunk_is_data(chunk))
-                       asoc->peer.last_data_from = chunk->transport;
-
-               if (chunk->transport)
-                       chunk->transport->last_time_heard = jiffies;
-
-               /* 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;
-
-               /* Check to see if the endpoint is freed in response to 
-                * the incoming chunk. If so, get out of the while loop.
-                */ 
-               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:
-}
-
-
-
-
diff --git a/net/sctp/sctp_hashdriver.c b/net/sctp/sctp_hashdriver.c
deleted file mode 100644 (file)
index 6f2a4d0..0000000
+++ /dev/null
@@ -1,129 +0,0 @@
-/* SCTP reference Implementation Copyright (C) 1999 Cisco And Motorola
- * 
- * This file origiantes from Randy Stewart's SCTP reference Implementation.
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    Randy Stewart  rstewar1@email.mot.com
- *    Ken Morneau    kmorneau@cisco.com
- *    Qiaobing Xie   qxie1@email.mot.com
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorperated into the next SCTP release.
- * 
- * There are still LOTS of bugs in this code... I always run on the motto
- * "it is a wonder any code ever works :)"
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_hashdriver.c,v 1.2 2002/07/19 22:00:33 jgrimm Exp $";
-
-#include <linux/types.h>
-#include <asm/string.h>
-#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.
- */
-
-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;
-       struct SLA_1_Context context;
-
-       __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;
-
-       /* 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
-        */
-
-       /* 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 */
-}
-
diff --git a/net/sctp/sctp_input.c b/net/sctp/sctp_input.c
deleted file mode 100644 (file)
index 77ec39d..0000000
+++ /dev/null
@@ -1,665 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 International Business Machines, Corp.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001 Nokia, Inc.
- * Copyright (c) 2001 La Monte H.P. Yarroll
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_input.c,v 1.24 2002/07/24 12:26:20 jgrimm Exp $
- *
- * These functions handle all input from the IP layer into SCTP. 
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson <karl@athena.chicago.il.us>
- *    Xingang Guo <xingang.guo@intel.com>
- *    Jon Grimm <jgrimm@us.ibm.com>
- *    Hui Huang <hui.huang@nokia.com>
- *    Daisy Chang <daisyc@us.ibm.com>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_input.c,v 1.24 2002/07/24 12:26:20 jgrimm Exp $";
-
-#include <linux/types.h>
-#include <linux/list.h> /* For struct list_head */
-#include <linux/socket.h>
-#include <linux/ip.h>
-#include <linux/time.h> /* For struct timeval */
-#include <net/sock.h>
-#include <linux/ipsec.h>
-#include <net/sctp/sctp.h>
-#include <net/sctp/sctp_sm.h>
-
-/* Forward declarations for internal helpers. */
-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);
-
-/* 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)
-{
-       sockaddr_storage_t *ret = NULL;
-       void *to, *saddr, *daddr;
-       __u16 *port;
-       size_t len;
-       struct sctphdr *sh;
-
-       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;
-               break;
-
-       case 6:
-               SCTP_V6(
-                       to = &addr->v6.sin6_addr;
-                       port = &addr->v6.sin6_port;
-                       saddr = &skb->nh.ipv6h->saddr;
-                       daddr = &skb->nh.ipv6h->daddr;
-                       len = sizeof(struct in6_addr);
-                       addr->v6.sin6_family = AF_INET6;
-                       addr->v6.sin6_flowinfo = 0; /* FIXME */
-                       addr->v6.sin6_scope_id = 0; /* FIXME */
-                       break;
-                       )
-
-       default:
-               goto out;
-       };
-
-       sh = (struct sctphdr *) skb->h.raw;
-       if (is_saddr) {
-               *port  = ntohs(sh->source);
-               memcpy(to, saddr, len);
-       } else {
-               *port = ntohs(sh->dest);
-               memcpy(to, daddr, len);
-       }
-
-       ret = addr;
-out:
-       return ret;
-}
-
-/* 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)
-{
-       struct sock *sk;
-       sctp_association_t *asoc;
-       sctp_endpoint_t *ep = NULL;
-       sctp_endpoint_common_t *rcvr;
-       sctp_transport_t *transport = NULL;
-       sctp_chunk_t *chunk;
-       struct sctphdr *sh;
-       sockaddr_storage_t src;
-       sockaddr_storage_t dest;
-       int ret = 0;
-
-       if (skb->pkt_type!=PACKET_HOST)
-               goto discard_it;
-
-       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))
-               goto bad_packet;
-       if (sctp_rcv_checksum(skb) < 0)
-               goto bad_packet;
-
-       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
-        * 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))
-               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.
-        */
-       if (!asoc) {
-               ep = __sctp_rcv_lookup_endpoint(&dest);
-               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))
-               goto discard_release;
-
-       /* 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;
-
-       /* Set the source address.  */
-       sctp_init_source(chunk);
-
-       /* Remember where we came from.  */
-       chunk->transport = transport;
-
-       /* Acquire access to the sock lock. Note: We are safe from other
-        * bottom halves on this lock, but a user may be in the lock too,
-        * so check if it is busy.
-        */
-       sctp_bh_lock_sock(sk);
-
-       if (__sctp_sock_busy(sk)) {
-               sk_add_backlog(sk, (struct sk_buff *) chunk);
-       } else {
-               sctp_backlog_rcv(sk, (struct sk_buff *) chunk);
-       }
-
-       /* Release the sock and any reference counts we took in the
-        * lookup calls.
-        */
-       sctp_bh_unlock_sock(sk);
-       if (asoc) {
-               sctp_association_put(asoc);
-       } else {
-               sctp_endpoint_put(ep);
-       }
-       sock_put(sk);
-       return ret;
-
-bad_packet:
-#if 0 /* FIXME */
-       SCTP_INC_STATS(SctpInErrs);
-#endif /* FIXME*/
-
-discard_it:
-       kfree_skb(skb);
-       return ret;
-
-discard_release:
-       /* Release any structures we may be holding. */
-       if (asoc) {
-               sock_put(asoc->base.sk);
-               sctp_association_put(asoc);
-       } else {
-               sock_put(ep->base.sk);
-               sctp_endpoint_put(ep);
-       }
-
-       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.
- */
-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;
-       inqueue = &chunk->rcvr->inqueue;
-
-       sctp_push_inqueue(inqueue, chunk);
-        return 0;
-}
-
-/*
- * This routine is called by the ICMP module when it gets some
- * sort of error condition.  If err < 0 then the socket should
- * be closed and the error returned to the user.  If err > 0
- * it's just the icmp type << 8 | icmp code.  After adjustment
- * header points to the first 8 bytes of the sctp header.  We need
- * to find the appropriate port.
- *
- * The locking strategy used here is very "optimistic". When
- * someone else accesses the socket the ICMP is just dropped
- * and for some paths there is no check at all.
- * A more general error queue to queue errors for later handling
- * is probably better.
- *
- */
-void sctp_v4_err(struct sk_buff *skb, u32 info)
-{
-       /* This should probably involve a call to SCTPhandleICMP().  */
-}
-
-/*
- * RFC 2960, 8.4 - Handle "Out of the blue" Packets.
- *
- * This function scans all the chunks in the OOTB packet to determine if
- * the packet should be discarded right away.  If a response might be needed
- * for this packet, or, if further processing is possible, the packet will
- * be queued to a proper inqueue for the next phase of handling.
- *
- * Output:
- * 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)
-{
-       sctp_chunkhdr_t *ch;
-       __u8 *ch_end;
-
-       ch = (sctp_chunkhdr_t *) skb->data;
-
-       /* Scan through all the chunks in the packet.  */
-       do {
-               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
-                * further action.
-                */
-               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
-                * and take no further action.
-                */
-               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)
-                       goto discard;
-
-               if (ch->type == SCTP_CID_ERROR) {
-                       /* FIXME - Need to check the "Stale cookie" ERROR. */
-                       goto discard;
-               }
-
-               ch = (sctp_chunkhdr_t *) ch_end;
-       } while (ch_end < skb->tail);
-
-       return 0;
-
-discard:
-       return 1;
-}
-
-/* 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;
-
-       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)
-               (*epp)->pprev = &epb->next;
-       *epp = epb;
-       epb->pprev = epp;
-       sctp_write_unlock(&head->lock);
-}
-
-/* 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();
-}
-
-/* Remove endpoint from the hash table.  */
-void __sctp_unhash_endpoint(sctp_endpoint_t *ep)
-{
-       sctp_hashbucket_t *head;
-       sctp_endpoint_common_t *epb;
-
-       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)
-                       epb->next->pprev = epb->pprev;
-               *epb->pprev = epb->next;
-               epb->pprev = NULL;
-       }
-
-       sctp_write_unlock(&head->lock);
-}
-
-/* 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();
-}
-
-/* Look up an endpoint. */
-sctp_endpoint_t *__sctp_rcv_lookup_endpoint(const sockaddr_storage_t *laddr)
-{
-       sctp_hashbucket_t *head;
-       sctp_endpoint_common_t *epb;
-       sctp_endpoint_t *ep;
-       int hash;
-
-       hash = sctp_ep_hashfn(laddr->v4.sin_port);
-       head = &sctp_proto.ep_hashbucket[hash];
-       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;
-       }
-
-       ep = sctp_sk((sctp_get_ctl_sock()))->ep;
-       epb = &ep->base;
-
-hit:
-       sctp_endpoint_hold(ep);
-       sock_hold(epb->sk);
-       read_unlock(&head->lock);
-       return ep;
-}
-
-/* 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();
-}
-
-/* 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)
-               (*epp)->pprev = &epb->next;
-       *epp = epb;
-       epb->pprev = epp;
-       sctp_write_unlock(&head->lock);
-}
-
-/* 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();
-}
-
-/* Remove association from the hash table.  */
-void __sctp_unhash_established(sctp_association_t *asoc)
-{
-       sctp_hashbucket_t *head;
-       sctp_endpoint_common_t *epb;
-
-       epb = &asoc->base;
-
-       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)
-                       epb->next->pprev = epb->pprev;
-               *epb->pprev = epb->next;
-               epb->pprev = NULL;
-       }
-
-       sctp_write_unlock(&head->lock);
-}
-
-/* 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_hashbucket_t *head;
-       sctp_endpoint_common_t *epb;
-       sctp_association_t *asoc;
-       sctp_transport_t *transport;
-       int hash;
-
-       /* Optimize here for direct hit, only listening connections can
-        * 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;
-       }
-
-       read_unlock(&head->lock);
-
-       return NULL;
-
-hit:
-       *transportp = transport;
-       sctp_association_hold(asoc);
-       sock_hold(epb->sk);
-       read_unlock(&head->lock);
-       return asoc;
-}
-
-/*
- * 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
- *    or INIT-ACK chunk the receiver SHOULD use not only the
- *    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
- * to look within the INIT or INIT-ACK. Any implementation that
- * does not do this, may not be able to establish associations
- * in certain circumstances.
- *
- */
-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;
-       sctp_chunkhdr_t *ch;
-       __u8 *ch_end, *data;
-       sctpParam_t parm;
-
-       ch = (sctp_chunkhdr_t *) skb->data;
-
-       ch_end = ((__u8 *) ch) + WORD_ROUND(ntohs(ch->length));
-
-       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
-        * 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) {
-               parm.v  = data;
-
-               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))
-                       continue;
-
-               sctp_param2sockaddr(paddr, parm, ntohs(sh->source));
-
-               asoc = __sctp_rcv_lookup_association(laddr, paddr, transportp);
-               if (asoc)
-                       return asoc;
-       }
-
-       return NULL;
-}
-
-/* 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 *asoc;
-
-       asoc = __sctp_rcv_lookup_association(laddr, paddr, transportp);
-
-       /* Further lookup for INIT-ACK packet.
-        * SCTP Implementors Guide, 2.18 Handling of address
-        * parameters within the INIT or INIT-ACK.
-        */
-       if (!asoc)
-               asoc = __sctp_rcv_initack_lookup(skb, laddr, transportp);
-
-       return asoc;
-}
-
-
-
-
-
diff --git a/net/sctp/sctp_inqueue.c b/net/sctp/sctp_inqueue.c
deleted file mode 100644 (file)
index 457ed52..0000000
+++ /dev/null
@@ -1,202 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2002 International Business Machines, Corp.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_inqueue.c,v 1.10 2002/05/20 22:05:54 jgrimm Exp $
- * 
- * These functions are the methods for accessing the SCTP inqueue.
- *
- * An SCTP inqueue is a queue into which you push SCTP packets
- * (which might be bundles or fragments of chunks) and out of which you
- * pop SCTP whole chunks.
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson <karl@athena.chicago.il.us>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_inqueue.c,v 1.10 2002/05/20 22:05:54 jgrimm Exp $";
-
-#include <net/sctp/sctp.h>
-#include <net/sctp/sctp_sm.h>
-#include <linux/interrupt.h>
-
-/* Initialize an SCTP_inqueue.  */
-void sctp_inqueue_init(sctp_inqueue_t *queue)
-{
-       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;
-
-       queue->malloced = 0;
-}
-
-/* Create an initialized SCTP_inqueue.  */
-sctp_inqueue_t *sctp_inqueue_new(void)
-{
-       sctp_inqueue_t *retval;
-
-       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)
-{
-       sctp_chunk_t *chunk;
-
-       /* 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,
-        * free it as well.
-        */
-       if (queue->in_progress)
-               sctp_free_chunk(queue->in_progress);
-
-       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.
- */
-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);
-       q->immediate.routine(q->immediate.data);
-}
-
-/* Extract a chunk from an SCTP inqueue.
- *
- * 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 *chunk;
-       sctp_chunkhdr_t *ch = NULL;
-
-       /* The assumption is that we are safe to process the chunks
-        * at this time.
-        */
-
-       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) {
-                       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;
-
-                       /* Force chunk->skb->data to chunk->chunk_end.  */
-                       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 (!chunk) {
-               /* Is the queue empty?  */
-               if (skb_queue_empty(&queue->in))
-                       return NULL;
-
-               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;
-       }
-
-        chunk->chunk_hdr = 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.  */
-
-       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;
-       }
-
-       SCTP_DEBUG_PRINTK("+++sctp_pop_inqueue+++ chunk %p[%s],"
-                         " 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;
-}
-
-/* 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
- * inqueue and process it.
- */
-void sctp_inqueue_set_th_handler(sctp_inqueue_t *q,
-                                void (*callback)(void *), void *arg)
-{
-       q->immediate.routine = callback;
-       q->immediate.data = arg;
-}
-
diff --git a/net/sctp/sctp_ipv6.c b/net/sctp/sctp_ipv6.c
deleted file mode 100644 (file)
index f76777e..0000000
+++ /dev/null
@@ -1,267 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 2001 Nokia, Inc.
- * Copyright (c) 2001 La Monte H.P. Yarroll
- * Copyright (c) 2002 International Business Machines, Corp.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_ipv6.c,v 1.12 2002/08/16 19:30:49 jgrimm Exp $
- * 
- * SCTP over IPv6.
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    Le Yanqun             <yanqun.le@nokia.com>
- *    Hui Huang                    <hui.huang@nokia.com>
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Sridhar Samudrala            <sri@us.ibm.com>
- *    Jon Grimm             <jgrimm@us.ibm.com>
- *
- * Based on:
- *      linux/net/ipv6/tcp_ipv6.c
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_ipv6.c,v 1.12 2002/08/16 19:30:49 jgrimm Exp $";
-
-#define __NO_VERSION__
-#include <linux/module.h>
-#include <linux/errno.h>
-#include <linux/types.h>
-#include <linux/socket.h>
-#include <linux/sockios.h>
-#include <linux/net.h>
-#include <linux/sched.h>
-#include <linux/in.h>
-#include <linux/in6.h>
-#include <linux/netdevice.h>
-#include <linux/init.h>
-#include <linux/ipsec.h>
-
-#include <linux/ipv6.h>
-#include <linux/icmpv6.h>
-#include <linux/random.h>
-
-#include <net/protocol.h>
-#include <net/tcp.h>
-#include <net/ndisc.h>
-#include <net/ipv6.h>
-#include <net/transp_v6.h>
-#include <net/addrconf.h>
-#include <net/ip6_route.h>
-#include <net/inet_common.h>
-#include <net/inet_ecn.h>
-#include <net/sctp/sctp.h>
-#include <net/sctp/sctp.h>
-
-#include <asm/uaccess.h>
-
-/* FIXME: Cleanup so we don't need TEST_FRAME here. */
-#ifndef TEST_FRAME
-/* FIXME: Comments. */
-static inline void sctp_v6_err(struct sk_buff *skb,
-                              struct inet6_skb_parm *opt,
-                              int type, int code, int offset, __u32 info)
-{
-       /* BUG.  WRITE ME.  */
-}
-
-/* Based on tcp_v6_xmit() in tcp_ipv6.c. */
-static inline int sctp_v6_xmit(struct sk_buff *skb)
-{
-       struct sock *sk = skb->sk;
-       struct ipv6_pinfo *np = inet6_sk(sk);
-       struct flowi fl;
-       struct dst_entry *dst;
-       struct in6_addr saddr;
-       int err = 0;
-
-       fl.proto = sk->protocol;
-       fl.fl6_dst = &np->daddr;
-       fl.fl6_src = NULL;
-
-       fl.fl6_flowlabel = np->flow_label;
-       IP6_ECN_flow_xmit(sk, fl.fl6_flowlabel);
-       fl.oif = sk->bound_dev_if;
-       fl.uli_u.ports.sport = inet_sk(sk)->sport;
-       fl.uli_u.ports.dport = inet_sk(sk)->dport;
-
-       if (np->opt && np->opt->srcrt) {
-               struct rt0_hdr *rt0 = (struct rt0_hdr *) np->opt->srcrt;
-               fl.nl_u.ip6_u.daddr = rt0->addr;
-       }
-
-       dst = __sk_dst_check(sk, np->dst_cookie);
-
-       if (dst == NULL) {
-               dst = ip6_route_output(sk, &fl);
-
-               if (dst->error) {
-                       sk->err_soft = -dst->error;
-                       dst_release(dst);
-                       return -sk->err_soft;
-               }
-               ip6_dst_store(sk, dst, NULL);
-       }
-
-       skb->dst = dst_clone(dst);
-
-       /* FIXME: This is all temporary until real source address
-        * selection is done.
-        */
-       if (ipv6_addr_any(&np->saddr)) {
-               err = ipv6_get_saddr(dst, fl.fl6_dst, &saddr);
-
-               if (err)
-                       printk(KERN_ERR "sctp_v6_xmit: no saddr available\n");
-
-               /* 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.
-                */
-               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;
-
-       return ip6_xmit(sk, skb, &fl, np->opt);
-}
-#endif /* TEST_FRAME */
-
-/* Returns the mtu for the given v6 destination 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;
-
-       fl.proto = 0;
-       fl.fl6_dst = (struct in6_addr *)&address->v6.sin6_addr;
-       fl.fl6_src = NULL;
-       fl.fl6_flowlabel = 0;
-       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;
-               SCTP_DEBUG_PRINTK("sctp_v6_get_dst_mtu: "
-                                 "ip6_route_output: dev:%s pmtu:%d\n", 
-                                 dst->dev->name, dst_mtu);
-               dst_release(dst);
-       } else {
-               SCTP_DEBUG_PRINTK("sctp_v6_get_dst_mtu: "
-                                 "ip6_route_output failed, returning "
-                                 "%d as dst_mtu\n", 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,
-       .getname    = inet6_getname,
-       .poll       = sctp_poll,
-       .ioctl      = inet6_ioctl,
-       .listen     = sctp_inet_listen,
-       .shutdown   = inet_shutdown,
-       .setsockopt = inet_setsockopt,
-       .getsockopt = inet_getsockopt,
-       .sendmsg    = inet_sendmsg,
-       .recvmsg    = inet_recvmsg,
-       .mmap       = sock_no_mmap,
-};
-
-static struct inet_protosw sctpv6_protosw = {
-       .type          = SOCK_SEQPACKET,
-       .protocol      = IPPROTO_SCTP,
-       .prot          = &sctp_prot,
-       .ops           = &inet6_seqpacket_ops,
-       .capability    = -1,
-       .no_check      = 0,
-       .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 sctp_func_t sctp_ipv6_specific = {
-       .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),
-       .sa_family       = AF_INET6,
-};
-
-/* Initialize IPv6 support and register with inet6 stack.  */
-int sctp_v6_init(void)
-{
-       /* Add SCTPv6 to inetsw6 linked list. */
-       inet6_register_protosw(&sctpv6_protosw);
-
-       /* 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;
-}
-
-/* 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);
-}
diff --git a/net/sctp/sctp_objcnt.c b/net/sctp/sctp_objcnt.c
deleted file mode 100644 (file)
index c4f3f17..0000000
+++ /dev/null
@@ -1,136 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 2001 International Business Machines Corp.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_objcnt.c,v 1.5 2002/07/12 14:50:25 jgrimm Exp $
- * 
- * Support for memory object debugging.  This allows one to monitor the
- * object allocations/deallocations for types instrumented for this 
- * via the proc fs. 
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    Jon Grimm             <jgrimm@us.ibm.com>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_objcnt.c,v 1.5 2002/07/12 14:50:25 jgrimm Exp $";
-
-#include <net/sctp/sctp.h>
-
-/*
- * 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.
- */
-
-SCTP_DBG_OBJCNT(sock);
-SCTP_DBG_OBJCNT(ep);
-SCTP_DBG_OBJCNT(transport);
-SCTP_DBG_OBJCNT(assoc);
-SCTP_DBG_OBJCNT(bind_addr);
-SCTP_DBG_OBJCNT(chunk);
-SCTP_DBG_OBJCNT(addr);
-
-/* An array to make it easy to pretty print the debug information
- * to the proc fs.
- */
-sctp_dbg_objcnt_entry_t sctp_dbg_objcnt[] = {
-       SCTP_DBG_OBJCNT_ENTRY(sock),
-       SCTP_DBG_OBJCNT_ENTRY(ep),
-       SCTP_DBG_OBJCNT_ENTRY(assoc),
-       SCTP_DBG_OBJCNT_ENTRY(transport),
-       SCTP_DBG_OBJCNT_ENTRY(chunk),
-       SCTP_DBG_OBJCNT_ENTRY(bind_addr),
-       SCTP_DBG_OBJCNT_ENTRY(addr),
-};
-
-/* 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)
-{
-       int len = 0;
-       off_t pos = 0;
-       int entries;
-       int i;
-       char temp[128];
-
-       /* How many entries? */
-       entries = sizeof(sctp_dbg_objcnt)/sizeof(sctp_dbg_objcnt[0]);
-
-       /* Walk the entries and print out the debug information
-        * for proc fs.
-        */
-       for (i = 0; i < entries; i++) {
-               pos += 128;
-
-               /* Skip ahead. */
-               if (pos <= offset) {
-                       len = 0;
-                       continue;
-               }
-               /* Print out each entry. */
-               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)
-                       goto done;
-       }
-
-done:
-       *start = buffer + len - (pos - offset);
-       len = pos - offset;
-       if (len > length)
-               len = length;
-
-       return len;
-}
-
-/* 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);
-}
-
-/* Cleanup the objcount entry in the proc filesystem.  */
-void sctp_dbg_objcnt_exit(void)
-{
-       remove_proc_entry("sctp_dbg_objcount", proc_net_sctp);
-}
-
-
diff --git a/net/sctp/sctp_output.c b/net/sctp/sctp_output.c
deleted file mode 100644 (file)
index 3a976d5..0000000
+++ /dev/null
@@ -1,561 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 International Business Machines, Corp.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_output.c,v 1.22 2002/07/12 14:39:05 jgrimm Exp $
- * 
- * These functions handle output processing.
- *  
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson          <karl@athena.chicago.il.us>
- *    Jon Grimm             <jgrimm@austin.ibm.com>
- *    Sridhar Samudrala     <sri@us.ibm.com>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_output.c,v 1.22 2002/07/12 14:39:05 jgrimm Exp $";
-
-#include <linux/types.h>
-#include <linux/kernel.h>
-#include <linux/wait.h>
-#include <linux/time.h>
-#include <linux/ip.h>
-#include <linux/ipv6.h>
-#include <linux/init.h>
-#include <net/inet_ecn.h> 
-#include <net/icmp.h>
-
-#ifndef TEST_FRAME
-#include <net/tcp.h>
-#endif /* TEST_FRAME (not defined) */
-
-#include <linux/socket.h> /* for sa_family_t */
-#include <linux/types.h> /* For NULL, etc... */
-#include <net/sock.h>
-
-#include <net/sctp/sctp.h>
-#include <net/sctp/sctp_sm.h>
-
-/* Forward declarations for private helpers. */
-__u32 count_crc(__u8 *ptr, __u16  count);
-static void sctp_packet_reset(sctp_packet_t *packet);
-static sctp_xmit_t sctp_packet_append_data(sctp_packet_t *packet,
-                                          sctp_chunk_t *chunk);
-
-/* Config a packet.
- * This appears to be a followup set of initializations.)
- */
-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->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)
-               sctp_packet_reset(packet);
-       return packet;
-}
-
-/* Initialize the packet structure. */
-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->destination_port = dport;
-       skb_queue_head_init(&packet->chunks);
-       packet->vtag = 0;
-       packet->ecn_capable = 0;
-       packet->get_prepend_chunk = NULL;
-       packet->has_cookie_echo = 0;
-       packet->malloced = 0;
-       sctp_packet_reset(packet);
-       return packet;
-}
-
-/* Free a packet.  */
-void sctp_packet_free(sctp_packet_t *packet)
-{
-       sctp_chunk_t *chunk;
-
-        while (NULL !=
-              (chunk = (sctp_chunk_t *)skb_dequeue(&packet->chunks))) {
-               sctp_free_chunk(chunk);
-       }
-
-       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
- * is not present in the packet, it transmits the input packet.
- * Data can be bundled with a packet containing a COOKIE_ECHO chunk as long
- * 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 retval;
-       int error = 0;
-
-       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)
-                               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;
-       };
-
-       return retval;
-}
-
-/* Append a chunk to the offered packet reporting back any inability to do
- * so.
- */
-sctp_xmit_t sctp_packet_append_chunk(sctp_packet_t *packet, sctp_chunk_t *chunk)
-{
-       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) ?
-                (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);
-
-               /* Both control chunks and data chunks with TSNs are
-                * 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;
-               }
-       } 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).
-        */
-
-       /* 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)
-                       goto finish;
-       } 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,
-                      (struct sk_buff *)chunk);
-       packet->size += chunk_len;
-
-finish:
-       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)
-{
-       sctp_transport_t *transport = packet->transport;
-       sctp_association_t *asoc = transport->asoc;
-       struct sctphdr *sh;
-       __u32 crc32;
-       struct sk_buff *nskb;
-       sctp_chunk_t *chunk;
-       struct sock *sk;
-       int err = 0;
-       int padding;            /* How much padding do we need?  */
-       __u8 packet_has_data = 0;
-
-       /* 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;
-
-       /* Allocate the new skb.  */
-       nskb = dev_alloc_skb(packet->size);
-       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.
-        */
-       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
-        *
-        * 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 multiple of 4 bytes, the sender MUST
-        * 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");
-       while (NULL != (chunk = (sctp_chunk_t *)
-                       skb_dequeue(&packet->chunks))) {
-               chunk->num_times_sent++;
-               chunk->sent_at = jiffies;
-               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
-                        * 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)) {
-                               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);
-               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, "
-                                 "%s %d\n",
-                                 "*** Chunk", chunk,
-                                 sctp_cname(SCTP_ST_CHUNK(
-                                         chunk->chunk_hdr->type)),
-                                 chunk->has_tsn ? "TSN" : "No 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,
-                                 "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
-                * acknowledged or have failed.
-                */
-               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.
-        *
-        * 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((__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
-        *   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
-        *
-        * 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...
-        */
-
-       /* 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)
-               sk->dst_cache->obsolete = 1;
-
-       if (packet_has_data) {
-               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) &&
-                   (asoc->autoclose)) {
-                       timer = &asoc->timers[SCTP_EVENT_TIMEOUT_AUTOCLOSE];
-                       timeout = asoc->timeouts[SCTP_EVENT_TIMEOUT_AUTOCLOSE];
-
-                       if (!mod_timer(timer, jiffies + timeout))
-                               sctp_association_hold(asoc);
-               }
-       }
-
-       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;
-}
-
-/********************************************************************
- * 2nd Level Abstractions
- ********************************************************************/
-
-/*
- * This private function resets the packet to a fresh state.
- */
-static void sctp_packet_reset(sctp_packet_t *packet)
-{
-       sctp_chunk_t *chunk = NULL;
-
-       packet->size = SCTP_IP_OVERHEAD;
-
-       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
-        * any other chunks get appended.
-        */
-       if (chunk)
-               sctp_packet_append_chunk(packet, 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;
-       size_t datasize, rwnd, inflight;
-       sctp_transport_t *transport = packet->transport;
-       __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).
-                        */
-                       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.
-        * 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, "
-                                 "ssthresh: %d, flight_size: %d, "
-                                 "pba: %d\n",
-                                  __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.
-        */
-       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;
-
-       /* Keep track of how many bytes are in flight to the receiver. */
-       transport->asoc->outqueue.outstanding_bytes += datasize;
-
-       /* Update our view of the receiver's rwnd. */
-       if (datasize < rwnd) {
-               rwnd -= datasize;
-       } else {
-               rwnd = 0;
-       }
-
-       transport->asoc->peer.rwnd = rwnd;
-
-finish:
-       return retval;
-}
-
-
-
-
diff --git a/net/sctp/sctp_outqueue.c b/net/sctp/sctp_outqueue.c
deleted file mode 100644 (file)
index 62fbf28..0000000
+++ /dev/null
@@ -1,1441 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001-2002 International Business Machines Corp.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_outqueue.c,v 1.35 2002/08/05 02:58:05 jgrimm Exp $
- * 
- * These functions implement the outqueue class.   The outqueue handles
- * bundling and queueing of outgoing SCTP chunks.  
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson          <karl@athena.chicago.il.us> 
- *    Perry Melange         <pmelange@null.cc.uic.edu>
- *    Xingang Guo           <xingang.guo@intel.com>
- *    Hui Huang            <hui.huang@nokia.com>
- *    Sridhar Samudrala     <sri@us.ibm.com>
- *    Jon Grimm             <jgrimm@us.ibm.com>
- *     
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_outqueue.c,v 1.35 2002/08/05 02:58:05 jgrimm Exp $";
-
-#include <linux/types.h>
-#include <linux/list.h> /* For struct list_head */
-#include <linux/socket.h>
-#include <linux/ip.h>
-#include <net/sock.h>          /* For skb_set_owner_w */
-
-#include <net/sctp/sctp.h>
-
-/* Declare internal functions here.  */
-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 *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)
-{
-       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;
-
-       /* Throw away unacknowledged chunks. */
-       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,
-                                          transmitted_list);
-                       sctp_free_chunk(chunk);
-               }
-       }
-
-       /* Throw away any leftover chunks. */
-       while ((chunk = (sctp_chunk_t *) skb_dequeue(&q->out)))
-               sctp_free_chunk(chunk);
-}
-
-/* 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);
-}
-
-/* Transmit any pending partial chunks.  */
-void sctp_force_outqueue(sctp_outqueue_t *q)
-{
-       /* Do we really need this? */
-       /* BUG */
-}
-
-/* Put a new chunk in an SCTP_outqueue.  */
-int sctp_push_outqueue(sctp_outqueue_t *q, sctp_chunk_t *chunk)
-{
-       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:
-                       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");
-
-                       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;
-}
-
-/* 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,
-                    __u8 fast_retransmit)
-{
-       struct list_head *lchunk;
-       sctp_chunk_t *chunk;
-       int error = 0;
-       struct list_head tlist;
-
-       if (fast_retransmit) {
-               sctp_transport_lower_cwnd(transport, SCTP_LOWER_CWND_FAST_RTX);
-       } else {
-               sctp_transport_lower_cwnd(transport, SCTP_LOWER_CWND_T3_RTX);
-       }
-
-       INIT_LIST_HEAD(&tlist);
-
-       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
-                * not yet acked should be added to the retransmit queue.
-                */  
-               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.
-                        */
-                       q->asoc->peer.rwnd += 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
-                        * 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
-                        * data chunk is sent.
-                        */
-                       if (chunk->rtt_in_progress) {
-                               chunk->rtt_in_progress = 0;
-                               transport->rto_pending = 0;
-                       }
-                       list_add_tail(lchunk, &q->retransmit);
-               }
-       }
-
-       /* Reconstruct the transmitted queue with chunks that are not
-        * eligible for retransmission.
-        */
-       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,
-                         transport->partial_bytes_acked);
-
-       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.
- * We assume that pkt->transport has already been set.
- *
- * 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)
-{
-       struct list_head *lqueue;
-       struct list_head *lchunk;
-       sctp_transport_t *transport = pkt->transport;
-       sctp_xmit_t status;
-       sctp_chunk_t *chunk;
-       sctp_association_t *asoc;
-       int error = 0;
-
-       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)
-        * 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.
-        *
-        * [Just to be painfully clear, if we are retransmitting
-        * because a timeout just happened, we should send only ONE
-        * packet of retransmitted data.]
-        */
-       lchunk = sctp_list_dequeue(lqueue);
-
-       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) {
-                       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)
-                               *start_timer = 1;
-
-                       /* If we are retransmitting, we should only
-                        * send a single packet.
-                        */
-                       if (rtx_timeout) {
-                               list_add(lchunk, lqueue);
-                               lchunk = NULL;
-                       }
-
-                       /* Bundle lchunk in the next round.  */
-                       break;
-
-               case SCTP_XMIT_RWND_FULL:
-                       /* Send this packet. */
-                       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.
-                        */
-                       list_add_tail(lchunk,
-                                     &transport->transmitted);
-                       *start_timer = 1;
-                       q->empty = 0;
-
-                       /* Retrieve a new chunk to bundle. */
-                       lchunk = sctp_list_dequeue(lqueue);
-                       break;
-               };
-       }
-
-       return error;
-}
-
-/* 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, __u32 tsn)
-{
-       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.
-        */
-       if (!skb_queue_empty(queue)) {
-               SCTP_DEBUG_PRINTK("sctp_xmit_frag: q not empty. "
-                                 "adding 0x%x to outqueue\n",
-                                  ntohl(frag->subh.data_hdr->tsn));
-               if (pos) {
-                       skb_insert(pos, (struct sk_buff *) frag);
-               } else {
-                       skb_queue_tail(queue, (struct sk_buff *) frag);
-               }
-               return;
-       }
-
-       /* Add the chunk fragment to the packet.  */
-       status = (*q->build_output)(packet, frag);
-       switch (status) {
-       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",
-                                 ntohl(frag->subh.data_hdr->tsn));
-               if (pos) {
-                       skb_insert(pos, (struct sk_buff *) frag);
-               } else {
-                       skb_queue_tail(queue, (struct sk_buff *) frag);
-               }
-               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
-                        */
-                       SCTP_DEBUG_PRINTK("sctp_xmit_frag: force output "
-                                         "failed. adding 0x%x to outqueue\n",
-                                         ntohl(frag->subh.data_hdr->tsn));
-                       if (pos) {
-                               skb_insert(pos, (struct sk_buff *) frag);
-                       } else {
-                               skb_queue_tail(queue, (struct sk_buff *) frag);
-                       }
-               } else {
-                       SCTP_DEBUG_PRINTK("sctp_xmit_frag: force output "
-                                         "success. 0x%x sent\n",
-                                         ntohl(frag->subh.data_hdr->tsn));
-                       list_add_tail(&frag->transmitted_list,
-                                     &transport->transmitted);
-
-                       sctp_transport_reset_timers(transport);
-               }
-               break;
-
-       default:
-               BUG();
-       };
-}
-
-/* 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)
-{
-       sctp_association_t *asoc = frag->asoc;
-       struct list_head *lfrag, *frag_list;
-       __u32 tsn;
-       int nfrags = 1;
-       struct sk_buff *pos;
-
-       /* Count the number of fragments. */
-       frag_list = &frag->frag_list;
-       list_for_each(lfrag, frag_list) {
-               nfrags++;
-       }
-
-       /* Get a TSN block of nfrags TSNs. */
-       tsn = __sctp_association_get_tsn_block(asoc, nfrags);
-
-       pos = skb_peek(&q->out);
-       /* Transmit the first fragment. */
-       sctp_xmit_frag(q, pos, packet, frag, tsn++);
-
-       /* Transmit the rest of fragments. */
-       frag_list = &frag->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++);
-       }
-}
-
-/* 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;
-       __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;
-
-       /* 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));
-
-       /* 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,
-                                       data_ptr, SCTP_DATA_FIRST_FRAG, ssn); 
-
-       if (!first_frag)
-               goto err;
-
-       /* All the fragments are added to the frag_list of the first chunk. */
-       frag_list = &first_frag->frag_list;
-
-       chunk_data_len -= max_frag_data_len;
-       data_ptr += max_frag_data_len;
-
-       /* 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;
-
-               /* 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;
-
-       /* Add the last fragment to the first fragment's frag_list. */
-       list_add_tail(&frag->frag_list, frag_list);
-
-       /* Free the original chunk. */
-       sctp_free_chunk(chunk);
-
-       return first_frag;
-
-err:
-       /* 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. */
-               flist = &first_frag->frag_list;
-               while (NULL != (lfrag = sctp_list_dequeue(flist))) {
-                       frag = list_entry(lfrag, sctp_chunk_t, frag_list);
-                       sctp_free_chunk(frag);
-               }
-
-               /* Free the first fragment. */
-               sctp_free_chunk(first_frag);
-       }
-
-       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.
- */
-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;
-       __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;
-       struct sk_buff_head *queue;
-       sctp_transport_t *transport = NULL;
-       sctp_transport_t *new_transport;
-       sctp_chunk_t *chunk;
-       sctp_xmit_t status;
-       int error = 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;
-
-       /*
-        * 6.10 Bundling
-        *   ...
-        *   When bundling control chunks with DATA chunks, an
-        *   endpoint MUST place control chunks first in the outbound
-        *   SCTP packet.  The transmitter MUST transmit DATA chunks
-        *   within a SCTP packet in increasing order of TSN.
-        *   ...
-        */
-       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) {
-                       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)
-                               new_transport = asoc->peer.active_path;     
-               }
-
-               /* Are we switching transports?
-                * Take care of transport locks.
-                */
-               if (new_transport != transport) {
-                       transport = new_transport;
-                       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);
-               }
-
-               switch (chunk->chunk_hdr->type) {
-               /*
-                * 6.10 Bundling
-                *   ...
-                *   An endpoint MUST NOT bundle INIT, INIT ACK or SHUTDOWN
-                *   COMPLETE with any other chunks.  [Send them immediately.]
-                */
-               case SCTP_CID_INIT:
-               case SCTP_CID_INIT_ACK:
-               case SCTP_CID_SHUTDOWN_COMPLETE:
-                       (*q->init_output)(&singleton, transport, sport, dport);
-                       (*q->config_output)(&singleton, vtag, ecn_capable,
-                                           s_ecne_handler);
-                       (void) (*q->build_output)(&singleton, chunk);
-                       error = (*q->force_output)(&singleton);
-                       if (error < 0)
-                               return(error);
-                       break;
-
-               case SCTP_CID_ABORT:
-               case SCTP_CID_SACK:
-               case SCTP_CID_HEARTBEAT:
-               case SCTP_CID_HEARTBEAT_ACK:
-               case SCTP_CID_SHUTDOWN:
-               case SCTP_CID_SHUTDOWN_ACK:
-               case SCTP_CID_ERROR:
-               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);
-                       break;
-
-               case SCTP_CID_ASCONF:
-               case SCTP_CID_ASCONF_ACK:
-                       (void) (*q->build_output)(packet, chunk);
-                       break;
-
-               default:
-                       /* We built a chunk with an illegal type! */
-                       BUG();                  
-               };
-       }
-
-       /* 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)
-                       break;
-
-               /* fallthru */
-       case SCTP_STATE_ESTABLISHED:
-       case SCTP_STATE_SHUTDOWN_PENDING:
-       case SCTP_STATE_SHUTDOWN_RECEIVED:
-               /*
-                * RFC 2960 6.1  Transmission of DATA Chunks
-                *
-                * C) When the time comes for the sender to transmit,
-                * before sending new DATA chunks, the sender MUST
-                * first transmit any outstanding DATA chunks which
-                * are marked for retransmission (limited by the
-                * current cwnd).
-                */
-               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)) {
-                               list_add_tail(&transport->send_ready,
-                                             &transport_list);
-                       }
-
-                       packet = &transport->packet;
-                       (*q->config_output)(packet, vtag,
-                                           ecn_capable, ecne_handler);
-               retran:
-                       error = sctp_flush_retran_queue(q,
-                                                       packet,
-                                                       rtx_timeout,
-                                                       &start_timer);
-
-                       if (start_timer)
-                               sctp_transport_reset_timers(transport);
-               }
-
-               /* Finally, transmit new packets.  */
-               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.
-                        */
-                       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,
-                                               GFP_ATOMIC);
-                               if (event) {
-                                       sctp_ulpqueue_tail_event(&asoc->ulpq,
-                                                                event);
-                               }
-
-                               /* 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)
-                               new_transport = asoc->peer.active_path;
-
-                       /* Change packets if necessary.  */
-                       if (new_transport != transport) {
-                               transport = new_transport;
-
-                               /* Schedule to have this transport's
-                                * packet flushed.
-                                */
-                               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);
-                       }
-
-                       SCTP_DEBUG_PRINTK("sctp_transmit_packet(%p, %p[%s]), ",
-                                         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 ?
-                                         atomic_read(&chunk->skb->users) :
-                                         -1);
-
-                       /* Add the chunk to the packet.  */
-                       status = (*q->build_output)(packet, chunk);
-
-                       switch (status) {
-                       case SCTP_XMIT_PMTU_FULL:
-                       case SCTP_XMIT_RWND_FULL:
-                               /* We could not append this chunk, so put
-                                * the chunk back on the output queue.
-                                */
-                               SCTP_DEBUG_PRINTK("sctp_flush_outqueue: could"
-                                  "not transmit TSN: 0x%x, status: %d\n",
-                                  ntohl(chunk->subh.data_hdr->tsn), status);
-                               skb_queue_head(queue, (struct sk_buff *)chunk);
-                               goto sctp_flush_out;
-                               break;
-
-                       case SCTP_XMIT_MUST_FRAG: {
-                               sctp_chunk_t *frag;
-
-                               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.
-                                        */
-                                       sctp_free_chunk(chunk);
-                                       error = -ENOMEM;
-                                       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
-                        * conditions.
-                        *
-                        * Is this really a problem?  Won't this behave
-                        * like a lost TSN?
-                        */
-                       list_add_tail(&chunk->transmitted_list,
-                                     &transport->transmitted);
-
-                       sctp_transport_reset_timers(transport);
-
-                       q->empty = 0;
-               }
-               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 ) {
-               sctp_transport_t *t = list_entry(ltransport,
-                                                sctp_transport_t, send_ready);
-               if (t != transport)
-                       transport = t;
-
-               packet = &transport->packet;
-               if (packet->size != SCTP_IP_OVERHEAD)
-                       error = (*q->force_output)(packet);
-       }
-
-       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)
-{
-       q->init_output = init;
-       q->config_output = config;
-       q->append_output = append;
-       q->build_output = build;
-       q->force_output = force;
-       return 0;
-}
-
-/* Update unack_data based on the incoming SACK chunk */
-static void sctp_sack_update_unack_data(sctp_association_t *assoc,
-                                       sctp_sackhdr_t *sack)
-{
-       sctp_sack_variable_t *frags;
-       __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) -
-                               ntohs(frags[i].gab.start) + 1));
-       }
-
-       assoc->unack_data = unack_data;
-}
-
-/* This is where we REALLY process a SACK.
- *
- * 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)
-{
-       sctp_chunk_t *tchunk;
-       list_t *lchunk, *transport_list, *pos;
-       __u32 tsn;
-       __u32 sack_ctsn;
-       __u32 ctsn;
-       sctp_transport_t *transport;
-       int outstanding;
-       __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.
-        */
-       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;
-
-       /* Update unack_data field in the assoc. */
-       sctp_sack_update_unack_data(q->asoc, sack);
-
-       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);
-       outstanding = q->outstanding_bytes;
-
-       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;
-
-       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;
-       }
-
-       SCTP_DEBUG_PRINTK("sack queue is empty.\n");      
-finish:
-       return q->empty;
-}
-
-/* Is the outqueue empty?  */
-int sctp_outqueue_is_empty(const sctp_outqueue_t *q)
-{
-       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
- * retransmit parameters are also updated and if needed a fast retransmit
- * process is started.
- *
- * I added coherent debug information output.  --xguo
- *
- * Instead of printing 'sacked' or 'kept' for each TSN on the
- * 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)
-{
-       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
-       __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).
-        * -1: We need to initialize.
-        */
-       int dbg_prt_state = -1;
-#endif /* SCTP_DEBUG */
-
-       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);
-               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 ((!tchunk->tsn_gap_acked) &&
-                                   (1 == tchunk->num_times_sent) &&
-                                   (tchunk->rtt_in_progress)) {
-                                       rtt = jiffies - tchunk->sent_at;
-                                       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;
-
-                               if (!tchunk->tsn_gap_acked) {
-                                       tchunk->tsn_gap_acked = 1;
-                                       bytes_acked += sctp_data_size(tchunk);
-                               }
-
-                               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'.
-                                */
-                               if (!tchunk->tsn_gap_acked) {
-                                       tchunk->tsn_gap_acked = 1;
-                                       bytes_acked += sctp_data_size(tchunk);
-                                       if (TSN_lt(highest_new_tsn_in_sack,
-                                                  tsn)) {
-                                               highest_new_tsn_in_sack = tsn;
-                                       }
-                               }
-                               list_add_tail(lchunk, &tlist);
-                       }
-
-#if SCTP_DEBUG
-                       switch (dbg_prt_state) {
-                       case 0: /* last TSN was ACKed */
-                               if (dbg_last_ack_tsn + 1 == tsn) {
-                                       /* This TSN belongs to the
-                                        * current ACK range.
-                                        */
-                                       break;
-                               }
-
-                               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);
-                               dbg_ack_tsn = tsn;
-                               break;
-
-                       case 1: /* The last TSN was NOT ACKed. */
-                               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);
-                               dbg_prt_state = 0;
-                               dbg_ack_tsn = tsn;
-                       };
-
-                       dbg_last_ack_tsn = tsn;
-#endif /* SCTP_DEBUG */
-
-               } else {
-                       if (tchunk->tsn_gap_acked) {
-                               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.
-                                */
-                               restart_timer = 1;
-                       }
-
-                       list_add_tail(lchunk, &tlist);
-
-#if SCTP_DEBUG
-                       /* See the above comments on ACK-ed TSNs. */
-                       switch (dbg_prt_state) {
-                       case 1:
-                               if (dbg_last_kept_tsn + 1 == tsn)
-                                       break;
-
-                               if (dbg_last_kept_tsn != dbg_kept_tsn)
-                                       SCTP_DEBUG_PRINTK("-%08x",
-                                                         dbg_last_kept_tsn);
-
-                               SCTP_DEBUG_PRINTK(",%08x", tsn);
-                               dbg_kept_tsn = tsn;
-                               break;
-
-                       case 0:
-                               if (dbg_last_ack_tsn != dbg_ack_tsn)
-                                       SCTP_DEBUG_PRINTK("-%08x",
-                                                         dbg_last_ack_tsn);
-                               SCTP_DEBUG_PRINTK("\n");
-
-                               /* FALL THROUGH... */
-                       default:
-                               SCTP_DEBUG_PRINTK("KEPT: %08x",tsn);
-                               dbg_prt_state = 1;
-                               dbg_kept_tsn = tsn;
-                       };
-
-                       dbg_last_kept_tsn = tsn;
-#endif /* SCTP_DEBUG */
-               }
-       }
-
-#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");
-               }
-       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");
-               }
-       };
-#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
-                        * DATA chunk was last sent.
-                        * 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
-                        * 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_transport_raise_cwnd(transport, sack_ctsn,
-                                                 bytes_acked);
-
-                       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
-                        * closed for an indefinite time.
-                        * 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)) {
-                               SCTP_DEBUG_PRINTK("%s: SACK received for zero "
-                                                 "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) {
-                       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))
-                               sctp_transport_hold(transport);
-               }
-       }
-
-       /* 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);
-
-               /* 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
-                * 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.
-                */
-               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,
-                                         tchunk->tsn_missing_report);
-               }
-               if (tchunk->tsn_missing_report >= 4) {
-                       tchunk->fast_retransmit = 1;
-                       do_fast_retransmit = 1;
-               }
-
-               list_add_tail(lchunk, transmitted_queue);
-       }
-
-       if (transport) {
-               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,
-                                 transport->partial_bytes_acked);
-       }
-}
-
-/* Is the given TSN acked by this packet?  */
-static int sctp_acked(sctp_sackhdr_t *sack, __u32 tsn)
-{
-       int i;
-       sctp_sack_variable_t *frags;
-       __u16 gap;
-       __u32 ctsn = ntohl(sack->cum_tsn_ack);
-
-        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;
-}
diff --git a/net/sctp/sctp_primitive.c b/net/sctp/sctp_primitive.c
deleted file mode 100644 (file)
index 71eeeaa..0000000
+++ /dev/null
@@ -1,205 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_primitive.c,v 1.6 2002/08/21 18:34:04 jgrimm Exp $
- * 
- * These functions implement the SCTP primitive functions from Section 10.
- * 
- * Note that the descriptions from the specification are USER level
- * functions--this file is the functions which populate the struct proto
- * for SCTP which is the BOTTOM of the sockets interface.
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Narasimha Budihal     <narasimha@refcode.org>
- *    Karl Knutson          <karl@athena.chicago.il.us>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_primitive.c,v 1.6 2002/08/21 18:34:04 jgrimm Exp $";
-
-#include <linux/types.h>
-#include <linux/list.h> /* For struct list_head */
-#include <linux/socket.h>
-#include <linux/ip.h>
-#include <linux/time.h> /* For struct timeval */
-#include <net/sock.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; \
-       sctp_state_t state; \
-       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; \
-       \
-       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)
- * -> association id [,destination transport addr list] [,outbound stream
- *    count]
- *
- * This primitive allows the upper layer to initiate an association to a
- * 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.
- */
-
-/* This is called in the code as sctp_primitive_ASSOCIATE.  */
-
-DECLARE_PRIMITIVE(ASSOCIATE)
-
-/* 10.1 ULP-to-SCTP
- * C) Shutdown
- *
- * Format: SHUTDOWN(association id)
- * -> 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.
- */
-
-DECLARE_PRIMITIVE(SHUTDOWN);
-
-/* 10.1 ULP-to-SCTP
- * C) Abort
- *
- * Format: Abort(association id [, cause code])
- * -> 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
- * attempting to abort the association results in a failure, an error
- * code shall be returned.
- */
-
-DECLARE_PRIMITIVE(ABORT);
-
-/* 10.1 ULP-to-SCTP
- * E) Send
- *
- * Format: SEND(association id, buffer address, byte count [,context]
- *         [,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.
- *
- * 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
- *    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.
- *
- *  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
- *    message).
- *
- *  o no-bundle flag - instructs SCTP not to bundle this user data with
- *    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
- *    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 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);
-
-       return error;
-}
diff --git a/net/sctp/sctp_protocol.c b/net/sctp/sctp_protocol.c
deleted file mode 100644 (file)
index bf39984..0000000
+++ /dev/null
@@ -1,594 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 International Business Machines, Corp.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001 Nokia, Inc.
- * Copyright (c) 2001 La Monte H.P. Yarroll
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_protocol.c,v 1.35 2002/08/16 19:30:49 jgrimm Exp $
- * 
- * Initialization/cleanup for SCTP protocol support.  
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson <karl@athena.chicago.il.us>
- *    Jon Grimm <jgrimm@us.ibm.com>
- *    Sridhar Samudrala <sri@us.ibm.com>
- *    Daisy Chang <daisyc@us.ibm.com>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_protocol.c,v 1.35 2002/08/16 19:30:49 jgrimm Exp $";
-
-#include <linux/module.h>
-#include <linux/init.h>
-#include <linux/netdevice.h>
-#include <linux/inetdevice.h>
-#include <net/protocol.h>
-#include <net/ip.h>
-#include <net/ipv6.h>
-#include <net/sctp/sctp.h>
-#include <net/addrconf.h>
-#include <net/inet_common.h>
-
-/* Global data structures. */
-sctp_protocol_t sctp_proto;
-struct proc_dir_entry  *proc_net_sctp;
-
-/* This is the global socket data structure used for responding to
- * the Out-of-the-blue (OOTB) packets.  A control sock will be created
- * for this socket at the initialization time.
- */
-static struct socket *sctp_ctl_socket;
-
-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;
-}
-
-/* Set up the proc fs entry for the SCTP protocol. */
-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;
-               }
-       }
-}
-
-/* Clean up the proc fs entry for the SCTP protocol. */
-void sctp_proc_exit(void)
-{
-       if (proc_net_sctp) {
-               proc_net_sctp= NULL;
-               remove_proc_entry("net/sctp", 0);
-       }
-}
-
-/* Private helper to extract ipv4 address and stash them in
- * the protocol structure.
- */
-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 sockaddr_storage_list *addr;
-
-       read_lock(&inetdev_lock);
-       if ((in_dev = __in_dev_get(dev)) == NULL) {
-               read_unlock(&inetdev_lock);
-               return;
-       }
-
-       read_lock(&in_dev->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);
-                       addr->a.v4.sin_family = AF_INET;
-                       addr->a.v4.sin_port = 0;
-                       addr->a.v4.sin_addr.s_addr = ifa->ifa_local;
-                       list_add_tail(&addr->list, &proto->local_addr_list);
-               }
-       }
-
-       read_unlock(&in_dev->lock);
-       read_unlock(&inetdev_lock);
-}
-
-/* Private helper to extract ipv6 address and stash them in
- * the protocol structure.
- * FIXME: Make this an address family function.
- */
-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
-       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_unlock(&addrconf_lock);
-               return;
-       }
-
-       read_lock_bh(&in6_dev->lock);
-       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_port = 0;
-                       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 */
-#endif /* SCTP_V6_SUPPORT */
-}
-
-/* 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)
-{
-       struct net_device *dev;
-
-       read_lock(&dev_base_lock);
-       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);
-}
-
-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);
-}
-
-/* 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;
-
-       list_for_each_safe(pos, temp, &proto->local_addr_list) {
-               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);
-}
-
-/* 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)
-{
-       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
-                        * 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,
-                                                          priority);
-                               if (error)
-                                       goto end_copy;
-                       }
-               }
-       }
-
-end_copy:
-       sctp_spin_unlock_irqrestore(&proto->local_addr_lock, flags);
-
-       return error;
-}
-
-/* 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 = {
-               .dst   = address->v4.sin_addr.s_addr,
-               .src   = 0,
-               .iif   = 0,
-               .oif   = 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",
-                                 dst_mtu);
-       } else {
-               dst_mtu = rt->u.dst.pmtu;
-               SCTP_DEBUG_PRINTK("sctp_v4_get_dst_mtu: "
-                                 "ip_route_output_key: dev:%s pmtu:%d\n",
-                                 rt->u.dst.dev->name, dst_mtu);
-               ip_rt_put(rt);
-       }
-
-       return dst_mtu;
-}
-
-/* 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)
-{
-       long flags __attribute__ ((unused));
-
-       sctp_spin_lock_irqsave(&sctp_proto.local_addr_lock, flags);
-       __sctp_free_local_addr_list(&sctp_proto);
-       __sctp_get_local_addr_list(&sctp_proto);
-       sctp_spin_unlock_irqrestore(&sctp_proto.local_addr_lock, flags);
-
-       return NOTIFY_DONE;
-}
-
-/*
- * 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 err = 0;
-       int family = PF_INET;
-
-       SCTP_V6(family = PF_INET6;)
-
-       err = sock_create(family, SOCK_SEQPACKET, IPPROTO_SCTP,
-                         &sctp_ctl_socket);
-       if (err < 0) {
-               printk(KERN_ERR 
-                       "SCTP: Failed to create the SCTP control socket.\n");
-               return err;
-       }
-       sctp_ctl_socket->sk->allocation = GFP_ATOMIC;
-       inet_sk(sctp_ctl_socket->sk)->ttl = MAXTTL;
-
-       return 0;
-}
-
-/* Get the table of functions for manipulating a particular address
- * family.
- */
-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;
-
-       retval = NULL;
-
-       /* 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;
-               }
-       }
-
-       return retval;
-}
-
-/* Registration for netdev events.  */
-struct notifier_block sctp_netdev_notifier = {
-       .notifier_call = sctp_netdev_event,
-};
-
-/* 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,
-       .socketpair  = sock_no_socketpair,
-       .accept      = inet_accept,
-       .getname     = inet_getname,      /* Semantics are different.  */
-       .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,
-       .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,
-       .protocol   = IPPROTO_SCTP,
-       .prot       = &sctp_prot,
-       .ops        = &inet_seqpacket_ops,
-       .capability = -1,
-       .no_check   = 0,
-       .flags      = SCTP_PROTOSW_FLAG
-};
-
-/* 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.  */
-sctp_func_t sctp_ipv4_specific = {
-       .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),
-       .sa_family      = AF_INET,
-};
-
-/* Initialize the universe into something sensible.  */
-int sctp_init(void)
-{
-       int i;
-       int status = 0;
-
-       /* Add SCTP to inetsw linked list.  */
-       inet_register_protosw(&sctp_protosw);
-
-       /* Add SCTP to inet_protos hash table.  */
-       inet_add_protocol(&sctp_protocol);
-
-       /* Initialize proc fs directory.  */
-       sctp_proc_init();
-
-       /* 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;
-
-       /* 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;
-
-       /* 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.
-        */
-       sctp_proto.max_instreams    = SCTP_MAX_STREAM;
-       sctp_proto.max_outstreams   = SCTP_MAX_STREAM;
-
-       /* 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");
-               status = -ENOMEM;
-               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.  */
-       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");
-               status = -ENOMEM;
-               goto err_ehash_alloc;
-       }
-
-       for (i = 0; i < sctp_proto.ep_hashsize; i++) {
-               sctp_proto.ep_hashbucket[i].lock = RW_LOCK_UNLOCKED;
-               sctp_proto.ep_hashbucket[i].chain = NULL;
-       }
-
-       /* Allocate and initialize the SCTP port hash table.  */
-       sctp_proto.port_hashsize = 4096;
-       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.");
-               status = -ENOMEM;
-               goto err_bhash_alloc;
-       }
-
-       sctp_proto.port_alloc_lock = SPIN_LOCK_UNLOCKED;
-       sctp_proto.port_rover = sysctl_local_port_range[0] - 1;
-       for (i = 0; i < sctp_proto.port_hashsize; i++) {
-               sctp_proto.port_hashtable[i].lock = SPIN_LOCK_UNLOCKED;
-               sctp_proto.port_hashtable[i].chain = NULL;
-       }
-
-       sctp_sysctl_register();
-
-       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)
-               goto err_v6_init;
-
-       /* 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;
-       }
-
-       /* Initialize the local address list. */
-       INIT_LIST_HEAD(&sctp_proto.local_addr_list);
-       sctp_proto.local_addr_lock = SPIN_LOCK_UNLOCKED;
-
-       register_inetaddr_notifier(&sctp_netdev_notifier);
-       sctp_get_local_addr_list(&sctp_proto);
-
-       return 0;
-
-err_ctl_sock_init:
-       sctp_v6_exit();
-err_v6_init:
-       sctp_sysctl_unregister();
-       list_del(&sctp_ipv4_specific.list);
-       kfree(sctp_proto.port_hashtable);
-err_bhash_alloc:
-       kfree(sctp_proto.ep_hashbucket);
-err_ehash_alloc:
-       kfree(sctp_proto.assoc_hashbucket);
-err_ahash_alloc:
-       sctp_dbg_objcnt_exit();
-       sctp_proc_exit();
-       inet_del_protocol(&sctp_protocol);
-       inet_unregister_protosw(&sctp_protosw);
-       return status;
-}
-
-/* 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.  */
-       unregister_inetaddr_notifier(&sctp_netdev_notifier);
-       sctp_free_local_addr_list(&sctp_proto);
-
-       /* Free the control endpoint.  */
-       sock_release(sctp_ctl_socket);
-
-       sctp_v6_exit();
-       sctp_sysctl_unregister();
-       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);
-}
-
-module_init(sctp_init);
-module_exit(sctp_exit);
-
-MODULE_AUTHOR("Linux Kernel SCTP developers <lksctp-developers@lists.sourceforge.net>");
-MODULE_DESCRIPTION("Support for the SCTP protocol (RFC2960)");
-MODULE_LICENSE("GPL");
-
diff --git a/net/sctp/sctp_sla1.c b/net/sctp/sctp_sla1.c
deleted file mode 100644 (file)
index 9151f51..0000000
+++ /dev/null
@@ -1,284 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_sla1.c,v 1.4 2002/07/19 22:00:33 jgrimm Exp $
- * 
- * (It's really SHA-1 but Hey I was tired when I created this 
- * file, and on a plane to France :-)
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    Randall Stewart <rstewar1@email.mot.com>
- *    kmorneau@cisco.com
- *    qxie1@email.mot.com
- *
- * Based on:
- *    Randy Stewart, et al. SCTP Reference Implementation which is licenced
- *    under the GPL. 
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_sla1.c,v 1.4 2002/07/19 22:00:33 jgrimm Exp $";
-
-#include <linux/types.h>
-#include <linux/kernel.h>
-#include <linux/fcntl.h>
-#include <asm/string.h>                /* for memcpy */
-#include <linux/sched.h>       /* dead chicken for in.h */
-#include <linux/in.h>          /* for htonl and ntohl */
-
-#include <net/sctp/sctp_sla1.h>
-
-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;
-}
-
-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) */
-
-       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;
-
-       /* 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_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;
-               }
-       }
-}
-
-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);
-}
diff --git a/net/sctp/sctp_sm_make_chunk.c b/net/sctp/sctp_sm_make_chunk.c
deleted file mode 100644 (file)
index 51850cb..0000000
+++ /dev/null
@@ -1,1765 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001 International Business Machines Corp.
- *
- * This file is part of the SCTP kernel reference Implementation
- *
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_sm_make_chunk.c,v 1.38 2002/07/26 22:52:32 jgrimm Exp $
- *
- * This file includes part of the implementation of the add-IP extension,
- * based on <draft-ietf-tsvwg-addip-sctp-02.txt> June 29, 2001,
- * for the SCTP kernel reference Implementation.
- * 
- * These functions work with the state functions in sctp_sm_statefuns.c
- * to implement the state operations.  These functions implement the
- * steps which require modifying existing data structures.
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson          <karl@athena.chicago.il.us>
- *    C. Robin              <chris@hundredacre.ac.uk>
- *    Jon Grimm             <jgrimm@us.ibm.com>
- *    Xingang Guo           <xingang.guo@intel.com>
- *    Dajiang Zhang        <dajiang.zhang@nokia.com>
- *    Sridhar Samudrala            <sri@us.ibm.com>
- *    Daisy Chang          <daisyc@us.ibm.com>
- *
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-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/types.h>
-#include <linux/kernel.h>
-#include <linux/ip.h>
-#include <linux/ipv6.h>
-#include <linux/net.h>
-#include <linux/inet.h>
-#include <net/sock.h>
-
-#include <linux/skbuff.h>
-#include <linux/random.h>      /* for get_random_bytes */
-#include <net/sctp/sctp.h>
-#include <net/sctp/sctp_sm.h>
-
-/* RFC 2960 3.3.2 Initiation (INIT) (1)
- *
- * Note 4: This parameter, when present, specifies all the
- * address types the sending endpoint can support. The absence
- * of this parameter indicates that the sending endpoint can
- * support any address type.
- */
-static const sctp_supported_addrs_param_t sat_param = {
-       {           
-               SCTP_PARAM_SUPPORTED_ADDRESS_TYPES,
-               __constant_htons(SCTP_SAT_LEN),
-       },
-       {               /* types[] */
-               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)),
-       }
-};
-
-/* 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, __u16 cause_code,
-                     const void *payload, size_t paylen)
-{
-       sctp_errhdr_t err;
-       int padlen;
-       __u16 len;
-
-        /* Cause code constants are now defined in network order.  */
-       err.cause = cause_code;
-       len = sizeof(sctp_errhdr_t) + paylen;
-       padlen = len % 4;
-       len += padlen;
-       err.length  = htons(len);
-       sctp_addto_chunk(chunk, sizeof(sctp_errhdr_t), &err);
-       chunk->subh.err_hdr = sctp_addto_chunk(chunk, paylen, payload);
-}
-
-/* 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
- *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
- *    |   Type = 1    |  Chunk Flags  |      Chunk Length             |
- *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
- *    |                         Initiate Tag                          |
- *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
- *    |           Advertised Receiver Window Credit (a_rwnd)          |
- *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
- *    |  Number of Outbound Streams   |  Number of Inbound Streams    |
- *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
- *    |                          Initial TSN                          |
- *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
- *    \                                                               \
- *    /              Optional/Variable-Length Parameters              /
- *    \                                                               \
- *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
- *
- *
- * 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
- * Advertised Receiver Window Credit   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
- * IPv6 Address (Note 1)               Optional    6
- * Cookie Preservative                 Optional    9
- * Reserved for ECN Capable (Note 2)   Optional    32768 (0x8000)
- * 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_inithdr_t init;
-       sctpParam_t addrs;
-       size_t chunksize;
-       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.
-        */
-       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 (!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.
-        *
-        * PLEASE DO NOT FIXME [This version does not support Host Name.]
-        */
-
-       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);
-       sctp_addto_chunk(retval, sizeof(ecap_param), &ecap_param);
-
-nodata:
-       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_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 (!bp)
-               goto nomem_bindaddr;
-
-       /* 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)
-               flags |= SCTP_ADDR4_PEERSUPP;
-       if (asoc->peer.ipv6_address)
-               flags |= SCTP_ADDR6_PEERSUPP;
-       error = sctp_bind_addr_copy(bp, &asoc->ep->base.bind_addr,
-                                   scope, priority, flags);
-       if (error)
-               goto nomem_copyaddr;
-
-       addrs = sctp_bind_addrs_to_raw(bp, &addrs_len, priority);
-       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.
-        */
-       cookie = sctp_pack_cookie(asoc->ep, asoc, chunk, &cookie_len,
-                                 addrs.v, addrs_len);
-       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)
-               chunksize += sizeof(ecap_param);
-
-       /* Now allocate and fill out the chunk.  */
-       retval = sctp_make_chunk(asoc, SCTP_CID_INIT_ACK, 0, chunksize);
-       if (!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)
-               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;
-
-       /* 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.
-        *
-        * [INIT ACK back to where the INIT came from.]
-        */
-       if (chunk)
-               retval->transport = chunk->transport;
-
-nomem_chunk:
-       kfree(cookie);
-nomem_cookie:
-       kfree(addrs.v);
-nomem_rawaddr:
-nomem_copyaddr:
-       sctp_bind_addr_free(bp);
-nomem_bindaddr:
-       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
- * 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
- *     +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
- *     |   Type = 10   |Chunk  Flags   |         Length                |
- *     +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
- *     /                     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
- *   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 *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 (!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
-        * to which it is replying.
-        *
-        * [COOKIE ECHO back to where the INIT ACK came from.]
-        */
-       if (chunk)
-               retval->transport = chunk->transport;
-
-nodata:
-       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
- * 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 *retval;
-
-       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
-        * to which it is replying.
-        *
-        * [COOKIE ACK back to where the COOKIE ECHO came from.]
-        */
-       if (retval && chunk)
-               retval->transport = chunk->transport;
-
-       return retval;
-}
-
-/*
- *  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
- *  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.
- *
- *     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
- *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
- *    | Chunk Type=13 | Flags=00000000|    Chunk Length = 8           |
- *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
- *    |                      Lowest TSN Number                        |
- *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
- *
- *     Note: The CWR is considered a Control 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_cwrhdr_t cwr;
-
-       cwr.lowest_tsn = htonl(lowest_tsn);
-       retval = sctp_make_chunk(asoc, SCTP_CID_ECN_CWR, 0,
-                                sizeof(sctp_cwrhdr_t));
-
-       if (!retval)
-               goto nodata;
-
-       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
-        * to which it is replying.
-        *
-        * [Report a reduced congestion window back to where the ECNE
-        * came from.]
-        */
-       if (chunk)
-               retval->transport = chunk->transport;
-
-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 __u32 lowest_tsn)
-{
-       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 (!retval)
-               goto nodata;
-       retval->subh.ecne_hdr =
-               sctp_addto_chunk(retval, sizeof(ecne), &ecne);
-
-nodata:
-       return retval;
-}
-
-/* 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, __u8 flags, __u16 ssn)
-{
-       sctp_chunk_t *retval;
-       sctp_datahdr_t dp;
-       int chunk_len;
-
-       /* We assign the TSN as LATE as possible, not here when
-        * creating the chunk.
-        */
-       dp.tsn= 1000000;       /* This marker is a debugging aid. */
-       dp.stream = htons(sinfo->sinfo_stream);
-       dp.ppid   = htonl(sinfo->sinfo_ppid);
-       dp.ssn    = htons(ssn);
-
-       /* 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 (!retval)
-               goto nodata;
-
-       retval->subh.data_hdr = sctp_addto_chunk(retval, sizeof(dp), &dp);
-       memcpy(&retval->sinfo, sinfo, sizeof(struct sctp_sndrcvinfo));
-
-nodata:
-       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 __u8 *data,
-                                __u8 flags, __u16 ssn)
-{
-       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;
-}
-
-/* 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 __u8 *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;
-}
-
-/* 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)
-{
-       __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().
-        */
-       if (sinfo->sinfo_flags & MSG_UNORDERED) {
-               ssn = 0;
-       } else {
-               ssn = __sctp_association_get_next_ssn(asoc,
-                                                     sinfo->sinfo_stream);
-       }
-
-       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 *retval;
-       sctp_sackhdr_t sack;
-       sctp_gap_ack_block_t gab;
-       int length;
-       __u32 ctsn;
-       sctp_tsnmap_iter_t iter;
-       __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);
-       SCTP_DEBUG_PRINTK("make_sack: sackCTSNAck sent is 0x%x.\n",
-                         ctsn);
-
-       /* 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);
-            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);
-
-       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 (!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
-        * destination transport addresses from which the DATA or
-        * control chunks being acknowledged were received.
-        *
-        * [BUG:  We do not implement the following paragraph.
-        * 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
-        * DATA chunk.  The reason being that receiving a duplicate
-        * 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.  */
-       sctp_tsnmap_iter_init(map, &iter);
-       while(sctp_tsnmap_next_gap_ack(map, &iter, &gab.start, &gab.end)) {
-               gab.start = htons(gab.start);
-               gab.end = htons(gab.end);
-               sctp_addto_chunk(retval,
-                                sizeof(sctp_gap_ack_block_t),
-                                &gab);
-       }
-
-       /* Register the duplicates.  */
-       sctp_addto_chunk(retval,
-                        sizeof(sctp_dup_tsn_t) * num_dup_tsns,
-                        &asoc->peer.dup_tsns);
-
-nodata:
-       return retval;
-}
-
-sctp_chunk_t *sctp_make_shutdown(const sctp_association_t *asoc)
-{
-       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);
-
-       retval = sctp_make_chunk(asoc, SCTP_CID_SHUTDOWN, 0,
-                                sizeof(sctp_shutdownhdr_t));
-       if (!retval)
-               goto nodata;
-
-       retval->subh.shutdown_hdr =
-               sctp_addto_chunk(retval, sizeof(shut), &shut);
-
-nodata:
-       return retval;
-}
-
-sctp_chunk_t *sctp_make_shutdown_ack(const sctp_association_t *asoc,
-                                    const sctp_chunk_t *chunk)
-{
-       sctp_chunk_t *retval;
-
-       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
-        * to which it is replying.
-        *
-        * [ACK back to where the SHUTDOWN came from.]
-        */
-       if (retval && chunk)
-               retval->transport = chunk->transport;
-
-       return retval;
-}
-
-sctp_chunk_t *sctp_make_shutdown_complete(const sctp_association_t *asoc,
-                                         const sctp_chunk_t *chunk)
-{
-       sctp_chunk_t *retval;
-       __u8 flags = 0;
-
-       /* 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);
-
-       /* 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.
-        *
-        * [Report SHUTDOWN COMPLETE back to where the SHUTDOWN ACK
-        * came from.]
-        */
-       if (retval && chunk)
-               retval->transport = chunk->transport;
-
-        return retval;
-}
-
-/* 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 *retval;
-       __u8 flags = 0;
-
-       /* 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);
-
-       /* 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.
-        *
-        * [ABORT back to where the offender came from.]
-        */
-       if (retval && chunk)
-               retval->transport = chunk->transport;
-
-       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, __u32 tsn)
-{
-       sctp_chunk_t *retval;
-       __u32 payload;
-
-       retval = sctp_make_abort(asoc, chunk, sizeof(sctp_errhdr_t)
-                                + sizeof(tsn));
-
-       if (!retval)
-               goto no_mem;
-
-       /* Put the tsn back into network byte order.  */
-       payload = htonl(tsn);
-       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
-        * to which it is replying.
-        *
-        * [ABORT back to where the offender came from.]
-        */
-       if (chunk)
-               retval->transport = chunk->transport;
-
-no_mem:
-       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)
-{
-       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->subh.hbs_hdr = sctp_addto_chunk(retval, paylen, payload);
-
-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 *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
-        * to which it is replying.
-        *
-        * [HBACK back to where the HEARTBEAT came from.]
-        */
-       if (chunk)
-               retval->transport = chunk->transport;
-
-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,
-                                __u16 cause_code, const void *payload,
-                                size_t 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 (chunk)
-               retval->transport = chunk->transport;
-
-nodata:
-       return retval;
-}
-
-/********************************************************************
- * 2nd Level Abstractions
- ********************************************************************/
-
-/* 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 *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
-        * 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);
-       INIT_LIST_HEAD(&retval->frag_list);
-       SCTP_DBG_OBJCNT_INC(chunk);
-
-nodata:
-       return retval;
-}
-
-/* Set chunk->source based on the IP header in chunk->skb.  */
-void sctp_init_source(sctp_chunk_t *chunk)
-{
-       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;
-
-       switch (ih4->version) {
-       case 4:
-               source->v4.sin_family = AF_INET;
-               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. ? */
-                       break;
-               )
-
-       default:
-               /* This is a bogus address type, just bail.  */
-               break;
-       };
-}
-
-/* Extract the source address from a 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.  */
-               return &chunk->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,
-                             __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 (!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 = ((__u8 *)chunk_hdr) + sizeof(sctp_chunkhdr_t);
-
-       /* Set the skb to the belonging sock for accounting.  */
-       skb->sk = sk;
-
-       return retval;
-
-nodata:
-       return NULL;
-}
-
-/* Release the memory occupied by a 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);
-
-       /* Free the chunk skb data and the SCTP_chunk stub itself. */
-       dev_kfree_skb(chunk->skb);
-
-       kfree(chunk);
-       SCTP_DBG_OBJCNT_DEC(chunk);
-}
-
-/* Do a deep copy of a chunk.  */
-sctp_chunk_t *sctp_copy_chunk(sctp_chunk_t *chunk, const int priority)
-{
-       sctp_chunk_t *retval;
-       long offset;
-
-       retval = t_new(sctp_chunk_t, priority);
-       if (!retval)
-               goto nodata;
-
-       /* 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);
-       INIT_LIST_HEAD(&retval->frag_list);
-
-       /* Now we copy the deep structure.  */
-       retval->skb = skb_copy(chunk->skb, priority);
-       if (!retval->skb) {
-               kfree(retval);
-               goto nodata;
-       }
-
-       /* 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 *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);
-
-       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;
-
-       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)
-{
-       __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)))
-               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;
-}
-
-/* Helper function to assign a TSN if needed.  This assumes that both
- * the data_hdr and association have already been assigned.
- */
-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 =
-                       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 *asoc;
-       sctp_scope_t scope;
-
-       /* Create the bare association.  */
-       scope = sctp_scope(sctp_source(chunk));
-       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 
-                       = 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;
-
-        default:
-               /* Yikes!  I never heard of this kind of address.  */
-               goto fail;
-       };
-
-nodata:
-       return asoc;
-
-fail:
-       sctp_association_free(asoc);
-       return NULL;
-}
-
-/* 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 __u8 *raw_addrs, int addrs_len)
-{
-       sctp_cookie_param_t *retval;
-       sctp_signed_cookie_t *cookie;
-       int headersize, bodysize;
-
-       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)
-               bodysize += SCTP_COOKIE_MULTIPLE 
-                       - (bodysize % SCTP_COOKIE_MULTIPLE);
-       *cookie_len = headersize + bodysize;
-
-       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;
-
-       /* 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;
-       /* 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));
-
-       /* Copy the raw local address list of the association. */
-       memcpy((__u8 *)&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,
-                        (__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 *retval = NULL;
-       sctp_signed_cookie_t *cookie;
-       sctp_cookie_t *bear_cookie;
-       int headersize, bodysize;
-       int fixed_size, var_size1, var_size2, var_size3;
-       __u8 digest_buf[SCTP_SIGNATURE_SIZE];
-       int secret;
-       sctp_scope_t scope;
-       __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.
-        * 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)))
-               goto malformed;
-
-       /* Verify that the cookie has been padded out. */
-       if (bodysize % SCTP_COOKIE_MULTIPLE)
-               goto malformed;
-
-       /* Process the cookie.  */
-       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,
-                        (__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;
-               }
-       }
-
-       /* 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;
-               goto fail;
-       }
-
-       /* Make a new base association.  */
-       scope = sctp_scope(sctp_source(chunk));
-       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 = (__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,
-                                  priority)) {
-               *error = -SCTP_IERROR_NOMEM;
-               goto fail;
-       }
-
-       retval->next_tsn = retval->c.initial_tsn;
-       retval->ctsn_ack_point = retval->next_tsn - 1;
-
-       /* The INIT stuff will be done by the side effects.  */
-       return retval;
-
-fail:
-       if (retval)
-               sctp_association_free(retval);
-
-       return NULL;
-
-malformed:
-       /* Yikes!  The packet is either corrupt or deliberately
-        * malformed.
-        */
-       *error = -SCTP_IERROR_MALFORMED;
-       goto fail;
-}
-
-/********************************************************************
- * 3rd Level Abstractions
- ********************************************************************/
-
-/* Unpack the parameters in an INIT packet.
- * FIXME:  There is no return status to allow callers to do
- * 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)
-{
-       sctpParam_t param;
-       __u8 *end;
-       sctp_transport_t *transport;
-       list_t *pos, *temp;
-
-       /* 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.
-        */
-
-       /* This implementation defaults to making the first transport
-        * added as the primary transport.  The source address seems to
-        * be a a better choice than any of the embedded addresses.
-        */
-       if (peer_addr)
-               sctp_assoc_add_peer(asoc, peer_addr, priority);
-
-       /* 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))
-                        goto clean_up;
-       }
-
-       /* The fixed INIT headers are always in network byte
-        * order.
-        */
-       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 =
-               ntohs(peer_init->init_hdr.num_outbound_streams);
-       asoc->peer.i.num_inbound_streams =
-               ntohs(peer_init->init_hdr.num_inbound_streams);
-       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.
-        */
-       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.   */
-       asoc->c.peer_vtag = asoc->peer.i.init_tag;
-
-       /* 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
-        * high (for example, implementations MAY use the size of the receiver
-        * advertised window).
-        */
-       list_for_each(pos, &asoc->peer.transport_addr_list) {
-               transport = list_entry(pos, sctp_transport_t, transports);
-               transport->ssthresh = asoc->peer.i.a_rwnd;
-       }
-
-       /* Set up the TSN tracking pieces.  */
-       sctp_tsnmap_init(&asoc->peer.tsn_map, SCTP_TSN_MAP_SIZE,
-                        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:
-        * ...
-        * 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.
-        */
-       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) {
-               transport = list_entry(pos, sctp_transport_t, transports);
-               list_del(pos);
-               sctp_transport_free(transport);
-       }
-}
-
-/* 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.
- * param is the variable length parameter to use for update.
- * cid tells us if this is an INIT, INIT ACK or COOKIE ECHO.
- * If the current packet is an INIT we want to minimize the amount of
- * 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)
-{
-       sockaddr_storage_t addr;
-       int j;
-       int i;
-       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.
-        */
-       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))
-                               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,
-                                                   asoc->peer.port);
-                               scope = sctp_scope(peer_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);
-               break;
-
-       case SCTP_PARAM_HOST_NAME_ADDRESS:
-               SCTP_DEBUG_PRINTK("unimplmented SCTP_HOST_NAME_ADDRESS\n");
-               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.
-                */
-               asoc->peer.ipv4_address = 0;
-               asoc->peer.ipv6_address = 0;
-
-               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);
-               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;
-               break;
-
-       default:
-               SCTP_DEBUG_PRINTK("Ignoring param: %d for association %p.\n",
-                                 ntohs(param.p->type), asoc);
-               /* FIXME:  The entire parameter processing really needs
-                * redesigned.  For now, always return success as doing
-                * otherwise craters the system.
-                */
-               retval = 1;
-
-               break;
-       };
-
-       return retval;
-}
-
-/* Select a new verification tag.  */
-__u32 sctp_generate_tag(const sctp_endpoint_t *ep)
-{
-       /* I believe that this random number generator complies with RFC1750.
-        * A tag of 0 is reserved for special cases (e.g. INIT).
-        */
-       __u32 x;
-
-       do {
-               get_random_bytes(&x, sizeof(__u32));
-       } while (x == 0);
-
-       return x;
-}
-
-/* Select an initial TSN to send during startup.  */
-__u32 sctp_generate_tsn(const sctp_endpoint_t *ep)
-{
-       /* I believe that this random number generator complies with RFC1750.  */
-       __u32 retval;
-
-       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, __u16 port)
-{
-       switch(param.p->type) {
-       case SCTP_PARAM_IPV4_ADDRESS:
-               addr->v4.sin_family = AF_INET;
-               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_flowinfo = 0; /* BUG */
-               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;
-       };
-}
-
-/* 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)
-{
-        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;
-}
-
-/* Convert from an IP version number to an Address Family symbol.  */
-int ipver2af(__u8 ipver)
-{
-       int family;
-
-       switch (ipver) {
-       case 4:
-               family = AF_INET;
-               break;
-
-       case 6:
-               family = AF_INET6;
-               break;
-
-       default:
-               family = 0;
-               break;
-       };
-
-       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 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;
-}
diff --git a/net/sctp/sctp_sm_sideeffect.c b/net/sctp/sctp_sm_sideeffect.c
deleted file mode 100644 (file)
index e206b83..0000000
+++ /dev/null
@@ -1,1170 +0,0 @@
-/* SCTP kernel reference Implementation 
- * Copyright (c) 1999 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001-2002 International Business Machines Corp.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_sm_sideeffect.c,v 1.44 2002/08/16 19:30:50 jgrimm Exp $
- * 
- * These functions work with the state functions in sctp_sm_statefuns.c
- * to implement that state operations.  These functions implement the
- * steps which require modifying existing data structures.
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson          <karl@athena.chicago.il.us>
- *    Jon Grimm             <jgrimm@austin.ibm.com>
- *    Hui Huang                    <hui.huang@nokia.com>
- *    Dajiang Zhang        <dajiang.zhang@nokia.com>
- *    Daisy Chang          <daisyc@us.ibm.com>
- *    Sridhar Samudrala            <sri@us.ibm.com>
- *
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-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/types.h>
-#include <linux/socket.h>
-#include <linux/ip.h>
-#include <net/sock.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,
-                               __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 *);
-
-/* 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
- * functionality there.
- */
-#define DEBUG_PRE \
-       SCTP_DEBUG_PRINTK("sctp_do_sm prefn: " \
-                         "ep %p, %s, %s, asoc %p[%s], %s\n", \
-                         ep, sctp_evttype_tbl[event_type], \
-                         (*debug_fn)(subtype), asoc, \
-                         sctp_state_tbl[state], state_fn->name)
-
-#define DEBUG_POST \
-       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)
-{
-       sctp_cmd_seq_t commands;
-       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[] = {
-               NULL, sctp_cname, sctp_tname, sctp_oname, sctp_pname,
-       };
-       printfn_t *debug_fn  __attribute__ ((unused)) = table[event_type];
-
-       /* 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);
-
-       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;
-}
-
-#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 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)))
-               goto bail;
-
-       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;
-
-       case SCTP_DISPOSITION_NOMEM:
-               /* We ran out of memory, so we need to discard this
-                * packet.
-                */
-               /* BUG--we should now recover some memory, probably by
-                * 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
-                * real work has been done as explicit commands above.
-                */
-               break;
-
-       case SCTP_DISPOSITION_VIOLATION:
-               printk(KERN_ERR "sctp protocol violation state %d "
-                      "chunkid %d\n", state, subtype.chunk);
-               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;
-
-       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:
-               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;
-       };
-
-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 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;
-
-       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:
-        *     while (cmds)
-        *         cmd->handle(x, y, z)
-        * --jgrimm
-        */
-       while (NULL != (command = sctp_next_cmd(commands))) {
-               switch (command->verb) {
-               case SCTP_CMD_NOP:
-                       /* Do nothing. */
-                       break;
-
-               case SCTP_CMD_NEW_ASOC:
-                       /* Register a new association.  */
-                       asoc = command->obj.ptr;
-                       /* Register with the endpoint.  */
-                       sctp_endpoint_add_asoc(ep, asoc);
-                       sctp_hash_established(asoc);
-                       break;
-
-               case SCTP_CMD_UPDATE_ASSOC:
-                      sctp_assoc_update(asoc, command->obj.ptr);
-                      break;
-
-               case SCTP_CMD_PURGE_OUTQUEUE:
-                      sctp_outqueue_teardown(&asoc->outqueue);
-                      break;
-
-               case SCTP_CMD_DELETE_TCB:
-                       /* Delete the current association.  */
-                       sctp_unhash_established(asoc);
-                       sctp_association_free(asoc);
-                       asoc = NULL;
-                       break;
-
-               case SCTP_CMD_NEW_STATE:
-                       /* Enter a new state.  */
-                       asoc->state = command->obj.state;
-                       asoc->state_timestamp = jiffies;
-                       break;
-
-               case SCTP_CMD_REPORT_TSN:
-                       /* Record the arrival of a TSN.  */
-                       sctp_tsnmap_mark(&asoc->peer.tsn_map,
-                                        command->obj.u32);
-                       break;
-
-               case SCTP_CMD_GEN_SACK:
-                       /* Generate a Selective ACK.
-                        * The argument tells us whether to just count
-                        * the packet and MAYBE generate a SACK, or
-                        * force a SACK out.
-                        */
-                       force = command->obj.i32;
-                       error = sctp_gen_sack(asoc, force, commands);
-                       break;
-
-               case SCTP_CMD_PROCESS_SACK:
-                       /* Process an inbound SACK.  */
-                       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;
-
-                       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,
-                                             asoc, chunk, command->obj.ptr,
-                                             priority);
-                       break;
-
-               case SCTP_CMD_GEN_COOKIE_ECHO:
-                       /* 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,
-                                       SCTP_CHUNK(new_obj));
-                       break;
-
-               case SCTP_CMD_GEN_SHUTDOWN:
-                       /* Generate SHUTDOWN when in SHUTDOWN_SENT state.
-                        * Reset error counts.
-                        */
-                       asoc->overall_error_count = 0;
-
-                       /* Generate a SHUTDOWN chunk.  */
-                       new_obj = sctp_make_shutdown(asoc);
-                       if (!new_obj)
-                               goto nomem;
-                       sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
-                                       SCTP_CHUNK(new_obj));
-                       break;
-
-               case SCTP_CMD_CHUNK_ULP:
-                       /* Send a chunk to the sockets layer.  */
-                       SCTP_DEBUG_PRINTK("sm_sideff: %s %p, %s %p.\n",
-                                         "chunk_up:",
-                                         command->obj.ptr,
-                                         "ulpq:",
-                                         &asoc->ulpq);
-                       sctp_ulpqueue_tail_data(&asoc->ulpq,
-                                               command->obj.ptr,
-                                               GFP_ATOMIC);
-                       break;
-
-               case SCTP_CMD_EVENT_ULP:
-                       /* 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,
-                                                command->obj.ptr);
-                       break;
-
-               case SCTP_CMD_REPLY:
-                       /* Send a chunk to our peer.  */
-                       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);
-                       break;
-
-               case SCTP_CMD_RETRAN:
-                       /* Mark a transport for retransmission.  */
-                       sctp_retransmit(&asoc->outqueue,
-                                       command->obj.transport, 0);
-                       break;
-
-               case SCTP_CMD_TRANSMIT:
-                       /* Kick start transmission. */
-                       error = sctp_flush_outqueue(&asoc->outqueue, 0);
-                       break;
-
-               case SCTP_CMD_ECN_CE:
-                       /* Do delayed CE processing.   */
-                       sctp_do_ecn_ce_work(asoc, command->obj.u32);
-                       break;
-
-               case SCTP_CMD_ECN_ECNE:
-                       /* Do delayed ECNE processing. */
-                       new_obj = sctp_do_ecn_ecne_work(asoc,
-                                                       command->obj.u32,
-                                                       chunk);
-                       if (new_obj) {
-                               sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
-                                               SCTP_CHUNK(new_obj));
-                       }
-                       break;
-
-               case SCTP_CMD_ECN_CWR:
-                       /* Do delayed CWR processing.  */
-                       sctp_do_ecn_cwr_work(asoc, command->obj.u32);
-                       break;
-
-               case SCTP_CMD_SETUP_T2:
-                       sctp_cmd_setup_t2(commands, asoc, command->obj.ptr);
-                       break;
-
-               case SCTP_CMD_TIMER_START:
-                       timer = &asoc->timers[command->obj.to];
-                       timeout = asoc->timeouts[command->obj.to];
-                       if (!timeout)
-                               BUG();
-
-                       timer->expires = jiffies + timeout;
-                       sctp_association_hold(asoc);
-                       add_timer(timer);
-                       break;
-
-               case SCTP_CMD_TIMER_RESTART:
-                       timer = &asoc->timers[command->obj.to];
-                       timeout = asoc->timeouts[command->obj.to];
-                       if (!mod_timer(timer, jiffies + timeout))
-                               sctp_association_hold(asoc);
-                       break;
-
-               case SCTP_CMD_TIMER_STOP:
-                       timer = &asoc->timers[command->obj.to];
-                       if (timer_pending(timer) && del_timer(timer))
-                               sctp_association_put(asoc);
-                       break;
-
-               case SCTP_CMD_INIT_RESTART:
-                       /* Do the needed accounting and updates
-                        * associated with restarting an initialization
-                        * 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;
-                       }
-
-                       sctp_add_cmd_sf(commands,
-                                       SCTP_CMD_TIMER_RESTART,
-                                       SCTP_TO(command->obj.to));
-                       break;
-
-               case SCTP_CMD_INIT_FAILED:
-                       sctp_cmd_init_failed(commands, asoc);
-                       break;
-
-               case SCTP_CMD_ASSOC_FAILED:
-                       sctp_cmd_assoc_failed(commands, asoc);
-                       break;
-
-               case SCTP_CMD_COUNTER_INC:
-                       asoc->counters[command->obj.counter]++;
-                       break;
-
-               case SCTP_CMD_COUNTER_RESET:
-                       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);
-                       }
-                       break;
-
-               case SCTP_CMD_REPORT_BIGGAP:
-                       SCTP_DEBUG_PRINTK("Big gap: %x to %x\n",
-                                         sctp_tsnmap_get_ctsn(
-                                                 &asoc->peer.tsn_map),
-                                         command->obj.u32);
-                       break;
-
-               case SCTP_CMD_REPORT_BAD_TAG:
-                       SCTP_DEBUG_PRINTK("vtag mismatch!\n");
-                       break;
-
-               case SCTP_CMD_SET_BIND_ADDR:
-                       sctp_cmd_set_bind_addrs(commands, asoc,
-                                               command->obj.bp);
-                       break;
-
-               case SCTP_CMD_STRIKE:
-                       /* Mark one strike against a transport.  */
-                       sctp_do_8_2_transport_strike(asoc,
-                                                    command->obj.transport);
-                       break;
-
-               case SCTP_CMD_TRANSPORT_RESET:
-                       t = command->obj.transport;
-                       sctp_cmd_transport_reset(commands, asoc, t);
-                       break;
-
-               case SCTP_CMD_TRANSPORT_ON:
-                       t = command->obj.transport;
-                       sctp_cmd_transport_on(commands, asoc, t, chunk);
-                       break;
-
-               case SCTP_CMD_HB_TIMERS_START:
-                       sctp_cmd_hb_timers_start(commands, asoc);
-                       break;
-
-               case SCTP_CMD_REPORT_ERROR:
-                       error = command->obj.error;
-                       break;
-
-               case SCTP_CMD_PROCESS_CTSN:
-                       /* Dummy up a SACK for processing. */
-                       sackh.cum_tsn_ack = command->obj.u32;
-                       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_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;
-               };
-       }
-
-       return error;
-
-nomem:
-       error = -ENOMEM;
-       return error;
-}
-
-/* A helper function for delayed processing of INET ECN CE bit. */
-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
-        */
-
-       asoc->last_ecne_tsn = lowest_tsn;
-       asoc->need_ecne = 1;
-}
-
-/* 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
- * that was originally marked with the CE bit.
- */
-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
-        * sending a CWR.
-        */
-
-       /* Find which transport's congestion variables
-        * need to be adjusted.
-        */
-
-       transport = sctp_assoc_lookup_tsn(asoc, lowest_tsn);
-
-       /* Update the congestion variables. */
-       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
-        * 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.
-        *
-        * Here's what RFC 2960 has to say about CWR.  This is NOT
-        * what we do.
-        *
-        * RFC 2960 Appendix A
-        *
-        *    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 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.
-        */
-       asoc->last_cwr_tsn = asoc->next_tsn - 1;
-
-       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;
-}
-
-/* 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;
-}
-
-/* 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)
-{
-       __u32 ctsn, max_tsn_seen;
-       sctp_chunk_t *sack;
-       int error = 0;
-
-       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) {
-               /* We will need a SACK for the next packet.  */
-               asoc->peer.sack_needed = 1;
-               goto out;
-       } else {
-               sack = sctp_make_sack(asoc);
-               if (!sack)
-                       goto nomem;
-
-               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));
-       }
-
-out:
-       return error;
-
-nomem:
-       error = -ENOMEM;
-       return error;
-}
-
-/* Handle a duplicate TSN.  */
-void sctp_do_TSNdup(sctp_association_t *asoc, sctp_chunk_t *chunk, long gap)
-{
-#if 0
-       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;
-#endif /* 0 */
-} /* sctp_do_TSNdup() */
-
-#undef AP
-
-/* 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)
-{
-       int error;
-       sctp_transport_t *transport = (sctp_transport_t *) peer;
-       sctp_association_t *asoc = transport->asoc;
-
-       /* 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)))
-                       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)
-               goto out_unlock;
-
-       /* Run through the state machine.  */
-       error = sctp_do_sm(SCTP_EVENT_T_TIMEOUT,
-                          SCTP_ST_TIMEOUT(SCTP_EVENT_TIMEOUT_T3_RTX),
-                          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);
-}
-
-/* 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)
-{
-       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)))
-                       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)
-               goto out_unlock;
-
-       /* Run through the state machine.  */
-       error = sctp_do_sm(SCTP_EVENT_T_TIMEOUT,
-                          SCTP_ST_TIMEOUT(timeout_type),
-                          asoc->state, asoc->ep, asoc,
-                          (void *)timeout_type,
-                          GFP_ATOMIC);
-
-       if (error)
-               asoc->base.sk->err = -error;
-
-out_unlock:
-       sctp_bh_unlock_sock(asoc->base.sk);
-       sctp_association_put(asoc);
-}
-
-void sctp_generate_t1_cookie_event(unsigned long data)
-{
-       sctp_association_t *asoc = (sctp_association_t *) data;
-       sctp_generate_timeout_event(asoc, SCTP_EVENT_TIMEOUT_T1_COOKIE);
-}
-
-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);
-}
-
-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);
-}
-
-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);
-}
-
-/* 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)
-{
-       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)))
-                       sctp_transport_hold(transport);
-               goto out_unlock;
-       } 
-
-       /* Is this structure just waiting around for us to actually
-        * get destroyed?
-        */
-       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;
-
-out_unlock:
-       sctp_bh_unlock_sock(asoc->base.sk);
-       sctp_transport_put(transport);
-}
-
-/* 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);
-}
-
-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_timer_event_t *sctp_timer_events[SCTP_NUM_TIMEOUT_TYPES] = {
-       NULL,
-       sctp_generate_t1_cookie_event,
-       sctp_generate_t1_init_event,
-       sctp_generate_t2_shutdown_event,
-       NULL,
-       NULL,
-       sctp_generate_heartbeat_event,
-       sctp_generate_sack_event,
-       sctp_generate_autoclose_event,
-       sctp_generate_pmtu_raise_event,
-};
-
-/********************************************************************
- * 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.
- * When the value in the error counter exceeds the protocol parameter
- * '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)
-{
-       /* 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)) {
-               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_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.
-        */
-       transport->rto = min((transport->rto * 2), transport->asoc->rto_max);
-}
-
-/* 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;
-
-       event = sctp_ulpevent_make_assoc_change(asoc,
-                                               0,
-                                               SCTP_CANT_STR_ASSOC,
-                                               0, 0, 0,
-                                               GFP_ATOMIC);
-
-       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
-        * the transmit queue, if the user has enabled
-        * SEND_FAILED notifications.
-        */
-       sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
-}
-
-/* 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;
-
-       event = sctp_ulpevent_make_assoc_change(asoc,
-                                               0,
-                                               SCTP_COMM_LOST,
-                                               0, 0, 0,
-                                               GFP_ATOMIC);
-
-       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());
-}
-
-/* Process an init chunk (may be real INIT/INIT-ACK or an embedded INIT
- * 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)
-{
-       /* 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.
-        */
-       sctp_process_init(asoc, chunk->chunk_hdr->type,
-                         sctp_source(chunk), peer_init,
-                         priority);
-}
-
-/* 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)
-{
-       sctp_transport_t *t;
-       list_t *pos;
-
-       /* Start a heartbeat timer for each transport on the association.
-        * hold a reference on the transport to make sure none of
-        * the needed data structures go away.
-        */
-       list_for_each(pos, &asoc->peer.transport_addr_list) {
-               t = list_entry(pos, sctp_transport_t, transports);
-               if (!mod_timer(&t->hb_timer,
-                              t->hb_interval + t->rto + jiffies)) {
-                       sctp_transport_hold(t);
-               }
-       }
-}
-
-/* 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;
-
-       list_for_each_safe(pos, temp, &bp->address_list) {
-               list_del_init(pos);
-               list_add_tail(pos, &asoc->base.bind_addr.address_list);
-       }
-
-       /* Free the temporary bind addr header, otherwise
-        * there will a memory leak.
-        */
-       sctp_bind_addr_free(bp);
-}
-
-/* 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;
-
-       /* 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;
-
-       /* Mark the destination transport address as active if it is not so
-        * marked.
-        */
-       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;
-       sctp_transport_update_rto(t, (jiffies - hbinfo->sent_at));
-}
-
-/* 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))
-               sctp_transport_hold(t);
-}
-
-/* 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.  */
-               err = sctp_do_sm(SCTP_EVENT_T_OTHER,
-                                SCTP_ST_OTHER(SCTP_EVENT_NO_PENDING_TSN),
-                                asoc->state, asoc->ep, asoc, NULL,
-                                GFP_ATOMIC);
-       } else {
-               /* Windows may have opened, so we need
-                * to check if we have DATA to transmit
-                */
-               err = sctp_flush_outqueue(&asoc->outqueue, 0);
-       }
-
-       return err;
-}
-
-/* Helper function to set the timeout value for T2-SHUTDOWN timer and to set
- * 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)
-{
-       sctp_transport_t *t;
-
-       t = sctp_assoc_choose_shutdown_transport(asoc);
-       asoc->shutdown_last_sent_to = t;
-       asoc->timeouts[SCTP_EVENT_TIMEOUT_T2_SHUTDOWN] = t->rto;
-       chunk->transport = t;
-}
diff --git a/net/sctp/sctp_sm_statefuns.c b/net/sctp/sctp_sm_statefuns.c
deleted file mode 100644 (file)
index 49cf5c4..0000000
+++ /dev/null
@@ -1,3700 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001-2002 International Business Machines, Corp.
- * Copyright (c) 2002      Nokia Corp.
- *
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_sm_statefuns.c,v 1.49 2002/08/21 18:34:04 jgrimm Exp $
- * 
- * This is part of the SCTP Linux Kernel Reference Implementation.
- *
- * These are the state functions for the state machine.
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson          <karl@athena.chicago.il.us>
- *    Mathew Kotowsky       <kotowsky@sctp.org>
- *    Sridhar Samudrala     <samudrala@us.ibm.com>
- *    Jon Grimm             <jgrimm@us.ibm.com>
- *    Hui Huang            <hui.huang@nokia.com>
- *    Dajiang Zhang        <dajiang.zhang@nokia.com>
- *    Daisy Chang          <daisyc@us.ibm.com>
- *
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_sm_statefuns.c,v 1.49 2002/08/21 18:34:04 jgrimm Exp $";
-
-#include <linux/types.h>
-#include <linux/kernel.h>
-#include <linux/ip.h>
-#include <linux/ipv6.h>
-#include <linux/net.h>
-#include <linux/inet.h>
-#include <net/sock.h>
-#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
- * 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).
- *
- * Verification Tag: 8.5.1(C)
- * C) Rules for packet carrying SHUTDOWN COMPLETE:
- * ...
- * - 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.
- *
- * 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_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)
-               return SCTP_DISPOSITION_VIOLATION;
-
-       /* 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
-        * 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);
-
-       /* 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.
-        */
-       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));
-
-       /* 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,
-                       SCTP_STATE(SCTP_STATE_CLOSED));
-
-       sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
-
-       return SCTP_DISPOSITION_DELETE_TCB;
-
-nomem:
-       return SCTP_DISPOSITION_NOMEM;
-}
-
-/*
- * 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.
- *    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_add_cmd_sf(commands, SCTP_CMD_DISCARD_PACKET, SCTP_NULL());
-       return SCTP_DISPOSITION_CONSUME;
-}
-
-/*
- * 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
- *    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: 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_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
-        * control endpoint, responding with an ABORT.
-        */
-       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)
-               return SCTP_DISPOSITION_VIOLATION;
-
-        /* Grab the INIT header.  */
-       chunk->subh.init_hdr = (sctp_inithdr_t *)chunk->skb->data;
-
-       /* 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 (!new_asoc)
-               goto nomem;
-
-       /* 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_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.  */
-       repl = sctp_make_init_ack(new_asoc, chunk, GFP_ATOMIC);
-       if (!repl)
-               goto nomem_ack;
-
-       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, nor keep any states for the
-        * new association.  Otherwise, "Z" will be vulnerable to resource
-        * attacks.
-        */
-       sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
-
-       return SCTP_DISPOSITION_DELETE_TCB;
-
-nomem_ack:
-       sctp_association_free(new_asoc);
-nomem:
-       return SCTP_DISPOSITION_NOMEM;
-}
-
-/*
- * 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
- *    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
- *   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_chunk_t *chunk = arg;
-       sctp_init_chunk_t *initchunk;
-       __u32 init_tag;
-
-       /* 6.10 Bundling
-        * An endpoint MUST NOT bundle INIT, INIT ACK or
-        * SHUTDOWN COMPLETE with any other chunks.
-        */
-       if (!chunk->singleton)
-               return SCTP_DISPOSITION_VIOLATION;
-
-       /* Grab the INIT header.  */
-       chunk->subh.init_hdr = (sctp_inithdr_t *) chunk->skb->data;     
-
-       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 (!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_STATE(SCTP_STATE_CLOSED));
-               sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB,
-                               SCTP_NULL());
-               return SCTP_DISPOSITION_DELETE_TCB;
-       }
-
-       /* 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;
-
-       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,
-                       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_TO(SCTP_EVENT_TIMEOUT_T1_COOKIE));
-       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());
-       return SCTP_DISPOSITION_CONSUME;
-
-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
- *    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_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
-        * control endpoint, responding with an ABORT.
-        */
-       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.
-        */
-        chunk->subh.cookie_hdr =
-               (sctp_signed_cookie_t *)chunk->skb->data;
-       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.
-        */
-       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,
-                                               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_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.
-        */
-       /* This 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,
-                         &chunk->subh.cookie_hdr->c.peer_addr, peer_init,
-                         GFP_ATOMIC);
-
-       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
-        *
-        * 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.
-        */
-       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;
-
-       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;
-}
-
-/*
- * 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
- *    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).
- *
- * 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_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.
-        */
-       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_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).
-        */
-       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;
-
-       sctp_add_cmd_sf(commands, SCTP_CMD_EVENT_ULP, SCTP_ULPEVENT(ev));
-
-       return SCTP_DISPOSITION_CONSUME;
-
-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_transport_t *transport = (sctp_transport_t *) arg;
-       sctp_chunk_t *reply;
-       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.
-        * 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;
-
-       /* 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);
-       reply = sctp_make_heartbeat(asoc, transport, &hbinfo, paylen);
-       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_TRANSPORT(transport));
-
-        return SCTP_DISPOSITION_CONSUME;
-
-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
- * 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
- * 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_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.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);
-
-       reply = sctp_make_heartbeat_ack(asoc, chunk,
-                                       chunk->subh.hb_hdr, paylen);
-       if (!reply)
-               goto nomem;
-
-       sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(reply));
-       return SCTP_DISPOSITION_CONSUME;
-
-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
- * address to which the HEARTBEAT was sent, and mark the destination
- * transport address as active if it is not so marked. The endpoint may
- * optionally report to the upper layer when an inactive destination
- * address is marked as active due to the reception of the latest
- * 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_chunk_t *chunk = arg;
-       sockaddr_storage_t from_addr;
-       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);
-
-       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.  */
-       if (!link) {
-               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)) {
-               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));
-
-       return SCTP_DISPOSITION_CONSUME;
-}
-
-/* Populate the verification/tie tags based on overlapping INIT
- * 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)
-{
-       switch (asoc->state) {
-
-       /* 5.2.1 INIT received in COOKIE-WAIT or COOKIE-ECHOED State */
-
-       case SCTP_STATE_COOKIE_WAIT:
-               new_asoc->c.my_vtag     = asoc->c.my_vtag;
-               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;
-
-       /* 5.2.2 Unexpected INIT in States Other than CLOSED, COOKIE-ECHOED,
-        *   COOKIE-WAIT and SHUTDOWN-ACK-SENT
-        */
-       default:
-               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.
-        */
-       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;
-}
-
-/*
- * 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.
- */
-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))
-               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))
-               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)))
-               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))
-               return 'C';
-
-       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)
-{
-       sctp_chunk_t *chunk = arg;
-       sctp_chunk_t *repl;
-       sctp_association_t *new_asoc;
-
-       /* 6.10 Bundling
-        * An endpoint MUST NOT bundle INIT, INIT ACK or
-        * SHUTDOWN COMPLETE with any other chunks.
-        */
-       if (!chunk->singleton)
-               return SCTP_DISPOSITION_VIOLATION;
-
-       /* Grab the INIT header.  */
-       chunk->subh.init_hdr = (sctp_inithdr_t *) chunk->skb->data;
-
-       /* 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.
-        */
-       new_asoc = sctp_make_temp_asoc(ep, chunk, GFP_ATOMIC);
-       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.
-        */
-       sctp_process_init(new_asoc, chunk->chunk_hdr->type,
-                         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;
-
-       sctp_add_cmd_sf(commands, SCTP_CMD_NEW_ASOC, SCTP_ASOC(new_asoc));
-       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;
-
-nomem:
-       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,
- * unchanged). These original parameters are combined with those from the
- * 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
- * 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)
-{
-       /* Call helper to do the real work for both simulataneous and
-        * duplicate INIT chunk handling.
-        */
-       return sctp_sf_do_unexpected_init(ep, asoc, type, arg, commands);
-}
-
-/*
- * 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
- *
- * Unless otherwise stated, upon reception of an unexpected INIT for
- * this association, the endpoint shall generate an INIT ACK with a
- * State Cookie.  In the outbound INIT ACK the endpoint MUST copy its
- * current Verification Tag and peer's Verification Tag into a reserved
- * place within the state cookie.  We shall refer to these locations as
- * the Peer's-Tie-Tag and the Local-Tie-Tag.  The outbound SCTP packet
- * containing this INIT ACK MUST carry a Verification Tag value equal to
- * the Initiation Tag found in the unexpected INIT.  And the INIT ACK
- * MUST contain a new Initiation Tag (randomly generated see Section
- * 5.3.1).  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.
- *
- * After sending out the INIT ACK, the endpoint shall take no further
- * actions, i.e., the existing association, including its current state,
- * and the corresponding TCB MUST NOT be changed.
- *
- * Note: Only when a TCB exists and the association is not in a COOKIE-
- * WAIT state are the Tie-Tags populated.  For a normal association INIT
- * (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)
-{
-       /* Call helper to do the real work for both simulataneous and
-        * duplicate INIT chunk handling.
-        */
-       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
- *  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)
-{
-       sctp_init_chunk_t *peer_init;
-       sctp_ulpevent_t *ev;
-       sctp_chunk_t *repl;
-       sctp_transport_t *new_addr, *addr;
-       list_t *pos, *pos2, *temp;
-       int found, error;
-
-       /* 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_source(chunk), peer_init, GFP_ATOMIC);
-
-       /* 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,
-                                  &asoc->peer.transport_addr_list) {
-                       addr = list_entry(pos2, sctp_transport_t, transports);
-                       if (!sctp_cmp_addr_exact(&new_addr->ipaddr,
-                                                &addr->ipaddr)) {
-                               found = 0;
-                               break;
-                       }
-               }
-               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;
-
-               error = sctp_add_bind_addr(bp, &new_addr->ipaddr, GFP_ATOMIC);
-               if (error)
-                       goto nomem_add;
-
-               rawaddr = sctp_bind_addrs_to_raw(bp, &len, GFP_ATOMIC);
-               if (!rawaddr.v)
-                       goto nomem_raw;
-
-               repl = sctp_make_abort(asoc, chunk, len+sizeof(sctp_errhdr_t));
-               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;
-       }
-
-       /* For now, fail any unsent/unacked data.  Consider the optional
-        * choice of resending of this data.
-        */
-       sctp_add_cmd_sf(commands, SCTP_CMD_PURGE_OUTQUEUE, SCTP_NULL());
-
-       /* 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));
-
-       /* Report association restart to upper layer. */
-       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 (!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;
-}
-
-/* Unexpected COOKIE-ECHO handler for setup collision (Table 2, action 'B')
- *
- * 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)
-{
-       sctp_init_chunk_t *peer_init;
-       sctp_ulpevent_t *ev;
-       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_source(chunk), peer_init, GFP_ATOMIC);
-
-       /* 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_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 (!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
-        * send the Communication Up notification to the SCTP user
-        * 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 (!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;
-}
-
-/* Unexpected COOKIE-ECHO handler for setup collision (Table 2, action 'C')
- *
- * 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)
-{
-       /* The cookie should be silently discarded.
-        * The endpoint SHOULD NOT change states and should leave
-        * any timers running.
-        */
-       return SCTP_DISPOSITION_DISCARD;
-}
-
-/* Unexpected COOKIE-ECHO handler lost chunk (Table 2, action 'D')
- *
- * 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)
-{
-       sctp_ulpevent_t *ev = NULL;
-       sctp_chunk_t *repl;
-
-       /* The local endpoint cannot use any value from the received
-        * state cookie and need to immediately resend a COOKIE-ACK
-        * and move into ESTABLISHED if it hasn't done so.
-        */
-       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
-                * to 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,
-                                            new_asoc->c.sinit_num_ostreams,
-                                            new_asoc->c.sinit_max_instreams,
-                                             GFP_ATOMIC);
-               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 (!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);
-       return SCTP_DISPOSITION_NOMEM;
-}
-
-/*
- * 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 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,
-                                                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. */
-               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. */
-               retval = sctp_sf_do_dupcook_c(ep, asoc, chunk, commands,
-                                             new_asoc);
-               break;
-
-       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");
-               retval = SCTP_DISPOSITION_DISCARD;
-               break;
-        };
-
-       /* 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;
-}
-
-#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.
- * ...
- * 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
- *    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.
- *
- * 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_chunk_t *in_packet;
-       stp_chunk_t *reply;
-       sctp_inithdr_t initack;
-       __u8 *addrs;
-       int addrs_len;
-       time_t rtt;
-       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 (!retval->event_up)
-                       goto nomem;
-               sctp_add_cmd_sf(retval->commands, SCTP_CMD_DELETE_TCB,
-                               SCTP_NULL());
-               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.  */
-       in_packet = chunk;
-       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);
-
-       /* Build that new INIT chunk.  */
-       reply = sctp_make_chunk(SCTP_INITIATION, 0,
-                               sizeof(initack)
-                               + sizeof(bht)
-                               + addrs_len);
-       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;
-}
-#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
- * 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.
- *
- *  - 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.
- *
- * 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_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;
-}
-
-/*
- * 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_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
-                       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;
-}
-
-/*
- * 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)
-{
-       /* 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_abort(ep, asoc, type, arg, commands);
-}
-
-#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.
- *...
- * While in SHUTDOWN-SENT state ... If the timer expires, the endpoint
- * must re-send the SHUTDOWN ACK.
- *
- * Verification Tag:  Neither the INIT nor the timeout will have a
- * valid verification tag, so it is 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 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) {
-                       sctp_add_cmd(commands, SCTP_CMD_DELETE_TCB,
-                                    SCTP_NULL());
-                       return SCTP_DISPOSITION_DELETE_TCB;
-               }
-               retval->counters[0] = SCTP_COUNTER_INCR;
-               retval->counters[0] = SctpCounterRetran;
-               retval->counters[1] = 0;
-               retval->counters[1] = 0;
-       }
-
-       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;
-}
-#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,
- *
- *  - 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
- *    SHUTDOWN sender.
- *
- * 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);
-
-       /* 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));
-       disposition = SCTP_DISPOSITION_CONSUME;
-
-       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.
-        */
-       sctp_add_cmd_sf(commands, SCTP_CMD_PROCESS_CTSN,
-                       SCTP_U32(chunk->subh.shutdown_hdr->cum_tsn_ack));
-       return disposition;
-}
-
-/*
- * sctp_sf_do_ecn_cwr
- *
- * Section:  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
- * 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.
- *
- * 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_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);
-
-       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
- *
- * Section:  Appendix A: Explicit Congestion Notification
- *
- * ECN-Echo
- *
- * RFC 2481 details a specific bit for a receiver to send back in its
- * TCP acknowledgements to notify the sender of the Congestion
- * Experienced (CE) bit having arrived from the network.  For SCTP this
- * same indication is made by including the ECNE chunk.  This chunk
- * contains one data element, i.e. the lowest TSN associated with the IP
- * datagram marked with the CE bit.....
- *
- * 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_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;
-       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)) {
-               sctp_add_cmd_sf(commands, SCTP_CMD_ECN_ECNE,
-                               SCTP_U32(ecne->lowest_tsn));
-       }       
-       return SCTP_DISPOSITION_CONSUME;
-}
-
-/*
- * Section: 6.2  Acknowledgement on Reception of DATA Chunks
- *
- * 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
- * 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_chunk_t *err;
-       size_t datalen;
-       int tmp;
-       __u32 tsn;
-
-       /* 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.
-        */
-
-       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);
-        }
-
-       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.  */
-
-       /* 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) {
-                       /* Do real work as sideffect. */
-                       sctp_add_cmd_sf(commands, SCTP_CMD_ECN_CE,
-                                       SCTP_U32(tsn));
-               }
-       }
-
-       tmp = sctp_tsnmap_check(&asoc->peer.tsn_map, tsn);
-       if (tmp < 0) {
-               /* The TSN is too high--silently discard the chunk and
-                * count on it getting retransmitted later.
-                */
-               goto discard_noforce;
-       } else if (tmp > 0) {
-               /* This is a duplicate.  Record it.  */
-               sctp_add_cmd_sf(commands, SCTP_CMD_REPORT_DUP, SCTP_U32(tsn));
-               goto discard_force;
-       }
-
-       /* This is a new TSN.  */
-
-       /* 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)) {
-               SCTP_DEBUG_PRINTK("Discarding tsn: %u datalen: %Zd, "
-                                 "rwnd: %d\n", tsn, datalen, asoc->rwnd);
-               goto discard_noforce;
-       }
-
-       /*
-        * Section 3.3.10.9 No User Data (9)
-        *
-        * Cause of error
-        * ---------------
-        * No User Data:  This error cause is returned to the originator of a
-        * DATA chunk if a received DATA chunk has no user data.
-        */
-       if (unlikely(0 == datalen)) {
-               err = sctp_make_abort_no_data(asoc, chunk, tsn);
-               if (err) {
-                       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_STATE(SCTP_STATE_CLOSED));
-               sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
-               return SCTP_DISPOSITION_CONSUME;
-       }
-
-       /* We are accepting this DATA chunk. */
-
-       /* 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
-        * 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.
-        */
-       if (ntohs(data_hdr->stream) >= asoc->c.sinit_max_instreams) {
-               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_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
-        * 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.
-        *
-        * RFC 2960 6.2 Acknowledgement on Reception of DATA Chunks
-        *
-        * 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 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:
-       /* RFC 2960 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.
-        */
-       /* In our case, we split the MAY SACK advice up whether or not
-        * the last chunk is a duplicate.'
-        */
-       if (chunk->end_of_packet)
-               sctp_add_cmd_sf(commands, SCTP_CMD_GEN_SACK, SCTP_FORCE());
-       return SCTP_DISPOSITION_DISCARD;
-
-discard_noforce:
-       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_DISCARD;
-}
-
-/*
- * 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_chunk_t *err;
-       size_t datalen;
-       int tmp;
-       __u32 tsn;
-
-       /* 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.
-        */
-       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);
-       }
-
-       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.
-        *
-        * 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) {
-                       /* Do real work as sideffect. */
-                       sctp_add_cmd_sf(commands, SCTP_CMD_ECN_CE,
-                                       SCTP_U32(tsn));
-               }
-       }
-
-       tmp = sctp_tsnmap_check(&asoc->peer.tsn_map, tsn);
-       if (tmp < 0) {
-               /* The TSN is too high--silently discard the chunk and
-                * count on it getting retransmitted later.
-                */
-               goto gen_shutdown;
-       } else if (tmp > 0) {
-               /* This is a duplicate.  Record it.  */
-               sctp_add_cmd_sf(commands, SCTP_CMD_REPORT_DUP, SCTP_U32(tsn));
-               goto gen_shutdown;
-       }
-
-       /* This is a new TSN.  */
-
-       datalen = ntohs(chunk->chunk_hdr->length);
-       datalen -= sizeof(sctp_data_chunk_t);
-
-       /*
-        * Section 3.3.10.9 No User Data (9)
-        *
-        * Cause of error
-        * ---------------
-        * No User Data:  This error cause is returned to the originator of a
-        * DATA chunk if a received DATA chunk has no user data.
-        */
-       if (unlikely(0 == datalen)) {
-               err = sctp_make_abort_no_data(asoc, chunk, tsn);
-               if (err) {
-                       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_STATE(SCTP_STATE_CLOSED));
-               sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
-               return SCTP_DISPOSITION_CONSUME;
-       }
-
-       /* We are accepting this DATA chunk. */
-
-       /* 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
-        * 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.
-        */
-       if (ntohs(data_hdr->stream) >= asoc->c.sinit_max_instreams) {
-               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_CHUNK(err));
-               }
-       }
-
-       /* Go a head and force a SACK, since we are shutting down. */
-gen_shutdown:
-       /* 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)
-        * with a SACK, a SHUTDOWN chunk, and restart the T2-shutdown timer
-        */
-       if (chunk->end_of_packet) {
-               /* We must delay the chunk creation since the cumulative
-                * 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_add_cmd_sf(commands, SCTP_CMD_TIMER_RESTART,
-                               SCTP_TO(SCTP_EVENT_TIMEOUT_T2_SHUTDOWN));
-       }
-       return SCTP_DISPOSITION_CONSUME;
-}
-
-/*
- * 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
- *     as available for retransmit:  Mark it as missing for fast
- *     retransmit as described in Section 7.2.4 and if no retransmit
- *     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;
-       __u32 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);
-
-       /* 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",
-                                 asoc->ctsn_ack_point);
-               return SCTP_DISPOSITION_DISCARD;
-       }
-
-       /* 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
-        * sideeffect.
-        */
-       return SCTP_DISPOSITION_CONSUME;
-}
-
-/*
- * 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
- *    with an ABORT.  When sending the ABORT, the receiver of the
- *    OOTB packet MUST fill in the Verification Tag field of the
- *    outbound packet with the value found in the Verification Tag
- *    field of the OOTB packet and set the T-bit in the Chunk Flags
- *    to indicate that no TCB was found.  After sending this ABORT,
- *    the receiver of the OOTB packet shall discard the OOTB packet
- *    and take no further action.
- *
- * 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;
-       __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;
-
-       /* 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_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 (!abort)
-               goto nomem_chunk;
-
-       /* Set the skb to the belonging sock for accounting.  */
-       abort->skb->sk = ep->base.sk;
-
-       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;
-}
-
-/*
- * 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)
-{
-       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 (!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;
-}
-
-/*
- * Process an inbound SHUTDOWN ACK.
- *
- * 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;
-       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.
-        */
-       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));
-
-       /* 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;
-
-       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. */
-       sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
-       return SCTP_DISPOSITION_DELETE_TCB;
-
-nomem:
-       return SCTP_DISPOSITION_NOMEM;
-}
-
-/*
- * RFC 2960, 8.4 - Handle "Out of the blue" Packets
- * 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. Otherwise,
- *
- * 8) The receiver should respond to the sender of the OOTB packet with
- *    an ABORT.  When sending the ABORT, the receiver of the OOTB packet
- *    MUST fill in the Verification Tag field of the outbound packet
- *    with the value found in the Verification Tag field of the OOTB
- *    packet and set the T-bit in the Chunk Flags to indicate that no
- *    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_chunk_t *chunk = arg;
-       struct sk_buff *skb = chunk->skb;
-       sctp_chunkhdr_t *ch;
-       __u8 *ch_end;
-       int ootb_shut_ack = 0;
-
-       ch = (sctp_chunkhdr_t *) chunk->chunk_hdr;
-       do {
-               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;
-       } while (ch_end < skb->tail);
-
-       if (ootb_shut_ack)
-               sctp_sf_shut_8_4_5(ep, asoc, type, arg, commands);
-       else
-               sctp_sf_tabort_8_4_8(ep, asoc, type, arg, commands);
-
-       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.
- *
- * 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
- *   should be treated as an Out Of The Blue packet.
- *   [This means that we do NOT check the Verification Tag on these
- *   chunks. --piggy ]
- *
- * 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_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);
-       if (!transport)
-               goto nomem;
-
-       /* 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_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;
-
-       /* Set the skb to the belonging sock for accounting.  */
-       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;
-}
-
-#if 0
-/*
- * We did something stupid but got lucky.  Namely, we sent a HEARTBEAT
- * before the association was all the way up and we did NOT get an
- * 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);
-
-       return SCTP_DISPOSITION_CONSUME;
-
-nomem:
-       return SCTP_DISPOSITION_NOMEM;
-}
-#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);
-
-       return SCTP_DISPOSITION_CONSUME;
-
-nomem:
-       return SCTP_DISPOSITION_NOMEM;
-}
-#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)
-{
-       return SCTP_DISPOSITION_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)
- * -> association id [,destination transport addr list] [,outbound stream
- * count]
- *
- * This primitive allows the upper layer to initiate an association to a
- * 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.
- * [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
- * address from the returned destination addresses will be selected by
- * the local endpoint as default primary path for sending SCTP packets
- * to this peer.  The returned "destination transport addr list" can
- * be used by the ULP to change the default primary path or to force
- * 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
- * addresses of the peer endpoint with which the association is to be
- * established.
- *  [This is asoc->peer.active_path.]
- * o outbound stream count - the number of outbound streams the ULP
- * 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_chunk_t *repl;
-       sctp_bind_addr_t *bp;
-       sctp_scope_t scope;
-       int error;
-       int flags;
-
-       /* The comment below says that we enter COOKIE-WAIT AFTER
-        * sending the INIT, but that doesn't actually work in our
-        * implementation...
-        */
-       sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
-                       SCTP_STATE(SCTP_STATE_COOKIE_WAIT));
-
-       /* 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(GFP_ATOMIC);
-       if (!bp)
-               goto nomem;
-
-       /* 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)
-               flags |= SCTP_ADDR4_PEERSUPP;
-       if (asoc->peer.ipv6_address)
-               flags |= SCTP_ADDR6_PEERSUPP;
-       error = sctp_bind_addr_copy(bp, &ep->base.bind_addr, scope,
-                                   GFP_ATOMIC, flags);
-       if (error)
-               goto nomem;
-
-       /* FIXME: Either move address assignment out of this function
-        * or else move the association allocation/init into this function.
-        * The association structure is brand new before calling this
-        * 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);
-
-       /* 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). ...
-        */
-
-       repl = sctp_make_init(asoc, bp, GFP_ATOMIC);
-       if (!repl)
-               goto nomem;
-
-       /* Cast away the const modifier, as we want to just
-        * rerun it through as a sideffect.
-        */
-       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,
-                       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)
-               sctp_bind_addr_free(bp);
-
-       return SCTP_DISPOSITION_NOMEM;
-}
-
-/*
- * 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]
- *         [,unorder flag] [,no-bundle flag] [,payload protocol-id] )
- * -> result
- *
- * 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
- *    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.
- *
- *  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
- *    message).
- *
- *  o no-bundle flag - instructs SCTP not to bundle this user data with
- *    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
- *    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_chunk_t *chunk = arg;
-
-       sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(chunk));
-       return SCTP_DISPOSITION_CONSUME;
-}
-
-/*
- * Process the SHUTDOWN primitive.
- *
- * Section: 10.1:
- * C) Shutdown
- *
- * Format: SHUTDOWN(association id)
- * -> 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_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,
-                                                     arg, commands);
-       }
-       return disposition;
-}
-
-/*
- * Process the ABORT primitive.
- *
- * Section: 10.1:
- * C) Abort
- *
- * Format: Abort(association id [, cause code])
- * -> 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
- * attempting to abort 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:
- *
- *  o cause code - reason of the abort to be passed to the peer
- *
- * None.
- *
- * 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)
-{
-       /* 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.  */
-       abort = sctp_make_abort(asoc, NULL, 0);
-       if (!abort)
-               retval = SCTP_DISPOSITION_NOMEM;
-       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
-        * TCB.  This is a departure from our typical NOMEM handling.
-        */
-
-       /* Change to CLOSED state.  */
-       sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
-                       SCTP_STATE(SCTP_STATE_CLOSED));
-
-       /* Delete the established association.  */
-       sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
-       return retval;
-}
-
-/* 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_add_cmd_sf(commands, SCTP_CMD_REPORT_ERROR, SCTP_ERROR(-EINVAL));
-       return SCTP_DISPOSITION_CONSUME;
-}
-
-/* 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_add_cmd_sf(commands, SCTP_CMD_REPORT_ERROR,
-                       SCTP_ERROR(-ESHUTDOWN));
-       return SCTP_DISPOSITION_CONSUME;
-}
-
-/*
- * 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_TO(SCTP_EVENT_TIMEOUT_T1_INIT));
-
-       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;
-}
-
-/*
- * 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)
-{
-       /* 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_shutdown(ep, asoc, type, arg, commands);
-}
-
-/*
- * sctp_cookie_wait_prm_abort
- *
- * 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 an abort while in COOKIE_WAIT state.
- *
- * 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)
-{
-       /* Stop T1-init timer */
-       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_cookie_echoed_prm_abort
- *
- * Section: 4 Note: 3
- * Verification Tag:
- * Inputs
- * (endpoint, asoc)
- *
- * The RFC does not explcitly address this issue, but is the route through the
- * state table when someone issues an abort while in COOKIE_ECHOED state.
- *
- * 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)
-{
-       /* 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);
-}
-
-/*
- * 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_DEBUG_PRINTK("Primitive type %d is ignored.\n", type.primitive);
-       return SCTP_DISPOSITION_DISCARD;
-}
-
-/***************************************************************************
- * 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
- * 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_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.
-        */
-       reply = sctp_make_shutdown(asoc);
-       if (!reply)
-               goto nomem;
-
-       /* Set the transport for the SHUTDOWN chunk and the timeout for the
-        * 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,
-                       SCTP_TO(SCTP_EVENT_TIMEOUT_T2_SHUTDOWN));
-
-       if (asoc->autoclose)
-               sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
-                               SCTP_TO(SCTP_EVENT_TIMEOUT_AUTOCLOSE));
-
-       /* and enter the SHUTDOWN-SENT state.  */
-       sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
-                       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;
-}
-
-/*
- * Generate a SHUTDOWN ACK now that everything is SACK'd.
- *
- * 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)
-{
-       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 (!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,
-                               SCTP_TO(SCTP_EVENT_TIMEOUT_AUTOCLOSE));
-
-       /* 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));
-
-       return SCTP_DISPOSITION_CONSUME;
-
-nomem:
-       return SCTP_DISPOSITION_NOMEM;
-}
-
-/*
- * 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_DEBUG_PRINTK("The event other type %d is ignored\n",
-                         type.other);
-       return SCTP_DISPOSITION_DISCARD;
-}
-
-/************************************************************
- * 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;
-
-       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.  */
-       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;
-}
-
-/*
- * Generate delayed SACK on timeout
- *
- * Section: 6.2  Acknowledgement on Reception of DATA Chunks
- *
- * 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 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_add_cmd_sf(commands, SCTP_CMD_GEN_SACK, SCTP_FORCE());
-       return SCTP_DISPOSITION_CONSUME;
-}
-
-/*
- * sctp_sf_t1_timer_expire
- *
- * Section: 4 Note: 2
- * Verification Tag:
- * Inputs
- * (endpoint, asoc)
- *
- *  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_bind_addr_t *bp;
-       sctp_event_timeout_t timer = (sctp_event_timeout_t) arg;
-       int timeout;
-       int attempts;
-
-       timeout = asoc->timeouts[timer];
-       attempts = asoc->counters[SCTP_COUNTER_INIT_ERROR] + 1;
-       repl = NULL;
-
-       SCTP_DEBUG_PRINTK("Timer T1 expired.\n");
-
-       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;
-                       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;
-               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_TO(timer));
-       } else {
-               sctp_add_cmd_sf(commands, SCTP_CMD_INIT_FAILED, SCTP_NULL());
-               return SCTP_DISPOSITION_DELETE_TCB;
-       }
-
-       return SCTP_DISPOSITION_CONSUME;
-
-nomem:
-       return SCTP_DISPOSITION_NOMEM;
-}
-
-/* RFC2960 9.2 If the timer expires, the endpoint must re-send the SHUTDOWN
- * with the updated last sequential TSN received from its peer.
- *
- * An endpoint should limit the number of retransmissions of the
- * SHUTDOWN chunk to the protocol parameter 'Association.Max.Retrans'.
- * If this threshold is exceeded the endpoint should destroy the TCB and
- * MUST report the peer endpoint unreachable to the upper layer (and
- * thus the association enters the CLOSED state).  The reception of any
- * packet from its peer (i.e. as the peer sends all of its queued DATA
- * chunks) should clear the endpoint's retransmission count and restart
- * 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_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) {
-       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;
-
-       /* Do some failure management (Section 8.2). */
-       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.
-        */
-       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;
-}
-
-/* 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_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,
-                                                     arg, commands);
-       }
-       return disposition;
-}
-
-/*****************************************************************************
- * These are sa state functions which could apply to all types of events.
- ****************************************************************************/
-
-/*
- * This table entry is not implemented.
- *
- * 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)
-{
-       return SCTP_DISPOSITION_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)
-{
-       return SCTP_DISPOSITION_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.
- *
- * 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_DEBUG_PRINTK("Timer %d ignored.\n", type.chunk);
-       return SCTP_DISPOSITION_CONSUME;
-}
-
-/*
- * Discard the chunk.
- *
- * Section: 0.2, 5.2.3, 5.2.5, 5.2.6, 6.0, 8.4.6, 8.5.1c, 9.2
- * [Too numerous to mention...]
- * 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_DEBUG_PRINTK("Chunk %d is discarded\n", type.chunk);
-       return SCTP_DISPOSITION_DISCARD;
-}
-
-/********************************************************************
- * 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 *sack;
-       __u16 num_blocks;
-       __u16 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);
-
-       skb_pull(chunk->skb, (num_blocks + num_dup_tsns) * sizeof(__u32));
-       return sack;
-}
diff --git a/net/sctp/sctp_sm_statetable.c b/net/sctp/sctp_sm_statetable.c
deleted file mode 100644 (file)
index 8902949..0000000
+++ /dev/null
@@ -1,1149 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 International Business Machines, Corp.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001 Nokia, Inc.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_sm_statetable.c,v 1.21 2002/08/22 02:25:33 jgrimm Exp $
- * 
- * These are the state tables for the SCTP state machine.
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson          <karl@athena.chicago.il.us>
- *    Jon Grimm             <jgrimm@us.ibm.com>
- *    Hui Huang                    <hui.huang@nokia.com>
- *    Daisy Chang          <daisyc@us.ibm.com>
- *
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_sm_statetable.c,v 1.21 2002/08/22 02:25:33 jgrimm Exp $";
-
-#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,
-                            name: "sctp_sf_discard_chunk"};
-sctp_sm_table_entry_t bug = {fn: sctp_sf_bug, name: "sctp_sf_bug"};
-
-#define DO_LOOKUP(_max, _type, _table) \
-       if ((event_subtype._type > (_max))) { \
-               printk(KERN_WARNING \
-                      "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]);
-
-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,
-                         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;
-       };
-}
-
-#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 */ \
-       {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 */ \
-       {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"}, \
-} /* 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 */ \
-       {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"}, \
-} /* 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 */ \
-       {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 */ \
-       {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 */ \
-       /* This should not happen, but we are nice.  */ \
-       {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 */ \
-       {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"}, \
-} /* 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 */ \
-       {fn: sctp_sf_cookie_wait_abort, name: "sctp_sf_cookie_wait_abort"}, \
-       /* SCTP_STATE_COOKIE_ECHOED */ \
-       {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"}, \
-       /* SCTP_STATE_SHUTDOWN_PENDING */ \
-       {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"}, \
-       /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
-       {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"}, \
-} /* 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 */ \
-       {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, \
-        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"}, \
-} /* 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"}, \
-} /* 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"}, \
-} /* 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 */ \
-       {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 */ \
-       {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 */ \
-       {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 */ \
-       {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"}, \
-} /* 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 */ \
-       {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"}, \
-} /* TYPE_SCTP_SHUTDOWN_COMPLETE */
-
-/* The primary index for this table is the chunk type.
- * The secondary index for this table is the state.
- *
- * 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,
-}; /* 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"},
-       /* 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 (will be sctp_addip_do_asconf)"},
-
-       /* 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"},
-};     /* 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"},
-       /* 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 (will be sctp_addip_do_asconf_ack)"},
-
-       /* 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"},
-};     /* 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"}, \
-} /* 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"}, \
-} /* 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 */ \
-       {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_do_9_2_prm_shutdown"}, \
-       /* 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"}, \
-} /* 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 */ \
-       {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"}, \
-} /* 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 */ \
-       {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"}, \
-} /* 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 */
-
-#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"}, \
-} /* 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"}, \
-} /* 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"}, \
-} /* 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"}, \
-} /* 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"}, \
-} /* 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"}, \
-} /* 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"}, \
-} /* 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"}, \
-} /* 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"}, \
-} /* 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"}, \
-} /* 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,
-};
-
-#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 */ \
-       {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, \
-        name: "sctp_do_9_2_start_shutdown"}, \
-       /* SCTP_STATE_SHUTDOWN_SENT */ \
-       {fn: sctp_sf_ignore_other, name: "sctp_sf_ignore_other"}, \
-       /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
-       {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"}, \
-}
-
-#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"}, \
-}
-
-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,
-};
-
-#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 */ \
-       {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"}, \
-}
-
-#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 */ \
-       {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 */ \
-       {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"}, \
-}
-
-#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 */ \
-       {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 */ \
-       {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_SHUTDOWN_PENDING */ \
-       {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"}, \
-       /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
-       {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"}, \
-}
-
-#define TYPE_SCTP_EVENT_TIMEOUT_T4_RTO { \
-       /* 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"}, \
-       /* 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"}, \
-}
-
-#define TYPE_SCTP_EVENT_TIMEOUT_HEARTBEAT { \
-       /* 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_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-       /* SCTP_STATE_ESTABLISHED */ \
-       {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"}, \
-       /* 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"}, \
-}
-
-#define TYPE_SCTP_EVENT_TIMEOUT_SACK { \
-       /* 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_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-       /* SCTP_STATE_ESTABLISHED */ \
-       {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"}, \
-       /* SCTP_STATE_SHUTDOWN_SENT */ \
-       {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"}, \
-       /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
-       {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_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_timer_ignore, name: "sctp_sf_timer_ignore"}, \
-       /* SCTP_STATE_ESTABLISHED */ \
-       {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"}, \
-       /* 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_PMTU_RAISE { \
-       /* 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 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_T4_RTO,
-       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)
-{
-       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;
-}
diff --git a/net/sctp/sctp_socket.c b/net/sctp/sctp_socket.c
deleted file mode 100644 (file)
index a619bd4..0000000
+++ /dev/null
@@ -1,2705 +0,0 @@
-/* Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 International Business Machines, Corp.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001 Nokia, Inc.
- * Copyright (c) 2001 La Monte H.P. Yarroll
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_socket.c,v 1.64 2002/08/21 23:06:28 jgrimm Exp $
- * 
- * These functions interface with the sockets layer to implement the
- * SCTP Extensions for the Sockets API.
- * 
- * Note that the descriptions from the specification are USER level
- * functions--this file is the functions which populate the struct proto
- * for SCTP which is the BOTTOM of the sockets interface.
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Narasimha Budihal     <narsi@refcode.org>
- *    Karl Knutson          <karl@athena.chicago.il.us>
- *    Jon Grimm             <jgrimm@us.ibm.com>
- *    Xingang Guo           <xingang.guo@intel.com>
- *    Daisy Chang           <daisyc@us.ibm.com>
- *    Sridhar Samudrala     <samudrala@us.ibm.com>
- *    Inaky Perez-Gonzalez  <inaky.gonzalez@intel.com>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_socket.c,v 1.64 2002/08/21 23:06:28 jgrimm Exp $";
-
-#include <linux/config.h>
-#include <linux/types.h>
-#include <linux/compiler.h>
-#include <linux/kernel.h>
-#include <linux/wait.h>
-#include <linux/time.h>
-#include <linux/ip.h>
-#include <linux/fcntl.h>
-#include <linux/poll.h>
-#include <linux/init.h>
-
-#include <net/ip.h>
-#include <net/icmp.h>
-#include <net/route.h>
-#include <net/ipv6.h>
-#include <net/inet_common.h> 
-
-#include <linux/socket.h> /* for sa_family_t */
-#include <net/sock.h>
-#include <net/sctp/sctp.h> 
-
-/* Forward declarations for internal helper functions. */
-static void __sctp_write_space(sctp_association_t *asoc);
-static int sctp_writeable(struct sock *sk);
-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,
-                               int msg_len);
-static int sctp_wait_for_packet(struct sock * sk, int *err, long *timeo_p);
-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 *,
-                                   const sockaddr_storage_t *newaddr,
-                                   sockaddr_storage_t *saveaddr);
-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,
-                                             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]),
- *   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 retval = 0;
-
-       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 (!sctp_sk(sk)->ep->base.bind_addr.port)
-               retval = sctp_do_bind(sk, (sockaddr_storage_t *)uaddr, 
-                                     addr_len);
-       else
-               retval = -EINVAL;
-
-       sctp_release_sock(sk);
-
-       return retval;
-}
-
-static long sctp_get_port_local(struct sock *, unsigned short);
-
-/* 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;
-       sctp_bind_addr_t *bp = &ep->base.bind_addr;
-       unsigned short sa_family = newaddr->sa.sa_family;
-       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",
-                         sk, newaddr, addr_len);
-
-       /* FIXME: This function needs to handle v4-mapped-on-v6
-        * addresses!
-        */
-       if (PF_INET == sk->family) {
-               if (sa_family != AF_INET)
-                       return -EINVAL;
-       }       
-
-       /* Make a local copy of the new address.  */
-       tmpaddr = *newaddr;
-
-       switch (sa_family) {
-       case AF_INET:
-               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"?
-                */
-               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))
-                               return -EINVAL;
-
-                       /* 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);
-                       snum = &tmpaddr.v6.sin6_port;
-                       break;
-               )
-
-       default:
-               return -EINVAL;
-       };
-
-       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 (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;
-
-       /* 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_local() does duplicate address
-        * detection.
-        */
-       if ((ret = sctp_get_port_local(sk, *snum))) {
-               sctp_sk_addr_restore(sk, &saveaddr);
-               if (ret == (long) sk) {
-                       /* This endpoint has a conflicting address. */
-                       return -EINVAL;
-               } else {
-                       return -EADDRINUSE;
-               }
-       }
-
-       /* 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.  */
-       sctp_local_bh_disable();
-       sctp_write_lock(&ep->base.addr_lock);
-
-       /* 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 (!bp->port) {
-               bp->port = *snum;
-       }
-
-       sctp_write_unlock(&ep->base.addr_lock);
-       sctp_local_bh_enable();
-       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.
- *
- * NOTE: This could be integrated into sctp_setsockopt_bindx(),
- * but keeping it this way makes it easier if sometime sys_bindx is
- * added.
- */
-
-/* Unprotected by locks. Call only with socket lock sk->lock held! See
- * sctp_bindx() for a lock-protected call.
- */ 
-
-static int __sctp_bindx(struct sock *sk, struct sockaddr_storage *addrs, 
-                       int addrcnt, int flags)
-{
-       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;
-}
-
-/* BINDX with locks.
- *
- * NOTE: Currently unused at all ...
- */
-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);
-
-       return retval;
-}
-
-/* Add a list of addresses as bind addresses to local endpoint or
- * association.
- *
- * Basically run through each address specified in the addrs/addrcnt
- * array/length pair, determine if it is IPv6 or IPv4 and call
- * sctp_do_bind() on it.
- *
- * If any of them fails, then the operation will be reversed and the
- * ones that were added will be removed.
- *
- * Only __sctp_bindx() is supposed to call this function.
- */
-int sctp_bindx_add(struct sock *sk, struct sockaddr_storage *addrs, int addrcnt)
-{
-       int cnt;
-       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++) {
-               /* 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;
-               };
-
-               retval = sctp_do_bind(sk, (sockaddr_storage_t *)&addrs[cnt],
-                                     addr_len);
-
-err_bindx_add:
-               if (retval < 0) {
-                       /* Failed. Cleanup the ones that has been added */
-                       if (cnt > 0)
-                               sctp_bindx_rem(sk, addrs, cnt);
-                       return retval;
-               }
-        }
-
-       /* Notify the peer(s), assuming we have (an) association(s).
-        * FIXME: for UDP, we have a 1-1-many mapping amongst sk, ep and asoc,
-        *        so we don't have to do much work on locating associations.
-        *
-        * However, when the separation of ep and asoc kicks in, especially
-        * for TCP style connection, it becomes n-1-n mapping.  We will need
-        * to do more fine work.  Until then, hold my peace.
-        *                                                      --xguo
-        *
-        * Really, I don't think that will be a problem.  The bind()
-        * call on a socket will either know the endpoint
-        * (e.g. TCP-style listen()ing socket, or UDP-style socket),
-        * or exactly one association.  The former case is EXACTLY
-        * 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
-        * ADDIP code.
-        *                                                      --daisy
-        */
-
-#if CONFIG_IP_SCTP_ADDIP
-       /* Add these addresses to all associations on this endpoint.  */
-       if (retval >= 0) {
-               list_t *pos;
-               sctp_endpoint_t *ep;
-               sctp_association_t *asoc;
-               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_ADD_IP,
-                                              addrs, addrcnt);
-               }
-       }
-#endif
-
-       return retval;
-}
-
-/* Remove a list of addresses from bind addresses list.  Do not remove the
- * last address.
- *
- * Basically run through each address specified in the addrs/addrcnt
- * array/length pair, determine if it is IPv6 or IPv4 and call
- * sctp_del_bind() on it.
- *
- * If any of them fails, then the operation will be reversed and the
- * ones that were removed will be added back.
- *
- * At least one address has to be left; if only one address is
- * available, the operation will return -EBUSY.
- *
- * Only __sctp_bindx() is supposed to call this function.
- */
-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;
-
-       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
-                * 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;
-                * 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:
-                       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);
-
-               sctp_write_unlock(&ep->base.addr_lock);
-               sctp_local_bh_enable();
-
-err_bindx_rem:
-               if (retval < 0) {
-                       /* Failed. Add the ones that has been removed back */
-                       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
-        * ADDIP code.
-        *                                                      --daisy
-        */
-#if CONFIG_IP_SCTP_ADDIP
-       /* Remove these addresses from all associations on this endpoint.  */
-       if (retval >= 0) {
-               list_t *pos;
-               sctp_endpoint_t *ep;
-               sctp_association_t *asoc;
-
-               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);
-               }
-       }
-#endif
-       return retval;
-}
-
-/* Helper for tunneling sys_bindx() requests through sctp_setsockopt()
- *
- * Basically do nothing but copying the addresses from user to kernel
- * land and invoking sctp_bindx on the sk. This is used for tunneling
- * the sctp_bindx() [sys_bindx()] request through sctp_setsockopt()
- * from userspace.
- *
- * Note I don't use move_addr_to_kernel(): the reason is we would be
- * iterating over an array of struct sockaddr_storage passing always
- * what we know is a good size (sizeof (struct sock...)), so it is
- * pointless. Instead check the whole area for read access and copy
- * it.
- *
- * We don't use copy_from_user() for optimization: we first do the
- * sanity checks (buffer size -fast- and access check-healthy
- * pointer); if all of those succeed, then we can alloc the memory
- * (expensive operation) needed to copy the data to kernel. Then we do
- * the copying without checking the user space area
- * (__copy_from_user()).
- *
- * On exit there is no need to do sockfd_put(), sys_setsockopt() does
- * it.
- *
- * sk        The sk of the socket
- * addrs     The pointer to the addresses in user land
- * addrssize Size of the addrs buffer
- * op        Operation to perform (add or remove, see the flags of
- *           sctp_bindx)
- *
- * 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)
-{
-       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);
-
-       /* 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)))
-               return -EFAULT;
-
-       /* Alloc space for the address array in kernel memory.  */
-       kaddrs = (struct sockaddr_storage *) kmalloc(addrssize, GFP_KERNEL);
-       if (unlikely(NULL == kaddrs))
-               return -ENOMEM;
-
-       if (copy_from_user(kaddrs, addrs, addrssize)) {
-               kfree(kaddrs);
-               return -EFAULT;
-       }
-
-       addrcnt = addrssize / sizeof(struct sockaddr_storage);
-       err = __sctp_bindx(sk, kaddrs, addrcnt, op);   /* Do the work. */
-       kfree(kaddrs);
-
-       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.
- *
- * 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.
- */
-static void sctp_close(struct sock *sk, long timeout)
-{
-       sctp_endpoint_t *ep;
-       sctp_association_t *asoc;
-       list_t *pos, *temp;
-
-       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.  */
-       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.  */
-       skb_queue_purge(&sk->receive_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);
-       inet_sock_release(sk);
-
-       sctp_bh_unlock_sock(sk);
-       sctp_local_bh_enable();
-
-       sock_put(sk);
-
-       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.
- *
- *  ssize_t sendmsg(int socket, const struct msghdr *message,
- *                  int flags);
- *
- *  socket  - the socket descriptor of the endpoint.
- *  message - pointer to the msghdr structure which contains a single
- *            user message and possibly some ancillary data.
- *
- *            See Section 5 for complete description of the data
- *            structures.
- *
- *  flags   - flags sent or received with the user message, see Section
- *            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(). */
-
-static int sctp_msghdr_parse(const struct msghdr *, sctp_cmsgs_t *);
-
-static 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_transport_t *transport;
-       sctp_chunk_t *chunk = NULL;
-       sockaddr_storage_t to;
-       struct sockaddr *msg_name = NULL;
-       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_scope_t scope;
-       long timeo;
-       __u16 sinfo_flags = 0;
-
-       SCTP_DEBUG_PRINTK("sctp_sendmsg(sk: %p, msg: %p, "
-                         "size: %d)\n", sk, msg, size);
-
-       err = 0;
-       sp = sctp_sk(sk);
-       ep = sp->ep;
-
-       SCTP_DEBUG_PRINTK("Using endpoint: %s.\n", ep->debug_name);
-
-       /* Parse out the SCTP CMSGs.  */
-       err = sctp_msghdr_parse(msg, &cmsgs);
-
-       if (err) {
-               SCTP_DEBUG_PRINTK("msghdr parse err = %x\n", 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.
-        */
-       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);
-               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?  */
-       if (sinfo) {
-               sinfo_flags = sinfo->sinfo_flags;
-               associd = sinfo->sinfo_assoc_id;
-       }
-
-       SCTP_DEBUG_PRINTK("msg_len: %Zd, sinfo_flags: 0x%x\n",
-                         msg_len, sinfo_flags);
-
-       /* 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;
-               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 (msg_name) {
-               asoc = sctp_endpoint_lookup_assoc(ep, &to, &transport);
-       } else {
-               /* For a peeled-off socket, ignore any associd specified by
-                * the user with SNDRCVINFO.
-                */ 
-               if (SCTP_SOCKET_UDP_HIGH_BANDWIDTH == sp->type) {
-                       if (list_empty(&ep->asocs)) {
-                               err = -EINVAL;
-                               goto out_unlock;
-                       }
-                       asoc = list_entry(ep->asocs.next, sctp_association_t,
-                                         asocs);
-               } else if (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);
-               if (sinfo_flags & MSG_EOF) {
-                       SCTP_DEBUG_PRINTK("Shutting down association: %p\n",
-                                         asoc);
-                       sctp_primitive_SHUTDOWN(asoc, NULL);
-                       err = 0;
-                       goto out_unlock;
-               }
-               if (sinfo_flags & MSG_ABORT) {
-                       SCTP_DEBUG_PRINTK("Aborting association: %p\n",asoc);
-                       sctp_primitive_ABORT(asoc, NULL);
-                       err = 0;
-                       goto out_unlock;
-               }
-       }
-
-       /* 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.
-                */
-               if (sinfo) {
-                       if (!sinit ||
-                           (sinit && !sinit->sinit_num_ostreams)) {
-                               /* Check against the defaults. */
-                               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) {
-                                       err = -EINVAL;
-                                       goto out_unlock;
-                               }
-
-                               /* 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.
-                */
-               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 (!asoc) {
-                       err = -ENOMEM;
-                       goto out_unlock;
-               }
-
-               /* If the SCTP_INIT ancillary data is specified, set all
-                * the association init values accordingly.
-                */
-               if (sinit) {
-                       if (sinit->sinit_num_ostreams) {
-                               asoc->c.sinit_num_ostreams =
-                                       sinit->sinit_num_ostreams;
-                       }
-                       if (sinit->sinit_max_instreams) {
-                               if (sinit->sinit_max_instreams <=
-                                   SCTP_MAX_STREAM) {
-                                       asoc->c.sinit_max_instreams =
-                                               sinit->sinit_max_instreams;
-                               } else {
-                                       asoc->c.sinit_max_instreams =
-                                               SCTP_MAX_STREAM;
-                               }
-                       }
-                       if (sinit->sinit_max_attempts) {
-                               asoc->max_init_attempts
-                                       = sinit->sinit_max_attempts;
-                       }
-                       if (sinit->sinit_max_init_timeo) {
-                               asoc->max_init_timeo 
-                                       = sinit->sinit_max_init_timeo * HZ;
-                       }
-               }
-
-               /* Prime the peer's transport structures.  */
-               transport = sctp_assoc_add_peer(asoc, &to, GFP_KERNEL);
-       }
-
-       /* 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
-        * maximum size of data that can be sent in a single send call.
-        */
-       if (msg_len > sk->sndbuf) {
-               err = -EMSGSIZE;
-               goto out_free;
-       }
-
-       /* 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(__u16).
-        * The real fix is to fragment the message before creating the chunks.
-        */
-       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.
-        */
-       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;
-               }
-       } else {
-               /* If the user didn't specify SNDRCVINFO, make up one with
-                * some defaults.
-                */
-               default_sinfo.sinfo_stream  = asoc->defaults.stream;
-               default_sinfo.sinfo_ppid = asoc->defaults.ppid;
-               sinfo = &default_sinfo;
-       }
-
-       timeo = sock_sndtimeo(sk, msg->msg_flags & MSG_DONTWAIT);
-       if (!sctp_wspace(asoc)) {
-               err = sctp_wait_for_sndbuf(asoc, &timeo, msg_len);
-               if (err)
-                       goto out_free;
-       }
-
-       /* 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
-        * primary address override!  --jgrimm
-        */ 
-       /* If the user gave us an address, copy it in.  */
-       if (msg->msg_name) {
-               chunk->transport = sctp_assoc_lookup_paddr(asoc, &to);
-               if (!chunk->transport) {
-                       err = -EINVAL;
-                       goto out_free;
-               }
-       }
-#endif /* 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, (__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)
-                       goto out_free;
-               SCTP_DEBUG_PRINTK("We associated primitively.\n");
-       }
-
-       /* 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 (!err) {
-               err = msg_len;
-               goto out_unlock;
-       }
-
-out_free:
-       if (SCTP_STATE_CLOSED == asoc->state)
-               sctp_association_free(asoc);
-       if (chunk)
-               sctp_free_chunk(chunk);
-
-out_unlock:
-       sctp_release_sock(sk);
-
-out_nounlock:
-       return err;
-
-#if 0
-do_sock_err:
-       if (msg_len)
-               err = msg_len;
-       else
-               err = sock_error(sk);
-       goto out;
-
-do_interrupted:
-       if (msg_len)
-               err = msg_len;
-       goto out;
-#endif /* 0 */
-}
-
-/* API 3.1.3  recvmsg() - UDP Style Syntax
- *
- *  ssize_t recvmsg(int socket, struct msghdr *message,
- *                    int flags);
- *
- *  socket  - the socket descriptor of the endpoint.
- *  message - pointer to the msghdr structure which contains a single
- *            user message and possibly some ancillary data.
- *
- *            See Section 5 for complete description of the data
- *            structures.
- *
- *  flags   - flags sent or received with the user message, see Section
- *            5 for complete description of the flags.
- */
-static struct sk_buff * sctp_skb_recv_datagram(struct sock *, int, int, int *);
-
-static 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;
-
-       SCTP_DEBUG_PRINTK("sctp_recvmsg("
-                         "%s: %p, %s: %p, %s: %d, %s: %d, %s: "
-                         "0x%x, %s: %p)\n",
-                         "sk", sk,
-                         "msghdr", msg,
-                         "len", len,
-                         "knoblauch", noblock,
-                         "flags", flags,
-                         "addr_len", addr_len);
-
-       sctp_lock_sock(sk);
-       skb = sctp_skb_recv_datagram(sk, flags, noblock, &err);
-       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)
-                       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;
-
-       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,
-                                               addr_len, skb);
-       }
-
-       /* Check if we allow SCTP_SNDRCVINFO. */
-       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.  */
-       if (sk->protinfo.af_inet.cmsg_flags)
-               ip_cmsg_recv(msg, skb);
-#endif
-
-       err = copied;
-
-       /* FIXME: We need to support MSG_EOR correctly. */
-       msg->msg_flags |= MSG_EOR;
-
-out_free:
-       sctp_ulpevent_free(event); /* Free the skb. */
-out:
-       sctp_release_sock(sk);
-       return err;
-}
-
-static inline int sctp_setsockopt_disable_fragments(struct sock *sk, char *optval, int optlen)
-{
-       int val;
-
-       if (optlen < sizeof(int))
-               return -EINVAL;
-
-       if (get_user(val, (int *)optval))
-               return -EFAULT;
-
-       sctp_sk(sk)->disable_fragments = (val == 0) ? 0 : 1;
-
-       return 0;
-}
-
-static inline int sctp_setsockopt_set_events(struct sock *sk, char *optval, int optlen)
-{
-       if (optlen != sizeof(struct sctp_event_subscribe))
-               return -EINVAL;
-       if (copy_from_user(&sctp_sk(sk)->subscribe, optval, optlen))
-               return -EFAULT;
-       return 0;
-}
-
-static inline int sctp_setsockopt_autoclose(struct sock *sk, char *optval, int optlen)
-{
-       sctp_opt_t *sp = sctp_sk(sk);
-
-       if (optlen != sizeof(int))
-               return -EINVAL;
-       if (copy_from_user(&sp->autoclose, optval, optlen))
-               return -EFAULT;
-
-       sp->ep->timeouts[SCTP_EVENT_TIMEOUT_AUTOCLOSE] = sp->autoclose * HZ;
-       return 0;
-}
-
-/* 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);
- *   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.
- */
-static 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();
-       list_t *pos;
-       sctp_func_t *af;
-
-       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.
-        */
-       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,
-                                               optlen);
-                       if (retval < 0)
-                               goto out_nounlock;
-               }
-       }
-
-       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 *)
-                                              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 *)
-                                              optval, optlen, BINDX_REM_ADDR);
-               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;
-       };
-
-out_unlock:
-       sctp_release_sock(sk);
-
-out_nounlock:
-       return retval;
-}
-
-/* FIXME: Write comments. */
-static int sctp_connect(struct sock *sk, struct sockaddr *uaddr, int addr_len)
-{
-       return -EOPNOTSUPP; /* STUB */
-}
-
-/* FIXME: Write comments. */
-static int sctp_disconnect(struct sock *sk, int flags)
-{
-       return -EOPNOTSUPP; /* STUB */
-}
-
-/* FIXME: Write comments. */
-static struct sock *sctp_accept(struct sock *sk, int flags, int *err)
-{
-       int error = -EOPNOTSUPP;
-
-       *err = error; 
-       return NULL;
-}
-
-/* FIXME: Write Comments. */
-static int sctp_ioctl(struct sock *sk, int cmd, unsigned long arg)
-{
-       return -EOPNOTSUPP; /* STUB */
-}
-
-/* This is the function which gets called during socket creation to
- * initialized the SCTP-specific portion of the sock.
- * The sock structure should already be zero-filled memory.
- */
-static int sctp_init_sock(struct sock *sk)
-{
-       sctp_endpoint_t *ep;
-       sctp_protocol_t *proto;
-       sctp_opt_t *sp;
-
-       SCTP_DEBUG_PRINTK("sctp_init_sock(sk: %p)\n", sk);
-
-       proto = sctp_get_protocol();
-
-       /* 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 (!ep)
-               return -ENOMEM;
-
-       sp = sctp_sk(sk);
-
-       /* Initialize the SCTP per socket area.  */
-
-       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).
-        */
-       sp->default_stream = 0;
-       sp->default_ppid = 0;
-
-       /* Initialize default setup parameters. These parameters
-        * can be modified with the SCTP_INITMSG socket option or
-        * overridden by the SCTP_INIT CMSG.
-        */
-       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_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.
-        */
-       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.
-        * 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) {
-               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;
-
-       /* 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.  */
-       sp->nodelay           = 0;
-
-       /* 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.
-        */
-       sp->autoclose         = 0;
-
-       SCTP_DBG_OBJCNT_INC(sock);
-       return 0;
-}
-
-/* Cleanup any SCTP per socket resources.  */
-static int sctp_destroy_sock(struct sock *sk)
-{
-       sctp_endpoint_t *ep;
-
-       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;
-}
-
-/* FIXME: Comments needed.  */
-static void sctp_shutdown(struct sock *sk, int how)
-{
-       /* UDP-style sockets do not support shutdown. */
-       /* STUB */
-}
-
-static int sctp_getsockopt_sctp_status(struct sock *sk, int len, char *optval,
-                                      int *optlen)
-{
-       struct sctp_status status;
-       sctp_endpoint_t *ep;
-       sctp_association_t *assoc = NULL;
-       sctp_transport_t *transport;
-       sctp_assoc_t associd;
-       int retval = 0;
-
-       if (len != sizeof(status)) {
-               retval = -EINVAL;
-               goto out_nounlock;
-       }
-
-       if (copy_from_user(&status, optval, sizeof(status))) {
-               retval = -EFAULT;
-               goto out_nounlock;
-       }
-
-       sctp_lock_sock(sk);
-
-       associd = status.sstat_assoc_id;
-       if ((SCTP_SOCKET_UDP_HIGH_BANDWIDTH != sctp_sk(sk)->type) && associd) {
-               assoc = sctp_id2assoc(sk, associd);
-               if (!assoc) {
-                       retval = -EINVAL;
-                       goto out_unlock;
-               }
-       } else {
-               ep = sctp_sk(sk)->ep;
-               if (list_empty(&ep->asocs)) {
-                       retval = -EINVAL;
-                       goto out_unlock;
-               }
-
-               assoc = list_entry(ep->asocs.next, sctp_association_t, asocs);
-       }
-
-       transport = assoc->peer.primary_path;
-
-       status.sstat_assoc_id = sctp_assoc2id(assoc);
-       status.sstat_state = assoc->state;
-       status.sstat_rwnd =  assoc->peer.rwnd;
-       status.sstat_unackdata = assoc->unack_data;
-       status.sstat_penddata = assoc->peer.tsn_map.pending_data;
-       status.sstat_instrms = assoc->c.sinit_max_instreams;
-       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));
-       status.sstat_primary.spinfo_state = transport->state.active;
-       status.sstat_primary.spinfo_cwnd = transport->cwnd;
-       status.sstat_primary.spinfo_srtt = transport->srtt;
-       status.sstat_primary.spinfo_rto = transport->rto;
-       status.sstat_primary.spinfo_mtu = transport->pmtu;
-
-       if (put_user(len, optlen)) {
-               retval = -EFAULT;
-               goto out_unlock;
-       }
-
-       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)) {
-               retval = -EFAULT;
-               goto out_unlock;
-       }
-
-out_unlock:
-       sctp_release_sock(sk);
-
-out_nounlock:
-       return (retval);
-}
-
-static inline int sctp_getsockopt_disable_fragments(struct sock *sk, int len,
-                                                   char *optval, int *optlen)
-{
-       int val;
-
-       if (len < sizeof(int))
-               return -EINVAL;
-
-       len = sizeof(int);
-       val = (sctp_sk(sk)->disable_fragments == 1);
-       if (put_user(len, optlen))
-               return -EFAULT;
-       if (copy_to_user(optval, &val, len))
-               return -EFAULT;
-       return 0;
-}
-
-static inline int sctp_getsockopt_set_events(struct sock *sk, int len, char *optval, int *optlen)
-{
-       if (len != sizeof(struct sctp_event_subscribe))
-               return -EINVAL;
-       if (copy_to_user(optval, &sctp_sk(sk)->subscribe, len))
-               return -EFAULT;
-       return 0;
-}
-
-static inline int sctp_getsockopt_autoclose(struct sock *sk, int len, char *optval, int *optlen)
-{
-       if (len != sizeof(int))
-               return -EINVAL;
-       if (copy_to_user(optval, &sctp_sk(sk)->autoclose, len))
-               return -EFAULT;
-       return 0;
-}
-
-/* Helper routine to branch off an association to a new socket.  */
-static int sctp_do_peeloff(sctp_association_t *assoc, struct socket **newsock)
-{
-       struct sock *oldsk = assoc->base.sk;
-       struct sock *newsk;
-       struct socket *tmpsock;
-       sctp_endpoint_t *newep;
-       sctp_opt_t *oldsp = sctp_sk(oldsk);
-       sctp_opt_t *newsp;
-       int err = 0;
-
-       /* An association cannot be branched off from an already peeled-off
-        * socket.
-        */
-       if (SCTP_SOCKET_UDP_HIGH_BANDWIDTH == sctp_sk(oldsk)->type)
-               return -EINVAL;
-
-       /* Create a new socket.  */
-       err = sock_create(PF_INET, SOCK_SEQPACKET, IPPROTO_SCTP, &tmpsock);
-       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.
-        */
-       newsk->sndbuf = oldsk->sndbuf;
-       newsk->rcvbuf = oldsk->rcvbuf;
-       *newsp = *oldsp;
-
-       /* Restore the ep value that was overwritten with the above structure
-        * copy.
-        */
-       newsp->ep = newep;
-
-       /* Set the type of socket to indicate that it is peeled off from the
-        * original socket.
-        */
-       newsp->type = SCTP_SOCKET_UDP_HIGH_BANDWIDTH;
-
-       /* Migrate the association to the new socket.  */
-       sctp_assoc_migrate(assoc, newsk);
-
-       *newsock = tmpsock;
-
-       return err;
-}
-
-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))
-               return -EINVAL;
-       if (copy_from_user(&peeloff, optval, len))
-               return -EFAULT;
-       assoc = sctp_id2assoc(sk, peeloff.associd);
-       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;
-
-       /* 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;
-       }
-
-       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.  */
-       peeloff.sd = sd;
-       if (copy_to_user(optval, &peeloff, len))
-               return -EFAULT;
-
-       return 0;
-}
-
-static 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;
-       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.
-        */
-       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,
-                                               optval, optlen);
-                       if (retval < 0)
-                               return retval;
-               }
-       }
-
-       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;
-}
-
-static void sctp_hash(struct sock *sk)
-{
-       /* STUB */
-}
-
-static void sctp_unhash(struct sock *sk)
-{
-       /* STUB */
-}
-
-/* Check if port is acceptable.  Possibly find first available port.
- *
- * The port hash table (contained in the 'global' SCTP protocol storage
- * returned by sctp_protocol_t * sctp_get_protocol()). The hash
- * table is an array of 4096 lists (sctp_bind_hashbucket_t). Each
- * list (the list number is the port number hashed out, so as you
- * would expect from a hash function, all the ports in a given list have
- * such a number that hashes out to the same list number; you were
- * expecting that, right?); so each list has a set of ports, with a
- * link to the socket (struct sock) that uses it, the port number and
- * a fastreuse flag (FIXME: NPI ipg).
- */
-static long sctp_get_port_local(struct sock *sk, unsigned short snum)
-{
-       sctp_bind_hashbucket_t *head; /* hash list */
-       sctp_bind_bucket_t *pp; /* hash list port iterator */
-       sctp_protocol_t *sctp = sctp_get_protocol();
-       int ret;
-
-       SCTP_DEBUG_PRINTK("sctp_get_port() begins, snum=%d\n", snum);
-
-       sctp_local_bh_disable();
-
-       if (snum == 0) {
-               /* Search for an available port.
-                *
-                * 'sctp->port_rover' was the last port assigned, so
-                * we start to search from 'sctp->port_rover +
-                * 1'. What we do is first check if port 'rover' is
-                * 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 {
-                       rover++;
-                       if ((rover < low) || (rover > high))
-                               rover = low;
-                       index = sctp_phashfn(rover);
-                       head = &sctp->port_hashtable[index];
-                       sctp_spin_lock(&head->lock);
-                       for (pp = head->chain; pp; pp = pp->next)
-                               if (pp->port == rover)
-                                       goto next;
-                       break;
-               next:
-                       sctp_spin_unlock(&head->lock);
-               } while (--remaining > 0);
-               sctp->port_rover = rover;
-               sctp_spin_unlock(&sctp->port_alloc_lock);
-
-               /* Exhausted local port range during search? */
-               ret = 1;
-               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
-                * mutex.
-                */
-               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
-                * to the port number (snum) - we detect that with the
-                * port iterator, pp being NULL.
-                */
-               head = &sctp->port_hashtable[sctp_phashfn(snum)];
-               sctp_spin_lock(&head->lock);
-               for (pp = head->chain; pp; pp = pp->next) {
-                       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.
-                */
-               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)
-                       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;
-                       )
-                       break;
-
-               default:
-                       break;
-               };
-
-               /* Run through the list of sockets bound to the port
-                * (pp->port) [via the pointers bind_next and
-                * bind_pprev in the struct sock *sk2 (pp->sk)]. On each one,
-                * we get the endpoint they describe and run through
-                * the endpoint's list of IP (v4 or v6) addresses,
-                * 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.
-                */
-               for( ; sk2 != NULL; sk2 = sk2->bind_next) {
-                       sctp_endpoint_t *ep2;
-                       ep2 = sctp_sk(sk2)->ep;
-
-                       if (!sk_reuse || !sk2->reuse) {
-                               if (sctp_bind_addr_has_addr(
-                                       &ep2->base.bind_addr, &tmpaddr)) {
-                                       goto found;
-                               }
-                       }
-               }
-
-       found:
-               /* If we found a conflict, fail.  */
-               if (sk2 != NULL) {
-                       ret = (long) sk2;
-                       goto fail_unlock;
-               }
-               SCTP_DEBUG_PRINTK("sctp_get_port(): Found a match\n");
-       }
-
-       /* If there was a hash table miss, create a new port.  */
-       ret = 1;
-
-       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-).
-        */
-       if (pp->sk == NULL) {
-               pp->fastreuse = sk->reuse? 1 : 0;
-       } else if (pp->fastreuse && sk->reuse == 0) {
-               pp->fastreuse = 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).
-        */
-success:
-       inet_sk(sk)->num = snum;
-       if (sk->prev == 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;
-       }
-       ret = 0;
-
-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;
-}
-
-static int sctp_get_port(struct sock *sk, unsigned short snum)
-{
-       long ret = sctp_get_port_local(sk, snum);
-
-       return (ret ? 1 : 0);
-}
-
-/*
- * 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.
- */
-static 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;
-
-       /*
-        * 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.
-        *
-        * This is not currently spelled out in the SCTP sockets
-        * extensions draft, but follows the practice as seen in TCP
-        * sockets.
-        */
-       if (!ep->base.bind_addr.port) {
-               if (sctp_autobind(sk))
-                       return -EAGAIN;
-       }
-       sk->state = SCTP_SS_LISTENING;
-       sctp_hash_endpoint(ep);
-       return 0;
-}
-
-/*
- *  Move a socket to LISTENING state.
- */
-int sctp_inet_listen(struct socket *sock, int backlog)
-{
-       struct sock *sk = sock->sk;
-       int err;
-
-       sctp_lock_sock(sk);
-
-       err = -EINVAL;
-       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;
-}
-
-/*
- * 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.
- */
-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))
-               mask |= POLLERR;
-       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))
-               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.
-        */
-       if (SCTP_SOCKET_UDP != sctp_sk(sk)->type) {
-               /* The association is going away.  */
-               if (SCTP_SS_DISCONNECTING == sk->state)
-                       mask |= POLLHUP;
-               /* The association is either gone or not ready.  */
-               if (SCTP_SS_CLOSED == sk->state)
-                       return mask;
-       }
-
-       /* 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
-                * signal.  tcp_poll() has a race breaker for this race
-                * condition.  Based on their implementation, we put
-                * in the following code to cover it as well.
-                */
-               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)
-{
-       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) {
-               pp->port = snum;
-               pp->fastreuse = 0;
-               pp->sk = NULL;
-               if ((pp->next = head->chain) != 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;
-}
-
-/* FIXME: Commments! */
-static __inline__ void __sctp_put_port(struct sock *sk)
-{
-       sctp_protocol_t *sctp_proto = sctp_get_protocol();
-       sctp_bind_hashbucket_t *head =
-               &sctp_proto->port_hashtable[sctp_phashfn(inet_sk(sk)->num)];
-       sctp_bind_bucket_t *pp;
-
-       sctp_spin_lock(&head->lock);
-       pp = (sctp_bind_bucket_t *) sk->prev;
-       if (sk->bind_next)
-               sk->bind_next->bind_pprev = sk->bind_pprev;
-       *(sk->bind_pprev) = sk->bind_next;
-       sk->prev = NULL;
-       inet_sk(sk)->num = 0;
-       if (pp->sk) {
-               if (pp->next)
-                       pp->next->pprev = pp->pprev;
-               *(pp->pprev) = pp->next;
-               kfree(pp);
-       }
-       sctp_spin_unlock(&head->lock);
-}
-
-void sctp_put_port(struct sock *sk)
-{
-       sctp_local_bh_disable();
-       __sctp_put_port(sk);
-       sctp_local_bh_enable();
-}
-
-/*
- * 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.
- */
-static int sctp_autobind(struct sock *sk)
-{
-       sockaddr_storage_t autoaddr;
-       int addr_len = 0;
-
-       memset(&autoaddr, 0, sizeof(sockaddr_storage_t));
-
-       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);
-               addr_len = sizeof(struct sockaddr_in);
-               break;
-
-       case PF_INET6:
-               SCTP_V6(
-                       /* 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.  */
-               break;
-       };
-
-       return sctp_do_bind(sk, &autoaddr, addr_len);
-}
-
-/* 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
- * 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
- * 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.
- *
- *   |<--------------------------- msg_controllen -------------------------->|
- *   |                                                                       |
- *
- *   |<----- ancillary data object ----->|<----- ancillary data object ----->|
- *
- *   |<---------- CMSG_SPACE() --------->|<---------- CMSG_SPACE() --------->|
- *   |                                   |                                   |
- *
- *   |<---------- cmsg_len ---------->|  |<--------- cmsg_len ----------->|  |
- *
- *   |<--------- CMSG_LEN() --------->|  |<-------- CMSG_LEN() ---------->|  |
- *   |                                |  |                                |  |
- *
- *   +-----+-----+-----+--+-----------+--+-----+-----+-----+--+-----------+--+
- *   |cmsg_|cmsg_|cmsg_|XX|           |XX|cmsg_|cmsg_|cmsg_|XX|           |XX|
- *
- *   |len  |level|type |XX|cmsg_data[]|XX|len  |level|type |XX|cmsg_data[]|XX|
- *
- *   +-----+-----+-----+--+-----------+--+-----+-----+-----+--+-----------+--+
- *    ^
- *    |
- *
- * msg_control
- * points here
- */
-static int sctp_msghdr_parse(const struct msghdr *msg, sctp_cmsgs_t *cmsgs)
-{
-       struct cmsghdr *cmsg;
-
-       for (cmsg = CMSG_FIRSTHDR(msg);
-            cmsg != NULL;
-            cmsg = CMSG_NXTHDR((struct msghdr*)msg, cmsg)) {
-               /* 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)
-                       continue;
-
-               /* Strictly check lengths following example in SCM code.  */
-               switch (cmsg->cmsg_type) {
-               case SCTP_INIT:
-                       /* 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
-                        * structure.  This structure is not used for
-                        * recvmsg().
-                        *
-                        * cmsg_level    cmsg_type      cmsg_data[]
-                        * ------------  ------------   ----------------------
-                        * IPPROTO_SCTP  SCTP_INIT      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)
-                        * 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().
-                        *
-                        * cmsg_level    cmsg_type      cmsg_data[]
-                        * ------------  ------------   ----------------------
-                        * IPPROTO_SCTP  SCTP_SNDRCV    struct sctp_sndrcvinfo
-                        */
-                       if (cmsg->cmsg_len !=
-                           CMSG_LEN(sizeof(struct sctp_sndrcvinfo)))
-                               return -EINVAL;
-
-                       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))
-                               return -EINVAL;
-                       break;
-
-               default:
-                       return -EINVAL;
-               };
-       }
-       return 0;
-}
-
-/* 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) {
-       case AF_INET:
-               saveaddr->v4.sin_addr.s_addr = inet->rcv_saddr;
-               inet->rcv_saddr = inet->saddr = newaddr->v4.sin_addr.s_addr;
-               break;
-
-       case AF_INET6:
-               SCTP_V6({
-                       struct ipv6_pinfo *np = inet6_sk(sk);
-
-                       saveaddr->v6.sin6_addr = np->rcv_saddr;
-                       np->rcv_saddr = np->saddr = newaddr->v6.sin6_addr;
-                       break;
-               })
-
-       default:
-               break;
-       };
-}
-
-/* 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) {
-       case AF_INET:
-               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;
-       };
-}
-
-/*
- * 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)
-{
-       int error;
-       DECLARE_WAITQUEUE(wait, current);
-
-       __set_current_state(TASK_INTERRUPTIBLE);
-       add_wait_queue_exclusive(sk->sleep, &wait);
-
-       /* Socket errors? */
-       error = sock_error(sk);
-       if (error)
-               goto out;
-
-       if (!skb_queue_empty(&sk->receive_queue))
-               goto ready;
-
-       /* Socket shut down?  */
-       if (sk->shutdown & RCV_SHUTDOWN)
-               goto out;
-
-       /* 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))
-               goto out;
-
-       /* Handle signals.  */
-       if (signal_pending(current))
-               goto interrupted;
-
-       /* 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.
-        */
-       sctp_release_sock(sk);
-       *timeo_p = schedule_timeout(*timeo_p);
-       sctp_lock_sock(sk);
-
-ready:
-       remove_wait_queue(sk->sleep, &wait);
-       __set_current_state(TASK_RUNNING);
-       return 0;
-
-interrupted:
-       error = sock_intr_errno(*timeo_p);
-
-out:
-       remove_wait_queue(sk->sleep, &wait);
-       __set_current_state(TASK_RUNNING);
-       *err = error;
-       return error;
-}
-
-/* Receive a datagram.
- * Note: This is pretty much the same routine as in core/datagram.c
- * with a few changes to make lksctp work.
- */
-static 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()  */
-       error = sock_error(sk);
-       if (error)
-               goto no_packet;
-
-       timeo = sock_rcvtimeo(sk, noblock);
-
-       SCTP_DEBUG_PRINTK("Timeout: timeo: %ld, MAX: %ld.\n",
-                         timeo, MAX_SCHEDULE_TIMEOUT);
-
-       do {
-               /* 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,
-                                              cpu_flags);
-                       skb = skb_peek(&sk->receive_queue);
-                       if (skb)
-                               atomic_inc(&skb->users);
-                       sctp_spin_unlock_irqrestore(&sk->receive_queue.lock,
-                                                   cpu_flags);
-               } else {
-                       skb = skb_dequeue(&sk->receive_queue);
-               }
-
-               if (skb)
-                       return skb;
-
-               /* 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;
-}
-
-/* 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.  */
-       switch (sk->family) {
-       case PF_INET:
-               sin = (struct sockaddr_in *)msgname;
-               if (addr_len)
-                       *addr_len = sizeof(struct sockaddr_in);
-               sin->sin_family = AF_INET;
-               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
-                        * for an example. --jgrimm
-                        */
-               );
-               break;
-
-       default: /* Should not get here. */
-               break;
-       };
-}
-
-static inline int sctp_sendmsg_verify_name(struct sock *sk, struct msghdr *msg)
-{
-       sockaddr_storage_t *sa;
-
-       if (msg->msg_namelen < sizeof (struct sockaddr))
-               return -EINVAL;
-
-       sa = (sockaddr_storage_t *) msg->msg_name;
-       switch (sa->sa.sa_family) {
-       case AF_INET:
-               if (msg->msg_namelen < sizeof(struct sockaddr_in))
-                       return -EINVAL;
-               break;
-
-       case AF_INET6:
-               if (PF_INET == sk->family)
-                       return -EINVAL;
-               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))
-               return -EINVAL;
-
-       return 0;
-}
-
-/* Get the sndbuf space available at the time on the association.  */
-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)
-               amt = 0;
-       return amt;
-}
-
-/* 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.
- *
- * 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.
- */
-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.  */
-       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;
-
-       asoc->sndbuf_used += SCTP_DATA_SNDSIZE(chunk);
-       sk->wmem_queued += SCTP_DATA_SNDSIZE(chunk);
-}
-
-/* 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)
-{
-       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;
-       sk = asoc->base.sk;
-       asoc->sndbuf_used -= SCTP_DATA_SNDSIZE(chunk);
-       sk->wmem_queued -= SCTP_DATA_SNDSIZE(chunk);
-       __sctp_write_space(asoc);
-
-       sctp_association_put(asoc);
-}
-
-/* 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",
-                         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)
-                       goto do_nonblock;
-               if (sk->err || asoc->state >= SCTP_STATE_SHUTDOWN_PENDING ||
-                   asoc->base.dead)
-                       goto do_error;
-               if (signal_pending(current))
-                       goto do_interrupted;
-               if (msg_len <= sctp_wspace(asoc))
-                       break;
-
-               /* 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.  */
-       sctp_association_put(asoc);
-
-       __set_current_state(TASK_RUNNING);
-       return err;
-
-do_error:
-       err = -EPIPE;
-       goto out;
-
-do_interrupted:
-       err = sock_intr_errno(*timeo_p);
-       goto out;
-
-do_nonblock:
-       err = -EAGAIN;
-       goto out;
-}
-
-/* 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))
-                       wake_up_interruptible(&asoc->wait);
-
-               if (sctp_writeable(sk)) {
-                       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.
-                        */
-                       if (sock->fasync_list &&
-                           !(sk->shutdown & SEND_SHUTDOWN))
-                               sock_wake_async(sock, 2, POLL_OUT);
-               }
-       }
-}
-
-/* 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;
-
-       /* Wake up the tasks in each wait queue.  */
-       list_for_each(pos, &((sctp_sk(sk))->ep->asocs)) {
-               asoc = list_entry(pos, sctp_association_t, asocs);
-               __sctp_write_space(asoc);
-       }
-}
-
-/* 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"
- * 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
- * UDP-style sockets or TCP-style sockets, this code should work.
- *  - Daisy
- */
-static int sctp_writeable(struct sock *sk)
-{
-       int amt = 0;
-
-       amt = sk->sndbuf - sk->wmem_queued;
-       if (amt < 0)
-               amt = 0;
-       return amt;
-}
-
-/* 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,
-};
diff --git a/net/sctp/sctp_sysctl.c b/net/sctp/sctp_sysctl.c
deleted file mode 100644 (file)
index 951e667..0000000
+++ /dev/null
@@ -1,107 +0,0 @@
-/* SCTP kernel reference Implementation 
- * Copyright (c) 2002 International Business Machines Corp.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_sysctl.c,v 1.2 2002/07/12 14:50:25 jgrimm Exp $
- * 
- * Sysctl related interfaces for SCTP. 
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    Mingqin Liu           <liuming@us.ibm.com>
- *    Jon Grimm             <jgrimm@us.ibm.com>
- *
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_sysctl.c,v 1.2 2002/07/12 14:50:25 jgrimm Exp $";
-
-#include <net/sctp/sctp_structs.h>
-#include <linux/sysctl.h>
-
-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 }
-};
-
-static ctl_table sctp_net_table[] = {
-       { 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 }
-};
-
-static struct ctl_table_header * sctp_sysctl_header;
-
-/* Sysctl registration.  */
-void sctp_sysctl_register(void)
-{
-       sctp_sysctl_header = register_sysctl_table(sctp_root_table, 0);
-}
-
-/* Sysctl deregistration.  */
-void sctp_sysctl_unregister(void)
-{
-       unregister_sysctl_table(sctp_sysctl_header);
-}
diff --git a/net/sctp/sctp_transport.c b/net/sctp/sctp_transport.c
deleted file mode 100644 (file)
index fcfa7c5..0000000
+++ /dev/null
@@ -1,442 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 International Business Machines Corp.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001 La Monte H.P. Yarroll
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_transport.c,v 1.11 2002/06/20 05:57:01 samudrala Exp $
- * 
- * This module provides the abstraction for an SCTP tranport representing
- * a remote transport address.  For local transport addresses, we just use 
- * sockaddr_storage_t.
- *
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Karl Knutson          <karl@athena.chicago.il.us>
- *    Jon Grimm             <jgrimm@us.ibm.com>
- *    Xingang Guo           <xingang.guo@intel.com>
- *    Hui Huang             <hui.huang@nokia.com>
- *    Sridhar Samudrala            <sri@us.ibm.com>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_transport.c,v 1.11 2002/06/20 05:57:01 samudrala Exp $";
-
-#include <linux/types.h>
-#include <net/sctp/sctp.h>
-
-/* 1st Level Abstractions.  */
-
-/* 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 (!transport)
-               goto fail;
-
-       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;
-}
-
-/* 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();
-
-       /* 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);
-
-       /* 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->srtt = 0;
-       peer->rto_pending = 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->max_retrans = proto->max_retrans_path;
-       peer->error_threshold = 0;
-       peer->error_count = 0;
-
-       peer->debug_name = "unnamedtransport";
-
-       INIT_LIST_HEAD(&peer->transmitted);
-       INIT_LIST_HEAD(&peer->send_ready);
-       INIT_LIST_HEAD(&peer->transports);
-
-       /* Set up the retransmission timer.  */
-       init_timer(&peer->T3_rtx_timer);
-       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;
-
-       atomic_set(&peer->refcnt, 1);
-       peer->dead = 0;
-
-       peer->malloced = 0;
-       return peer;
-}
-
-/* This transport is no longer needed.  Free up if possible, or
- * delay until it last reference count.
- */
-void sctp_transport_free(sctp_transport_t *transport)
-{
-       transport->dead = 1;
-
-       /* Try to delete the heartbeat timer.  */
-       if (del_timer(&transport->hb_timer))
-               sctp_transport_put(transport);
-
-       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)
-{
-       SCTP_ASSERT(transport->dead, "Transport is not dead", return);
-
-       if (transport->asoc)
-               sctp_association_put(transport->asoc);
-
-       kfree(transport);
-       SCTP_DBG_OBJCNT_DEC(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
-        * 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))
-                       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))
-               sctp_transport_hold(transport);
-}
-
-/* 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,
-                             sctp_association_t *asoc)
-{
-       transport->asoc = asoc;
-       sctp_association_hold(asoc);
-}
-
-/* Hold a reference to a transport.  */
-void sctp_transport_hold(sctp_transport_t *transport)
-{
-       atomic_inc(&transport->refcnt);
-}
-
-/* Release a reference to a transport and clean up
- * if there are no more references.
- */
-void sctp_transport_put(sctp_transport_t *transport)
-{
-       if (atomic_dec_and_test(&transport->refcnt))
-               sctp_transport_destroy(transport);
-}
-
-/* Update transport's RTO based on the newly calculated RTT. */ 
-void sctp_transport_update_rto(sctp_transport_t *tp, __u32 rtt)
-{
-       sctp_protocol_t *proto = sctp_get_protocol();
-
-       /* Check for valid transport.  */
-       SCTP_ASSERT(tp, "NULL transport", return);
-
-       /* 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'|
-                * SRTT <- (1 - RTO.Alpha) * SRTT + RTO.Alpha * R'
-                */
-
-               /* Note:  The above algorithm has been rewritten to
-                * express rto_beta and rto_alpha as inverse powers
-                * of two.
-                * For example, assuming the default value of RTO.Alpha of
-                * 1/8, rto_alpha would be expressed as 3.
-                */
-               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)
-                       + (rtt >> proto->rto_alpha);
-       } else {
-               /* 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
-        * adjust RTTVAR <- G, where G is the CLOCK GRANULARITY.
-        */
-       if (tp->rttvar == 0)
-               tp->rttvar = SCTP_CLOCK_GRANULARITY;
-
-       /* 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.
-        */
-       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
-        * at least RTO.max seconds.
-        */
-       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
-        * new data chunk is sent.
-        */
-       tp->rto_pending = 0;
-
-       SCTP_DEBUG_PRINTK(__FUNCTION__ ": transport: %p, rtt: %d, srtt: %d "
-                         "rttvar: %d, rto: %d\n",
-                         tp, rtt, tp->srtt, tp->rttvar, tp->rto);
-}
-
-/* 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, __u32 sack_ctsn,
-                              __u32 bytes_acked)
-{
-       __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.
-        */
-       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
-                * 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.
-                */
-               if (bytes_acked > pmtu)
-                       cwnd += pmtu;
-               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,
-                                 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
-                * 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
-                * (partial_bytes_acked - cwnd).
-                */
-               pba += bytes_acked;
-               if (pba >= cwnd) {
-                       cwnd += pmtu;
-                       pba = ((cwnd < pba) ? (pba - cwnd) : 0);
-               }
-               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, flight_size, pba);
-       }
-
-       transport->cwnd = cwnd;
-       transport->partial_bytes_acked = pba;
-}
-
-/* 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) {
-       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
-                * perform slow start by:
-                *      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 
-                * were last sent, according to the formula described in 
-                * Section 7.2.3.
-                *
-                * 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
-                *      partial_bytes_acked = 0
-                */
-               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.
-                * 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
-                * 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
-                * 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,
-                                                 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.
-                */
-               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,
-                         transport->cwnd, transport->ssthresh);
-}
diff --git a/net/sctp/sctp_tsnmap.c b/net/sctp/sctp_tsnmap.c
deleted file mode 100644 (file)
index c4d1970..0000000
+++ /dev/null
@@ -1,386 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 International Business Machines, Corp.
- * Copyright (c) 2001 Intel Corp.
- * 
- * This file is part of the SCTP kernel reference Implementation
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_tsnmap.c,v 1.8 2002/07/26 22:52:32 jgrimm Exp $
- * 
- * These functions manipulate sctp tsn mapping array.
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Jon Grimm             <jgrimm@us.ibm.com>
- *    Karl Knutson          <karl@athena.chicago.il.us>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_tsnmap.c,v 1.8 2002/07/26 22:52:32 jgrimm Exp $";
-
-#include <linux/types.h>
-#include <net/sctp/sctp.h>
-#include <net/sctp/sctp_sm.h>
-
-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(__u16 len, __u32 initial_tsn, int priority)
-{
-       sctp_tsnmap_t *retval;
-
-       retval = kmalloc(sizeof(sctp_tsnmap_t) +
-                        sctp_tsnmap_storage_size(len),
-                        priority);
-       if (!retval)
-               goto fail;
-
-       if (!sctp_tsnmap_init(retval, len, initial_tsn))
-               goto fail_map;
-       retval->malloced = 1;
-       return retval;
-
-fail_map:
-       kfree(retval);
-
-fail:
-       return NULL;
-}
-
-/* Initialize a block of memory as a tsnmap.  */
-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;
-}
-
-/* Test the tracking state of this TSN.
- * Returns:
- *   0 if the TSN has not yet been seen
- *  >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, __u32 tsn)
-{
-       __s32 gap;
-       int dup;
-
-       /* 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)) {
-               dup = -1;
-               goto out;
-       }
-
-       /* Honk if we've already seen this TSN.
-        * We have three cases:
-        *      1. The TSN is ancient or belongs to a previous tsn_map.
-        *      2. The TSN is already marked in the tsn_map.
-        *      3. The TSN is already marked in the tsn_map_overflow.
-        */
-       if (gap < 0 ||
-           (gap < map->len && map->tsn_map[gap]) ||
-           (gap >= map->len && map->overflow_map[gap - map->len]))
-               dup = 1;
-       else
-               dup = 0;
-
-out:
-       return dup;
-}
-
-/* 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;
-}
-
-/* Mark this TSN as seen.  */
-void sctp_tsnmap_mark(sctp_tsnmap_t *map, __u32 tsn)
-{
-       __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))
-               return;
-       if (!TSN_lt(tsn, map->base_tsn + map->len + map->len))
-               return;
-
-       /* Bump the max.  */
-       if (TSN_lt(map->max_tsn_seen, tsn))
-               map->max_tsn_seen = tsn;
-
-       /* Assert: TSN is in range.  */
-       gap = tsn - map->base_tsn;
-
-       /* Mark the TSN as received.  */
-       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);
-}
-
-/* Retrieve the Cumulative TSN Ack Point. */
-__u32 sctp_tsnmap_get_ctsn(const sctp_tsnmap_t *map)
-{
-       return map->cumulative_tsn_ack_point;
-}
-
-/* Retrieve the highest TSN we've seen.  */
-__u32 sctp_tsnmap_get_max_tsn_seen(const sctp_tsnmap_t *map)
-{
-       return map->max_tsn_seen;
-}
-
-/* Dispose of a tsnmap.  */
-void sctp_tsnmap_free(sctp_tsnmap_t *map)
-{
-       if (map->malloced)
-               kfree(map);
-}
-
-/* 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)
-{
-       /* Only start looking one past the Cumulative TSN Ack Point.  */
-       iter->start = map->cumulative_tsn_ack_point + 1;
-}
-
-/* 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,
-                            __u16 *start, __u16 *end)
-{
-       int started, ended;
-       __u16 _start, _end, offset;
-
-       /* 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;
-               _sctp_tsnmap_find_gap_ack(map->tsn_map,
-                                         offset,
-                                         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)
-                       offset = 0;
-               else
-                       offset = iter->start - map->base_tsn - map->len;
-
-               /* Search the overflow map.  */
-               _sctp_tsnmap_find_gap_ack(map->overflow_map,
-                                         offset,
-                                         map->len,
-                                         map->len,
-                                         &started, &_start,
-                                         &ended, &_end);
-       }
-
-       /* The Gap Ack Block happens to end at the end of the
-        * overflow map.
-        */
-       if (started & !ended) {
-               ended++;
-               _end = map->len + map->len - 1;
-       }
-
-       /* If we found a Gap Ack Block, return the start and end and
-        * bump the iterator forward.
-        */
-       if (ended) {
-               /* 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;
-
-               *start = _start - gap;
-               *end = _end - gap;
-
-               /* Move the iterator forward.  */
-               iter->start = map->cumulative_tsn_ack_point + *end + 1;
-       }
-
-       return ended;
-}
-
-/********************************************************************
- * 2nd Level Abstractions
- ********************************************************************/
-
-/* This private helper function updates the tsnmap buffers and
- * the Cumulative TSN Ack Point.
- */
-static void _sctp_tsnmap_update(sctp_tsnmap_t *map)
-{
-       __u32 ctsn;
-
-       ctsn = map->cumulative_tsn_ack_point;
-       do {
-               ctsn++;
-               if (ctsn == map->overflow_tsn) {
-                       /* Now tsn_map must have been all '1's,
-                        * so we swap the map and check the overflow table
-                        */
-                       __u8 *tmp = map->tsn_map;
-                       memset(tmp, 0, map->len);
-                       map->tsn_map = map->overflow_map;
-                       map->overflow_map = tmp;
-
-                       /* Update the tsn_map boundaries.  */
-                       map->base_tsn += map->len;
-                       map->overflow_tsn += map->len;
-               }
-       } while (map->tsn_map[ctsn - map->base_tsn]);
-
-       map->cumulative_tsn_ack_point = ctsn - 1; /* Back up one. */
-       _sctp_tsnmap_update_pending_data(map);
-}
-
-static void _sctp_tsnmap_update_pending_data(sctp_tsnmap_t *map)
-{
-       __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))
-               goto out;
-
-       start = ((cum_tsn >= base_tsn) ? (cum_tsn - base_tsn + 1) : 0);
-       end = ((gap > map->len ) ? map->len : gap + 1);
-
-       for (i = start; i < end; i++) {
-               if (map->tsn_map[i])
-                       pending_data--;
-       } 
-
-       if (gap >= map->len) {
-               start = 0;
-               end = gap - map->len + 1;
-               for (i = start; i < end; i++) {
-                       if (map->overflow_map[i])
-                               pending_data--;
-               }
-       }
-
-out:
-       map->pending_data = pending_data;
-}
-
-/* This is a private helper for finding Gap Ack Blocks.  It searches a
- * single array for the start and end of a Gap Ack Block.
- *
- * The flags "started" and "ended" tell is if we found the beginning
- * or (respectively) the end of a Gap Ack Block.
- */
-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 start. */
-       if (!(*started)) {
-               for (; i < len; i++) {
-                       if (map[i]) {
-                               (*started)++;
-                               *start = base + i;
-                               break;
-                       }
-               }
-       }
-
-       /* 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]) {
-                               (*ended)++;
-                               *end = base + i - 1;
-                               break;
-                       }
-               }
-       }
-}
diff --git a/net/sctp/sctp_ulpevent.c b/net/sctp/sctp_ulpevent.c
deleted file mode 100644 (file)
index 5bbb91b..0000000
+++ /dev/null
@@ -1,843 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001 International Business Machines, Corp.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001 Nokia, Inc.
- * Copyright (c) 2001 La Monte H.P. Yarroll
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_ulpevent.c,v 1.16 2002/08/21 18:34:04 jgrimm Exp $
- * 
- * These functions manipulate an sctp event.   The sctp_ulpevent_t is used
- * to carry notifications and data to the ULP (sockets).  
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    Jon Grimm             <jgrimm@us.ibm.com>
- *    La Monte H.P. Yarroll <piggy@acm.org>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_ulpevent.c,v 1.16 2002/08/21 18:34:04 jgrimm Exp $";
-
-#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);
-
-/* 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 (!skb)
-               goto fail;
-
-       event = (sctp_ulpevent_t *) skb->cb;
-       event = sctp_ulpevent_init(event, skb, msg_flags);
-       if (!event)
-               goto fail_init;
-
-       event->malloced = 1;
-       return event;
-
-fail_init:
-       kfree_skb(event->parent);
-
-fail:
-       return NULL;
-}
-
-/* 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;
-}
-
-/* Dispose of an event.  */
-void sctp_ulpevent_free(sctp_ulpevent_t *event)
-{
-       if (event->malloced)
-               kfree_skb(event->parent);
-}
-
-/* Is this a MSG_NOTIFICATION?  */
-int sctp_ulpevent_is_notification(const sctp_ulpevent_t *event)
-{
-       return event->msg_flags & MSG_NOTIFICATION;
-}
-
-/* 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,
-                                                __u16 flags,
-                                                __u16 state,
-                                                __u16 error,
-                                                __u16 outbound,
-                                                __u16 inbound,
-                                                int priority)
-{
-       sctp_ulpevent_t *event;
-       struct sctp_assoc_change *sac;
-
-       event = sctp_ulpevent_new(sizeof(struct sctp_assoc_change),
-                                 MSG_NOTIFICATION,
-                                 priority);
-       if (!event)
-               goto fail;
-
-       sac = (struct sctp_assoc_change *)
-               skb_put(event->parent,
-                       sizeof(struct sctp_assoc_change));
-
-       /* 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
-        * 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.
-        */
-       sac->sac_state = state;
-
-       /* 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
-        * 5.3.1.1 SCTP_ASSOC_CHANGE
-        *
-        * 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
-        * 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
-        * 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
-        * 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;    
-}
-
-/* Create and initialize an SCTP_PEER_ADDR_CHANGE event.
- *
- * Socket Extensions for SCTP - draft-01
- * 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.
- */
-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 (!event)
-               goto fail;
-
-       spc = (struct sctp_paddr_change *)
-               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
-        * Section 5.3.1.2 SCTP_PEER_ADDR_CHANGE
-        *
-        * 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);
-
-       /* Sockets API Extensions for SCTP
-        * Section 5.3.1.2 SCTP_PEER_ADDR_CHANGE
-        *
-        * spc_flags: 16 bits (unsigned integer)
-        * Currently unused.
-        */
-       spc->spc_flags = 0;
-
-       /* Sockets API Extensions for SCTP
-        * Section 5.3.1.2 SCTP_PEER_ADDR_CHANGE
-        *
-        * spc_state:  32 bits (signed integer)
-        * 
-        * This field holds one of a number of values that communicate the
-        * event that happened to the address.
-        */
-       spc->spc_state = state;
-
-       /* Sockets API Extensions for SCTP
-        * Section 5.3.1.2 SCTP_PEER_ADDR_CHANGE
-        *
-        * spc_error:  32 bits (signed integer)
-        *
-        * If the state was reached due to any error condition (e.g.
-        * ADDRESS_UNREACHABLE) any relevant error information is available in
-        * this field.
-        */
-       spc->spc_error = error;
-
-       /* 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
-        * Section 5.3.1.2 SCTP_PEER_ADDR_CHANGE
-        *
-        * spc_aaddr: sizeof (struct sockaddr_storage)
-        *
-        * The affected address field, holds the remote peer's address that is
-        * encountering the change of state.
-        */
-       memcpy(&spc->spc_aaddr, aaddr, sizeof(struct sockaddr_storage));
-
-       return event;
-
-fail:
-       return NULL;
-}
-
-/* Create and initialize an SCTP_REMOTE_ERROR notification.
- *
- * Note: This assumes that the chunk->skb->data already points to the
- * operation error payload.
- *
- * Socket Extensions for SCTP - draft-01
- * 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
- * association. The entire error TLV as it appears on the wire is
- * 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,
-                                                __u16 flags,
-                                                int priority)
-{
-       sctp_ulpevent_t *event;
-       struct sctp_remote_error *sre;
-       struct sk_buff *skb;
-       sctp_errhdr_t *ch;
-       __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.  */
-       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,
-                             sizeof(struct sctp_remote_error), /* headroom */
-                             0,                                /* tailroom */
-                             priority);
-
-       /* Pull off the rest of the cause TLV from the chunk.  */
-       skb_pull(chunk->skb, elen);
-       if (!skb)
-               goto fail;
-
-       /* Embed the event fields inside the cloned skb.  */
-       event = (sctp_ulpevent_t *) skb->cb;
-       event = sctp_ulpevent_init(event,
-                                  skb,
-                                  MSG_NOTIFICATION);
-       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.  */
-       skb_trim(skb, sizeof(struct sctp_remote_error) + elen);
-
-       /* 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
-        *
-        * 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 (__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
-        *
-        * 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;
-
-       /* 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);
-
-       return event;
-
-fail:
-       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,
-                                               __u16 flags,
-                                               __u32 error,
-                                               int priority)
-{
-       sctp_ulpevent_t *event;
-       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,
-                             sizeof(struct sctp_send_failed), /* headroom */
-                             0,                               /* tailroom */
-                             priority);
-       if (!skb)
-               goto fail;
-
-       /* 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;
-       event = sctp_ulpevent_init(event, skb, MSG_NOTIFICATION);
-       if (!event)
-               goto fail;
-
-       /* 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
-        * 5.3.1.4 SCTP_SEND_FAILED
-        *
-        * ssf_type:
-        * It should be SCTP_SEND_FAILED.
-        */
-       ssf->ssf_type = SCTP_SEND_FAILED;
-
-       /* 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.
-        *
-        * 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.
-        */
-       ssf->ssf_flags = flags;
-
-       /* Socket Extensions for SCTP
-        * 5.3.1.4 SCTP_SEND_FAILED
-        *
-        * 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
-        * 5.3.1.4 SCTP_SEND_FAILED
-        *
-        * ssf_error: 16 bits (unsigned integer)
-        * This value represents the reason why the send failed, and if set,
-        * will be a SCTP protocol error code as defined in [SCTP] section
-        * 3.3.10.
-        */
-       ssf->ssf_error = error;
-
-       /* 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.
-        */
-       memcpy(&ssf->ssf_info,
-              &chunk->sinfo,
-              sizeof(struct sctp_sndrcvinfo));
-
-       /* Socket Extensions for SCTP
-        * 5.3.1.4 SCTP_SEND_FAILED
-        *
-        * ssf_assoc_id: sizeof (sctp_assoc_t)
-        * The association id field, sf_assoc_id, 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.
-        */
-       ssf->ssf_assoc_id = sctp_assoc2id(asoc);
-
-       return event;
-
-fail:
-       return NULL;
-}
-
-/* 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,
-                                                  __u16 flags,
-                                                  int priority)
-{
-       sctp_ulpevent_t *event;
-       struct sctp_shutdown_event *sse;
-
-       event = sctp_ulpevent_new(sizeof(struct sctp_assoc_change),
-                                 MSG_NOTIFICATION,
-                                 priority);
-       if (!event)
-               goto fail;
-
-       sse = (struct sctp_shutdown_event *)
-               skb_put(event->parent,
-                       sizeof(struct sctp_shutdown_event));
-
-       /* 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
-        * 5.3.1.5 SCTP_SHUTDOWN_EVENT
-        *
-        * sse_flags: 16 bits (unsigned integer)
-        * Currently unused.
-        */
-       sse->sse_flags = 0;
-
-       /* Socket Extensions for SCTP
-        * 5.3.1.5 SCTP_SHUTDOWN_EVENT
-        *
-        * 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
-        * 5.3.1.5 SCTP_SHUTDOWN_EVENT
-        *
-        * sse_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.
-        */
-       sse->sse_assoc_id = sctp_assoc2id(asoc);
-
-       return 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 *event;
-       struct sctp_sndrcvinfo *info;
-       struct sk_buff *skb;
-       size_t padding, len;
-
-       /* Clone the original skb, sharing the data.  */
-       skb = skb_clone(chunk->skb, priority);
-       if (!skb)
-               goto fail;
-               
-       /* 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
-        *
-        * 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
-        * multiple of 4 bytes, the sender MUST 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.  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.  */
-       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); 
-       if (!event)
-               goto fail_init;
-
-       event->malloced = 1;
-
-       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);
-
-       /* Sockets API Extensions for SCTP
-        * Section 5.2.2 SCTP Header Information Structure (SCTP_SNDRCV)
-        *
-        * sinfo_ssn: 16 bits (unsigned integer)
-        *
-        * For recvmsg() this value contains the stream sequence number that
-        * the remote endpoint placed in the DATA chunk.  For fragmented
-        * 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);
-
-        /* Sockets API Extensions for SCTP
-        * Section 5.2.2 SCTP Header Information Structure (SCTP_SNDRCV)
-        *
-        * sinfo_ppid: 32 bits (unsigned integer)
-        *
-        * In recvmsg() this value is
-        * the same information that was passed by the upper layer in the peer
-        * application.  Please note that byte order issues are NOT accounted
-        * for and this information is passed opaquely by the SCTP stack from
-        * one end to the other.
-        */
-       info->sinfo_ppid = ntohl(chunk->subh.data_hdr->ppid);
-
-       /* Sockets API Extensions for SCTP
-        * Section 5.2.2 SCTP Header Information Structure (SCTP_SNDRCV)
-        *
-        * 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.
-        */
-       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
-        * its difficult to fix everything at the same time.   Eventually,
-        * we should create and skb based chunk structure.   This structure
-        * storage can be converted to an event.  --jgrimm
-        */
-       event->chunk_flags = chunk->chunk_hdr->flags;
-
-       /* 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;
-
-       /* 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);
-
-       return event;
-
-fail_init:
-       kfree_skb(skb);
-
-fail:
-       return NULL;
-}
-
-/* Return the notification type, assuming this is a notification
- * event.
- */
-__u16 sctp_ulpevent_get_notification_type(const sctp_ulpevent_t *event)
-{
-       union sctp_notification *notification;
-
-       notification = (union sctp_notification *) event->parent->data;
-       return notification->h.sn_type;
-}
-
-/* Copy out the sndrcvinfo into a 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);
-       }
-}
-
-/* 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;
-
-       /* 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;
-       asoc = event->asoc;
-       if (asoc->rwnd_over) {
-               if (asoc->rwnd_over >= skb->len) {
-                       asoc->rwnd_over -= skb->len;
-               } else {
-                       asoc->rwnd += (skb->len - asoc->rwnd_over);
-                       asoc->rwnd_over = 0;
-               }
-       } else {
-               asoc->rwnd += skb->len;
-       }
-
-       SCTP_DEBUG_PRINTK("rwnd increased by %d to (%u, %u)\n",
-                         skb->len, asoc->rwnd, asoc->rwnd_over);
-
-       sctp_association_put(asoc);
-}
-
-/* Charge receive window for bytes recieved.  */
-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
-        * association.
-        */
-       sctp_association_hold(asoc);
-       skb->sk = asoc->base.sk;
-       event = (sctp_ulpevent_t *) skb->cb;
-       event->asoc = asoc;
-
-       skb->destructor = sctp_rcvmsg_rfree;
-
-       SCTP_ASSERT(asoc->rwnd, "rwnd zero", return);
-       SCTP_ASSERT(!asoc->rwnd_over, "rwnd_over not zero", return);
-       if (asoc->rwnd >= skb->len) {
-               asoc->rwnd -= skb->len;
-       } else {
-               asoc->rwnd_over = skb->len - asoc->rwnd;
-               asoc->rwnd = 0;
-       }
-
-       SCTP_DEBUG_PRINTK("rwnd decreased by %d to (%u, %u)\n",
-                         skb->len, asoc->rwnd, asoc->rwnd_over);
-}
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
diff --git a/net/sctp/sctp_ulpqueue.c b/net/sctp/sctp_ulpqueue.c
deleted file mode 100644 (file)
index ed1e700..0000000
+++ /dev/null
@@ -1,479 +0,0 @@
-/* SCTP kernel reference Implementation
- * Copyright (c) 1999-2000 Cisco, Inc.
- * Copyright (c) 1999-2001 Motorola, Inc.
- * Copyright (c) 2001-2002 International Business Machines, Corp.
- * Copyright (c) 2001 Intel Corp.
- * Copyright (c) 2001 Nokia, Inc.
- * Copyright (c) 2001 La Monte H.P. Yarroll
- * 
- * $Header: /cvsroot/lksctp/lksctp/sctp_cvs/net/sctp/sctp_ulpqueue.c,v 1.14 2002/08/21 18:34:04 jgrimm Exp $
- * 
- * This abstraction carries sctp events to the ULP (sockets).
- * 
- * The SCTP reference implementation is free software; 
- * you can redistribute it and/or modify it under the terms of 
- * the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- * 
- * The SCTP reference implementation is distributed in the hope that it 
- * will be useful, but WITHOUT ANY WARRANTY; without even the implied
- *                 ************************
- * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * 
- * You should have received a copy of the GNU General Public License
- * along with GNU CC; see the file COPYING.  If not, write to
- * the Free Software Foundation, 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.  
- * 
- * Please send any bug reports or fixes you make to the
- * email address(es):
- *    lksctp developers <lksctp-developers@lists.sourceforge.net>
- * 
- * Or submit a bug report through the following website:
- *    http://www.sf.net/projects/lksctp
- *
- * Written or modified by: 
- *    Jon Grimm             <jgrimm@us.ibm.com>
- *    La Monte H.P. Yarroll <piggy@acm.org>
- *    Sridhar Samudrala     <sri@us.ibm.com>
- * 
- * Any bugs reported given to us we will try to fix... any fixes shared will
- * be incorporated into the next SCTP release.
- */
-static char *cvs_id __attribute__ ((unused)) = "$Id: sctp_ulpqueue.c,v 1.14 2002/08/21 18:34:04 jgrimm Exp $";
-
-#include <linux/types.h>
-#include <linux/skbuff.h>
-#include <net/sock.h>
-#include <net/sctp/sctp_structs.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);
-
-/* 1st Level Abstractions */
-
-/* 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
-        * the future.
-        */
-       size = sizeof(sctp_ulpqueue_t)+sctp_ulpqueue_storage_size(inbound);
-       ulpq = kmalloc(size, priority);
-       if (!ulpq)
-               goto fail;
-       if (!sctp_ulpqueue_init(ulpq, asoc, inbound))
-               goto fail_init;
-       ulpq->malloced = 1;
-       return ulpq;
-
-fail_init:
-       kfree(ulpq);
-
-fail:
-       return NULL;
-}
-
-/* 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),
-              0x00);
-
-       ulpq->asoc = asoc;
-       spin_lock_init(&ulpq->lock);
-       skb_queue_head_init(&ulpq->reasm);
-       skb_queue_head_init(&ulpq->lobby);
-       ulpq->malloced = 0;
-
-       return 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);
-       }
-}
-
-/* Dispose of a ulpqueue.  */
-void sctp_ulpqueue_free(sctp_ulpqueue_t *ulpq)
-{
-       sctp_ulpqueue_flush(ulpq);
-       if (ulpq->malloced)
-               kfree(ulpq);
-}
-
-/* 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
-        * 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)
-               return -ENOMEM;
-
-       /* Do reassembly if needed.  */
-       event = sctp_ulpqueue_reasm(ulpq, event);
-
-       /* 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 (event)
-               sctp_ulpqueue_tail_event(ulpq, event);
-
-       return 0;
-}
-
-/* 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 = 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))
-               goto out_free;
-
-       /* 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 (event->parent->list)
-               sctp_skb_list_tail(event->parent->list, &sk->receive_queue);
-       else
-               skb_queue_tail(&sk->receive_queue, event->parent);
-
-       wake_up_interruptible(sk->sleep);
-       return 1;
-
-out_free:
-       if (event->parent->list)
-               skb_queue_purge(event->parent->list);
-       else
-               kfree_skb(event->parent);
-       return 0;
-}
-
-/* 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)
-{
-       struct sk_buff *pos, *tmp;
-       sctp_ulpevent_t *cevent;
-       __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.  */
-       sctp_skb_for_each(pos, &ulpq->reasm, tmp) {
-               cevent = (sctp_ulpevent_t *)pos->cb;
-               ctsn = cevent->sndrcvinfo.sinfo_tsn;
-
-               if (TSN_lt(tsn, ctsn))
-                       break;
-       }
-
-       /* 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
-               __skb_queue_tail(&ulpq->reasm, event->parent);
-
-       sctp_spin_unlock_irqrestore(&ulpq->reasm.lock, flags);
-}
-
-/* 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)
-{
-       struct sk_buff *pos;
-       sctp_ulpevent_t *event;
-       struct sk_buff *pnext;
-
-       pos = f_frag->next;
-
-       /* 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.  */
-       __skb_unlink(f_frag, f_frag->list);
-       do {
-               pnext = pos->next;
-
-               /* 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;
-
-               pos->next = pnext;
-               pos = pnext;
-       } while (1);
-
-       event = (sctp_ulpevent_t *) f_frag->cb;
-
-       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.
- */
-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;
-       __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
-        * 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.
-        * Walk through the queue sequentially and look for a sequence of
-        * 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.
-        */
-       sctp_skb_for_each(pos, &ulpq->reasm, tmp) {
-               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;
-                       break;
-
-               case SCTP_DATA_MIDDLE_FRAG:
-                       if ((first_frag) && (ctsn == next_tsn))
-                               next_tsn++;
-                       else
-                               first_frag = NULL;
-                       break;
-
-               case SCTP_DATA_LAST_FRAG:
-                       if ((first_frag) && (ctsn == next_tsn))
-                               retval = sctp_make_reassembled_event(
-                                               first_frag, pos);
-                       else
-                               first_frag = NULL;
-                       break;
-               };
-
-               /* We have the reassembled event. There is no need to look
-                * further.
-                */
-               if (retval)
-                       break;
-       }
-       sctp_spin_unlock_irqrestore(&ulpq->reasm.lock, flags);
-
-       return retval;
-}
-
-/* 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)
-{
-       sctp_ulpevent_t *retval = NULL;
-
-       /* 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
-        * at once.
-        */
-
-       /* 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;
-}
-
-/* Helper function to gather skbs that have possibly become
- * ordered by an an incoming chunk.
- */
-static inline void sctp_ulpqueue_retrieve_ordered(sctp_ulpqueue_t *ulpq,
-                                                 sctp_ulpevent_t *event)
-{
-       struct sk_buff *pos, *tmp;
-       sctp_ulpevent_t *cevent;
-       __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.  */
-       sctp_spin_lock_irqsave(&ulpq->lobby.lock, flags);
-       sctp_skb_for_each(pos, &ulpq->lobby, tmp) {
-               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;
-
-               if (cssn != ulpq->ssn[sid])
-                       break;
-
-               ulpq->ssn[sid]++;
-               __skb_unlink(pos, pos->list);
-
-               /* Attach all gathered skbs to the event.  */
-               __skb_queue_tail(event->parent->list, pos);
-       }
-       sctp_spin_unlock_irqrestore(&ulpq->lobby.lock, flags);
-}
-
-/* Helper function to store chunks needing ordering.  */
-static inline void sctp_ulpqueue_store_ordered(sctp_ulpqueue_t *ulpq,
-                                              sctp_ulpevent_t *event)
-{
-       struct sk_buff *pos, *tmp;
-       sctp_ulpevent_t *cevent;
-       __u16 sid, csid;
-       __u16 ssn, cssn;
-       unsigned long flags __attribute ((unused));
-
-       sid = event->sndrcvinfo.sinfo_stream;
-       ssn = event->sndrcvinfo.sinfo_ssn;
-
-       sctp_spin_lock_irqsave(&ulpq->lobby.lock, flags);
-
-       /* Find the right place in this list.  We store them by
-        * stream ID and then by SSN.
-        */
-       sctp_skb_for_each(pos, &ulpq->lobby, tmp) {
-               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 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
-               __skb_queue_tail(&ulpq->lobby, event->parent);
-
-       sctp_spin_unlock_irqrestore(&ulpq->lobby.lock, flags);
-}
-
-static inline sctp_ulpevent_t *sctp_ulpqueue_order(sctp_ulpqueue_t *ulpq,
-                                                  sctp_ulpevent_t *event)
-{
-       __u16 sid, ssn;
-
-       /* 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
-        * 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);
-
-       return event;
-}
diff --git a/net/sctp/sla1.c b/net/sctp/sla1.c
new file mode 100644 (file)
index 0000000..eb967a3
--- /dev/null
@@ -0,0 +1,281 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * (It's really SHA-1 but Hey I was tired when I created this 
+ * file, and on a plane to France :-)
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    Randall Stewart <rstewar1@email.mot.com>
+ *    kmorneau@cisco.com
+ *    qxie1@email.mot.com
+ *
+ * Based on:
+ *    Randy Stewart, et al. SCTP Reference Implementation which is licenced
+ *    under the GPL. 
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/types.h>
+#include <linux/kernel.h>
+#include <linux/fcntl.h>
+#include <asm/string.h>                /* for memcpy */
+#include <linux/sched.h>       /* dead chicken for in.h */
+#include <linux/in.h>          /* for htonl and ntohl */
+
+#include <net/sctp/sla1.h>
+
+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;
+}
+
+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) */
+
+       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;
+
+       /* 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_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;
+               }
+       }
+}
+
+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);
+}
diff --git a/net/sctp/sm_make_chunk.c b/net/sctp/sm_make_chunk.c
new file mode 100644 (file)
index 0000000..e1e9d9e
--- /dev/null
@@ -0,0 +1,1762 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001 International Business Machines Corp.
+ *
+ * This file is part of the SCTP kernel reference Implementation
+ *
+ * This file includes part of the implementation of the add-IP extension,
+ * based on <draft-ietf-tsvwg-addip-sctp-02.txt> June 29, 2001,
+ * for the SCTP kernel reference Implementation.
+ * 
+ * These functions work with the state functions in sctp_sm_statefuns.c
+ * to implement the state operations.  These functions implement the
+ * steps which require modifying existing data structures.
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson          <karl@athena.chicago.il.us>
+ *    C. Robin              <chris@hundredacre.ac.uk>
+ *    Jon Grimm             <jgrimm@us.ibm.com>
+ *    Xingang Guo           <xingang.guo@intel.com>
+ *    Dajiang Zhang        <dajiang.zhang@nokia.com>
+ *    Sridhar Samudrala            <sri@us.ibm.com>
+ *    Daisy Chang          <daisyc@us.ibm.com>
+ *
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/types.h>
+#include <linux/kernel.h>
+#include <linux/ip.h>
+#include <linux/ipv6.h>
+#include <linux/net.h>
+#include <linux/inet.h>
+#include <net/sock.h>
+
+#include <linux/skbuff.h>
+#include <linux/random.h>      /* for get_random_bytes */
+#include <net/sctp/sctp.h>
+#include <net/sctp/sm.h>
+
+/* RFC 2960 3.3.2 Initiation (INIT) (1)
+ *
+ * Note 4: This parameter, when present, specifies all the
+ * address types the sending endpoint can support. The absence
+ * of this parameter indicates that the sending endpoint can
+ * support any address type.
+ */
+static const sctp_supported_addrs_param_t sat_param = {
+       {           
+               SCTP_PARAM_SUPPORTED_ADDRESS_TYPES,
+               __constant_htons(SCTP_SAT_LEN),
+       },
+       {               /* types[] */
+               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)),
+       }
+};
+
+/* 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, __u16 cause_code,
+                     const void *payload, size_t paylen)
+{
+       sctp_errhdr_t err;
+       int padlen;
+       __u16 len;
+
+        /* Cause code constants are now defined in network order.  */
+       err.cause = cause_code;
+       len = sizeof(sctp_errhdr_t) + paylen;
+       padlen = len % 4;
+       len += padlen;
+       err.length  = htons(len);
+       sctp_addto_chunk(chunk, sizeof(sctp_errhdr_t), &err);
+       chunk->subh.err_hdr = sctp_addto_chunk(chunk, paylen, payload);
+}
+
+/* 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
+ *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *    |   Type = 1    |  Chunk Flags  |      Chunk Length             |
+ *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *    |                         Initiate Tag                          |
+ *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *    |           Advertised Receiver Window Credit (a_rwnd)          |
+ *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *    |  Number of Outbound Streams   |  Number of Inbound Streams    |
+ *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *    |                          Initial TSN                          |
+ *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *    \                                                               \
+ *    /              Optional/Variable-Length Parameters              /
+ *    \                                                               \
+ *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *
+ *
+ * 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
+ * Advertised Receiver Window Credit   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
+ * IPv6 Address (Note 1)               Optional    6
+ * Cookie Preservative                 Optional    9
+ * Reserved for ECN Capable (Note 2)   Optional    32768 (0x8000)
+ * 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_inithdr_t init;
+       sctpParam_t addrs;
+       size_t chunksize;
+       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.
+        */
+       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 (!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.
+        *
+        * PLEASE DO NOT FIXME [This version does not support Host Name.]
+        */
+
+       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);
+       sctp_addto_chunk(retval, sizeof(ecap_param), &ecap_param);
+
+nodata:
+       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_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 (!bp)
+               goto nomem_bindaddr;
+
+       /* 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)
+               flags |= SCTP_ADDR4_PEERSUPP;
+       if (asoc->peer.ipv6_address)
+               flags |= SCTP_ADDR6_PEERSUPP;
+       error = sctp_bind_addr_copy(bp, &asoc->ep->base.bind_addr,
+                                   scope, priority, flags);
+       if (error)
+               goto nomem_copyaddr;
+
+       addrs = sctp_bind_addrs_to_raw(bp, &addrs_len, priority);
+       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.
+        */
+       cookie = sctp_pack_cookie(asoc->ep, asoc, chunk, &cookie_len,
+                                 addrs.v, addrs_len);
+       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)
+               chunksize += sizeof(ecap_param);
+
+       /* Now allocate and fill out the chunk.  */
+       retval = sctp_make_chunk(asoc, SCTP_CID_INIT_ACK, 0, chunksize);
+       if (!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)
+               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;
+
+       /* 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.
+        *
+        * [INIT ACK back to where the INIT came from.]
+        */
+       if (chunk)
+               retval->transport = chunk->transport;
+
+nomem_chunk:
+       kfree(cookie);
+nomem_cookie:
+       kfree(addrs.v);
+nomem_rawaddr:
+nomem_copyaddr:
+       sctp_bind_addr_free(bp);
+nomem_bindaddr:
+       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
+ * 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
+ *     +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *     |   Type = 10   |Chunk  Flags   |         Length                |
+ *     +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *     /                     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
+ *   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 *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 (!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
+        * to which it is replying.
+        *
+        * [COOKIE ECHO back to where the INIT ACK came from.]
+        */
+       if (chunk)
+               retval->transport = chunk->transport;
+
+nodata:
+       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
+ * 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 *retval;
+
+       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
+        * to which it is replying.
+        *
+        * [COOKIE ACK back to where the COOKIE ECHO came from.]
+        */
+       if (retval && chunk)
+               retval->transport = chunk->transport;
+
+       return retval;
+}
+
+/*
+ *  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
+ *  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.
+ *
+ *     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
+ *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *    | Chunk Type=13 | Flags=00000000|    Chunk Length = 8           |
+ *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *    |                      Lowest TSN Number                        |
+ *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *
+ *     Note: The CWR is considered a Control 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_cwrhdr_t cwr;
+
+       cwr.lowest_tsn = htonl(lowest_tsn);
+       retval = sctp_make_chunk(asoc, SCTP_CID_ECN_CWR, 0,
+                                sizeof(sctp_cwrhdr_t));
+
+       if (!retval)
+               goto nodata;
+
+       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
+        * to which it is replying.
+        *
+        * [Report a reduced congestion window back to where the ECNE
+        * came from.]
+        */
+       if (chunk)
+               retval->transport = chunk->transport;
+
+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 __u32 lowest_tsn)
+{
+       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 (!retval)
+               goto nodata;
+       retval->subh.ecne_hdr =
+               sctp_addto_chunk(retval, sizeof(ecne), &ecne);
+
+nodata:
+       return retval;
+}
+
+/* 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, __u8 flags, __u16 ssn)
+{
+       sctp_chunk_t *retval;
+       sctp_datahdr_t dp;
+       int chunk_len;
+
+       /* We assign the TSN as LATE as possible, not here when
+        * creating the chunk.
+        */
+       dp.tsn= 1000000;       /* This marker is a debugging aid. */
+       dp.stream = htons(sinfo->sinfo_stream);
+       dp.ppid   = htonl(sinfo->sinfo_ppid);
+       dp.ssn    = htons(ssn);
+
+       /* 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 (!retval)
+               goto nodata;
+
+       retval->subh.data_hdr = sctp_addto_chunk(retval, sizeof(dp), &dp);
+       memcpy(&retval->sinfo, sinfo, sizeof(struct sctp_sndrcvinfo));
+
+nodata:
+       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 __u8 *data,
+                                __u8 flags, __u16 ssn)
+{
+       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;
+}
+
+/* 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 __u8 *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;
+}
+
+/* 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)
+{
+       __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().
+        */
+       if (sinfo->sinfo_flags & MSG_UNORDERED) {
+               ssn = 0;
+       } else {
+               ssn = __sctp_association_get_next_ssn(asoc,
+                                                     sinfo->sinfo_stream);
+       }
+
+       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 *retval;
+       sctp_sackhdr_t sack;
+       sctp_gap_ack_block_t gab;
+       int length;
+       __u32 ctsn;
+       sctp_tsnmap_iter_t iter;
+       __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);
+       SCTP_DEBUG_PRINTK("make_sack: sackCTSNAck sent is 0x%x.\n",
+                         ctsn);
+
+       /* 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);
+            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);
+
+       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 (!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
+        * destination transport addresses from which the DATA or
+        * control chunks being acknowledged were received.
+        *
+        * [BUG:  We do not implement the following paragraph.
+        * 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
+        * DATA chunk.  The reason being that receiving a duplicate
+        * 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.  */
+       sctp_tsnmap_iter_init(map, &iter);
+       while(sctp_tsnmap_next_gap_ack(map, &iter, &gab.start, &gab.end)) {
+               gab.start = htons(gab.start);
+               gab.end = htons(gab.end);
+               sctp_addto_chunk(retval,
+                                sizeof(sctp_gap_ack_block_t),
+                                &gab);
+       }
+
+       /* Register the duplicates.  */
+       sctp_addto_chunk(retval,
+                        sizeof(sctp_dup_tsn_t) * num_dup_tsns,
+                        &asoc->peer.dup_tsns);
+
+nodata:
+       return retval;
+}
+
+sctp_chunk_t *sctp_make_shutdown(const sctp_association_t *asoc)
+{
+       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);
+
+       retval = sctp_make_chunk(asoc, SCTP_CID_SHUTDOWN, 0,
+                                sizeof(sctp_shutdownhdr_t));
+       if (!retval)
+               goto nodata;
+
+       retval->subh.shutdown_hdr =
+               sctp_addto_chunk(retval, sizeof(shut), &shut);
+
+nodata:
+       return retval;
+}
+
+sctp_chunk_t *sctp_make_shutdown_ack(const sctp_association_t *asoc,
+                                    const sctp_chunk_t *chunk)
+{
+       sctp_chunk_t *retval;
+
+       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
+        * to which it is replying.
+        *
+        * [ACK back to where the SHUTDOWN came from.]
+        */
+       if (retval && chunk)
+               retval->transport = chunk->transport;
+
+       return retval;
+}
+
+sctp_chunk_t *sctp_make_shutdown_complete(const sctp_association_t *asoc,
+                                         const sctp_chunk_t *chunk)
+{
+       sctp_chunk_t *retval;
+       __u8 flags = 0;
+
+       /* 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);
+
+       /* 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.
+        *
+        * [Report SHUTDOWN COMPLETE back to where the SHUTDOWN ACK
+        * came from.]
+        */
+       if (retval && chunk)
+               retval->transport = chunk->transport;
+
+        return retval;
+}
+
+/* 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 *retval;
+       __u8 flags = 0;
+
+       /* 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);
+
+       /* 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.
+        *
+        * [ABORT back to where the offender came from.]
+        */
+       if (retval && chunk)
+               retval->transport = chunk->transport;
+
+       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, __u32 tsn)
+{
+       sctp_chunk_t *retval;
+       __u32 payload;
+
+       retval = sctp_make_abort(asoc, chunk, sizeof(sctp_errhdr_t)
+                                + sizeof(tsn));
+
+       if (!retval)
+               goto no_mem;
+
+       /* Put the tsn back into network byte order.  */
+       payload = htonl(tsn);
+       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
+        * to which it is replying.
+        *
+        * [ABORT back to where the offender came from.]
+        */
+       if (chunk)
+               retval->transport = chunk->transport;
+
+no_mem:
+       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)
+{
+       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->subh.hbs_hdr = sctp_addto_chunk(retval, paylen, payload);
+
+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 *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
+        * to which it is replying.
+        *
+        * [HBACK back to where the HEARTBEAT came from.]
+        */
+       if (chunk)
+               retval->transport = chunk->transport;
+
+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,
+                                __u16 cause_code, const void *payload,
+                                size_t 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 (chunk)
+               retval->transport = chunk->transport;
+
+nodata:
+       return retval;
+}
+
+/********************************************************************
+ * 2nd Level Abstractions
+ ********************************************************************/
+
+/* 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 *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
+        * 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);
+       INIT_LIST_HEAD(&retval->frag_list);
+       SCTP_DBG_OBJCNT_INC(chunk);
+
+nodata:
+       return retval;
+}
+
+/* Set chunk->source based on the IP header in chunk->skb.  */
+void sctp_init_source(sctp_chunk_t *chunk)
+{
+       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;
+
+       switch (ih4->version) {
+       case 4:
+               source->v4.sin_family = AF_INET;
+               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. ? */
+                       break;
+               )
+
+       default:
+               /* This is a bogus address type, just bail.  */
+               break;
+       };
+}
+
+/* Extract the source address from a 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.  */
+               return &chunk->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,
+                             __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 (!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 = ((__u8 *)chunk_hdr) + sizeof(sctp_chunkhdr_t);
+
+       /* Set the skb to the belonging sock for accounting.  */
+       skb->sk = sk;
+
+       return retval;
+
+nodata:
+       return NULL;
+}
+
+/* Release the memory occupied by a 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);
+
+       /* Free the chunk skb data and the SCTP_chunk stub itself. */
+       dev_kfree_skb(chunk->skb);
+
+       kfree(chunk);
+       SCTP_DBG_OBJCNT_DEC(chunk);
+}
+
+/* Do a deep copy of a chunk.  */
+sctp_chunk_t *sctp_copy_chunk(sctp_chunk_t *chunk, const int priority)
+{
+       sctp_chunk_t *retval;
+       long offset;
+
+       retval = t_new(sctp_chunk_t, priority);
+       if (!retval)
+               goto nodata;
+
+       /* 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);
+       INIT_LIST_HEAD(&retval->frag_list);
+
+       /* Now we copy the deep structure.  */
+       retval->skb = skb_copy(chunk->skb, priority);
+       if (!retval->skb) {
+               kfree(retval);
+               goto nodata;
+       }
+
+       /* 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 *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);
+
+       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;
+
+       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)
+{
+       __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)))
+               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;
+}
+
+/* Helper function to assign a TSN if needed.  This assumes that both
+ * the data_hdr and association have already been assigned.
+ */
+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 =
+                       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 *asoc;
+       sctp_scope_t scope;
+
+       /* Create the bare association.  */
+       scope = sctp_scope(sctp_source(chunk));
+       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 
+                       = 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;
+
+        default:
+               /* Yikes!  I never heard of this kind of address.  */
+               goto fail;
+       };
+
+nodata:
+       return asoc;
+
+fail:
+       sctp_association_free(asoc);
+       return NULL;
+}
+
+/* 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 __u8 *raw_addrs, int addrs_len)
+{
+       sctp_cookie_param_t *retval;
+       sctp_signed_cookie_t *cookie;
+       int headersize, bodysize;
+
+       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)
+               bodysize += SCTP_COOKIE_MULTIPLE 
+                       - (bodysize % SCTP_COOKIE_MULTIPLE);
+       *cookie_len = headersize + bodysize;
+
+       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;
+
+       /* 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;
+       /* 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));
+
+       /* Copy the raw local address list of the association. */
+       memcpy((__u8 *)&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,
+                        (__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 *retval = NULL;
+       sctp_signed_cookie_t *cookie;
+       sctp_cookie_t *bear_cookie;
+       int headersize, bodysize;
+       int fixed_size, var_size1, var_size2, var_size3;
+       __u8 digest_buf[SCTP_SIGNATURE_SIZE];
+       int secret;
+       sctp_scope_t scope;
+       __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.
+        * 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)))
+               goto malformed;
+
+       /* Verify that the cookie has been padded out. */
+       if (bodysize % SCTP_COOKIE_MULTIPLE)
+               goto malformed;
+
+       /* Process the cookie.  */
+       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,
+                        (__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;
+               }
+       }
+
+       /* 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;
+               goto fail;
+       }
+
+       /* Make a new base association.  */
+       scope = sctp_scope(sctp_source(chunk));
+       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 = (__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,
+                                  priority)) {
+               *error = -SCTP_IERROR_NOMEM;
+               goto fail;
+       }
+
+       retval->next_tsn = retval->c.initial_tsn;
+       retval->ctsn_ack_point = retval->next_tsn - 1;
+
+       /* The INIT stuff will be done by the side effects.  */
+       return retval;
+
+fail:
+       if (retval)
+               sctp_association_free(retval);
+
+       return NULL;
+
+malformed:
+       /* Yikes!  The packet is either corrupt or deliberately
+        * malformed.
+        */
+       *error = -SCTP_IERROR_MALFORMED;
+       goto fail;
+}
+
+/********************************************************************
+ * 3rd Level Abstractions
+ ********************************************************************/
+
+/* Unpack the parameters in an INIT packet.
+ * FIXME:  There is no return status to allow callers to do
+ * 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)
+{
+       sctpParam_t param;
+       __u8 *end;
+       sctp_transport_t *transport;
+       list_t *pos, *temp;
+
+       /* 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.
+        */
+
+       /* This implementation defaults to making the first transport
+        * added as the primary transport.  The source address seems to
+        * be a a better choice than any of the embedded addresses.
+        */
+       if (peer_addr)
+               sctp_assoc_add_peer(asoc, peer_addr, priority);
+
+       /* 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))
+                        goto clean_up;
+       }
+
+       /* The fixed INIT headers are always in network byte
+        * order.
+        */
+       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 =
+               ntohs(peer_init->init_hdr.num_outbound_streams);
+       asoc->peer.i.num_inbound_streams =
+               ntohs(peer_init->init_hdr.num_inbound_streams);
+       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.
+        */
+       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.   */
+       asoc->c.peer_vtag = asoc->peer.i.init_tag;
+
+       /* 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
+        * high (for example, implementations MAY use the size of the receiver
+        * advertised window).
+        */
+       list_for_each(pos, &asoc->peer.transport_addr_list) {
+               transport = list_entry(pos, sctp_transport_t, transports);
+               transport->ssthresh = asoc->peer.i.a_rwnd;
+       }
+
+       /* Set up the TSN tracking pieces.  */
+       sctp_tsnmap_init(&asoc->peer.tsn_map, SCTP_TSN_MAP_SIZE,
+                        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:
+        * ...
+        * 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.
+        */
+       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) {
+               transport = list_entry(pos, sctp_transport_t, transports);
+               list_del(pos);
+               sctp_transport_free(transport);
+       }
+}
+
+/* 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.
+ * param is the variable length parameter to use for update.
+ * cid tells us if this is an INIT, INIT ACK or COOKIE ECHO.
+ * If the current packet is an INIT we want to minimize the amount of
+ * 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)
+{
+       sockaddr_storage_t addr;
+       int j;
+       int i;
+       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.
+        */
+       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))
+                               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,
+                                                   asoc->peer.port);
+                               scope = sctp_scope(peer_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);
+               break;
+
+       case SCTP_PARAM_HOST_NAME_ADDRESS:
+               SCTP_DEBUG_PRINTK("unimplmented SCTP_HOST_NAME_ADDRESS\n");
+               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.
+                */
+               asoc->peer.ipv4_address = 0;
+               asoc->peer.ipv6_address = 0;
+
+               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);
+               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;
+               break;
+
+       default:
+               SCTP_DEBUG_PRINTK("Ignoring param: %d for association %p.\n",
+                                 ntohs(param.p->type), asoc);
+               /* FIXME:  The entire parameter processing really needs
+                * redesigned.  For now, always return success as doing
+                * otherwise craters the system.
+                */
+               retval = 1;
+
+               break;
+       };
+
+       return retval;
+}
+
+/* Select a new verification tag.  */
+__u32 sctp_generate_tag(const sctp_endpoint_t *ep)
+{
+       /* I believe that this random number generator complies with RFC1750.
+        * A tag of 0 is reserved for special cases (e.g. INIT).
+        */
+       __u32 x;
+
+       do {
+               get_random_bytes(&x, sizeof(__u32));
+       } while (x == 0);
+
+       return x;
+}
+
+/* Select an initial TSN to send during startup.  */
+__u32 sctp_generate_tsn(const sctp_endpoint_t *ep)
+{
+       /* I believe that this random number generator complies with RFC1750.  */
+       __u32 retval;
+
+       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, __u16 port)
+{
+       switch(param.p->type) {
+       case SCTP_PARAM_IPV4_ADDRESS:
+               addr->v4.sin_family = AF_INET;
+               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_flowinfo = 0; /* BUG */
+               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;
+       };
+}
+
+/* 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)
+{
+        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;
+}
+
+/* Convert from an IP version number to an Address Family symbol.  */
+int ipver2af(__u8 ipver)
+{
+       int family;
+
+       switch (ipver) {
+       case 4:
+               family = AF_INET;
+               break;
+
+       case 6:
+               family = AF_INET6;
+               break;
+
+       default:
+               family = 0;
+               break;
+       };
+
+       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 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;
+}
diff --git a/net/sctp/sm_sideeffect.c b/net/sctp/sm_sideeffect.c
new file mode 100644 (file)
index 0000000..547f452
--- /dev/null
@@ -0,0 +1,1168 @@
+/* SCTP kernel reference Implementation 
+ * Copyright (c) 1999 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001-2002 International Business Machines Corp.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * These functions work with the state functions in sctp_sm_statefuns.c
+ * to implement that state operations.  These functions implement the
+ * steps which require modifying existing data structures.
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson          <karl@athena.chicago.il.us>
+ *    Jon Grimm             <jgrimm@austin.ibm.com>
+ *    Hui Huang                    <hui.huang@nokia.com>
+ *    Dajiang Zhang        <dajiang.zhang@nokia.com>
+ *    Daisy Chang          <daisyc@us.ibm.com>
+ *    Sridhar Samudrala            <sri@us.ibm.com>
+ *
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/skbuff.h>
+#include <linux/types.h>
+#include <linux/socket.h>
+#include <linux/ip.h>
+#include <net/sock.h>
+#include <net/sctp/sctp.h>
+#include <net/sctp/sm.h>
+
+/* Do forward declarations of static functions.  */
+static void sctp_do_ecn_ce_work(sctp_association_t *asoc,
+                               __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 *);
+
+/* 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
+ * functionality there.
+ */
+#define DEBUG_PRE \
+       SCTP_DEBUG_PRINTK("sctp_do_sm prefn: " \
+                         "ep %p, %s, %s, asoc %p[%s], %s\n", \
+                         ep, sctp_evttype_tbl[event_type], \
+                         (*debug_fn)(subtype), asoc, \
+                         sctp_state_tbl[state], state_fn->name)
+
+#define DEBUG_POST \
+       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)
+{
+       sctp_cmd_seq_t commands;
+       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[] = {
+               NULL, sctp_cname, sctp_tname, sctp_oname, sctp_pname,
+       };
+       printfn_t *debug_fn  __attribute__ ((unused)) = table[event_type];
+
+       /* 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);
+
+       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;
+}
+
+#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 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)))
+               goto bail;
+
+       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;
+
+       case SCTP_DISPOSITION_NOMEM:
+               /* We ran out of memory, so we need to discard this
+                * packet.
+                */
+               /* BUG--we should now recover some memory, probably by
+                * 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
+                * real work has been done as explicit commands above.
+                */
+               break;
+
+       case SCTP_DISPOSITION_VIOLATION:
+               printk(KERN_ERR "sctp protocol violation state %d "
+                      "chunkid %d\n", state, subtype.chunk);
+               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;
+
+       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:
+               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;
+       };
+
+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 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;
+
+       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:
+        *     while (cmds)
+        *         cmd->handle(x, y, z)
+        * --jgrimm
+        */
+       while (NULL != (command = sctp_next_cmd(commands))) {
+               switch (command->verb) {
+               case SCTP_CMD_NOP:
+                       /* Do nothing. */
+                       break;
+
+               case SCTP_CMD_NEW_ASOC:
+                       /* Register a new association.  */
+                       asoc = command->obj.ptr;
+                       /* Register with the endpoint.  */
+                       sctp_endpoint_add_asoc(ep, asoc);
+                       sctp_hash_established(asoc);
+                       break;
+
+               case SCTP_CMD_UPDATE_ASSOC:
+                      sctp_assoc_update(asoc, command->obj.ptr);
+                      break;
+
+               case SCTP_CMD_PURGE_OUTQUEUE:
+                      sctp_outqueue_teardown(&asoc->outqueue);
+                      break;
+
+               case SCTP_CMD_DELETE_TCB:
+                       /* Delete the current association.  */
+                       sctp_unhash_established(asoc);
+                       sctp_association_free(asoc);
+                       asoc = NULL;
+                       break;
+
+               case SCTP_CMD_NEW_STATE:
+                       /* Enter a new state.  */
+                       asoc->state = command->obj.state;
+                       asoc->state_timestamp = jiffies;
+                       break;
+
+               case SCTP_CMD_REPORT_TSN:
+                       /* Record the arrival of a TSN.  */
+                       sctp_tsnmap_mark(&asoc->peer.tsn_map,
+                                        command->obj.u32);
+                       break;
+
+               case SCTP_CMD_GEN_SACK:
+                       /* Generate a Selective ACK.
+                        * The argument tells us whether to just count
+                        * the packet and MAYBE generate a SACK, or
+                        * force a SACK out.
+                        */
+                       force = command->obj.i32;
+                       error = sctp_gen_sack(asoc, force, commands);
+                       break;
+
+               case SCTP_CMD_PROCESS_SACK:
+                       /* Process an inbound SACK.  */
+                       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;
+
+                       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,
+                                             asoc, chunk, command->obj.ptr,
+                                             priority);
+                       break;
+
+               case SCTP_CMD_GEN_COOKIE_ECHO:
+                       /* 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,
+                                       SCTP_CHUNK(new_obj));
+                       break;
+
+               case SCTP_CMD_GEN_SHUTDOWN:
+                       /* Generate SHUTDOWN when in SHUTDOWN_SENT state.
+                        * Reset error counts.
+                        */
+                       asoc->overall_error_count = 0;
+
+                       /* Generate a SHUTDOWN chunk.  */
+                       new_obj = sctp_make_shutdown(asoc);
+                       if (!new_obj)
+                               goto nomem;
+                       sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
+                                       SCTP_CHUNK(new_obj));
+                       break;
+
+               case SCTP_CMD_CHUNK_ULP:
+                       /* Send a chunk to the sockets layer.  */
+                       SCTP_DEBUG_PRINTK("sm_sideff: %s %p, %s %p.\n",
+                                         "chunk_up:",
+                                         command->obj.ptr,
+                                         "ulpq:",
+                                         &asoc->ulpq);
+                       sctp_ulpqueue_tail_data(&asoc->ulpq,
+                                               command->obj.ptr,
+                                               GFP_ATOMIC);
+                       break;
+
+               case SCTP_CMD_EVENT_ULP:
+                       /* 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,
+                                                command->obj.ptr);
+                       break;
+
+               case SCTP_CMD_REPLY:
+                       /* Send a chunk to our peer.  */
+                       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);
+                       break;
+
+               case SCTP_CMD_RETRAN:
+                       /* Mark a transport for retransmission.  */
+                       sctp_retransmit(&asoc->outqueue,
+                                       command->obj.transport, 0);
+                       break;
+
+               case SCTP_CMD_TRANSMIT:
+                       /* Kick start transmission. */
+                       error = sctp_flush_outqueue(&asoc->outqueue, 0);
+                       break;
+
+               case SCTP_CMD_ECN_CE:
+                       /* Do delayed CE processing.   */
+                       sctp_do_ecn_ce_work(asoc, command->obj.u32);
+                       break;
+
+               case SCTP_CMD_ECN_ECNE:
+                       /* Do delayed ECNE processing. */
+                       new_obj = sctp_do_ecn_ecne_work(asoc,
+                                                       command->obj.u32,
+                                                       chunk);
+                       if (new_obj) {
+                               sctp_add_cmd_sf(commands, SCTP_CMD_REPLY,
+                                               SCTP_CHUNK(new_obj));
+                       }
+                       break;
+
+               case SCTP_CMD_ECN_CWR:
+                       /* Do delayed CWR processing.  */
+                       sctp_do_ecn_cwr_work(asoc, command->obj.u32);
+                       break;
+
+               case SCTP_CMD_SETUP_T2:
+                       sctp_cmd_setup_t2(commands, asoc, command->obj.ptr);
+                       break;
+
+               case SCTP_CMD_TIMER_START:
+                       timer = &asoc->timers[command->obj.to];
+                       timeout = asoc->timeouts[command->obj.to];
+                       if (!timeout)
+                               BUG();
+
+                       timer->expires = jiffies + timeout;
+                       sctp_association_hold(asoc);
+                       add_timer(timer);
+                       break;
+
+               case SCTP_CMD_TIMER_RESTART:
+                       timer = &asoc->timers[command->obj.to];
+                       timeout = asoc->timeouts[command->obj.to];
+                       if (!mod_timer(timer, jiffies + timeout))
+                               sctp_association_hold(asoc);
+                       break;
+
+               case SCTP_CMD_TIMER_STOP:
+                       timer = &asoc->timers[command->obj.to];
+                       if (timer_pending(timer) && del_timer(timer))
+                               sctp_association_put(asoc);
+                       break;
+
+               case SCTP_CMD_INIT_RESTART:
+                       /* Do the needed accounting and updates
+                        * associated with restarting an initialization
+                        * 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;
+                       }
+
+                       sctp_add_cmd_sf(commands,
+                                       SCTP_CMD_TIMER_RESTART,
+                                       SCTP_TO(command->obj.to));
+                       break;
+
+               case SCTP_CMD_INIT_FAILED:
+                       sctp_cmd_init_failed(commands, asoc);
+                       break;
+
+               case SCTP_CMD_ASSOC_FAILED:
+                       sctp_cmd_assoc_failed(commands, asoc);
+                       break;
+
+               case SCTP_CMD_COUNTER_INC:
+                       asoc->counters[command->obj.counter]++;
+                       break;
+
+               case SCTP_CMD_COUNTER_RESET:
+                       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);
+                       }
+                       break;
+
+               case SCTP_CMD_REPORT_BIGGAP:
+                       SCTP_DEBUG_PRINTK("Big gap: %x to %x\n",
+                                         sctp_tsnmap_get_ctsn(
+                                                 &asoc->peer.tsn_map),
+                                         command->obj.u32);
+                       break;
+
+               case SCTP_CMD_REPORT_BAD_TAG:
+                       SCTP_DEBUG_PRINTK("vtag mismatch!\n");
+                       break;
+
+               case SCTP_CMD_SET_BIND_ADDR:
+                       sctp_cmd_set_bind_addrs(commands, asoc,
+                                               command->obj.bp);
+                       break;
+
+               case SCTP_CMD_STRIKE:
+                       /* Mark one strike against a transport.  */
+                       sctp_do_8_2_transport_strike(asoc,
+                                                    command->obj.transport);
+                       break;
+
+               case SCTP_CMD_TRANSPORT_RESET:
+                       t = command->obj.transport;
+                       sctp_cmd_transport_reset(commands, asoc, t);
+                       break;
+
+               case SCTP_CMD_TRANSPORT_ON:
+                       t = command->obj.transport;
+                       sctp_cmd_transport_on(commands, asoc, t, chunk);
+                       break;
+
+               case SCTP_CMD_HB_TIMERS_START:
+                       sctp_cmd_hb_timers_start(commands, asoc);
+                       break;
+
+               case SCTP_CMD_REPORT_ERROR:
+                       error = command->obj.error;
+                       break;
+
+               case SCTP_CMD_PROCESS_CTSN:
+                       /* Dummy up a SACK for processing. */
+                       sackh.cum_tsn_ack = command->obj.u32;
+                       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_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;
+               };
+       }
+
+       return error;
+
+nomem:
+       error = -ENOMEM;
+       return error;
+}
+
+/* A helper function for delayed processing of INET ECN CE bit. */
+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
+        */
+
+       asoc->last_ecne_tsn = lowest_tsn;
+       asoc->need_ecne = 1;
+}
+
+/* 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
+ * that was originally marked with the CE bit.
+ */
+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
+        * sending a CWR.
+        */
+
+       /* Find which transport's congestion variables
+        * need to be adjusted.
+        */
+
+       transport = sctp_assoc_lookup_tsn(asoc, lowest_tsn);
+
+       /* Update the congestion variables. */
+       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
+        * 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.
+        *
+        * Here's what RFC 2960 has to say about CWR.  This is NOT
+        * what we do.
+        *
+        * RFC 2960 Appendix A
+        *
+        *    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 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.
+        */
+       asoc->last_cwr_tsn = asoc->next_tsn - 1;
+
+       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;
+}
+
+/* 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;
+}
+
+/* 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)
+{
+       __u32 ctsn, max_tsn_seen;
+       sctp_chunk_t *sack;
+       int error = 0;
+
+       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) {
+               /* We will need a SACK for the next packet.  */
+               asoc->peer.sack_needed = 1;
+               goto out;
+       } else {
+               sack = sctp_make_sack(asoc);
+               if (!sack)
+                       goto nomem;
+
+               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));
+       }
+
+out:
+       return error;
+
+nomem:
+       error = -ENOMEM;
+       return error;
+}
+
+/* Handle a duplicate TSN.  */
+void sctp_do_TSNdup(sctp_association_t *asoc, sctp_chunk_t *chunk, long gap)
+{
+#if 0
+       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;
+#endif /* 0 */
+} /* sctp_do_TSNdup() */
+
+#undef AP
+
+/* 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)
+{
+       int error;
+       sctp_transport_t *transport = (sctp_transport_t *) peer;
+       sctp_association_t *asoc = transport->asoc;
+
+       /* 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("%s:Sock is busy.\n", __FUNCTION__);
+
+               /* 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)
+               goto out_unlock;
+
+       /* Run through the state machine.  */
+       error = sctp_do_sm(SCTP_EVENT_T_TIMEOUT,
+                          SCTP_ST_TIMEOUT(SCTP_EVENT_TIMEOUT_T3_RTX),
+                          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);
+}
+
+/* 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)
+{
+       int error = 0;
+
+       sctp_bh_lock_sock(asoc->base.sk);
+       if (__sctp_sock_busy(asoc->base.sk)) {
+               SCTP_DEBUG_PRINTK("%s:Sock is busy: timer %d\n",
+                                 __FUNCTION__,
+                                 timeout_type);
+
+               /* 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)
+               goto out_unlock;
+
+       /* Run through the state machine.  */
+       error = sctp_do_sm(SCTP_EVENT_T_TIMEOUT,
+                          SCTP_ST_TIMEOUT(timeout_type),
+                          asoc->state, asoc->ep, asoc,
+                          (void *)timeout_type,
+                          GFP_ATOMIC);
+
+       if (error)
+               asoc->base.sk->err = -error;
+
+out_unlock:
+       sctp_bh_unlock_sock(asoc->base.sk);
+       sctp_association_put(asoc);
+}
+
+void sctp_generate_t1_cookie_event(unsigned long data)
+{
+       sctp_association_t *asoc = (sctp_association_t *) data;
+       sctp_generate_timeout_event(asoc, SCTP_EVENT_TIMEOUT_T1_COOKIE);
+}
+
+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);
+}
+
+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);
+}
+
+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);
+}
+
+/* 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)
+{
+       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("%s:Sock is busy.\n", __FUNCTION__);
+
+               /* 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
+        * get destroyed?
+        */
+       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;
+
+out_unlock:
+       sctp_bh_unlock_sock(asoc->base.sk);
+       sctp_transport_put(transport);
+}
+
+/* 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);
+}
+
+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_timer_event_t *sctp_timer_events[SCTP_NUM_TIMEOUT_TYPES] = {
+       NULL,
+       sctp_generate_t1_cookie_event,
+       sctp_generate_t1_init_event,
+       sctp_generate_t2_shutdown_event,
+       NULL,
+       NULL,
+       sctp_generate_heartbeat_event,
+       sctp_generate_sack_event,
+       sctp_generate_autoclose_event,
+       sctp_generate_pmtu_raise_event,
+};
+
+/********************************************************************
+ * 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.
+ * When the value in the error counter exceeds the protocol parameter
+ * '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)
+{
+       /* 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)) {
+               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_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.
+        */
+       transport->rto = min((transport->rto * 2), transport->asoc->rto_max);
+}
+
+/* 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;
+
+       event = sctp_ulpevent_make_assoc_change(asoc,
+                                               0,
+                                               SCTP_CANT_STR_ASSOC,
+                                               0, 0, 0,
+                                               GFP_ATOMIC);
+
+       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
+        * the transmit queue, if the user has enabled
+        * SEND_FAILED notifications.
+        */
+       sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
+}
+
+/* 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;
+
+       event = sctp_ulpevent_make_assoc_change(asoc,
+                                               0,
+                                               SCTP_COMM_LOST,
+                                               0, 0, 0,
+                                               GFP_ATOMIC);
+
+       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());
+}
+
+/* Process an init chunk (may be real INIT/INIT-ACK or an embedded INIT
+ * 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)
+{
+       /* 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.
+        */
+       sctp_process_init(asoc, chunk->chunk_hdr->type,
+                         sctp_source(chunk), peer_init,
+                         priority);
+}
+
+/* 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)
+{
+       sctp_transport_t *t;
+       list_t *pos;
+
+       /* Start a heartbeat timer for each transport on the association.
+        * hold a reference on the transport to make sure none of
+        * the needed data structures go away.
+        */
+       list_for_each(pos, &asoc->peer.transport_addr_list) {
+               t = list_entry(pos, sctp_transport_t, transports);
+               if (!mod_timer(&t->hb_timer,
+                              t->hb_interval + t->rto + jiffies)) {
+                       sctp_transport_hold(t);
+               }
+       }
+}
+
+/* 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;
+
+       list_for_each_safe(pos, temp, &bp->address_list) {
+               list_del_init(pos);
+               list_add_tail(pos, &asoc->base.bind_addr.address_list);
+       }
+
+       /* Free the temporary bind addr header, otherwise
+        * there will a memory leak.
+        */
+       sctp_bind_addr_free(bp);
+}
+
+/* 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;
+
+       /* 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;
+
+       /* Mark the destination transport address as active if it is not so
+        * marked.
+        */
+       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;
+       sctp_transport_update_rto(t, (jiffies - hbinfo->sent_at));
+}
+
+/* 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))
+               sctp_transport_hold(t);
+}
+
+/* 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.  */
+               err = sctp_do_sm(SCTP_EVENT_T_OTHER,
+                                SCTP_ST_OTHER(SCTP_EVENT_NO_PENDING_TSN),
+                                asoc->state, asoc->ep, asoc, NULL,
+                                GFP_ATOMIC);
+       } else {
+               /* Windows may have opened, so we need
+                * to check if we have DATA to transmit
+                */
+               err = sctp_flush_outqueue(&asoc->outqueue, 0);
+       }
+
+       return err;
+}
+
+/* Helper function to set the timeout value for T2-SHUTDOWN timer and to set
+ * 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)
+{
+       sctp_transport_t *t;
+
+       t = sctp_assoc_choose_shutdown_transport(asoc);
+       asoc->shutdown_last_sent_to = t;
+       asoc->timeouts[SCTP_EVENT_TIMEOUT_T2_SHUTDOWN] = t->rto;
+       chunk->transport = t;
+}
diff --git a/net/sctp/sm_statefuns.c b/net/sctp/sm_statefuns.c
new file mode 100644 (file)
index 0000000..99003bc
--- /dev/null
@@ -0,0 +1,3697 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001-2002 International Business Machines, Corp.
+ * Copyright (c) 2002      Nokia Corp.
+ *
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * This is part of the SCTP Linux Kernel Reference Implementation.
+ *
+ * These are the state functions for the state machine.
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson          <karl@athena.chicago.il.us>
+ *    Mathew Kotowsky       <kotowsky@sctp.org>
+ *    Sridhar Samudrala     <samudrala@us.ibm.com>
+ *    Jon Grimm             <jgrimm@us.ibm.com>
+ *    Hui Huang            <hui.huang@nokia.com>
+ *    Dajiang Zhang        <dajiang.zhang@nokia.com>
+ *    Daisy Chang          <daisyc@us.ibm.com>
+ *
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/types.h>
+#include <linux/kernel.h>
+#include <linux/ip.h>
+#include <linux/ipv6.h>
+#include <linux/net.h>
+#include <linux/inet.h>
+#include <net/sock.h>
+#include <net/inet_ecn.h>
+#include <linux/skbuff.h>
+#include <net/sctp/sctp.h>
+#include <net/sctp/sm.h>
+#include <net/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
+ * 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).
+ *
+ * Verification Tag: 8.5.1(C)
+ * C) Rules for packet carrying SHUTDOWN COMPLETE:
+ * ...
+ * - 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.
+ *
+ * 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_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)
+               return SCTP_DISPOSITION_VIOLATION;
+
+       /* 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
+        * 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);
+
+       /* 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.
+        */
+       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));
+
+       /* 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,
+                       SCTP_STATE(SCTP_STATE_CLOSED));
+
+       sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
+
+       return SCTP_DISPOSITION_DELETE_TCB;
+
+nomem:
+       return SCTP_DISPOSITION_NOMEM;
+}
+
+/*
+ * 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.
+ *    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_add_cmd_sf(commands, SCTP_CMD_DISCARD_PACKET, SCTP_NULL());
+       return SCTP_DISPOSITION_CONSUME;
+}
+
+/*
+ * 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
+ *    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: 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_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
+        * control endpoint, responding with an ABORT.
+        */
+       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)
+               return SCTP_DISPOSITION_VIOLATION;
+
+        /* Grab the INIT header.  */
+       chunk->subh.init_hdr = (sctp_inithdr_t *)chunk->skb->data;
+
+       /* 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 (!new_asoc)
+               goto nomem;
+
+       /* 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_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.  */
+       repl = sctp_make_init_ack(new_asoc, chunk, GFP_ATOMIC);
+       if (!repl)
+               goto nomem_ack;
+
+       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, nor keep any states for the
+        * new association.  Otherwise, "Z" will be vulnerable to resource
+        * attacks.
+        */
+       sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
+
+       return SCTP_DISPOSITION_DELETE_TCB;
+
+nomem_ack:
+       sctp_association_free(new_asoc);
+nomem:
+       return SCTP_DISPOSITION_NOMEM;
+}
+
+/*
+ * 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
+ *    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
+ *   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_chunk_t *chunk = arg;
+       sctp_init_chunk_t *initchunk;
+       __u32 init_tag;
+
+       /* 6.10 Bundling
+        * An endpoint MUST NOT bundle INIT, INIT ACK or
+        * SHUTDOWN COMPLETE with any other chunks.
+        */
+       if (!chunk->singleton)
+               return SCTP_DISPOSITION_VIOLATION;
+
+       /* Grab the INIT header.  */
+       chunk->subh.init_hdr = (sctp_inithdr_t *) chunk->skb->data;     
+
+       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 (!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_STATE(SCTP_STATE_CLOSED));
+               sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB,
+                               SCTP_NULL());
+               return SCTP_DISPOSITION_DELETE_TCB;
+       }
+
+       /* 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;
+
+       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,
+                       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_TO(SCTP_EVENT_TIMEOUT_T1_COOKIE));
+       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());
+       return SCTP_DISPOSITION_CONSUME;
+
+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
+ *    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_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
+        * control endpoint, responding with an ABORT.
+        */
+       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.
+        */
+        chunk->subh.cookie_hdr =
+               (sctp_signed_cookie_t *)chunk->skb->data;
+       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.
+        */
+       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,
+                                               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_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.
+        */
+       /* This 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,
+                         &chunk->subh.cookie_hdr->c.peer_addr, peer_init,
+                         GFP_ATOMIC);
+
+       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
+        *
+        * 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.
+        */
+       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;
+
+       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;
+}
+
+/*
+ * 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
+ *    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).
+ *
+ * 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_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.
+        */
+       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_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).
+        */
+       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;
+
+       sctp_add_cmd_sf(commands, SCTP_CMD_EVENT_ULP, SCTP_ULPEVENT(ev));
+
+       return SCTP_DISPOSITION_CONSUME;
+
+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_transport_t *transport = (sctp_transport_t *) arg;
+       sctp_chunk_t *reply;
+       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.
+        * 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;
+
+       /* 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);
+       reply = sctp_make_heartbeat(asoc, transport, &hbinfo, paylen);
+       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_TRANSPORT(transport));
+
+        return SCTP_DISPOSITION_CONSUME;
+
+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
+ * 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
+ * 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_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.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);
+
+       reply = sctp_make_heartbeat_ack(asoc, chunk,
+                                       chunk->subh.hb_hdr, paylen);
+       if (!reply)
+               goto nomem;
+
+       sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(reply));
+       return SCTP_DISPOSITION_CONSUME;
+
+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
+ * address to which the HEARTBEAT was sent, and mark the destination
+ * transport address as active if it is not so marked. The endpoint may
+ * optionally report to the upper layer when an inactive destination
+ * address is marked as active due to the reception of the latest
+ * 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_chunk_t *chunk = arg;
+       sockaddr_storage_t from_addr;
+       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);
+
+       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.  */
+       if (!link) {
+               printk(KERN_WARNING
+                      "%s: Could not find address %d.%d.%d.%d\n",
+                      __FUNCTION__, 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)) {
+               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));
+
+       return SCTP_DISPOSITION_CONSUME;
+}
+
+/* Populate the verification/tie tags based on overlapping INIT
+ * 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)
+{
+       switch (asoc->state) {
+
+       /* 5.2.1 INIT received in COOKIE-WAIT or COOKIE-ECHOED State */
+
+       case SCTP_STATE_COOKIE_WAIT:
+               new_asoc->c.my_vtag     = asoc->c.my_vtag;
+               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;
+
+       /* 5.2.2 Unexpected INIT in States Other than CLOSED, COOKIE-ECHOED,
+        *   COOKIE-WAIT and SHUTDOWN-ACK-SENT
+        */
+       default:
+               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.
+        */
+       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;
+}
+
+/*
+ * 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.
+ */
+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))
+               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))
+               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)))
+               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))
+               return 'C';
+
+       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)
+{
+       sctp_chunk_t *chunk = arg;
+       sctp_chunk_t *repl;
+       sctp_association_t *new_asoc;
+
+       /* 6.10 Bundling
+        * An endpoint MUST NOT bundle INIT, INIT ACK or
+        * SHUTDOWN COMPLETE with any other chunks.
+        */
+       if (!chunk->singleton)
+               return SCTP_DISPOSITION_VIOLATION;
+
+       /* Grab the INIT header.  */
+       chunk->subh.init_hdr = (sctp_inithdr_t *) chunk->skb->data;
+
+       /* 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.
+        */
+       new_asoc = sctp_make_temp_asoc(ep, chunk, GFP_ATOMIC);
+       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.
+        */
+       sctp_process_init(new_asoc, chunk->chunk_hdr->type,
+                         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;
+
+       sctp_add_cmd_sf(commands, SCTP_CMD_NEW_ASOC, SCTP_ASOC(new_asoc));
+       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;
+
+nomem:
+       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,
+ * unchanged). These original parameters are combined with those from the
+ * 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
+ * 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)
+{
+       /* Call helper to do the real work for both simulataneous and
+        * duplicate INIT chunk handling.
+        */
+       return sctp_sf_do_unexpected_init(ep, asoc, type, arg, commands);
+}
+
+/*
+ * 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
+ *
+ * Unless otherwise stated, upon reception of an unexpected INIT for
+ * this association, the endpoint shall generate an INIT ACK with a
+ * State Cookie.  In the outbound INIT ACK the endpoint MUST copy its
+ * current Verification Tag and peer's Verification Tag into a reserved
+ * place within the state cookie.  We shall refer to these locations as
+ * the Peer's-Tie-Tag and the Local-Tie-Tag.  The outbound SCTP packet
+ * containing this INIT ACK MUST carry a Verification Tag value equal to
+ * the Initiation Tag found in the unexpected INIT.  And the INIT ACK
+ * MUST contain a new Initiation Tag (randomly generated see Section
+ * 5.3.1).  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.
+ *
+ * After sending out the INIT ACK, the endpoint shall take no further
+ * actions, i.e., the existing association, including its current state,
+ * and the corresponding TCB MUST NOT be changed.
+ *
+ * Note: Only when a TCB exists and the association is not in a COOKIE-
+ * WAIT state are the Tie-Tags populated.  For a normal association INIT
+ * (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)
+{
+       /* Call helper to do the real work for both simulataneous and
+        * duplicate INIT chunk handling.
+        */
+       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
+ *  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)
+{
+       sctp_init_chunk_t *peer_init;
+       sctp_ulpevent_t *ev;
+       sctp_chunk_t *repl;
+       sctp_transport_t *new_addr, *addr;
+       list_t *pos, *pos2, *temp;
+       int found, error;
+
+       /* 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_source(chunk), peer_init, GFP_ATOMIC);
+
+       /* 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,
+                                  &asoc->peer.transport_addr_list) {
+                       addr = list_entry(pos2, sctp_transport_t, transports);
+                       if (!sctp_cmp_addr_exact(&new_addr->ipaddr,
+                                                &addr->ipaddr)) {
+                               found = 0;
+                               break;
+                       }
+               }
+               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;
+
+               error = sctp_add_bind_addr(bp, &new_addr->ipaddr, GFP_ATOMIC);
+               if (error)
+                       goto nomem_add;
+
+               rawaddr = sctp_bind_addrs_to_raw(bp, &len, GFP_ATOMIC);
+               if (!rawaddr.v)
+                       goto nomem_raw;
+
+               repl = sctp_make_abort(asoc, chunk, len+sizeof(sctp_errhdr_t));
+               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;
+       }
+
+       /* For now, fail any unsent/unacked data.  Consider the optional
+        * choice of resending of this data.
+        */
+       sctp_add_cmd_sf(commands, SCTP_CMD_PURGE_OUTQUEUE, SCTP_NULL());
+
+       /* 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));
+
+       /* Report association restart to upper layer. */
+       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 (!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;
+}
+
+/* Unexpected COOKIE-ECHO handler for setup collision (Table 2, action 'B')
+ *
+ * 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)
+{
+       sctp_init_chunk_t *peer_init;
+       sctp_ulpevent_t *ev;
+       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_source(chunk), peer_init, GFP_ATOMIC);
+
+       /* 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_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 (!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
+        * send the Communication Up notification to the SCTP user
+        * 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 (!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;
+}
+
+/* Unexpected COOKIE-ECHO handler for setup collision (Table 2, action 'C')
+ *
+ * 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)
+{
+       /* The cookie should be silently discarded.
+        * The endpoint SHOULD NOT change states and should leave
+        * any timers running.
+        */
+       return SCTP_DISPOSITION_DISCARD;
+}
+
+/* Unexpected COOKIE-ECHO handler lost chunk (Table 2, action 'D')
+ *
+ * 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)
+{
+       sctp_ulpevent_t *ev = NULL;
+       sctp_chunk_t *repl;
+
+       /* The local endpoint cannot use any value from the received
+        * state cookie and need to immediately resend a COOKIE-ACK
+        * and move into ESTABLISHED if it hasn't done so.
+        */
+       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
+                * to 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,
+                                            new_asoc->c.sinit_num_ostreams,
+                                            new_asoc->c.sinit_max_instreams,
+                                             GFP_ATOMIC);
+               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 (!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);
+       return SCTP_DISPOSITION_NOMEM;
+}
+
+/*
+ * 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 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,
+                                                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. */
+               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. */
+               retval = sctp_sf_do_dupcook_c(ep, asoc, chunk, commands,
+                                             new_asoc);
+               break;
+
+       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 "%s:unknown case\n", __FUNCTION__);
+               retval = SCTP_DISPOSITION_DISCARD;
+               break;
+        };
+
+       /* 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;
+}
+
+#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.
+ * ...
+ * 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
+ *    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.
+ *
+ * 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_chunk_t *in_packet;
+       stp_chunk_t *reply;
+       sctp_inithdr_t initack;
+       __u8 *addrs;
+       int addrs_len;
+       time_t rtt;
+       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 (!retval->event_up)
+                       goto nomem;
+               sctp_add_cmd_sf(retval->commands, SCTP_CMD_DELETE_TCB,
+                               SCTP_NULL());
+               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.  */
+       in_packet = chunk;
+       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);
+
+       /* Build that new INIT chunk.  */
+       reply = sctp_make_chunk(SCTP_INITIATION, 0,
+                               sizeof(initack)
+                               + sizeof(bht)
+                               + addrs_len);
+       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;
+}
+#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
+ * 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.
+ *
+ *  - 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.
+ *
+ * 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_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;
+}
+
+/*
+ * 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_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
+                       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;
+}
+
+/*
+ * 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)
+{
+       /* 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_abort(ep, asoc, type, arg, commands);
+}
+
+#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.
+ *...
+ * While in SHUTDOWN-SENT state ... If the timer expires, the endpoint
+ * must re-send the SHUTDOWN ACK.
+ *
+ * Verification Tag:  Neither the INIT nor the timeout will have a
+ * valid verification tag, so it is 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 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) {
+                       sctp_add_cmd(commands, SCTP_CMD_DELETE_TCB,
+                                    SCTP_NULL());
+                       return SCTP_DISPOSITION_DELETE_TCB;
+               }
+               retval->counters[0] = SCTP_COUNTER_INCR;
+               retval->counters[0] = SctpCounterRetran;
+               retval->counters[1] = 0;
+               retval->counters[1] = 0;
+       }
+
+       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;
+}
+#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,
+ *
+ *  - 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
+ *    SHUTDOWN sender.
+ *
+ * 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);
+
+       /* 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));
+       disposition = SCTP_DISPOSITION_CONSUME;
+
+       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.
+        */
+       sctp_add_cmd_sf(commands, SCTP_CMD_PROCESS_CTSN,
+                       SCTP_U32(chunk->subh.shutdown_hdr->cum_tsn_ack));
+       return disposition;
+}
+
+/*
+ * sctp_sf_do_ecn_cwr
+ *
+ * Section:  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
+ * 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.
+ *
+ * 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_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);
+
+       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
+ *
+ * Section:  Appendix A: Explicit Congestion Notification
+ *
+ * ECN-Echo
+ *
+ * RFC 2481 details a specific bit for a receiver to send back in its
+ * TCP acknowledgements to notify the sender of the Congestion
+ * Experienced (CE) bit having arrived from the network.  For SCTP this
+ * same indication is made by including the ECNE chunk.  This chunk
+ * contains one data element, i.e. the lowest TSN associated with the IP
+ * datagram marked with the CE bit.....
+ *
+ * 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_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;
+       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)) {
+               sctp_add_cmd_sf(commands, SCTP_CMD_ECN_ECNE,
+                               SCTP_U32(ecne->lowest_tsn));
+       }       
+       return SCTP_DISPOSITION_CONSUME;
+}
+
+/*
+ * Section: 6.2  Acknowledgement on Reception of DATA Chunks
+ *
+ * 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
+ * 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_chunk_t *err;
+       size_t datalen;
+       int tmp;
+       __u32 tsn;
+
+       /* 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.
+        */
+
+       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);
+        }
+
+       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.  */
+
+       /* 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) {
+                       /* Do real work as sideffect. */
+                       sctp_add_cmd_sf(commands, SCTP_CMD_ECN_CE,
+                                       SCTP_U32(tsn));
+               }
+       }
+
+       tmp = sctp_tsnmap_check(&asoc->peer.tsn_map, tsn);
+       if (tmp < 0) {
+               /* The TSN is too high--silently discard the chunk and
+                * count on it getting retransmitted later.
+                */
+               goto discard_noforce;
+       } else if (tmp > 0) {
+               /* This is a duplicate.  Record it.  */
+               sctp_add_cmd_sf(commands, SCTP_CMD_REPORT_DUP, SCTP_U32(tsn));
+               goto discard_force;
+       }
+
+       /* This is a new TSN.  */
+
+       /* 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)) {
+               SCTP_DEBUG_PRINTK("Discarding tsn: %u datalen: %Zd, "
+                                 "rwnd: %d\n", tsn, datalen, asoc->rwnd);
+               goto discard_noforce;
+       }
+
+       /*
+        * Section 3.3.10.9 No User Data (9)
+        *
+        * Cause of error
+        * ---------------
+        * No User Data:  This error cause is returned to the originator of a
+        * DATA chunk if a received DATA chunk has no user data.
+        */
+       if (unlikely(0 == datalen)) {
+               err = sctp_make_abort_no_data(asoc, chunk, tsn);
+               if (err) {
+                       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_STATE(SCTP_STATE_CLOSED));
+               sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
+               return SCTP_DISPOSITION_CONSUME;
+       }
+
+       /* We are accepting this DATA chunk. */
+
+       /* 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
+        * 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.
+        */
+       if (ntohs(data_hdr->stream) >= asoc->c.sinit_max_instreams) {
+               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_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
+        * 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.
+        *
+        * RFC 2960 6.2 Acknowledgement on Reception of DATA Chunks
+        *
+        * 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 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:
+       /* RFC 2960 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.
+        */
+       /* In our case, we split the MAY SACK advice up whether or not
+        * the last chunk is a duplicate.'
+        */
+       if (chunk->end_of_packet)
+               sctp_add_cmd_sf(commands, SCTP_CMD_GEN_SACK, SCTP_FORCE());
+       return SCTP_DISPOSITION_DISCARD;
+
+discard_noforce:
+       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_DISCARD;
+}
+
+/*
+ * 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_chunk_t *err;
+       size_t datalen;
+       int tmp;
+       __u32 tsn;
+
+       /* 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.
+        */
+       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);
+       }
+
+       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.
+        *
+        * 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) {
+                       /* Do real work as sideffect. */
+                       sctp_add_cmd_sf(commands, SCTP_CMD_ECN_CE,
+                                       SCTP_U32(tsn));
+               }
+       }
+
+       tmp = sctp_tsnmap_check(&asoc->peer.tsn_map, tsn);
+       if (tmp < 0) {
+               /* The TSN is too high--silently discard the chunk and
+                * count on it getting retransmitted later.
+                */
+               goto gen_shutdown;
+       } else if (tmp > 0) {
+               /* This is a duplicate.  Record it.  */
+               sctp_add_cmd_sf(commands, SCTP_CMD_REPORT_DUP, SCTP_U32(tsn));
+               goto gen_shutdown;
+       }
+
+       /* This is a new TSN.  */
+
+       datalen = ntohs(chunk->chunk_hdr->length);
+       datalen -= sizeof(sctp_data_chunk_t);
+
+       /*
+        * Section 3.3.10.9 No User Data (9)
+        *
+        * Cause of error
+        * ---------------
+        * No User Data:  This error cause is returned to the originator of a
+        * DATA chunk if a received DATA chunk has no user data.
+        */
+       if (unlikely(0 == datalen)) {
+               err = sctp_make_abort_no_data(asoc, chunk, tsn);
+               if (err) {
+                       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_STATE(SCTP_STATE_CLOSED));
+               sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
+               return SCTP_DISPOSITION_CONSUME;
+       }
+
+       /* We are accepting this DATA chunk. */
+
+       /* 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
+        * 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.
+        */
+       if (ntohs(data_hdr->stream) >= asoc->c.sinit_max_instreams) {
+               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_CHUNK(err));
+               }
+       }
+
+       /* Go a head and force a SACK, since we are shutting down. */
+gen_shutdown:
+       /* 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)
+        * with a SACK, a SHUTDOWN chunk, and restart the T2-shutdown timer
+        */
+       if (chunk->end_of_packet) {
+               /* We must delay the chunk creation since the cumulative
+                * 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_add_cmd_sf(commands, SCTP_CMD_TIMER_RESTART,
+                               SCTP_TO(SCTP_EVENT_TIMEOUT_T2_SHUTDOWN));
+       }
+       return SCTP_DISPOSITION_CONSUME;
+}
+
+/*
+ * 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
+ *     as available for retransmit:  Mark it as missing for fast
+ *     retransmit as described in Section 7.2.4 and if no retransmit
+ *     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;
+       __u32 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);
+
+       /* 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",
+                                 asoc->ctsn_ack_point);
+               return SCTP_DISPOSITION_DISCARD;
+       }
+
+       /* 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
+        * sideeffect.
+        */
+       return SCTP_DISPOSITION_CONSUME;
+}
+
+/*
+ * 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
+ *    with an ABORT.  When sending the ABORT, the receiver of the
+ *    OOTB packet MUST fill in the Verification Tag field of the
+ *    outbound packet with the value found in the Verification Tag
+ *    field of the OOTB packet and set the T-bit in the Chunk Flags
+ *    to indicate that no TCB was found.  After sending this ABORT,
+ *    the receiver of the OOTB packet shall discard the OOTB packet
+ *    and take no further action.
+ *
+ * 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;
+       __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;
+
+       /* 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_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 (!abort)
+               goto nomem_chunk;
+
+       /* Set the skb to the belonging sock for accounting.  */
+       abort->skb->sk = ep->base.sk;
+
+       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;
+}
+
+/*
+ * 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)
+{
+       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 (!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;
+}
+
+/*
+ * Process an inbound SHUTDOWN ACK.
+ *
+ * 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;
+       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.
+        */
+       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));
+
+       /* 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;
+
+       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. */
+       sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
+       return SCTP_DISPOSITION_DELETE_TCB;
+
+nomem:
+       return SCTP_DISPOSITION_NOMEM;
+}
+
+/*
+ * RFC 2960, 8.4 - Handle "Out of the blue" Packets
+ * 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. Otherwise,
+ *
+ * 8) The receiver should respond to the sender of the OOTB packet with
+ *    an ABORT.  When sending the ABORT, the receiver of the OOTB packet
+ *    MUST fill in the Verification Tag field of the outbound packet
+ *    with the value found in the Verification Tag field of the OOTB
+ *    packet and set the T-bit in the Chunk Flags to indicate that no
+ *    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_chunk_t *chunk = arg;
+       struct sk_buff *skb = chunk->skb;
+       sctp_chunkhdr_t *ch;
+       __u8 *ch_end;
+       int ootb_shut_ack = 0;
+
+       ch = (sctp_chunkhdr_t *) chunk->chunk_hdr;
+       do {
+               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;
+       } while (ch_end < skb->tail);
+
+       if (ootb_shut_ack)
+               sctp_sf_shut_8_4_5(ep, asoc, type, arg, commands);
+       else
+               sctp_sf_tabort_8_4_8(ep, asoc, type, arg, commands);
+
+       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.
+ *
+ * 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
+ *   should be treated as an Out Of The Blue packet.
+ *   [This means that we do NOT check the Verification Tag on these
+ *   chunks. --piggy ]
+ *
+ * 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_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);
+       if (!transport)
+               goto nomem;
+
+       /* 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_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;
+
+       /* Set the skb to the belonging sock for accounting.  */
+       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;
+}
+
+#if 0
+/*
+ * We did something stupid but got lucky.  Namely, we sent a HEARTBEAT
+ * before the association was all the way up and we did NOT get an
+ * 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);
+
+       return SCTP_DISPOSITION_CONSUME;
+
+nomem:
+       return SCTP_DISPOSITION_NOMEM;
+}
+#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);
+
+       return SCTP_DISPOSITION_CONSUME;
+
+nomem:
+       return SCTP_DISPOSITION_NOMEM;
+}
+#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)
+{
+       return SCTP_DISPOSITION_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)
+ * -> association id [,destination transport addr list] [,outbound stream
+ * count]
+ *
+ * This primitive allows the upper layer to initiate an association to a
+ * 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.
+ * [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
+ * address from the returned destination addresses will be selected by
+ * the local endpoint as default primary path for sending SCTP packets
+ * to this peer.  The returned "destination transport addr list" can
+ * be used by the ULP to change the default primary path or to force
+ * 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
+ * addresses of the peer endpoint with which the association is to be
+ * established.
+ *  [This is asoc->peer.active_path.]
+ * o outbound stream count - the number of outbound streams the ULP
+ * 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_chunk_t *repl;
+       sctp_bind_addr_t *bp;
+       sctp_scope_t scope;
+       int error;
+       int flags;
+
+       /* The comment below says that we enter COOKIE-WAIT AFTER
+        * sending the INIT, but that doesn't actually work in our
+        * implementation...
+        */
+       sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
+                       SCTP_STATE(SCTP_STATE_COOKIE_WAIT));
+
+       /* 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(GFP_ATOMIC);
+       if (!bp)
+               goto nomem;
+
+       /* 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)
+               flags |= SCTP_ADDR4_PEERSUPP;
+       if (asoc->peer.ipv6_address)
+               flags |= SCTP_ADDR6_PEERSUPP;
+       error = sctp_bind_addr_copy(bp, &ep->base.bind_addr, scope,
+                                   GFP_ATOMIC, flags);
+       if (error)
+               goto nomem;
+
+       /* FIXME: Either move address assignment out of this function
+        * or else move the association allocation/init into this function.
+        * The association structure is brand new before calling this
+        * 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);
+
+       /* 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). ...
+        */
+
+       repl = sctp_make_init(asoc, bp, GFP_ATOMIC);
+       if (!repl)
+               goto nomem;
+
+       /* Cast away the const modifier, as we want to just
+        * rerun it through as a sideffect.
+        */
+       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,
+                       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)
+               sctp_bind_addr_free(bp);
+
+       return SCTP_DISPOSITION_NOMEM;
+}
+
+/*
+ * 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]
+ *         [,unorder flag] [,no-bundle flag] [,payload protocol-id] )
+ * -> result
+ *
+ * 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
+ *    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.
+ *
+ *  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
+ *    message).
+ *
+ *  o no-bundle flag - instructs SCTP not to bundle this user data with
+ *    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
+ *    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_chunk_t *chunk = arg;
+
+       sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(chunk));
+       return SCTP_DISPOSITION_CONSUME;
+}
+
+/*
+ * Process the SHUTDOWN primitive.
+ *
+ * Section: 10.1:
+ * C) Shutdown
+ *
+ * Format: SHUTDOWN(association id)
+ * -> 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_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,
+                                                     arg, commands);
+       }
+       return disposition;
+}
+
+/*
+ * Process the ABORT primitive.
+ *
+ * Section: 10.1:
+ * C) Abort
+ *
+ * Format: Abort(association id [, cause code])
+ * -> 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
+ * attempting to abort 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:
+ *
+ *  o cause code - reason of the abort to be passed to the peer
+ *
+ * None.
+ *
+ * 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)
+{
+       /* 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.  */
+       abort = sctp_make_abort(asoc, NULL, 0);
+       if (!abort)
+               retval = SCTP_DISPOSITION_NOMEM;
+       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
+        * TCB.  This is a departure from our typical NOMEM handling.
+        */
+
+       /* Change to CLOSED state.  */
+       sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
+                       SCTP_STATE(SCTP_STATE_CLOSED));
+
+       /* Delete the established association.  */
+       sctp_add_cmd_sf(commands, SCTP_CMD_DELETE_TCB, SCTP_NULL());
+       return retval;
+}
+
+/* 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_add_cmd_sf(commands, SCTP_CMD_REPORT_ERROR, SCTP_ERROR(-EINVAL));
+       return SCTP_DISPOSITION_CONSUME;
+}
+
+/* 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_add_cmd_sf(commands, SCTP_CMD_REPORT_ERROR,
+                       SCTP_ERROR(-ESHUTDOWN));
+       return SCTP_DISPOSITION_CONSUME;
+}
+
+/*
+ * 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_TO(SCTP_EVENT_TIMEOUT_T1_INIT));
+
+       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;
+}
+
+/*
+ * 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)
+{
+       /* 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_shutdown(ep, asoc, type, arg, commands);
+}
+
+/*
+ * sctp_cookie_wait_prm_abort
+ *
+ * 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 an abort while in COOKIE_WAIT state.
+ *
+ * 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)
+{
+       /* Stop T1-init timer */
+       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_cookie_echoed_prm_abort
+ *
+ * Section: 4 Note: 3
+ * Verification Tag:
+ * Inputs
+ * (endpoint, asoc)
+ *
+ * The RFC does not explcitly address this issue, but is the route through the
+ * state table when someone issues an abort while in COOKIE_ECHOED state.
+ *
+ * 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)
+{
+       /* 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);
+}
+
+/*
+ * 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_DEBUG_PRINTK("Primitive type %d is ignored.\n", type.primitive);
+       return SCTP_DISPOSITION_DISCARD;
+}
+
+/***************************************************************************
+ * 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
+ * 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_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.
+        */
+       reply = sctp_make_shutdown(asoc);
+       if (!reply)
+               goto nomem;
+
+       /* Set the transport for the SHUTDOWN chunk and the timeout for the
+        * 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,
+                       SCTP_TO(SCTP_EVENT_TIMEOUT_T2_SHUTDOWN));
+
+       if (asoc->autoclose)
+               sctp_add_cmd_sf(commands, SCTP_CMD_TIMER_STOP,
+                               SCTP_TO(SCTP_EVENT_TIMEOUT_AUTOCLOSE));
+
+       /* and enter the SHUTDOWN-SENT state.  */
+       sctp_add_cmd_sf(commands, SCTP_CMD_NEW_STATE,
+                       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;
+}
+
+/*
+ * Generate a SHUTDOWN ACK now that everything is SACK'd.
+ *
+ * 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)
+{
+       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 (!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,
+                               SCTP_TO(SCTP_EVENT_TIMEOUT_AUTOCLOSE));
+
+       /* 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));
+
+       return SCTP_DISPOSITION_CONSUME;
+
+nomem:
+       return SCTP_DISPOSITION_NOMEM;
+}
+
+/*
+ * 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_DEBUG_PRINTK("The event other type %d is ignored\n",
+                         type.other);
+       return SCTP_DISPOSITION_DISCARD;
+}
+
+/************************************************************
+ * 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;
+
+       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.  */
+       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;
+}
+
+/*
+ * Generate delayed SACK on timeout
+ *
+ * Section: 6.2  Acknowledgement on Reception of DATA Chunks
+ *
+ * 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 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_add_cmd_sf(commands, SCTP_CMD_GEN_SACK, SCTP_FORCE());
+       return SCTP_DISPOSITION_CONSUME;
+}
+
+/*
+ * sctp_sf_t1_timer_expire
+ *
+ * Section: 4 Note: 2
+ * Verification Tag:
+ * Inputs
+ * (endpoint, asoc)
+ *
+ *  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_bind_addr_t *bp;
+       sctp_event_timeout_t timer = (sctp_event_timeout_t) arg;
+       int timeout;
+       int attempts;
+
+       timeout = asoc->timeouts[timer];
+       attempts = asoc->counters[SCTP_COUNTER_INIT_ERROR] + 1;
+       repl = NULL;
+
+       SCTP_DEBUG_PRINTK("Timer T1 expired.\n");
+
+       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;
+                       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;
+               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_TO(timer));
+       } else {
+               sctp_add_cmd_sf(commands, SCTP_CMD_INIT_FAILED, SCTP_NULL());
+               return SCTP_DISPOSITION_DELETE_TCB;
+       }
+
+       return SCTP_DISPOSITION_CONSUME;
+
+nomem:
+       return SCTP_DISPOSITION_NOMEM;
+}
+
+/* RFC2960 9.2 If the timer expires, the endpoint must re-send the SHUTDOWN
+ * with the updated last sequential TSN received from its peer.
+ *
+ * An endpoint should limit the number of retransmissions of the
+ * SHUTDOWN chunk to the protocol parameter 'Association.Max.Retrans'.
+ * If this threshold is exceeded the endpoint should destroy the TCB and
+ * MUST report the peer endpoint unreachable to the upper layer (and
+ * thus the association enters the CLOSED state).  The reception of any
+ * packet from its peer (i.e. as the peer sends all of its queued DATA
+ * chunks) should clear the endpoint's retransmission count and restart
+ * 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_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) {
+       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;
+
+       /* Do some failure management (Section 8.2). */
+       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.
+        */
+       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;
+}
+
+/* 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_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,
+                                                     arg, commands);
+       }
+       return disposition;
+}
+
+/*****************************************************************************
+ * These are sa state functions which could apply to all types of events.
+ ****************************************************************************/
+
+/*
+ * This table entry is not implemented.
+ *
+ * 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)
+{
+       return SCTP_DISPOSITION_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)
+{
+       return SCTP_DISPOSITION_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.
+ *
+ * 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_DEBUG_PRINTK("Timer %d ignored.\n", type.chunk);
+       return SCTP_DISPOSITION_CONSUME;
+}
+
+/*
+ * Discard the chunk.
+ *
+ * Section: 0.2, 5.2.3, 5.2.5, 5.2.6, 6.0, 8.4.6, 8.5.1c, 9.2
+ * [Too numerous to mention...]
+ * 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_DEBUG_PRINTK("Chunk %d is discarded\n", type.chunk);
+       return SCTP_DISPOSITION_DISCARD;
+}
+
+/********************************************************************
+ * 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 *sack;
+       __u16 num_blocks;
+       __u16 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);
+
+       skb_pull(chunk->skb, (num_blocks + num_dup_tsns) * sizeof(__u32));
+       return sack;
+}
diff --git a/net/sctp/sm_statetable.c b/net/sctp/sm_statetable.c
new file mode 100644 (file)
index 0000000..504e6b1
--- /dev/null
@@ -0,0 +1,1146 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 International Business Machines, Corp.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001 Nokia, Inc.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * These are the state tables for the SCTP state machine.
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson          <karl@athena.chicago.il.us>
+ *    Jon Grimm             <jgrimm@us.ibm.com>
+ *    Hui Huang                    <hui.huang@nokia.com>
+ *    Daisy Chang          <daisyc@us.ibm.com>
+ *
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/skbuff.h>
+#include <net/sctp/sctp.h>
+#include <net/sctp/sm.h>
+
+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"};
+
+#define DO_LOOKUP(_max, _type, _table) \
+       if ((event_subtype._type > (_max))) { \
+               printk(KERN_WARNING \
+                      "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]);
+
+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,
+                         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;
+       };
+}
+
+#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 */ \
+       {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 */ \
+       {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"}, \
+} /* 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 */ \
+       {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"}, \
+} /* 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 */ \
+       {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 */ \
+       {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 */ \
+       /* This should not happen, but we are nice.  */ \
+       {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 */ \
+       {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"}, \
+} /* 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 */ \
+       {fn: sctp_sf_cookie_wait_abort, name: "sctp_sf_cookie_wait_abort"}, \
+       /* SCTP_STATE_COOKIE_ECHOED */ \
+       {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"}, \
+       /* SCTP_STATE_SHUTDOWN_PENDING */ \
+       {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"}, \
+       /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+       {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"}, \
+} /* 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 */ \
+       {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, \
+        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"}, \
+} /* 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"}, \
+} /* 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"}, \
+} /* 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 */ \
+       {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 */ \
+       {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 */ \
+       {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 */ \
+       {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"}, \
+} /* 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 */ \
+       {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"}, \
+} /* TYPE_SCTP_SHUTDOWN_COMPLETE */
+
+/* The primary index for this table is the chunk type.
+ * The secondary index for this table is the state.
+ *
+ * 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,
+}; /* 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"},
+       /* 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 (will be sctp_addip_do_asconf)"},
+
+       /* 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"},
+};     /* 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"},
+       /* 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 (will be sctp_addip_do_asconf_ack)"},
+
+       /* 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"},
+};     /* 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"}, \
+} /* 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"}, \
+} /* 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 */ \
+       {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_do_9_2_prm_shutdown"}, \
+       /* 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"}, \
+} /* 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 */ \
+       {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"}, \
+} /* 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 */ \
+       {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"}, \
+} /* 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 */
+
+#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"}, \
+} /* 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"}, \
+} /* 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"}, \
+} /* 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"}, \
+} /* 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"}, \
+} /* 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"}, \
+} /* 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"}, \
+} /* 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"}, \
+} /* 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"}, \
+} /* 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"}, \
+} /* 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,
+};
+
+#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 */ \
+       {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, \
+        name: "sctp_do_9_2_start_shutdown"}, \
+       /* SCTP_STATE_SHUTDOWN_SENT */ \
+       {fn: sctp_sf_ignore_other, name: "sctp_sf_ignore_other"}, \
+       /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+       {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"}, \
+}
+
+#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"}, \
+}
+
+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,
+};
+
+#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 */ \
+       {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"}, \
+}
+
+#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 */ \
+       {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 */ \
+       {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"}, \
+}
+
+#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 */ \
+       {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 */ \
+       {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_SHUTDOWN_PENDING */ \
+       {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"}, \
+       /* SCTP_STATE_SHUTDOWN_RECEIVED */ \
+       {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"}, \
+}
+
+#define TYPE_SCTP_EVENT_TIMEOUT_T4_RTO { \
+       /* 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"}, \
+       /* 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"}, \
+}
+
+#define TYPE_SCTP_EVENT_TIMEOUT_HEARTBEAT { \
+       /* 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_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+       /* SCTP_STATE_ESTABLISHED */ \
+       {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"}, \
+       /* 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"}, \
+}
+
+#define TYPE_SCTP_EVENT_TIMEOUT_SACK { \
+       /* 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_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+       /* SCTP_STATE_ESTABLISHED */ \
+       {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"}, \
+       /* SCTP_STATE_SHUTDOWN_SENT */ \
+       {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"}, \
+       /* SCTP_STATE_SHUTDOWN_ACK_SENT */ \
+       {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_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_timer_ignore, name: "sctp_sf_timer_ignore"}, \
+       /* SCTP_STATE_ESTABLISHED */ \
+       {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"}, \
+       /* 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_PMTU_RAISE { \
+       /* 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 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_T4_RTO,
+       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)
+{
+       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;
+}
diff --git a/net/sctp/socket.c b/net/sctp/socket.c
new file mode 100644 (file)
index 0000000..1001142
--- /dev/null
@@ -0,0 +1,2702 @@
+/* Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 International Business Machines, Corp.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001 Nokia, Inc.
+ * Copyright (c) 2001 La Monte H.P. Yarroll
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * These functions interface with the sockets layer to implement the
+ * SCTP Extensions for the Sockets API.
+ * 
+ * Note that the descriptions from the specification are USER level
+ * functions--this file is the functions which populate the struct proto
+ * for SCTP which is the BOTTOM of the sockets interface.
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Narasimha Budihal     <narsi@refcode.org>
+ *    Karl Knutson          <karl@athena.chicago.il.us>
+ *    Jon Grimm             <jgrimm@us.ibm.com>
+ *    Xingang Guo           <xingang.guo@intel.com>
+ *    Daisy Chang           <daisyc@us.ibm.com>
+ *    Sridhar Samudrala     <samudrala@us.ibm.com>
+ *    Inaky Perez-Gonzalez  <inaky.gonzalez@intel.com>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/config.h>
+#include <linux/types.h>
+#include <linux/compiler.h>
+#include <linux/kernel.h>
+#include <linux/wait.h>
+#include <linux/time.h>
+#include <linux/ip.h>
+#include <linux/fcntl.h>
+#include <linux/poll.h>
+#include <linux/init.h>
+
+#include <net/ip.h>
+#include <net/icmp.h>
+#include <net/route.h>
+#include <net/ipv6.h>
+#include <net/inet_common.h> 
+
+#include <linux/socket.h> /* for sa_family_t */
+#include <net/sock.h>
+#include <net/sctp/sctp.h> 
+
+/* Forward declarations for internal helper functions. */
+static void __sctp_write_space(sctp_association_t *asoc);
+static int sctp_writeable(struct sock *sk);
+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,
+                               int msg_len);
+static int sctp_wait_for_packet(struct sock * sk, int *err, long *timeo_p);
+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 *,
+                                   const sockaddr_storage_t *newaddr,
+                                   sockaddr_storage_t *saveaddr);
+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,
+                                             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]),
+ *   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 retval = 0;
+
+       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 (!sctp_sk(sk)->ep->base.bind_addr.port)
+               retval = sctp_do_bind(sk, (sockaddr_storage_t *)uaddr, 
+                                     addr_len);
+       else
+               retval = -EINVAL;
+
+       sctp_release_sock(sk);
+
+       return retval;
+}
+
+static long sctp_get_port_local(struct sock *, unsigned short);
+
+/* 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;
+       sctp_bind_addr_t *bp = &ep->base.bind_addr;
+       unsigned short sa_family = newaddr->sa.sa_family;
+       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",
+                         sk, newaddr, addr_len);
+
+       /* FIXME: This function needs to handle v4-mapped-on-v6
+        * addresses!
+        */
+       if (PF_INET == sk->family) {
+               if (sa_family != AF_INET)
+                       return -EINVAL;
+       }       
+
+       /* Make a local copy of the new address.  */
+       tmpaddr = *newaddr;
+
+       switch (sa_family) {
+       case AF_INET:
+               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"?
+                */
+               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))
+                               return -EINVAL;
+
+                       /* 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);
+                       snum = &tmpaddr.v6.sin6_port;
+                       break;
+               )
+
+       default:
+               return -EINVAL;
+       };
+
+       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 (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;
+
+       /* 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_local() does duplicate address
+        * detection.
+        */
+       if ((ret = sctp_get_port_local(sk, *snum))) {
+               sctp_sk_addr_restore(sk, &saveaddr);
+               if (ret == (long) sk) {
+                       /* This endpoint has a conflicting address. */
+                       return -EINVAL;
+               } else {
+                       return -EADDRINUSE;
+               }
+       }
+
+       /* 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.  */
+       sctp_local_bh_disable();
+       sctp_write_lock(&ep->base.addr_lock);
+
+       /* 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 (!bp->port) {
+               bp->port = *snum;
+       }
+
+       sctp_write_unlock(&ep->base.addr_lock);
+       sctp_local_bh_enable();
+       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.
+ *
+ * NOTE: This could be integrated into sctp_setsockopt_bindx(),
+ * but keeping it this way makes it easier if sometime sys_bindx is
+ * added.
+ */
+
+/* Unprotected by locks. Call only with socket lock sk->lock held! See
+ * sctp_bindx() for a lock-protected call.
+ */ 
+
+static int __sctp_bindx(struct sock *sk, struct sockaddr_storage *addrs, 
+                       int addrcnt, int flags)
+{
+       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;
+}
+
+/* BINDX with locks.
+ *
+ * NOTE: Currently unused at all ...
+ */
+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);
+
+       return retval;
+}
+
+/* Add a list of addresses as bind addresses to local endpoint or
+ * association.
+ *
+ * Basically run through each address specified in the addrs/addrcnt
+ * array/length pair, determine if it is IPv6 or IPv4 and call
+ * sctp_do_bind() on it.
+ *
+ * If any of them fails, then the operation will be reversed and the
+ * ones that were added will be removed.
+ *
+ * Only __sctp_bindx() is supposed to call this function.
+ */
+int sctp_bindx_add(struct sock *sk, struct sockaddr_storage *addrs, int addrcnt)
+{
+       int cnt;
+       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++) {
+               /* 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;
+               };
+
+               retval = sctp_do_bind(sk, (sockaddr_storage_t *)&addrs[cnt],
+                                     addr_len);
+
+err_bindx_add:
+               if (retval < 0) {
+                       /* Failed. Cleanup the ones that has been added */
+                       if (cnt > 0)
+                               sctp_bindx_rem(sk, addrs, cnt);
+                       return retval;
+               }
+        }
+
+       /* Notify the peer(s), assuming we have (an) association(s).
+        * FIXME: for UDP, we have a 1-1-many mapping amongst sk, ep and asoc,
+        *        so we don't have to do much work on locating associations.
+        *
+        * However, when the separation of ep and asoc kicks in, especially
+        * for TCP style connection, it becomes n-1-n mapping.  We will need
+        * to do more fine work.  Until then, hold my peace.
+        *                                                      --xguo
+        *
+        * Really, I don't think that will be a problem.  The bind()
+        * call on a socket will either know the endpoint
+        * (e.g. TCP-style listen()ing socket, or UDP-style socket),
+        * or exactly one association.  The former case is EXACTLY
+        * 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
+        * ADDIP code.
+        *                                                      --daisy
+        */
+
+#if CONFIG_IP_SCTP_ADDIP
+       /* Add these addresses to all associations on this endpoint.  */
+       if (retval >= 0) {
+               list_t *pos;
+               sctp_endpoint_t *ep;
+               sctp_association_t *asoc;
+               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_ADD_IP,
+                                              addrs, addrcnt);
+               }
+       }
+#endif
+
+       return retval;
+}
+
+/* Remove a list of addresses from bind addresses list.  Do not remove the
+ * last address.
+ *
+ * Basically run through each address specified in the addrs/addrcnt
+ * array/length pair, determine if it is IPv6 or IPv4 and call
+ * sctp_del_bind() on it.
+ *
+ * If any of them fails, then the operation will be reversed and the
+ * ones that were removed will be added back.
+ *
+ * At least one address has to be left; if only one address is
+ * available, the operation will return -EBUSY.
+ *
+ * Only __sctp_bindx() is supposed to call this function.
+ */
+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;
+
+       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
+                * 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;
+                * 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:
+                       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);
+
+               sctp_write_unlock(&ep->base.addr_lock);
+               sctp_local_bh_enable();
+
+err_bindx_rem:
+               if (retval < 0) {
+                       /* Failed. Add the ones that has been removed back */
+                       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
+        * ADDIP code.
+        *                                                      --daisy
+        */
+#if CONFIG_IP_SCTP_ADDIP
+       /* Remove these addresses from all associations on this endpoint.  */
+       if (retval >= 0) {
+               list_t *pos;
+               sctp_endpoint_t *ep;
+               sctp_association_t *asoc;
+
+               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);
+               }
+       }
+#endif
+       return retval;
+}
+
+/* Helper for tunneling sys_bindx() requests through sctp_setsockopt()
+ *
+ * Basically do nothing but copying the addresses from user to kernel
+ * land and invoking sctp_bindx on the sk. This is used for tunneling
+ * the sctp_bindx() [sys_bindx()] request through sctp_setsockopt()
+ * from userspace.
+ *
+ * Note I don't use move_addr_to_kernel(): the reason is we would be
+ * iterating over an array of struct sockaddr_storage passing always
+ * what we know is a good size (sizeof (struct sock...)), so it is
+ * pointless. Instead check the whole area for read access and copy
+ * it.
+ *
+ * We don't use copy_from_user() for optimization: we first do the
+ * sanity checks (buffer size -fast- and access check-healthy
+ * pointer); if all of those succeed, then we can alloc the memory
+ * (expensive operation) needed to copy the data to kernel. Then we do
+ * the copying without checking the user space area
+ * (__copy_from_user()).
+ *
+ * On exit there is no need to do sockfd_put(), sys_setsockopt() does
+ * it.
+ *
+ * sk        The sk of the socket
+ * addrs     The pointer to the addresses in user land
+ * addrssize Size of the addrs buffer
+ * op        Operation to perform (add or remove, see the flags of
+ *           sctp_bindx)
+ *
+ * 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)
+{
+       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);
+
+       /* 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)))
+               return -EFAULT;
+
+       /* Alloc space for the address array in kernel memory.  */
+       kaddrs = (struct sockaddr_storage *) kmalloc(addrssize, GFP_KERNEL);
+       if (unlikely(NULL == kaddrs))
+               return -ENOMEM;
+
+       if (copy_from_user(kaddrs, addrs, addrssize)) {
+               kfree(kaddrs);
+               return -EFAULT;
+       }
+
+       addrcnt = addrssize / sizeof(struct sockaddr_storage);
+       err = __sctp_bindx(sk, kaddrs, addrcnt, op);   /* Do the work. */
+       kfree(kaddrs);
+
+       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.
+ *
+ * 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.
+ */
+static void sctp_close(struct sock *sk, long timeout)
+{
+       sctp_endpoint_t *ep;
+       sctp_association_t *asoc;
+       list_t *pos, *temp;
+
+       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.  */
+       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.  */
+       skb_queue_purge(&sk->receive_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);
+       inet_sock_release(sk);
+
+       sctp_bh_unlock_sock(sk);
+       sctp_local_bh_enable();
+
+       sock_put(sk);
+
+       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.
+ *
+ *  ssize_t sendmsg(int socket, const struct msghdr *message,
+ *                  int flags);
+ *
+ *  socket  - the socket descriptor of the endpoint.
+ *  message - pointer to the msghdr structure which contains a single
+ *            user message and possibly some ancillary data.
+ *
+ *            See Section 5 for complete description of the data
+ *            structures.
+ *
+ *  flags   - flags sent or received with the user message, see Section
+ *            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(). */
+
+static int sctp_msghdr_parse(const struct msghdr *, sctp_cmsgs_t *);
+
+static 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_transport_t *transport;
+       sctp_chunk_t *chunk = NULL;
+       sockaddr_storage_t to;
+       struct sockaddr *msg_name = NULL;
+       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_scope_t scope;
+       long timeo;
+       __u16 sinfo_flags = 0;
+
+       SCTP_DEBUG_PRINTK("sctp_sendmsg(sk: %p, msg: %p, "
+                         "size: %d)\n", sk, msg, size);
+
+       err = 0;
+       sp = sctp_sk(sk);
+       ep = sp->ep;
+
+       SCTP_DEBUG_PRINTK("Using endpoint: %s.\n", ep->debug_name);
+
+       /* Parse out the SCTP CMSGs.  */
+       err = sctp_msghdr_parse(msg, &cmsgs);
+
+       if (err) {
+               SCTP_DEBUG_PRINTK("msghdr parse err = %x\n", 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.
+        */
+       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);
+               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?  */
+       if (sinfo) {
+               sinfo_flags = sinfo->sinfo_flags;
+               associd = sinfo->sinfo_assoc_id;
+       }
+
+       SCTP_DEBUG_PRINTK("msg_len: %Zd, sinfo_flags: 0x%x\n",
+                         msg_len, sinfo_flags);
+
+       /* 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;
+               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 (msg_name) {
+               asoc = sctp_endpoint_lookup_assoc(ep, &to, &transport);
+       } else {
+               /* For a peeled-off socket, ignore any associd specified by
+                * the user with SNDRCVINFO.
+                */ 
+               if (SCTP_SOCKET_UDP_HIGH_BANDWIDTH == sp->type) {
+                       if (list_empty(&ep->asocs)) {
+                               err = -EINVAL;
+                               goto out_unlock;
+                       }
+                       asoc = list_entry(ep->asocs.next, sctp_association_t,
+                                         asocs);
+               } else if (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);
+               if (sinfo_flags & MSG_EOF) {
+                       SCTP_DEBUG_PRINTK("Shutting down association: %p\n",
+                                         asoc);
+                       sctp_primitive_SHUTDOWN(asoc, NULL);
+                       err = 0;
+                       goto out_unlock;
+               }
+               if (sinfo_flags & MSG_ABORT) {
+                       SCTP_DEBUG_PRINTK("Aborting association: %p\n",asoc);
+                       sctp_primitive_ABORT(asoc, NULL);
+                       err = 0;
+                       goto out_unlock;
+               }
+       }
+
+       /* 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.
+                */
+               if (sinfo) {
+                       if (!sinit ||
+                           (sinit && !sinit->sinit_num_ostreams)) {
+                               /* Check against the defaults. */
+                               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) {
+                                       err = -EINVAL;
+                                       goto out_unlock;
+                               }
+
+                               /* 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.
+                */
+               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 (!asoc) {
+                       err = -ENOMEM;
+                       goto out_unlock;
+               }
+
+               /* If the SCTP_INIT ancillary data is specified, set all
+                * the association init values accordingly.
+                */
+               if (sinit) {
+                       if (sinit->sinit_num_ostreams) {
+                               asoc->c.sinit_num_ostreams =
+                                       sinit->sinit_num_ostreams;
+                       }
+                       if (sinit->sinit_max_instreams) {
+                               if (sinit->sinit_max_instreams <=
+                                   SCTP_MAX_STREAM) {
+                                       asoc->c.sinit_max_instreams =
+                                               sinit->sinit_max_instreams;
+                               } else {
+                                       asoc->c.sinit_max_instreams =
+                                               SCTP_MAX_STREAM;
+                               }
+                       }
+                       if (sinit->sinit_max_attempts) {
+                               asoc->max_init_attempts
+                                       = sinit->sinit_max_attempts;
+                       }
+                       if (sinit->sinit_max_init_timeo) {
+                               asoc->max_init_timeo 
+                                       = sinit->sinit_max_init_timeo * HZ;
+                       }
+               }
+
+               /* Prime the peer's transport structures.  */
+               transport = sctp_assoc_add_peer(asoc, &to, GFP_KERNEL);
+       }
+
+       /* 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
+        * maximum size of data that can be sent in a single send call.
+        */
+       if (msg_len > sk->sndbuf) {
+               err = -EMSGSIZE;
+               goto out_free;
+       }
+
+       /* 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(__u16).
+        * The real fix is to fragment the message before creating the chunks.
+        */
+       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.
+        */
+       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;
+               }
+       } else {
+               /* If the user didn't specify SNDRCVINFO, make up one with
+                * some defaults.
+                */
+               default_sinfo.sinfo_stream  = asoc->defaults.stream;
+               default_sinfo.sinfo_ppid = asoc->defaults.ppid;
+               sinfo = &default_sinfo;
+       }
+
+       timeo = sock_sndtimeo(sk, msg->msg_flags & MSG_DONTWAIT);
+       if (!sctp_wspace(asoc)) {
+               err = sctp_wait_for_sndbuf(asoc, &timeo, msg_len);
+               if (err)
+                       goto out_free;
+       }
+
+       /* 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
+        * primary address override!  --jgrimm
+        */ 
+       /* If the user gave us an address, copy it in.  */
+       if (msg->msg_name) {
+               chunk->transport = sctp_assoc_lookup_paddr(asoc, &to);
+               if (!chunk->transport) {
+                       err = -EINVAL;
+                       goto out_free;
+               }
+       }
+#endif /* 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, (__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)
+                       goto out_free;
+               SCTP_DEBUG_PRINTK("We associated primitively.\n");
+       }
+
+       /* 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 (!err) {
+               err = msg_len;
+               goto out_unlock;
+       }
+
+out_free:
+       if (SCTP_STATE_CLOSED == asoc->state)
+               sctp_association_free(asoc);
+       if (chunk)
+               sctp_free_chunk(chunk);
+
+out_unlock:
+       sctp_release_sock(sk);
+
+out_nounlock:
+       return err;
+
+#if 0
+do_sock_err:
+       if (msg_len)
+               err = msg_len;
+       else
+               err = sock_error(sk);
+       goto out;
+
+do_interrupted:
+       if (msg_len)
+               err = msg_len;
+       goto out;
+#endif /* 0 */
+}
+
+/* API 3.1.3  recvmsg() - UDP Style Syntax
+ *
+ *  ssize_t recvmsg(int socket, struct msghdr *message,
+ *                    int flags);
+ *
+ *  socket  - the socket descriptor of the endpoint.
+ *  message - pointer to the msghdr structure which contains a single
+ *            user message and possibly some ancillary data.
+ *
+ *            See Section 5 for complete description of the data
+ *            structures.
+ *
+ *  flags   - flags sent or received with the user message, see Section
+ *            5 for complete description of the flags.
+ */
+static struct sk_buff * sctp_skb_recv_datagram(struct sock *, int, int, int *);
+
+static 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;
+
+       SCTP_DEBUG_PRINTK("sctp_recvmsg("
+                         "%s: %p, %s: %p, %s: %d, %s: %d, %s: "
+                         "0x%x, %s: %p)\n",
+                         "sk", sk,
+                         "msghdr", msg,
+                         "len", len,
+                         "knoblauch", noblock,
+                         "flags", flags,
+                         "addr_len", addr_len);
+
+       sctp_lock_sock(sk);
+       skb = sctp_skb_recv_datagram(sk, flags, noblock, &err);
+       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)
+                       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;
+
+       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,
+                                               addr_len, skb);
+       }
+
+       /* Check if we allow SCTP_SNDRCVINFO. */
+       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.  */
+       if (sk->protinfo.af_inet.cmsg_flags)
+               ip_cmsg_recv(msg, skb);
+#endif
+
+       err = copied;
+
+       /* FIXME: We need to support MSG_EOR correctly. */
+       msg->msg_flags |= MSG_EOR;
+
+out_free:
+       sctp_ulpevent_free(event); /* Free the skb. */
+out:
+       sctp_release_sock(sk);
+       return err;
+}
+
+static inline int sctp_setsockopt_disable_fragments(struct sock *sk, char *optval, int optlen)
+{
+       int val;
+
+       if (optlen < sizeof(int))
+               return -EINVAL;
+
+       if (get_user(val, (int *)optval))
+               return -EFAULT;
+
+       sctp_sk(sk)->disable_fragments = (val == 0) ? 0 : 1;
+
+       return 0;
+}
+
+static inline int sctp_setsockopt_set_events(struct sock *sk, char *optval, int optlen)
+{
+       if (optlen != sizeof(struct sctp_event_subscribe))
+               return -EINVAL;
+       if (copy_from_user(&sctp_sk(sk)->subscribe, optval, optlen))
+               return -EFAULT;
+       return 0;
+}
+
+static inline int sctp_setsockopt_autoclose(struct sock *sk, char *optval, int optlen)
+{
+       sctp_opt_t *sp = sctp_sk(sk);
+
+       if (optlen != sizeof(int))
+               return -EINVAL;
+       if (copy_from_user(&sp->autoclose, optval, optlen))
+               return -EFAULT;
+
+       sp->ep->timeouts[SCTP_EVENT_TIMEOUT_AUTOCLOSE] = sp->autoclose * HZ;
+       return 0;
+}
+
+/* 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);
+ *   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.
+ */
+static 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();
+       list_t *pos;
+       sctp_func_t *af;
+
+       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.
+        */
+       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,
+                                               optlen);
+                       if (retval < 0)
+                               goto out_nounlock;
+               }
+       }
+
+       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 *)
+                                              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 *)
+                                              optval, optlen, BINDX_REM_ADDR);
+               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;
+       };
+
+out_unlock:
+       sctp_release_sock(sk);
+
+out_nounlock:
+       return retval;
+}
+
+/* FIXME: Write comments. */
+static int sctp_connect(struct sock *sk, struct sockaddr *uaddr, int addr_len)
+{
+       return -EOPNOTSUPP; /* STUB */
+}
+
+/* FIXME: Write comments. */
+static int sctp_disconnect(struct sock *sk, int flags)
+{
+       return -EOPNOTSUPP; /* STUB */
+}
+
+/* FIXME: Write comments. */
+static struct sock *sctp_accept(struct sock *sk, int flags, int *err)
+{
+       int error = -EOPNOTSUPP;
+
+       *err = error; 
+       return NULL;
+}
+
+/* FIXME: Write Comments. */
+static int sctp_ioctl(struct sock *sk, int cmd, unsigned long arg)
+{
+       return -EOPNOTSUPP; /* STUB */
+}
+
+/* This is the function which gets called during socket creation to
+ * initialized the SCTP-specific portion of the sock.
+ * The sock structure should already be zero-filled memory.
+ */
+static int sctp_init_sock(struct sock *sk)
+{
+       sctp_endpoint_t *ep;
+       sctp_protocol_t *proto;
+       sctp_opt_t *sp;
+
+       SCTP_DEBUG_PRINTK("sctp_init_sock(sk: %p)\n", sk);
+
+       proto = sctp_get_protocol();
+
+       /* 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 (!ep)
+               return -ENOMEM;
+
+       sp = sctp_sk(sk);
+
+       /* Initialize the SCTP per socket area.  */
+
+       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).
+        */
+       sp->default_stream = 0;
+       sp->default_ppid = 0;
+
+       /* Initialize default setup parameters. These parameters
+        * can be modified with the SCTP_INITMSG socket option or
+        * overridden by the SCTP_INIT CMSG.
+        */
+       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_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.
+        */
+       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.
+        * 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) {
+               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;
+
+       /* 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.  */
+       sp->nodelay           = 0;
+
+       /* 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.
+        */
+       sp->autoclose         = 0;
+
+       SCTP_DBG_OBJCNT_INC(sock);
+       return 0;
+}
+
+/* Cleanup any SCTP per socket resources.  */
+static int sctp_destroy_sock(struct sock *sk)
+{
+       sctp_endpoint_t *ep;
+
+       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;
+}
+
+/* FIXME: Comments needed.  */
+static void sctp_shutdown(struct sock *sk, int how)
+{
+       /* UDP-style sockets do not support shutdown. */
+       /* STUB */
+}
+
+static int sctp_getsockopt_sctp_status(struct sock *sk, int len, char *optval,
+                                      int *optlen)
+{
+       struct sctp_status status;
+       sctp_endpoint_t *ep;
+       sctp_association_t *assoc = NULL;
+       sctp_transport_t *transport;
+       sctp_assoc_t associd;
+       int retval = 0;
+
+       if (len != sizeof(status)) {
+               retval = -EINVAL;
+               goto out_nounlock;
+       }
+
+       if (copy_from_user(&status, optval, sizeof(status))) {
+               retval = -EFAULT;
+               goto out_nounlock;
+       }
+
+       sctp_lock_sock(sk);
+
+       associd = status.sstat_assoc_id;
+       if ((SCTP_SOCKET_UDP_HIGH_BANDWIDTH != sctp_sk(sk)->type) && associd) {
+               assoc = sctp_id2assoc(sk, associd);
+               if (!assoc) {
+                       retval = -EINVAL;
+                       goto out_unlock;
+               }
+       } else {
+               ep = sctp_sk(sk)->ep;
+               if (list_empty(&ep->asocs)) {
+                       retval = -EINVAL;
+                       goto out_unlock;
+               }
+
+               assoc = list_entry(ep->asocs.next, sctp_association_t, asocs);
+       }
+
+       transport = assoc->peer.primary_path;
+
+       status.sstat_assoc_id = sctp_assoc2id(assoc);
+       status.sstat_state = assoc->state;
+       status.sstat_rwnd =  assoc->peer.rwnd;
+       status.sstat_unackdata = assoc->unack_data;
+       status.sstat_penddata = assoc->peer.tsn_map.pending_data;
+       status.sstat_instrms = assoc->c.sinit_max_instreams;
+       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));
+       status.sstat_primary.spinfo_state = transport->state.active;
+       status.sstat_primary.spinfo_cwnd = transport->cwnd;
+       status.sstat_primary.spinfo_srtt = transport->srtt;
+       status.sstat_primary.spinfo_rto = transport->rto;
+       status.sstat_primary.spinfo_mtu = transport->pmtu;
+
+       if (put_user(len, optlen)) {
+               retval = -EFAULT;
+               goto out_unlock;
+       }
+
+       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)) {
+               retval = -EFAULT;
+               goto out_unlock;
+       }
+
+out_unlock:
+       sctp_release_sock(sk);
+
+out_nounlock:
+       return (retval);
+}
+
+static inline int sctp_getsockopt_disable_fragments(struct sock *sk, int len,
+                                                   char *optval, int *optlen)
+{
+       int val;
+
+       if (len < sizeof(int))
+               return -EINVAL;
+
+       len = sizeof(int);
+       val = (sctp_sk(sk)->disable_fragments == 1);
+       if (put_user(len, optlen))
+               return -EFAULT;
+       if (copy_to_user(optval, &val, len))
+               return -EFAULT;
+       return 0;
+}
+
+static inline int sctp_getsockopt_set_events(struct sock *sk, int len, char *optval, int *optlen)
+{
+       if (len != sizeof(struct sctp_event_subscribe))
+               return -EINVAL;
+       if (copy_to_user(optval, &sctp_sk(sk)->subscribe, len))
+               return -EFAULT;
+       return 0;
+}
+
+static inline int sctp_getsockopt_autoclose(struct sock *sk, int len, char *optval, int *optlen)
+{
+       if (len != sizeof(int))
+               return -EINVAL;
+       if (copy_to_user(optval, &sctp_sk(sk)->autoclose, len))
+               return -EFAULT;
+       return 0;
+}
+
+/* Helper routine to branch off an association to a new socket.  */
+static int sctp_do_peeloff(sctp_association_t *assoc, struct socket **newsock)
+{
+       struct sock *oldsk = assoc->base.sk;
+       struct sock *newsk;
+       struct socket *tmpsock;
+       sctp_endpoint_t *newep;
+       sctp_opt_t *oldsp = sctp_sk(oldsk);
+       sctp_opt_t *newsp;
+       int err = 0;
+
+       /* An association cannot be branched off from an already peeled-off
+        * socket.
+        */
+       if (SCTP_SOCKET_UDP_HIGH_BANDWIDTH == sctp_sk(oldsk)->type)
+               return -EINVAL;
+
+       /* Create a new socket.  */
+       err = sock_create(PF_INET, SOCK_SEQPACKET, IPPROTO_SCTP, &tmpsock);
+       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.
+        */
+       newsk->sndbuf = oldsk->sndbuf;
+       newsk->rcvbuf = oldsk->rcvbuf;
+       *newsp = *oldsp;
+
+       /* Restore the ep value that was overwritten with the above structure
+        * copy.
+        */
+       newsp->ep = newep;
+
+       /* Set the type of socket to indicate that it is peeled off from the
+        * original socket.
+        */
+       newsp->type = SCTP_SOCKET_UDP_HIGH_BANDWIDTH;
+
+       /* Migrate the association to the new socket.  */
+       sctp_assoc_migrate(assoc, newsk);
+
+       *newsock = tmpsock;
+
+       return err;
+}
+
+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))
+               return -EINVAL;
+       if (copy_from_user(&peeloff, optval, len))
+               return -EFAULT;
+       assoc = sctp_id2assoc(sk, peeloff.associd);
+       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;
+
+       /* 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;
+       }
+
+       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.  */
+       peeloff.sd = sd;
+       if (copy_to_user(optval, &peeloff, len))
+               return -EFAULT;
+
+       return 0;
+}
+
+static 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;
+       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.
+        */
+       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,
+                                               optval, optlen);
+                       if (retval < 0)
+                               return retval;
+               }
+       }
+
+       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;
+}
+
+static void sctp_hash(struct sock *sk)
+{
+       /* STUB */
+}
+
+static void sctp_unhash(struct sock *sk)
+{
+       /* STUB */
+}
+
+/* Check if port is acceptable.  Possibly find first available port.
+ *
+ * The port hash table (contained in the 'global' SCTP protocol storage
+ * returned by sctp_protocol_t * sctp_get_protocol()). The hash
+ * table is an array of 4096 lists (sctp_bind_hashbucket_t). Each
+ * list (the list number is the port number hashed out, so as you
+ * would expect from a hash function, all the ports in a given list have
+ * such a number that hashes out to the same list number; you were
+ * expecting that, right?); so each list has a set of ports, with a
+ * link to the socket (struct sock) that uses it, the port number and
+ * a fastreuse flag (FIXME: NPI ipg).
+ */
+static long sctp_get_port_local(struct sock *sk, unsigned short snum)
+{
+       sctp_bind_hashbucket_t *head; /* hash list */
+       sctp_bind_bucket_t *pp; /* hash list port iterator */
+       sctp_protocol_t *sctp = sctp_get_protocol();
+       int ret;
+
+       SCTP_DEBUG_PRINTK("sctp_get_port() begins, snum=%d\n", snum);
+
+       sctp_local_bh_disable();
+
+       if (snum == 0) {
+               /* Search for an available port.
+                *
+                * 'sctp->port_rover' was the last port assigned, so
+                * we start to search from 'sctp->port_rover +
+                * 1'. What we do is first check if port 'rover' is
+                * 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 {
+                       rover++;
+                       if ((rover < low) || (rover > high))
+                               rover = low;
+                       index = sctp_phashfn(rover);
+                       head = &sctp->port_hashtable[index];
+                       sctp_spin_lock(&head->lock);
+                       for (pp = head->chain; pp; pp = pp->next)
+                               if (pp->port == rover)
+                                       goto next;
+                       break;
+               next:
+                       sctp_spin_unlock(&head->lock);
+               } while (--remaining > 0);
+               sctp->port_rover = rover;
+               sctp_spin_unlock(&sctp->port_alloc_lock);
+
+               /* Exhausted local port range during search? */
+               ret = 1;
+               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
+                * mutex.
+                */
+               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
+                * to the port number (snum) - we detect that with the
+                * port iterator, pp being NULL.
+                */
+               head = &sctp->port_hashtable[sctp_phashfn(snum)];
+               sctp_spin_lock(&head->lock);
+               for (pp = head->chain; pp; pp = pp->next) {
+                       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.
+                */
+               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)
+                       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;
+                       )
+                       break;
+
+               default:
+                       break;
+               };
+
+               /* Run through the list of sockets bound to the port
+                * (pp->port) [via the pointers bind_next and
+                * bind_pprev in the struct sock *sk2 (pp->sk)]. On each one,
+                * we get the endpoint they describe and run through
+                * the endpoint's list of IP (v4 or v6) addresses,
+                * 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.
+                */
+               for( ; sk2 != NULL; sk2 = sk2->bind_next) {
+                       sctp_endpoint_t *ep2;
+                       ep2 = sctp_sk(sk2)->ep;
+
+                       if (!sk_reuse || !sk2->reuse) {
+                               if (sctp_bind_addr_has_addr(
+                                       &ep2->base.bind_addr, &tmpaddr)) {
+                                       goto found;
+                               }
+                       }
+               }
+
+       found:
+               /* If we found a conflict, fail.  */
+               if (sk2 != NULL) {
+                       ret = (long) sk2;
+                       goto fail_unlock;
+               }
+               SCTP_DEBUG_PRINTK("sctp_get_port(): Found a match\n");
+       }
+
+       /* If there was a hash table miss, create a new port.  */
+       ret = 1;
+
+       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-).
+        */
+       if (pp->sk == NULL) {
+               pp->fastreuse = sk->reuse? 1 : 0;
+       } else if (pp->fastreuse && sk->reuse == 0) {
+               pp->fastreuse = 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).
+        */
+success:
+       inet_sk(sk)->num = snum;
+       if (sk->prev == 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;
+       }
+       ret = 0;
+
+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;
+}
+
+static int sctp_get_port(struct sock *sk, unsigned short snum)
+{
+       long ret = sctp_get_port_local(sk, snum);
+
+       return (ret ? 1 : 0);
+}
+
+/*
+ * 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.
+ */
+static 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;
+
+       /*
+        * 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.
+        *
+        * This is not currently spelled out in the SCTP sockets
+        * extensions draft, but follows the practice as seen in TCP
+        * sockets.
+        */
+       if (!ep->base.bind_addr.port) {
+               if (sctp_autobind(sk))
+                       return -EAGAIN;
+       }
+       sk->state = SCTP_SS_LISTENING;
+       sctp_hash_endpoint(ep);
+       return 0;
+}
+
+/*
+ *  Move a socket to LISTENING state.
+ */
+int sctp_inet_listen(struct socket *sock, int backlog)
+{
+       struct sock *sk = sock->sk;
+       int err;
+
+       sctp_lock_sock(sk);
+
+       err = -EINVAL;
+       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;
+}
+
+/*
+ * 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.
+ */
+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))
+               mask |= POLLERR;
+       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))
+               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.
+        */
+       if (SCTP_SOCKET_UDP != sctp_sk(sk)->type) {
+               /* The association is going away.  */
+               if (SCTP_SS_DISCONNECTING == sk->state)
+                       mask |= POLLHUP;
+               /* The association is either gone or not ready.  */
+               if (SCTP_SS_CLOSED == sk->state)
+                       return mask;
+       }
+
+       /* 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
+                * signal.  tcp_poll() has a race breaker for this race
+                * condition.  Based on their implementation, we put
+                * in the following code to cover it as well.
+                */
+               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)
+{
+       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) {
+               pp->port = snum;
+               pp->fastreuse = 0;
+               pp->sk = NULL;
+               if ((pp->next = head->chain) != 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;
+}
+
+/* FIXME: Commments! */
+static __inline__ void __sctp_put_port(struct sock *sk)
+{
+       sctp_protocol_t *sctp_proto = sctp_get_protocol();
+       sctp_bind_hashbucket_t *head =
+               &sctp_proto->port_hashtable[sctp_phashfn(inet_sk(sk)->num)];
+       sctp_bind_bucket_t *pp;
+
+       sctp_spin_lock(&head->lock);
+       pp = (sctp_bind_bucket_t *) sk->prev;
+       if (sk->bind_next)
+               sk->bind_next->bind_pprev = sk->bind_pprev;
+       *(sk->bind_pprev) = sk->bind_next;
+       sk->prev = NULL;
+       inet_sk(sk)->num = 0;
+       if (pp->sk) {
+               if (pp->next)
+                       pp->next->pprev = pp->pprev;
+               *(pp->pprev) = pp->next;
+               kfree(pp);
+       }
+       sctp_spin_unlock(&head->lock);
+}
+
+void sctp_put_port(struct sock *sk)
+{
+       sctp_local_bh_disable();
+       __sctp_put_port(sk);
+       sctp_local_bh_enable();
+}
+
+/*
+ * 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.
+ */
+static int sctp_autobind(struct sock *sk)
+{
+       sockaddr_storage_t autoaddr;
+       int addr_len = 0;
+
+       memset(&autoaddr, 0, sizeof(sockaddr_storage_t));
+
+       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);
+               addr_len = sizeof(struct sockaddr_in);
+               break;
+
+       case PF_INET6:
+               SCTP_V6(
+                       /* 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.  */
+               break;
+       };
+
+       return sctp_do_bind(sk, &autoaddr, addr_len);
+}
+
+/* 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
+ * 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
+ * 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.
+ *
+ *   |<--------------------------- msg_controllen -------------------------->|
+ *   |                                                                       |
+ *
+ *   |<----- ancillary data object ----->|<----- ancillary data object ----->|
+ *
+ *   |<---------- CMSG_SPACE() --------->|<---------- CMSG_SPACE() --------->|
+ *   |                                   |                                   |
+ *
+ *   |<---------- cmsg_len ---------->|  |<--------- cmsg_len ----------->|  |
+ *
+ *   |<--------- CMSG_LEN() --------->|  |<-------- CMSG_LEN() ---------->|  |
+ *   |                                |  |                                |  |
+ *
+ *   +-----+-----+-----+--+-----------+--+-----+-----+-----+--+-----------+--+
+ *   |cmsg_|cmsg_|cmsg_|XX|           |XX|cmsg_|cmsg_|cmsg_|XX|           |XX|
+ *
+ *   |len  |level|type |XX|cmsg_data[]|XX|len  |level|type |XX|cmsg_data[]|XX|
+ *
+ *   +-----+-----+-----+--+-----------+--+-----+-----+-----+--+-----------+--+
+ *    ^
+ *    |
+ *
+ * msg_control
+ * points here
+ */
+static int sctp_msghdr_parse(const struct msghdr *msg, sctp_cmsgs_t *cmsgs)
+{
+       struct cmsghdr *cmsg;
+
+       for (cmsg = CMSG_FIRSTHDR(msg);
+            cmsg != NULL;
+            cmsg = CMSG_NXTHDR((struct msghdr*)msg, cmsg)) {
+               /* 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)
+                       continue;
+
+               /* Strictly check lengths following example in SCM code.  */
+               switch (cmsg->cmsg_type) {
+               case SCTP_INIT:
+                       /* 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
+                        * structure.  This structure is not used for
+                        * recvmsg().
+                        *
+                        * cmsg_level    cmsg_type      cmsg_data[]
+                        * ------------  ------------   ----------------------
+                        * IPPROTO_SCTP  SCTP_INIT      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)
+                        * 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().
+                        *
+                        * cmsg_level    cmsg_type      cmsg_data[]
+                        * ------------  ------------   ----------------------
+                        * IPPROTO_SCTP  SCTP_SNDRCV    struct sctp_sndrcvinfo
+                        */
+                       if (cmsg->cmsg_len !=
+                           CMSG_LEN(sizeof(struct sctp_sndrcvinfo)))
+                               return -EINVAL;
+
+                       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))
+                               return -EINVAL;
+                       break;
+
+               default:
+                       return -EINVAL;
+               };
+       }
+       return 0;
+}
+
+/* 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) {
+       case AF_INET:
+               saveaddr->v4.sin_addr.s_addr = inet->rcv_saddr;
+               inet->rcv_saddr = inet->saddr = newaddr->v4.sin_addr.s_addr;
+               break;
+
+       case AF_INET6:
+               SCTP_V6({
+                       struct ipv6_pinfo *np = inet6_sk(sk);
+
+                       saveaddr->v6.sin6_addr = np->rcv_saddr;
+                       np->rcv_saddr = np->saddr = newaddr->v6.sin6_addr;
+                       break;
+               })
+
+       default:
+               break;
+       };
+}
+
+/* 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) {
+       case AF_INET:
+               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;
+       };
+}
+
+/*
+ * 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)
+{
+       int error;
+       DECLARE_WAITQUEUE(wait, current);
+
+       __set_current_state(TASK_INTERRUPTIBLE);
+       add_wait_queue_exclusive(sk->sleep, &wait);
+
+       /* Socket errors? */
+       error = sock_error(sk);
+       if (error)
+               goto out;
+
+       if (!skb_queue_empty(&sk->receive_queue))
+               goto ready;
+
+       /* Socket shut down?  */
+       if (sk->shutdown & RCV_SHUTDOWN)
+               goto out;
+
+       /* 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))
+               goto out;
+
+       /* Handle signals.  */
+       if (signal_pending(current))
+               goto interrupted;
+
+       /* 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.
+        */
+       sctp_release_sock(sk);
+       *timeo_p = schedule_timeout(*timeo_p);
+       sctp_lock_sock(sk);
+
+ready:
+       remove_wait_queue(sk->sleep, &wait);
+       __set_current_state(TASK_RUNNING);
+       return 0;
+
+interrupted:
+       error = sock_intr_errno(*timeo_p);
+
+out:
+       remove_wait_queue(sk->sleep, &wait);
+       __set_current_state(TASK_RUNNING);
+       *err = error;
+       return error;
+}
+
+/* Receive a datagram.
+ * Note: This is pretty much the same routine as in core/datagram.c
+ * with a few changes to make lksctp work.
+ */
+static 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()  */
+       error = sock_error(sk);
+       if (error)
+               goto no_packet;
+
+       timeo = sock_rcvtimeo(sk, noblock);
+
+       SCTP_DEBUG_PRINTK("Timeout: timeo: %ld, MAX: %ld.\n",
+                         timeo, MAX_SCHEDULE_TIMEOUT);
+
+       do {
+               /* 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,
+                                              cpu_flags);
+                       skb = skb_peek(&sk->receive_queue);
+                       if (skb)
+                               atomic_inc(&skb->users);
+                       sctp_spin_unlock_irqrestore(&sk->receive_queue.lock,
+                                                   cpu_flags);
+               } else {
+                       skb = skb_dequeue(&sk->receive_queue);
+               }
+
+               if (skb)
+                       return skb;
+
+               /* 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;
+}
+
+/* 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.  */
+       switch (sk->family) {
+       case PF_INET:
+               sin = (struct sockaddr_in *)msgname;
+               if (addr_len)
+                       *addr_len = sizeof(struct sockaddr_in);
+               sin->sin_family = AF_INET;
+               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
+                        * for an example. --jgrimm
+                        */
+               );
+               break;
+
+       default: /* Should not get here. */
+               break;
+       };
+}
+
+static inline int sctp_sendmsg_verify_name(struct sock *sk, struct msghdr *msg)
+{
+       sockaddr_storage_t *sa;
+
+       if (msg->msg_namelen < sizeof (struct sockaddr))
+               return -EINVAL;
+
+       sa = (sockaddr_storage_t *) msg->msg_name;
+       switch (sa->sa.sa_family) {
+       case AF_INET:
+               if (msg->msg_namelen < sizeof(struct sockaddr_in))
+                       return -EINVAL;
+               break;
+
+       case AF_INET6:
+               if (PF_INET == sk->family)
+                       return -EINVAL;
+               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))
+               return -EINVAL;
+
+       return 0;
+}
+
+/* Get the sndbuf space available at the time on the association.  */
+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)
+               amt = 0;
+       return amt;
+}
+
+/* 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.
+ *
+ * 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.
+ */
+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.  */
+       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;
+
+       asoc->sndbuf_used += SCTP_DATA_SNDSIZE(chunk);
+       sk->wmem_queued += SCTP_DATA_SNDSIZE(chunk);
+}
+
+/* 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)
+{
+       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;
+       sk = asoc->base.sk;
+       asoc->sndbuf_used -= SCTP_DATA_SNDSIZE(chunk);
+       sk->wmem_queued -= SCTP_DATA_SNDSIZE(chunk);
+       __sctp_write_space(asoc);
+
+       sctp_association_put(asoc);
+}
+
+/* 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",
+                         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)
+                       goto do_nonblock;
+               if (sk->err || asoc->state >= SCTP_STATE_SHUTDOWN_PENDING ||
+                   asoc->base.dead)
+                       goto do_error;
+               if (signal_pending(current))
+                       goto do_interrupted;
+               if (msg_len <= sctp_wspace(asoc))
+                       break;
+
+               /* 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.  */
+       sctp_association_put(asoc);
+
+       __set_current_state(TASK_RUNNING);
+       return err;
+
+do_error:
+       err = -EPIPE;
+       goto out;
+
+do_interrupted:
+       err = sock_intr_errno(*timeo_p);
+       goto out;
+
+do_nonblock:
+       err = -EAGAIN;
+       goto out;
+}
+
+/* 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))
+                       wake_up_interruptible(&asoc->wait);
+
+               if (sctp_writeable(sk)) {
+                       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.
+                        */
+                       if (sock->fasync_list &&
+                           !(sk->shutdown & SEND_SHUTDOWN))
+                               sock_wake_async(sock, 2, POLL_OUT);
+               }
+       }
+}
+
+/* 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;
+
+       /* Wake up the tasks in each wait queue.  */
+       list_for_each(pos, &((sctp_sk(sk))->ep->asocs)) {
+               asoc = list_entry(pos, sctp_association_t, asocs);
+               __sctp_write_space(asoc);
+       }
+}
+
+/* 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"
+ * 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
+ * UDP-style sockets or TCP-style sockets, this code should work.
+ *  - Daisy
+ */
+static int sctp_writeable(struct sock *sk)
+{
+       int amt = 0;
+
+       amt = sk->sndbuf - sk->wmem_queued;
+       if (amt < 0)
+               amt = 0;
+       return amt;
+}
+
+/* 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,
+};
diff --git a/net/sctp/sysctl.c b/net/sctp/sysctl.c
new file mode 100644 (file)
index 0000000..11b3bb6
--- /dev/null
@@ -0,0 +1,104 @@
+/* SCTP kernel reference Implementation 
+ * Copyright (c) 2002 International Business Machines Corp.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * Sysctl related interfaces for SCTP. 
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    Mingqin Liu           <liuming@us.ibm.com>
+ *    Jon Grimm             <jgrimm@us.ibm.com>
+ *
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <net/sctp/structs.h>
+#include <linux/sysctl.h>
+
+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 }
+};
+
+static ctl_table sctp_net_table[] = {
+       { 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 }
+};
+
+static struct ctl_table_header * sctp_sysctl_header;
+
+/* Sysctl registration.  */
+void sctp_sysctl_register(void)
+{
+       sctp_sysctl_header = register_sysctl_table(sctp_root_table, 0);
+}
+
+/* Sysctl deregistration.  */
+void sctp_sysctl_unregister(void)
+{
+       unregister_sysctl_table(sctp_sysctl_header);
+}
diff --git a/net/sctp/transport.c b/net/sctp/transport.c
new file mode 100644 (file)
index 0000000..ecc1ef4
--- /dev/null
@@ -0,0 +1,442 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 International Business Machines Corp.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001 La Monte H.P. Yarroll
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * This module provides the abstraction for an SCTP tranport representing
+ * a remote transport address.  For local transport addresses, we just use 
+ * sockaddr_storage_t.
+ *
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Karl Knutson          <karl@athena.chicago.il.us>
+ *    Jon Grimm             <jgrimm@us.ibm.com>
+ *    Xingang Guo           <xingang.guo@intel.com>
+ *    Hui Huang             <hui.huang@nokia.com>
+ *    Sridhar Samudrala            <sri@us.ibm.com>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/types.h>
+#include <net/sctp/sctp.h>
+
+/* 1st Level Abstractions.  */
+
+/* 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 (!transport)
+               goto fail;
+
+       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;
+}
+
+/* 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();
+
+       /* 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);
+
+       /* 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->srtt = 0;
+       peer->rto_pending = 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->max_retrans = proto->max_retrans_path;
+       peer->error_threshold = 0;
+       peer->error_count = 0;
+
+       peer->debug_name = "unnamedtransport";
+
+       INIT_LIST_HEAD(&peer->transmitted);
+       INIT_LIST_HEAD(&peer->send_ready);
+       INIT_LIST_HEAD(&peer->transports);
+
+       /* Set up the retransmission timer.  */
+       init_timer(&peer->T3_rtx_timer);
+       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;
+
+       atomic_set(&peer->refcnt, 1);
+       peer->dead = 0;
+
+       peer->malloced = 0;
+       return peer;
+}
+
+/* This transport is no longer needed.  Free up if possible, or
+ * delay until it last reference count.
+ */
+void sctp_transport_free(sctp_transport_t *transport)
+{
+       transport->dead = 1;
+
+       /* Try to delete the heartbeat timer.  */
+       if (del_timer(&transport->hb_timer))
+               sctp_transport_put(transport);
+
+       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)
+{
+       SCTP_ASSERT(transport->dead, "Transport is not dead", return);
+
+       if (transport->asoc)
+               sctp_association_put(transport->asoc);
+
+       kfree(transport);
+       SCTP_DBG_OBJCNT_DEC(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
+        * 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))
+                       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))
+               sctp_transport_hold(transport);
+}
+
+/* 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,
+                             sctp_association_t *asoc)
+{
+       transport->asoc = asoc;
+       sctp_association_hold(asoc);
+}
+
+/* Hold a reference to a transport.  */
+void sctp_transport_hold(sctp_transport_t *transport)
+{
+       atomic_inc(&transport->refcnt);
+}
+
+/* Release a reference to a transport and clean up
+ * if there are no more references.
+ */
+void sctp_transport_put(sctp_transport_t *transport)
+{
+       if (atomic_dec_and_test(&transport->refcnt))
+               sctp_transport_destroy(transport);
+}
+
+/* Update transport's RTO based on the newly calculated RTT. */ 
+void sctp_transport_update_rto(sctp_transport_t *tp, __u32 rtt)
+{
+       sctp_protocol_t *proto = sctp_get_protocol();
+
+       /* Check for valid transport.  */
+       SCTP_ASSERT(tp, "NULL transport", return);
+
+       /* 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'|
+                * SRTT <- (1 - RTO.Alpha) * SRTT + RTO.Alpha * R'
+                */
+
+               /* Note:  The above algorithm has been rewritten to
+                * express rto_beta and rto_alpha as inverse powers
+                * of two.
+                * For example, assuming the default value of RTO.Alpha of
+                * 1/8, rto_alpha would be expressed as 3.
+                */
+               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)
+                       + (rtt >> proto->rto_alpha);
+       } else {
+               /* 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
+        * adjust RTTVAR <- G, where G is the CLOCK GRANULARITY.
+        */
+       if (tp->rttvar == 0)
+               tp->rttvar = SCTP_CLOCK_GRANULARITY;
+
+       /* 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.
+        */
+       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
+        * at least RTO.max seconds.
+        */
+       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
+        * new data chunk is sent.
+        */
+       tp->rto_pending = 0;
+
+       SCTP_DEBUG_PRINTK("%s: transport: %p, rtt: %d, srtt: %d "
+                         "rttvar: %d, rto: %d\n", __FUNCTION__,
+                         tp, rtt, tp->srtt, tp->rttvar, tp->rto);
+}
+
+/* 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, __u32 sack_ctsn,
+                              __u32 bytes_acked)
+{
+       __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.
+        */
+       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
+                * 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.
+                */
+               if (bytes_acked > pmtu)
+                       cwnd += pmtu;
+               else
+                       cwnd += bytes_acked;
+               SCTP_DEBUG_PRINTK("%s: SLOW START: transport: %p, "
+                                 "bytes_acked: %d, cwnd: %d, ssthresh: %d, "
+                                 "flight_size: %d, pba: %d\n",
+                                 __FUNCTION__,
+                                 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
+                * 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
+                * (partial_bytes_acked - cwnd).
+                */
+               pba += bytes_acked;
+               if (pba >= cwnd) {
+                       cwnd += pmtu;
+                       pba = ((cwnd < pba) ? (pba - cwnd) : 0);
+               }
+               SCTP_DEBUG_PRINTK("%s: CONGESTION AVOIDANCE: "
+                                 "transport: %p, bytes_acked: %d, cwnd: %d, "
+                                 "ssthresh: %d, flight_size: %d, pba: %d\n",
+                                 __FUNCTION__,
+                                 transport, bytes_acked, cwnd,
+                                 ssthresh, flight_size, pba);
+       }
+
+       transport->cwnd = cwnd;
+       transport->partial_bytes_acked = pba;
+}
+
+/* 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) {
+       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
+                * perform slow start by:
+                *      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 
+                * were last sent, according to the formula described in 
+                * Section 7.2.3.
+                *
+                * 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
+                *      partial_bytes_acked = 0
+                */
+               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.
+                * 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
+                * 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
+                * 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,
+                                                 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.
+                */
+               if ((jiffies - transport->last_time_used) > transport->rto)
+                       transport->cwnd = 2*transport->asoc->pmtu;
+               break;
+       };
+
+       transport->partial_bytes_acked = 0;
+       SCTP_DEBUG_PRINTK("%s: transport: %p reason: %d cwnd: "
+                         "%d ssthresh: %d\n", __FUNCTION__,
+                         transport, reason,
+                         transport->cwnd, transport->ssthresh);
+}
diff --git a/net/sctp/tsnmap.c b/net/sctp/tsnmap.c
new file mode 100644 (file)
index 0000000..286fd18
--- /dev/null
@@ -0,0 +1,383 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 International Business Machines, Corp.
+ * Copyright (c) 2001 Intel Corp.
+ * 
+ * This file is part of the SCTP kernel reference Implementation
+ * 
+ * These functions manipulate sctp tsn mapping array.
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Jon Grimm             <jgrimm@us.ibm.com>
+ *    Karl Knutson          <karl@athena.chicago.il.us>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/types.h>
+#include <net/sctp/sctp.h>
+#include <net/sctp/sm.h>
+
+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(__u16 len, __u32 initial_tsn, int priority)
+{
+       sctp_tsnmap_t *retval;
+
+       retval = kmalloc(sizeof(sctp_tsnmap_t) +
+                        sctp_tsnmap_storage_size(len),
+                        priority);
+       if (!retval)
+               goto fail;
+
+       if (!sctp_tsnmap_init(retval, len, initial_tsn))
+               goto fail_map;
+       retval->malloced = 1;
+       return retval;
+
+fail_map:
+       kfree(retval);
+
+fail:
+       return NULL;
+}
+
+/* Initialize a block of memory as a tsnmap.  */
+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;
+}
+
+/* Test the tracking state of this TSN.
+ * Returns:
+ *   0 if the TSN has not yet been seen
+ *  >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, __u32 tsn)
+{
+       __s32 gap;
+       int dup;
+
+       /* 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)) {
+               dup = -1;
+               goto out;
+       }
+
+       /* Honk if we've already seen this TSN.
+        * We have three cases:
+        *      1. The TSN is ancient or belongs to a previous tsn_map.
+        *      2. The TSN is already marked in the tsn_map.
+        *      3. The TSN is already marked in the tsn_map_overflow.
+        */
+       if (gap < 0 ||
+           (gap < map->len && map->tsn_map[gap]) ||
+           (gap >= map->len && map->overflow_map[gap - map->len]))
+               dup = 1;
+       else
+               dup = 0;
+
+out:
+       return dup;
+}
+
+/* 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;
+}
+
+/* Mark this TSN as seen.  */
+void sctp_tsnmap_mark(sctp_tsnmap_t *map, __u32 tsn)
+{
+       __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))
+               return;
+       if (!TSN_lt(tsn, map->base_tsn + map->len + map->len))
+               return;
+
+       /* Bump the max.  */
+       if (TSN_lt(map->max_tsn_seen, tsn))
+               map->max_tsn_seen = tsn;
+
+       /* Assert: TSN is in range.  */
+       gap = tsn - map->base_tsn;
+
+       /* Mark the TSN as received.  */
+       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);
+}
+
+/* Retrieve the Cumulative TSN Ack Point. */
+__u32 sctp_tsnmap_get_ctsn(const sctp_tsnmap_t *map)
+{
+       return map->cumulative_tsn_ack_point;
+}
+
+/* Retrieve the highest TSN we've seen.  */
+__u32 sctp_tsnmap_get_max_tsn_seen(const sctp_tsnmap_t *map)
+{
+       return map->max_tsn_seen;
+}
+
+/* Dispose of a tsnmap.  */
+void sctp_tsnmap_free(sctp_tsnmap_t *map)
+{
+       if (map->malloced)
+               kfree(map);
+}
+
+/* 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)
+{
+       /* Only start looking one past the Cumulative TSN Ack Point.  */
+       iter->start = map->cumulative_tsn_ack_point + 1;
+}
+
+/* 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,
+                            __u16 *start, __u16 *end)
+{
+       int started, ended;
+       __u16 _start, _end, offset;
+
+       /* 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;
+               _sctp_tsnmap_find_gap_ack(map->tsn_map,
+                                         offset,
+                                         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)
+                       offset = 0;
+               else
+                       offset = iter->start - map->base_tsn - map->len;
+
+               /* Search the overflow map.  */
+               _sctp_tsnmap_find_gap_ack(map->overflow_map,
+                                         offset,
+                                         map->len,
+                                         map->len,
+                                         &started, &_start,
+                                         &ended, &_end);
+       }
+
+       /* The Gap Ack Block happens to end at the end of the
+        * overflow map.
+        */
+       if (started & !ended) {
+               ended++;
+               _end = map->len + map->len - 1;
+       }
+
+       /* If we found a Gap Ack Block, return the start and end and
+        * bump the iterator forward.
+        */
+       if (ended) {
+               /* 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;
+
+               *start = _start - gap;
+               *end = _end - gap;
+
+               /* Move the iterator forward.  */
+               iter->start = map->cumulative_tsn_ack_point + *end + 1;
+       }
+
+       return ended;
+}
+
+/********************************************************************
+ * 2nd Level Abstractions
+ ********************************************************************/
+
+/* This private helper function updates the tsnmap buffers and
+ * the Cumulative TSN Ack Point.
+ */
+static void _sctp_tsnmap_update(sctp_tsnmap_t *map)
+{
+       __u32 ctsn;
+
+       ctsn = map->cumulative_tsn_ack_point;
+       do {
+               ctsn++;
+               if (ctsn == map->overflow_tsn) {
+                       /* Now tsn_map must have been all '1's,
+                        * so we swap the map and check the overflow table
+                        */
+                       __u8 *tmp = map->tsn_map;
+                       memset(tmp, 0, map->len);
+                       map->tsn_map = map->overflow_map;
+                       map->overflow_map = tmp;
+
+                       /* Update the tsn_map boundaries.  */
+                       map->base_tsn += map->len;
+                       map->overflow_tsn += map->len;
+               }
+       } while (map->tsn_map[ctsn - map->base_tsn]);
+
+       map->cumulative_tsn_ack_point = ctsn - 1; /* Back up one. */
+       _sctp_tsnmap_update_pending_data(map);
+}
+
+static void _sctp_tsnmap_update_pending_data(sctp_tsnmap_t *map)
+{
+       __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))
+               goto out;
+
+       start = ((cum_tsn >= base_tsn) ? (cum_tsn - base_tsn + 1) : 0);
+       end = ((gap > map->len ) ? map->len : gap + 1);
+
+       for (i = start; i < end; i++) {
+               if (map->tsn_map[i])
+                       pending_data--;
+       } 
+
+       if (gap >= map->len) {
+               start = 0;
+               end = gap - map->len + 1;
+               for (i = start; i < end; i++) {
+                       if (map->overflow_map[i])
+                               pending_data--;
+               }
+       }
+
+out:
+       map->pending_data = pending_data;
+}
+
+/* This is a private helper for finding Gap Ack Blocks.  It searches a
+ * single array for the start and end of a Gap Ack Block.
+ *
+ * The flags "started" and "ended" tell is if we found the beginning
+ * or (respectively) the end of a Gap Ack Block.
+ */
+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 start. */
+       if (!(*started)) {
+               for (; i < len; i++) {
+                       if (map[i]) {
+                               (*started)++;
+                               *start = base + i;
+                               break;
+                       }
+               }
+       }
+
+       /* 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]) {
+                               (*ended)++;
+                               *end = base + i - 1;
+                               break;
+                       }
+               }
+       }
+}
diff --git a/net/sctp/ulpevent.c b/net/sctp/ulpevent.c
new file mode 100644 (file)
index 0000000..30a6fcf
--- /dev/null
@@ -0,0 +1,840 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001 International Business Machines, Corp.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001 Nokia, Inc.
+ * Copyright (c) 2001 La Monte H.P. Yarroll
+ * 
+ * These functions manipulate an sctp event.   The sctp_ulpevent_t is used
+ * to carry notifications and data to the ULP (sockets).  
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    Jon Grimm             <jgrimm@us.ibm.com>
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/types.h>
+#include <linux/skbuff.h>
+#include <net/sctp/structs.h>
+#include <net/sctp/sctp.h>
+#include <net/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);
+
+/* 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 (!skb)
+               goto fail;
+
+       event = (sctp_ulpevent_t *) skb->cb;
+       event = sctp_ulpevent_init(event, skb, msg_flags);
+       if (!event)
+               goto fail_init;
+
+       event->malloced = 1;
+       return event;
+
+fail_init:
+       kfree_skb(event->parent);
+
+fail:
+       return NULL;
+}
+
+/* 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;
+}
+
+/* Dispose of an event.  */
+void sctp_ulpevent_free(sctp_ulpevent_t *event)
+{
+       if (event->malloced)
+               kfree_skb(event->parent);
+}
+
+/* Is this a MSG_NOTIFICATION?  */
+int sctp_ulpevent_is_notification(const sctp_ulpevent_t *event)
+{
+       return event->msg_flags & MSG_NOTIFICATION;
+}
+
+/* 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,
+                                                __u16 flags,
+                                                __u16 state,
+                                                __u16 error,
+                                                __u16 outbound,
+                                                __u16 inbound,
+                                                int priority)
+{
+       sctp_ulpevent_t *event;
+       struct sctp_assoc_change *sac;
+
+       event = sctp_ulpevent_new(sizeof(struct sctp_assoc_change),
+                                 MSG_NOTIFICATION,
+                                 priority);
+       if (!event)
+               goto fail;
+
+       sac = (struct sctp_assoc_change *)
+               skb_put(event->parent,
+                       sizeof(struct sctp_assoc_change));
+
+       /* 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
+        * 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.
+        */
+       sac->sac_state = state;
+
+       /* 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
+        * 5.3.1.1 SCTP_ASSOC_CHANGE
+        *
+        * 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
+        * 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
+        * 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
+        * 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;    
+}
+
+/* Create and initialize an SCTP_PEER_ADDR_CHANGE event.
+ *
+ * Socket Extensions for SCTP - draft-01
+ * 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.
+ */
+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 (!event)
+               goto fail;
+
+       spc = (struct sctp_paddr_change *)
+               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
+        * Section 5.3.1.2 SCTP_PEER_ADDR_CHANGE
+        *
+        * 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);
+
+       /* Sockets API Extensions for SCTP
+        * Section 5.3.1.2 SCTP_PEER_ADDR_CHANGE
+        *
+        * spc_flags: 16 bits (unsigned integer)
+        * Currently unused.
+        */
+       spc->spc_flags = 0;
+
+       /* Sockets API Extensions for SCTP
+        * Section 5.3.1.2 SCTP_PEER_ADDR_CHANGE
+        *
+        * spc_state:  32 bits (signed integer)
+        * 
+        * This field holds one of a number of values that communicate the
+        * event that happened to the address.
+        */
+       spc->spc_state = state;
+
+       /* Sockets API Extensions for SCTP
+        * Section 5.3.1.2 SCTP_PEER_ADDR_CHANGE
+        *
+        * spc_error:  32 bits (signed integer)
+        *
+        * If the state was reached due to any error condition (e.g.
+        * ADDRESS_UNREACHABLE) any relevant error information is available in
+        * this field.
+        */
+       spc->spc_error = error;
+
+       /* 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
+        * Section 5.3.1.2 SCTP_PEER_ADDR_CHANGE
+        *
+        * spc_aaddr: sizeof (struct sockaddr_storage)
+        *
+        * The affected address field, holds the remote peer's address that is
+        * encountering the change of state.
+        */
+       memcpy(&spc->spc_aaddr, aaddr, sizeof(struct sockaddr_storage));
+
+       return event;
+
+fail:
+       return NULL;
+}
+
+/* Create and initialize an SCTP_REMOTE_ERROR notification.
+ *
+ * Note: This assumes that the chunk->skb->data already points to the
+ * operation error payload.
+ *
+ * Socket Extensions for SCTP - draft-01
+ * 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
+ * association. The entire error TLV as it appears on the wire is
+ * 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,
+                                                __u16 flags,
+                                                int priority)
+{
+       sctp_ulpevent_t *event;
+       struct sctp_remote_error *sre;
+       struct sk_buff *skb;
+       sctp_errhdr_t *ch;
+       __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.  */
+       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,
+                             sizeof(struct sctp_remote_error), /* headroom */
+                             0,                                /* tailroom */
+                             priority);
+
+       /* Pull off the rest of the cause TLV from the chunk.  */
+       skb_pull(chunk->skb, elen);
+       if (!skb)
+               goto fail;
+
+       /* Embed the event fields inside the cloned skb.  */
+       event = (sctp_ulpevent_t *) skb->cb;
+       event = sctp_ulpevent_init(event,
+                                  skb,
+                                  MSG_NOTIFICATION);
+       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.  */
+       skb_trim(skb, sizeof(struct sctp_remote_error) + elen);
+
+       /* 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
+        *
+        * 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 (__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
+        *
+        * 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;
+
+       /* 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);
+
+       return event;
+
+fail:
+       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,
+                                               __u16 flags,
+                                               __u32 error,
+                                               int priority)
+{
+       sctp_ulpevent_t *event;
+       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,
+                             sizeof(struct sctp_send_failed), /* headroom */
+                             0,                               /* tailroom */
+                             priority);
+       if (!skb)
+               goto fail;
+
+       /* 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;
+       event = sctp_ulpevent_init(event, skb, MSG_NOTIFICATION);
+       if (!event)
+               goto fail;
+
+       /* 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
+        * 5.3.1.4 SCTP_SEND_FAILED
+        *
+        * ssf_type:
+        * It should be SCTP_SEND_FAILED.
+        */
+       ssf->ssf_type = SCTP_SEND_FAILED;
+
+       /* 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.
+        *
+        * 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.
+        */
+       ssf->ssf_flags = flags;
+
+       /* Socket Extensions for SCTP
+        * 5.3.1.4 SCTP_SEND_FAILED
+        *
+        * 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
+        * 5.3.1.4 SCTP_SEND_FAILED
+        *
+        * ssf_error: 16 bits (unsigned integer)
+        * This value represents the reason why the send failed, and if set,
+        * will be a SCTP protocol error code as defined in [SCTP] section
+        * 3.3.10.
+        */
+       ssf->ssf_error = error;
+
+       /* 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.
+        */
+       memcpy(&ssf->ssf_info,
+              &chunk->sinfo,
+              sizeof(struct sctp_sndrcvinfo));
+
+       /* Socket Extensions for SCTP
+        * 5.3.1.4 SCTP_SEND_FAILED
+        *
+        * ssf_assoc_id: sizeof (sctp_assoc_t)
+        * The association id field, sf_assoc_id, 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.
+        */
+       ssf->ssf_assoc_id = sctp_assoc2id(asoc);
+
+       return event;
+
+fail:
+       return NULL;
+}
+
+/* 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,
+                                                  __u16 flags,
+                                                  int priority)
+{
+       sctp_ulpevent_t *event;
+       struct sctp_shutdown_event *sse;
+
+       event = sctp_ulpevent_new(sizeof(struct sctp_assoc_change),
+                                 MSG_NOTIFICATION,
+                                 priority);
+       if (!event)
+               goto fail;
+
+       sse = (struct sctp_shutdown_event *)
+               skb_put(event->parent,
+                       sizeof(struct sctp_shutdown_event));
+
+       /* 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
+        * 5.3.1.5 SCTP_SHUTDOWN_EVENT
+        *
+        * sse_flags: 16 bits (unsigned integer)
+        * Currently unused.
+        */
+       sse->sse_flags = 0;
+
+       /* Socket Extensions for SCTP
+        * 5.3.1.5 SCTP_SHUTDOWN_EVENT
+        *
+        * 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
+        * 5.3.1.5 SCTP_SHUTDOWN_EVENT
+        *
+        * sse_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.
+        */
+       sse->sse_assoc_id = sctp_assoc2id(asoc);
+
+       return 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 *event;
+       struct sctp_sndrcvinfo *info;
+       struct sk_buff *skb;
+       size_t padding, len;
+
+       /* Clone the original skb, sharing the data.  */
+       skb = skb_clone(chunk->skb, priority);
+       if (!skb)
+               goto fail;
+               
+       /* 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
+        *
+        * 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
+        * multiple of 4 bytes, the sender MUST 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.  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.  */
+       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); 
+       if (!event)
+               goto fail_init;
+
+       event->malloced = 1;
+
+       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);
+
+       /* Sockets API Extensions for SCTP
+        * Section 5.2.2 SCTP Header Information Structure (SCTP_SNDRCV)
+        *
+        * sinfo_ssn: 16 bits (unsigned integer)
+        *
+        * For recvmsg() this value contains the stream sequence number that
+        * the remote endpoint placed in the DATA chunk.  For fragmented
+        * 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);
+
+        /* Sockets API Extensions for SCTP
+        * Section 5.2.2 SCTP Header Information Structure (SCTP_SNDRCV)
+        *
+        * sinfo_ppid: 32 bits (unsigned integer)
+        *
+        * In recvmsg() this value is
+        * the same information that was passed by the upper layer in the peer
+        * application.  Please note that byte order issues are NOT accounted
+        * for and this information is passed opaquely by the SCTP stack from
+        * one end to the other.
+        */
+       info->sinfo_ppid = ntohl(chunk->subh.data_hdr->ppid);
+
+       /* Sockets API Extensions for SCTP
+        * Section 5.2.2 SCTP Header Information Structure (SCTP_SNDRCV)
+        *
+        * 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.
+        */
+       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
+        * its difficult to fix everything at the same time.   Eventually,
+        * we should create and skb based chunk structure.   This structure
+        * storage can be converted to an event.  --jgrimm
+        */
+       event->chunk_flags = chunk->chunk_hdr->flags;
+
+       /* 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;
+
+       /* 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);
+
+       return event;
+
+fail_init:
+       kfree_skb(skb);
+
+fail:
+       return NULL;
+}
+
+/* Return the notification type, assuming this is a notification
+ * event.
+ */
+__u16 sctp_ulpevent_get_notification_type(const sctp_ulpevent_t *event)
+{
+       union sctp_notification *notification;
+
+       notification = (union sctp_notification *) event->parent->data;
+       return notification->h.sn_type;
+}
+
+/* Copy out the sndrcvinfo into a 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);
+       }
+}
+
+/* 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;
+
+       /* 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;
+       asoc = event->asoc;
+       if (asoc->rwnd_over) {
+               if (asoc->rwnd_over >= skb->len) {
+                       asoc->rwnd_over -= skb->len;
+               } else {
+                       asoc->rwnd += (skb->len - asoc->rwnd_over);
+                       asoc->rwnd_over = 0;
+               }
+       } else {
+               asoc->rwnd += skb->len;
+       }
+
+       SCTP_DEBUG_PRINTK("rwnd increased by %d to (%u, %u)\n",
+                         skb->len, asoc->rwnd, asoc->rwnd_over);
+
+       sctp_association_put(asoc);
+}
+
+/* Charge receive window for bytes recieved.  */
+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
+        * association.
+        */
+       sctp_association_hold(asoc);
+       skb->sk = asoc->base.sk;
+       event = (sctp_ulpevent_t *) skb->cb;
+       event->asoc = asoc;
+
+       skb->destructor = sctp_rcvmsg_rfree;
+
+       SCTP_ASSERT(asoc->rwnd, "rwnd zero", return);
+       SCTP_ASSERT(!asoc->rwnd_over, "rwnd_over not zero", return);
+       if (asoc->rwnd >= skb->len) {
+               asoc->rwnd -= skb->len;
+       } else {
+               asoc->rwnd_over = skb->len - asoc->rwnd;
+               asoc->rwnd = 0;
+       }
+
+       SCTP_DEBUG_PRINTK("rwnd decreased by %d to (%u, %u)\n",
+                         skb->len, asoc->rwnd, asoc->rwnd_over);
+}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/net/sctp/ulpqueue.c b/net/sctp/ulpqueue.c
new file mode 100644 (file)
index 0000000..f2011e0
--- /dev/null
@@ -0,0 +1,476 @@
+/* SCTP kernel reference Implementation
+ * Copyright (c) 1999-2000 Cisco, Inc.
+ * Copyright (c) 1999-2001 Motorola, Inc.
+ * Copyright (c) 2001-2002 International Business Machines, Corp.
+ * Copyright (c) 2001 Intel Corp.
+ * Copyright (c) 2001 Nokia, Inc.
+ * Copyright (c) 2001 La Monte H.P. Yarroll
+ * 
+ * This abstraction carries sctp events to the ULP (sockets).
+ * 
+ * The SCTP reference implementation is free software; 
+ * you can redistribute it and/or modify it under the terms of 
+ * the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ * 
+ * The SCTP reference implementation is distributed in the hope that it 
+ * will be useful, but WITHOUT ANY WARRANTY; without even the implied
+ *                 ************************
+ * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ * 
+ * You should have received a copy of the GNU General Public License
+ * along with GNU CC; see the file COPYING.  If not, write to
+ * the Free Software Foundation, 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.  
+ * 
+ * Please send any bug reports or fixes you make to the
+ * email address(es):
+ *    lksctp developers <lksctp-developers@lists.sourceforge.net>
+ * 
+ * Or submit a bug report through the following website:
+ *    http://www.sf.net/projects/lksctp
+ *
+ * Written or modified by: 
+ *    Jon Grimm             <jgrimm@us.ibm.com>
+ *    La Monte H.P. Yarroll <piggy@acm.org>
+ *    Sridhar Samudrala     <sri@us.ibm.com>
+ * 
+ * Any bugs reported given to us we will try to fix... any fixes shared will
+ * be incorporated into the next SCTP release.
+ */
+
+#include <linux/types.h>
+#include <linux/skbuff.h>
+#include <net/sock.h>
+#include <net/sctp/structs.h>
+#include <net/sctp/sctp.h>
+#include <net/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);
+
+/* 1st Level Abstractions */
+
+/* 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
+        * the future.
+        */
+       size = sizeof(sctp_ulpqueue_t)+sctp_ulpqueue_storage_size(inbound);
+       ulpq = kmalloc(size, priority);
+       if (!ulpq)
+               goto fail;
+       if (!sctp_ulpqueue_init(ulpq, asoc, inbound))
+               goto fail_init;
+       ulpq->malloced = 1;
+       return ulpq;
+
+fail_init:
+       kfree(ulpq);
+
+fail:
+       return NULL;
+}
+
+/* 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),
+              0x00);
+
+       ulpq->asoc = asoc;
+       spin_lock_init(&ulpq->lock);
+       skb_queue_head_init(&ulpq->reasm);
+       skb_queue_head_init(&ulpq->lobby);
+       ulpq->malloced = 0;
+
+       return 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);
+       }
+}
+
+/* Dispose of a ulpqueue.  */
+void sctp_ulpqueue_free(sctp_ulpqueue_t *ulpq)
+{
+       sctp_ulpqueue_flush(ulpq);
+       if (ulpq->malloced)
+               kfree(ulpq);
+}
+
+/* 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
+        * 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)
+               return -ENOMEM;
+
+       /* Do reassembly if needed.  */
+       event = sctp_ulpqueue_reasm(ulpq, event);
+
+       /* 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 (event)
+               sctp_ulpqueue_tail_event(ulpq, event);
+
+       return 0;
+}
+
+/* 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 = 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))
+               goto out_free;
+
+       /* 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 (event->parent->list)
+               sctp_skb_list_tail(event->parent->list, &sk->receive_queue);
+       else
+               skb_queue_tail(&sk->receive_queue, event->parent);
+
+       wake_up_interruptible(sk->sleep);
+       return 1;
+
+out_free:
+       if (event->parent->list)
+               skb_queue_purge(event->parent->list);
+       else
+               kfree_skb(event->parent);
+       return 0;
+}
+
+/* 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)
+{
+       struct sk_buff *pos, *tmp;
+       sctp_ulpevent_t *cevent;
+       __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.  */
+       sctp_skb_for_each(pos, &ulpq->reasm, tmp) {
+               cevent = (sctp_ulpevent_t *)pos->cb;
+               ctsn = cevent->sndrcvinfo.sinfo_tsn;
+
+               if (TSN_lt(tsn, ctsn))
+                       break;
+       }
+
+       /* 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
+               __skb_queue_tail(&ulpq->reasm, event->parent);
+
+       sctp_spin_unlock_irqrestore(&ulpq->reasm.lock, flags);
+}
+
+/* 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)
+{
+       struct sk_buff *pos;
+       sctp_ulpevent_t *event;
+       struct sk_buff *pnext;
+
+       pos = f_frag->next;
+
+       /* 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.  */
+       __skb_unlink(f_frag, f_frag->list);
+       do {
+               pnext = pos->next;
+
+               /* 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;
+
+               pos->next = pnext;
+               pos = pnext;
+       } while (1);
+
+       event = (sctp_ulpevent_t *) f_frag->cb;
+
+       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.
+ */
+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;
+       __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
+        * 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.
+        * Walk through the queue sequentially and look for a sequence of
+        * 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.
+        */
+       sctp_skb_for_each(pos, &ulpq->reasm, tmp) {
+               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;
+                       break;
+
+               case SCTP_DATA_MIDDLE_FRAG:
+                       if ((first_frag) && (ctsn == next_tsn))
+                               next_tsn++;
+                       else
+                               first_frag = NULL;
+                       break;
+
+               case SCTP_DATA_LAST_FRAG:
+                       if ((first_frag) && (ctsn == next_tsn))
+                               retval = sctp_make_reassembled_event(
+                                               first_frag, pos);
+                       else
+                               first_frag = NULL;
+                       break;
+               };
+
+               /* We have the reassembled event. There is no need to look
+                * further.
+                */
+               if (retval)
+                       break;
+       }
+       sctp_spin_unlock_irqrestore(&ulpq->reasm.lock, flags);
+
+       return retval;
+}
+
+/* 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)
+{
+       sctp_ulpevent_t *retval = NULL;
+
+       /* 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
+        * at once.
+        */
+
+       /* 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;
+}
+
+/* Helper function to gather skbs that have possibly become
+ * ordered by an an incoming chunk.
+ */
+static inline void sctp_ulpqueue_retrieve_ordered(sctp_ulpqueue_t *ulpq,
+                                                 sctp_ulpevent_t *event)
+{
+       struct sk_buff *pos, *tmp;
+       sctp_ulpevent_t *cevent;
+       __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.  */
+       sctp_spin_lock_irqsave(&ulpq->lobby.lock, flags);
+       sctp_skb_for_each(pos, &ulpq->lobby, tmp) {
+               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;
+
+               if (cssn != ulpq->ssn[sid])
+                       break;
+
+               ulpq->ssn[sid]++;
+               __skb_unlink(pos, pos->list);
+
+               /* Attach all gathered skbs to the event.  */
+               __skb_queue_tail(event->parent->list, pos);
+       }
+       sctp_spin_unlock_irqrestore(&ulpq->lobby.lock, flags);
+}
+
+/* Helper function to store chunks needing ordering.  */
+static inline void sctp_ulpqueue_store_ordered(sctp_ulpqueue_t *ulpq,
+                                              sctp_ulpevent_t *event)
+{
+       struct sk_buff *pos, *tmp;
+       sctp_ulpevent_t *cevent;
+       __u16 sid, csid;
+       __u16 ssn, cssn;
+       unsigned long flags __attribute ((unused));
+
+       sid = event->sndrcvinfo.sinfo_stream;
+       ssn = event->sndrcvinfo.sinfo_ssn;
+
+       sctp_spin_lock_irqsave(&ulpq->lobby.lock, flags);
+
+       /* Find the right place in this list.  We store them by
+        * stream ID and then by SSN.
+        */
+       sctp_skb_for_each(pos, &ulpq->lobby, tmp) {
+               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 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
+               __skb_queue_tail(&ulpq->lobby, event->parent);
+
+       sctp_spin_unlock_irqrestore(&ulpq->lobby.lock, flags);
+}
+
+static inline sctp_ulpevent_t *sctp_ulpqueue_order(sctp_ulpqueue_t *ulpq,
+                                                  sctp_ulpevent_t *event)
+{
+       __u16 sid, ssn;
+
+       /* 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
+        * 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);
+
+       return event;
+}