diff options
Diffstat (limited to 'src/plugins/sctp/sctp_packet.h')
-rw-r--r-- | src/plugins/sctp/sctp_packet.h | 1470 |
1 files changed, 0 insertions, 1470 deletions
diff --git a/src/plugins/sctp/sctp_packet.h b/src/plugins/sctp/sctp_packet.h deleted file mode 100644 index 04995aa7ee3..00000000000 --- a/src/plugins/sctp/sctp_packet.h +++ /dev/null @@ -1,1470 +0,0 @@ -/* - * Copyright (c) 2017 SUSE LLC. - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at: - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef included_vnet_sctp_packet_h -#define included_vnet_sctp_packet_h - -#include <stdbool.h> - -#include <vnet/ip/ip4_packet.h> -#include <vnet/ip/ip6_packet.h> - -/* - * As per RFC 4960 - * https://tools.ietf.org/html/rfc4960 - */ - -/* - * 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 - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Source Port Number | Destination Port Number | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Verification Tag | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Checksum | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - /* - * This is the SCTP sender's port number. It can be used by the - * receiver in combination with the source IP address, the SCTP - * destination port, and possibly the destination IP address to - * identify the association to which this packet belongs. - * The port number 0 MUST NOT be used. - */ - u16 src_port; - - /* - * This is the SCTP port number to which this packet is destined. - * The receiving host will use this port number to de-multiplex the - * SCTP packet to the correct receiving endpoint/application. - * The port number 0 MUST NOT be used. - */ - u16 dst_port; - - /* - * The receiver of this packet uses the Verification Tag to validate - * the sender of this SCTP packet. On transmit, the value of this - * Verification Tag MUST be set to the value of the Initiate Tag - * received from the peer endpoint during the association - * initialization, with the following exceptions: - * - A packet containing an INIT chunk MUST have a zero Verification - * Tag. - * - A packet containing a SHUTDOWN COMPLETE chunk with the T bit - * set MUST have the Verification Tag copied from the packet with - * the SHUTDOWN ACK chunk. - * - A packet containing an ABORT chunk may have the verification tag - * copied from the packet that caused the ABORT to be sent. - * An INIT chunk MUST be the only chunk in the SCTP packet carrying it. - */ - u32 verification_tag; - - /* - * This field contains the checksum of this SCTP packet. - * SCTP uses the CRC32c algorithm. - */ - u32 checksum; - -} sctp_header_t; - -always_inline void -vnet_set_sctp_src_port (sctp_header_t * h, u16 src_port) -{ - h->src_port = clib_host_to_net_u16 (src_port); -} - -always_inline u16 -vnet_get_sctp_src_port (sctp_header_t * h) -{ - return (clib_net_to_host_u16 (h->src_port)); -} - -always_inline void -vnet_set_sctp_dst_port (sctp_header_t * h, u16 dst_port) -{ - h->dst_port = clib_host_to_net_u16 (dst_port); -} - -always_inline u16 -vnet_get_sctp_dst_port (sctp_header_t * h) -{ - return (clib_net_to_host_u16 (h->dst_port)); -} - -always_inline void -vnet_set_sctp_verification_tag (sctp_header_t * h, u32 verification_tag) -{ - h->verification_tag = clib_host_to_net_u32 (verification_tag); -} - -always_inline u32 -vnet_get_sctp_verification_tag (sctp_header_t * h) -{ - return (clib_net_to_host_u32 (h->verification_tag)); -} - -always_inline void -vnet_set_sctp_checksum (sctp_header_t * h, u32 checksum) -{ - h->checksum = clib_host_to_net_u32 (checksum); -} - -always_inline u32 -vnet_get_sctp_checksum (sctp_header_t * h) -{ - return (clib_net_to_host_u32 (h->checksum)); -} - -/* - * 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. - * - * - * 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 - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Common Header | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Chunk #1 | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | ... | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Chunk #n | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ - -typedef enum -{ - DATA = 0, - INIT, - INIT_ACK, - SACK, - HEARTBEAT, - HEARTBEAT_ACK, - ABORT, - SHUTDOWN, - SHUTDOWN_ACK, - OPERATION_ERROR, - COOKIE_ECHO, - COOKIE_ACK, - ECNE, - CWR, - SHUTDOWN_COMPLETE, - UNKNOWN -} sctp_chunk_type; - -/* - * 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 | Chunk Flags | Chunk Length | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - /* - * This field identifies the type of information contained in the - * Chunk Value field. It takes a value from 0 to 254. - * The value of 255 is reserved for future use as an extension field. - * - * The values of Chunk Types are defined as follows: - * ID Value Chunk Type - * ----- ---------- - * 0 - Payload Data (DATA) - * 1 - Initiation (INIT) - * 2 - Initiation Acknowledgement (INIT ACK) - * 3 - Selective Acknowledgement (SACK) - * 4 - Heartbeat Request (HEARTBEAT) - * 5 - Heartbeat Acknowledgement (HEARTBEAT ACK) - * 6 - Abort (ABORT) - * 7 - Shutdown (SHUTDOWN) - * 8 - Shutdown Acknowledgement (SHUTDOWN ACK) - * 9 - Operation Error (ERROR) - * 10 - State Cookie (COOKIE ECHO) - * 11 - Cookie Acknowledgement (COOKIE ACK) - * 12 - Reserved for Explicit Congestion Notification Echo (ECNE) - * 13 - Reserved for Congestion Window Reduced (CWR) - * 14 - Shutdown Complete (SHUTDOWN COMPLETE) - * 15 to 62 - available - * 63 - reserved for IETF-defined Chunk Extensions - * 64 to 126 - available - * 127 - reserved for IETF-defined Chunk Extensions - * 128 to 190 - available - * 191 - reserved for IETF-defined Chunk Extensions - * 192 to 254 - available - * 255 - reserved for IETF-defined Chunk Extensions - * - * Chunk Types are encoded such that the highest-order 2 bits specify - * the action that must be taken if the processing endpoint does not - * recognize the Chunk Type. - * 00 - Stop processing this SCTP packet and discard it, do not - * process any further chunks within it. - * 01 - Stop processing this SCTP packet and discard it, do not - * process any further chunks within it, and report the - * unrecognized chunk in an 'Unrecognized Chunk Type'. - * 10 - Skip this chunk and continue processing. - * 11 - Skip this chunk and continue processing, but report in an - * ERROR chunk using the 'Unrecognized Chunk Type' cause of error. - * - * Note: The ECNE and CWR chunk types are reserved for future use of - * Explicit Congestion Notification (ECN); - */ - //u8 type; - - /* - * The usage of these bits depends on the Chunk type as given by the - * Chunk Type field. Unless otherwise specified, they are set to 0 on - * transmit and are ignored on receipt. - */ - //u8 flags; - - /* - * This value represents the size of the chunk in bytes, including - * the Chunk Type, Chunk Flags, Chunk Length, and Chunk Value fields. - * Therefore, if the Chunk Value field is zero-length, the Length - * field will be set to 4. - * The Chunk Length field does not count any chunk padding. - * Chunks (including Type, Length, and Value fields) are padded out - * by the sender with all zero bytes to be a multiple of 4 bytes - * long. This padding MUST NOT be more than 3 bytes in total. The - * Chunk Length value does not include terminating padding of the - * chunk. However, it does include padding of any variable-length - * parameter except the last parameter in the chunk. The receiver - * MUST ignore the padding. - * - * Note: A robust implementation should accept the chunk whether or - * not the final padding has been included in the Chunk Length. - */ - //u16 length; - - u32 params; - -} sctp_chunks_common_hdr_t; - -typedef struct -{ - sctp_header_t hdr; - sctp_chunks_common_hdr_t common_hdr; - -} sctp_full_hdr_t; - -#define CHUNK_TYPE_MASK 0xFF000000 -#define CHUNK_TYPE_SHIFT 24 - -#define CHUNK_FLAGS_MASK 0x00FF0000 -#define CHUNK_FLAGS_SHIFT 16 - -#define CHUNK_UBIT_MASK 0x00040000 -#define CHUNK_UBIT_SHIFT 18 - -#define CHUNK_BBIT_MASK 0x00020000 -#define CHUNK_BBIT_SHIFT 17 - -#define CHUNK_EBIT_MASK 0x00010000 -#define CHUNK_EBIT_SHIFT 16 - -#define CHUNK_LENGTH_MASK 0x0000FFFF -#define CHUNK_LENGTH_SHIFT 0 - -always_inline void -vnet_sctp_common_hdr_params_host_to_net (sctp_chunks_common_hdr_t * h) -{ - h->params = clib_host_to_net_u32 (h->params); -} - -always_inline void -vnet_sctp_common_hdr_params_net_to_host (sctp_chunks_common_hdr_t * h) -{ - h->params = clib_net_to_host_u32 (h->params); -} - -always_inline void -vnet_sctp_set_ubit (sctp_chunks_common_hdr_t * h) -{ - h->params &= ~(CHUNK_UBIT_MASK); - h->params |= (1 << CHUNK_UBIT_SHIFT) & CHUNK_UBIT_MASK; -} - -always_inline u8 -vnet_sctp_get_ubit (sctp_chunks_common_hdr_t * h) -{ - return ((h->params & CHUNK_UBIT_MASK) >> CHUNK_UBIT_SHIFT); -} - -always_inline void -vnet_sctp_set_bbit (sctp_chunks_common_hdr_t * h) -{ - h->params &= ~(CHUNK_BBIT_MASK); - h->params |= (1 << CHUNK_BBIT_SHIFT) & CHUNK_BBIT_MASK; -} - -always_inline u8 -vnet_sctp_get_bbit (sctp_chunks_common_hdr_t * h) -{ - return ((h->params & CHUNK_BBIT_MASK) >> CHUNK_BBIT_SHIFT); -} - -always_inline void -vnet_sctp_set_ebit (sctp_chunks_common_hdr_t * h) -{ - h->params &= ~(CHUNK_EBIT_MASK); - h->params |= (1 << CHUNK_EBIT_SHIFT) & CHUNK_EBIT_MASK; -} - -always_inline u8 -vnet_sctp_get_ebit (sctp_chunks_common_hdr_t * h) -{ - return ((h->params & CHUNK_EBIT_MASK) >> CHUNK_EBIT_SHIFT); -} - -always_inline void -vnet_sctp_set_chunk_type (sctp_chunks_common_hdr_t * h, sctp_chunk_type t) -{ - h->params &= ~(CHUNK_TYPE_MASK); - h->params |= (t << CHUNK_TYPE_SHIFT) & CHUNK_TYPE_MASK; -} - -always_inline u8 -vnet_sctp_get_chunk_type (sctp_chunks_common_hdr_t * h) -{ - return ((h->params & CHUNK_TYPE_MASK) >> CHUNK_TYPE_SHIFT); -} - -always_inline void -vnet_sctp_set_chunk_length (sctp_chunks_common_hdr_t * h, u16 length) -{ - h->params &= ~(CHUNK_LENGTH_MASK); - h->params |= (length << CHUNK_LENGTH_SHIFT) & CHUNK_LENGTH_MASK; -} - -always_inline u16 -vnet_sctp_get_chunk_length (sctp_chunks_common_hdr_t * h) -{ - return ((h->params & CHUNK_LENGTH_MASK) >> CHUNK_LENGTH_SHIFT); -} - -/* - * Payload chunk - * - * 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 = 0 | Reserved|U|B|E| Length | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | TSN | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Stream Identifier S | Stream Sequence Number n | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Payload Protocol Identifier | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * \ \ - * / User Data (seq n of Stream S) / - * \ \ - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - sctp_header_t sctp_hdr; - /* - * Type (8 bits): 0 - * Flags (8 bits): - * -- Reserved (5 bits): all 0s - * -- U bit - * -- B bit - * -- E bit - * Length (16 bits): This field indicates the length of the DATA chunk in - * bytes from the beginning of the type field to the end of the User Data - * field excluding any padding. - * A DATA chunk with one byte of user data will have Length set to 17 - * (indicating 17 bytes). A DATA chunk with a User Data field of length L - * will have the Length field set to (16 + L) (indicating 16+L bytes) where - * L MUST be greater than 0. - */ - - /* - * Fragment Description Table: - * - * B E Description - * ============================================================ - * | 1 0 | First piece of a fragmented user message | - * +----------------------------------------------------------+ - * | 0 0 | Middle piece of a fragmented user message | - * +----------------------------------------------------------+ - * | 0 1 | Last piece of a fragmented user message | - * +----------------------------------------------------------+ - * | 1 1 | Unfragmented message | - * ============================================================ - */ - sctp_chunks_common_hdr_t chunk_hdr; - - /* - * This value represents the TSN for this DATA chunk. - * The valid range of TSN is from 0 to 4294967295 (2**32 - 1). - * TSN wraps back to 0 after reaching 4294967295. - */ - u32 tsn; - - /* - * Identifies the stream to which the following user data belongs. - */ - u16 stream_id; - - /* - * This value represents the Stream Sequence Number of the following user data - * within the stream S. Valid range is 0 to 65535. - * When a user message is fragmented by SCTP for transport, the same Stream - * Sequence Number MUST be carried in each of the fragments of the message. - */ - u16 stream_seq; - - /* - * This value represents an application (or upper layer) specified protocol - * identifier. This value is passed to SCTP by its upper layer and sent to its - * peer. This identifier is not used by SCTP but can be used by certain network - * entities, as well as by the peer application, to identify the type of - * information being carried in this DATA chunk. This field must be sent even - * in fragmented DATA chunks (to make sure it is available for agents in the - * middle of the network). Note that this field is NOT touched by an SCTP - * implementation; therefore, its byte order is NOT necessarily big endian. - * The upper layer is responsible for any byte order conversions to this field. - * The value 0 indicates that no application identifier is specified by the - * upper layer for this payload data. - */ - u32 payload_id; - - /* - * This is the payload user data. The implementation MUST pad the end of the - * data to a 4-byte boundary with all-zero bytes. Any padding MUST NOT be - * included in the Length field. A sender MUST never add more than 3 bytes of - * padding. - */ - u32 data[]; - -} sctp_payload_data_chunk_t; - -always_inline void -vnet_sctp_set_tsn (sctp_payload_data_chunk_t * p, u32 tsn) -{ - p->tsn = clib_host_to_net_u32 (tsn); -} - -always_inline u32 -vnet_sctp_get_tsn (sctp_payload_data_chunk_t * p) -{ - return (clib_net_to_host_u32 (p->tsn)); -} - -always_inline void -vnet_sctp_set_stream_id (sctp_payload_data_chunk_t * p, u16 stream_id) -{ - p->stream_id = clib_host_to_net_u16 (stream_id); -} - -always_inline u16 -vnet_sctp_get_stream_id (sctp_payload_data_chunk_t * p) -{ - return (clib_net_to_host_u16 (p->stream_id)); -} - -always_inline void -vnet_sctp_set_stream_seq (sctp_payload_data_chunk_t * p, u16 stream_seq) -{ - p->stream_seq = clib_host_to_net_u16 (stream_seq); -} - -always_inline u16 -vnet_sctp_get_stream_seq (sctp_payload_data_chunk_t * p) -{ - return (clib_net_to_host_u16 (p->stream_seq)); -} - -always_inline void -vnet_sctp_set_payload_id (sctp_payload_data_chunk_t * p, u32 payload_id) -{ - p->payload_id = clib_host_to_net_u32 (payload_id); -} - -always_inline u32 -vnet_sctp_get_payload_id (sctp_payload_data_chunk_t * p) -{ - return (clib_net_to_host_u32 (p->payload_id)); -} - -always_inline u16 -vnet_sctp_calculate_padding (u16 base_length) -{ - if (base_length % 4 == 0) - return 0; - - return (4 - base_length % 4); -} - -#define INBOUND_STREAMS_COUNT 1 -#define OUTBOUND_STREAMS_COUNT 1 - -/* - * INIT chunk - * - * This chunk is used to initiate an SCTP association between two - * endpoints. - * - * 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 - * - * Note 1: The INIT chunks can contain multiple addresses that can be - * IPv4 and/or IPv6 in any combination. - * - * Note 2: The ECN Capable field is reserved for future use of Explicit - * Congestion Notification. - * - * 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. - * - * 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. - * - * IMPLEMENTATION NOTE: If an INIT chunk is received with known parameters that - * are not optional parameters of the INIT chunk, then the receiver SHOULD - * process the INIT chunk and send back an INIT ACK. The receiver of the INIT - * chunk MAY bundle an ERROR chunk with the COOKIE ACK chunk later. - * However, restrictive implementations MAY send back an ABORT chunk in response - * to the INIT chunk. The Chunk Flags field in INIT is reserved, and all bits - * in it should be set to 0 by the sender and ignored by the receiver. - * The sequence of parameters within an INIT can be processed in any order. - */ -typedef struct -{ - sctp_header_t sctp_hdr; - sctp_chunks_common_hdr_t chunk_hdr; - - /* - * The receiver of the INIT (the responding end) records the value of - * the Initiate Tag parameter. - * This value MUST be placed into the Verification Tag field of every - * SCTP packet that the receiver of the INIT transmits within this association. - * The Initiate Tag is allowed to have any value except 0. - * - * If the value of the Initiate Tag in a received INIT chunk is found - * to be 0, the receiver MUST treat it as an error and close the - * association by transmitting an ABORT. - * - * The value of the INIT TAG is recommended to be random for security - * reasons. A good method is described in https://tools.ietf.org/html/rfc4086 - */ - u32 initiate_tag; - - /* - * This value represents the dedicated buffer space, in number of bytes, - * the sender of the INIT has reserved in association with this window. - * During the life of the association, this buffer space SHOULD NOT be - * lessened (i.e., dedicated buffers taken away from this association); - * however, an endpoint MAY change the value of a_rwnd it sends in SACK - * chunks. - */ - u32 a_rwnd; - - /* - * Defines the number of outbound streams the sender of this INIT chunk - * wishes to create in this association. - * The value of 0 MUST NOT be used. - * - * Note: A receiver of an INIT with the OS value set to 0 SHOULD abort - * the association. - */ - u16 outbound_streams_count; - - /* - * Defines the maximum number of streams the sender of this INIT - * chunk allows the peer end to create in this association. - * The value 0 MUST NOT be used. - * - * Note: There is no negotiation of the actual number of streams but - * instead the two endpoints will use the min(requested, offered). - * - * Note: A receiver of an INIT with the MIS value of 0 SHOULD abort - * the association. - */ - u16 inboud_streams_count; - - /* - * Defines the initial TSN that the sender will use. - * The valid range is from 0 to 4294967295. - * This field MAY be set to the value of the Initiate Tag field. - */ - u32 initial_tsn; - - /* The following field allows to have multiple optional fields which are: - * - sctp_ipv4_address - * - sctp_ipv6_address - * - sctp_cookie_preservative - * - sctp_hostname_address - * - sctp_supported_address_types - */ - u32 optional_fields[]; - -} sctp_init_chunk_t; - -/* - * INIT ACK chunk - * - * The INIT ACK chunk is used to acknowledge the initiation of an SCTP - * association. The parameter part of INIT ACK is formatted similarly to the - * INIT chunk. - * - * It uses two extra variable parameters: - * - the State Cookie and - * - the Unrecognized Parameter: - */ -/* - * 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 = 2 | Chunk Flags | Chunk Length | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Initiate Tag | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Advertised Receiver Window Credit | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Number of Outbound Streams | Number of Inbound Streams | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Initial TSN | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * \ \ - * / Optional/Variable-Length Parameters / - * \ \ - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef sctp_init_chunk_t sctp_init_ack_chunk_t; - -typedef struct -{ - u16 type; - u16 length; - -} sctp_opt_params_hdr_t; - -#define SHA1_OUTPUT_LENGTH 20 -/* - * 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 - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Parameter Type | Parameter Length | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * \ \ - * / Parameter Value / - * \ \ - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - sctp_opt_params_hdr_t param_hdr; - - unsigned char mac[SHA1_OUTPUT_LENGTH]; /* RFC 2104 */ - u64 creation_time; - u32 cookie_lifespan; - -} sctp_state_cookie_param_t; - -/* - * 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 / - * \ \ - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - sctp_header_t sctp_hdr; - sctp_chunks_common_hdr_t chunk_hdr; - - sctp_state_cookie_param_t cookie; - -} sctp_cookie_echo_chunk_t; - - -/* - * 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 | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - sctp_header_t sctp_hdr; - sctp_chunks_common_hdr_t chunk_hdr; - -} sctp_cookie_ack_chunk_t; - -/* - * 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 = 14 |Chunk Flags | Length = 4 | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - sctp_header_t sctp_hdr; - sctp_chunks_common_hdr_t chunk_hdr; - -} sctp_shutdown_complete_chunk_t; - -/* OPTIONAL or VARIABLE-LENGTH parameters for INIT */ -#define SCTP_IPV4_ADDRESS_TYPE 5 -#define SCTP_IPV4_ADDRESS_TYPE_LENGTH 8 -#define SCTP_IPV6_ADDRESS_TYPE 6 -#define SCTP_IPV6_ADDRESS_TYPE_LENGTH 20 -#define SCTP_STATE_COOKIE_TYPE 7 -#define SCTP_UNRECOGNIZED_TYPE 8 -#define SCTP_COOKIE_PRESERVATIVE_TYPE 9 -#define SCTP_COOKIE_PRESERVATIVE_TYPE_LENGTH 8 -#define SCTP_HOSTNAME_ADDRESS_TYPE 11 -#define SCTP_SUPPORTED_ADDRESS_TYPES 12 - -/* - * 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 = 5 | Length = 8 | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | IPv4 Address | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - sctp_opt_params_hdr_t param_hdr; - - /* - * Contains an IPv4 address of the sending endpoint. - * It is binary encoded. - */ - ip4_address_t address; - -} sctp_ipv4_addr_param_t; - -always_inline void -vnet_sctp_set_ipv4_address (sctp_ipv4_addr_param_t * a, ip4_address_t address) -{ - a->param_hdr.type = clib_host_to_net_u16 (SCTP_IPV4_ADDRESS_TYPE); - a->param_hdr.length = clib_host_to_net_u16 (8); - a->address.as_u32 = clib_host_to_net_u32 (address.as_u32); -} - -always_inline u32 -vnet_sctp_get_ipv4_address (sctp_ipv4_addr_param_t * a) -{ - return (clib_net_to_host_u32 (a->address.as_u32)); -} - -/* - * 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 = 6 | Length = 20 | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | | - * | IPv6 Address | - * | | - * | | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - sctp_opt_params_hdr_t param_hdr; - - /* - * Contains an IPv6 address of the sending endpoint. - * It is binary encoded. - */ - ip6_address_t address; - -} sctp_ipv6_addr_param_t; - -always_inline void -vnet_sctp_set_ipv6_address (sctp_ipv6_addr_param_t * a, ip6_address_t address) -{ - a->param_hdr.type = clib_host_to_net_u16 (SCTP_IPV6_ADDRESS_TYPE); - a->param_hdr.length = clib_host_to_net_u16 (20); - a->address.as_u64[0] = clib_host_to_net_u64 (address.as_u64[0]); - a->address.as_u64[1] = clib_host_to_net_u64 (address.as_u64[1]); -} - -always_inline ip6_address_t -vnet_sctp_get_ipv6_address (sctp_ipv6_addr_param_t * a) -{ - ip6_address_t ip6_address; - - ip6_address.as_u64[0] = clib_net_to_host_u64 (a->address.as_u64[0]); - ip6_address.as_u64[1] = clib_net_to_host_u64 (a->address.as_u64[1]); - - return ip6_address; -} - -/* - * The sender of the INIT shall use this parameter to suggest to the - * receiver of the INIT for a longer life-span of the State Cookie. - */ -/* - * 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 = 9 | Length = 8 | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Suggested Cookie Life-Span Increment (msec.) | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - sctp_opt_params_hdr_t param_hdr; - - /* - * This parameter indicates to the receiver how much increment in - * milliseconds the sender wishes the receiver to add to its default - * cookie life-span. - * - * This optional parameter should be added to the INIT chunk by the - * sender when it reattempts establishing an association with a peer - * to which its previous attempt of establishing the association - * failed due to a stale cookie operation error. The receiver MAY - * choose to ignore the suggested cookie life-span increase for its - * own security reasons. - */ - u32 life_span_inc; - -} sctp_cookie_preservative_param_t; - -always_inline void -vnet_sctp_set_cookie_preservative (sctp_cookie_preservative_param_t * c, - u32 life_span_inc) -{ - c->param_hdr.type = clib_host_to_net_u16 (SCTP_COOKIE_PRESERVATIVE_TYPE); - c->param_hdr.length = clib_host_to_net_u16 (8); - c->life_span_inc = clib_host_to_net_u32 (life_span_inc); -} - -always_inline u32 -vnet_sctp_get_cookie_preservative (sctp_cookie_preservative_param_t * c) -{ - return (clib_net_to_host_u32 (c->life_span_inc)); -} - -#define FQDN_MAX_LENGTH 256 - -/* - * The sender of INIT uses this parameter to pass its Host Name (in - * place of its IP addresses) to its peer. - * The peer is responsible for resolving the name. - * Using this parameter might make it more likely for the association to work - * across a NAT box. - */ -/* - * 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 | Length | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * / Host Name / - * \ \ - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - sctp_opt_params_hdr_t param_hdr; - - - /* - * This field contains a host name in "host name syntax" per RFC 1123 - * Section 2.1 - * - * Note: At least one null terminator is included in the Host Name - * string and must be included in the length. - */ - char hostname[FQDN_MAX_LENGTH]; - -} sctp_hostname_param_t; - -always_inline void -vnet_sctp_set_hostname_address (sctp_hostname_param_t * h, char *hostname) -{ - h->param_hdr.length = FQDN_MAX_LENGTH; - h->param_hdr.type = clib_host_to_net_u16 (SCTP_HOSTNAME_ADDRESS_TYPE); - clib_memset (h->hostname, '0', FQDN_MAX_LENGTH); - memcpy (h->hostname, hostname, FQDN_MAX_LENGTH); -} - -#define MAX_SUPPORTED_ADDRESS_TYPES 3 - -/* - * The sender of INIT uses this parameter to list all the address types - * it can support. - */ -/* - * 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 = 12 | Length | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Address Type #1 | Address Type #2 | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | ...... | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-++-+-+-+-+-+-+-+-+-+-+-+-+-+-++-+-+-+ - */ -typedef struct -{ - sctp_opt_params_hdr_t param_hdr; - - u16 address_type[MAX_SUPPORTED_ADDRESS_TYPES]; - -} sctp_supported_addr_types_param_t; - -always_inline void -vnet_sctp_set_supported_address_types (sctp_supported_addr_types_param_t * s) -{ - s->param_hdr.type = clib_host_to_net_u16 (SCTP_SUPPORTED_ADDRESS_TYPES); - s->param_hdr.length = 4 /* base = type + length */ + - MAX_SUPPORTED_ADDRESS_TYPES * 4; /* each address type is 4 bytes */ - - s->address_type[0] = clib_host_to_net_u16 (SCTP_IPV4_ADDRESS_TYPE); - s->address_type[1] = clib_host_to_net_u16 (SCTP_IPV6_ADDRESS_TYPE); - s->address_type[2] = clib_host_to_net_u16 (SCTP_HOSTNAME_ADDRESS_TYPE); -} - -/* - * Error cause codes to be used for the sctp_error_cause.cause_code field - */ -#define INVALID_STREAM_IDENTIFIER 1 -#define MISSING_MANDATORY_PARAMETER 2 -#define STALE_COOKIE_ERROR 3 -#define OUT_OF_RESOURCE 4 -#define UNRESOLVABLE_ADDRESS 5 -#define UNRECOGNIZED_CHUNK_TYPE 6 -#define INVALID_MANDATORY_PARAMETER 7 -#define UNRECOGNIZED_PARAMETER 8 -#define NO_USER_DATA 9 -#define COOKIE_RECEIVED_WHILE_SHUTTING_DOWN 10 -#define RESTART_OF_ASSOCIATION_WITH_NEW_ADDR 11 -#define USER_INITIATED_ABORT 12 -#define PROTOCOL_VIOLATION 13 - -always_inline void -vnet_sctp_set_state_cookie (sctp_state_cookie_param_t * s) -{ - s->param_hdr.type = clib_host_to_net_u16 (SCTP_STATE_COOKIE_TYPE); - - /* TODO: length & value to be populated */ -} - -typedef struct -{ - sctp_opt_params_hdr_t param_hdr; - - u32 value[]; - -} sctp_unrecognized_param_t; - -always_inline void -vnet_sctp_set_unrecognized_param (sctp_unrecognized_param_t * u) -{ - u->param_hdr.type = clib_host_to_net_u16 (UNRECOGNIZED_PARAMETER); - - /* TODO: length & value to be populated */ -} - -/* - * Selective ACK (SACK) chunk - * - * This chunk is sent to the peer endpoint to acknowledge received DATA - * chunks and to inform the peer endpoint of gaps in the received - * subsequences of DATA chunks as represented by their TSNs. - */ -/* - * 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 = 3 |Chunk Flags | Chunk Length | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Cumulative TSN Ack | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Advertised Receiver Window Credit (a_rwnd) | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Number of Gap Ack Blocks = N | Number of Duplicate TSNs = X | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Gap Ack Block #1 Start | Gap Ack Block #1 End | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * / / - * \ ... \ - * / / - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Gap Ack Block #N Start | Gap Ack Block #N End | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Duplicate TSN 1 | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * / / - * \ ... \ - * / / - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Duplicate TSN X | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - sctp_header_t sctp_hdr; - sctp_chunks_common_hdr_t chunk_hdr; - /* - * This parameter contains the TSN of the last DATA chunk received in - * sequence before a gap. In the case where no DATA chunk has been - * received, this value is set to the peer's Initial TSN minus one. - */ - u32 cumulative_tsn_ack; - - /* - * This field indicates the updated receive buffer space in bytes of - * the sender of this SACK. - */ - u32 a_rwnd; - - /* - * Indicates the number of Gap Ack Blocks included in this SACK. - */ - u16 gap_ack_blocks_count; - - /* - * This field contains the number of duplicate TSNs the endpoint has - * received. Each duplicate TSN is listed following the Gap Ack Block - * list. - */ - u16 duplicate_tsn_count; - - /* - * Indicates the Start offset TSN for this Gap Ack Block. To calculate - * the actual TSN number the Cumulative TSN Ack is added to this offset - * number. This calculated TSN identifies the first TSN in this Gap Ack - * Block that has been received. - */ - u16 *gap_ack_block_start; - - /* - * Indicates the End offset TSN for this Gap Ack Block. To calculate - * the actual TSN number, the Cumulative TSN Ack is added to this offset - * number. This calculated TSN identifies the TSN of the last DATA chunk - * received in this Gap Ack Block. - */ - u16 *gap_ack_block_end; - - /* - * Indicates the number of times a TSN was received in duplicate since - * the last SACK was sent. Every time a receiver gets a duplicate TSN - * (before sending the SACK), it adds it to the list of duplicates. - * The duplicate count is reinitialized to zero after sending each SACK. - */ - u32 duplicate_tsn; - -} sctp_selective_ack_chunk_t; - -always_inline void -vnet_sctp_set_cumulative_tsn_ack (sctp_selective_ack_chunk_t * s, - u32 cumulative_tsn_ack) -{ - vnet_sctp_set_chunk_type (&s->chunk_hdr, SACK); - s->cumulative_tsn_ack = clib_host_to_net_u32 (cumulative_tsn_ack); -} - -always_inline u32 -vnet_sctp_get_cumulative_tsn_ack (sctp_selective_ack_chunk_t * s) -{ - return clib_net_to_host_u32 (s->cumulative_tsn_ack); -} - -always_inline void -vnet_sctp_set_arwnd (sctp_selective_ack_chunk_t * s, u32 a_rwnd) -{ - vnet_sctp_set_chunk_type (&s->chunk_hdr, SACK); - s->a_rwnd = clib_host_to_net_u32 (a_rwnd); -} - -always_inline u32 -vnet_sctp_get_arwnd (sctp_selective_ack_chunk_t * s) -{ - return clib_net_to_host_u32 (s->a_rwnd); -} - -always_inline void -vnet_sctp_set_gap_ack_blocks_count (sctp_selective_ack_chunk_t * s, - u16 gap_ack_blocks_count) -{ - vnet_sctp_set_chunk_type (&s->chunk_hdr, SACK); - s->gap_ack_blocks_count = clib_host_to_net_u16 (gap_ack_blocks_count); - - if (s->gap_ack_block_start == NULL) - s->gap_ack_block_start = - clib_mem_alloc (sizeof (u16) * gap_ack_blocks_count); - if (s->gap_ack_block_end == NULL) - s->gap_ack_block_end = - clib_mem_alloc (sizeof (u16) * gap_ack_blocks_count); -} - -always_inline u16 -vnet_sctp_get_gap_ack_blocks_count (sctp_selective_ack_chunk_t * s) -{ - return clib_net_to_host_u32 (s->gap_ack_blocks_count); -} - -always_inline void -vnet_sctp_set_duplicate_tsn_count (sctp_selective_ack_chunk_t * s, - u16 duplicate_tsn_count) -{ - vnet_sctp_set_chunk_type (&s->chunk_hdr, SACK); - s->duplicate_tsn_count = clib_host_to_net_u16 (duplicate_tsn_count); -} - -always_inline u16 -vnet_sctp_get_duplicate_tsn_count (sctp_selective_ack_chunk_t * s) -{ - return clib_net_to_host_u16 (s->duplicate_tsn_count); -} - -/* - * Heartbeat Info - * - * 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 - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Heartbeat Info Type=1 | HB Info Length | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * / Sender-Specific Heartbeat Info / - * \ \ - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - sctp_opt_params_hdr_t param_hdr; - - /* - * 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. - * This information is simply reflected back by the receiver in the - * HEARTBEAT ACK message. - * - * Note also that the HEARTBEAT message is both for reachability - * checking and for path verification. - * When a HEARTBEAT chunk is being used for path verification purposes, - * it MUST hold a 64-bit random nonce. - */ - u64 hb_info; - -} sctp_hb_info_param_t; - -always_inline void -vnet_sctp_set_heartbeat_info (sctp_hb_info_param_t * h, u64 hb_info, - u16 hb_info_length) -{ - h->hb_info = clib_host_to_net_u16 (1); - h->param_hdr.length = clib_host_to_net_u16 (hb_info_length); - h->hb_info = clib_host_to_net_u64 (hb_info); -} - -/* - * Heartbeat Request - * - * An endpoint should send this chunk to its peer endpoint to probe the - * reachability of a particular destination transport address defined in - * the present association. - * The parameter field contains the Heartbeat Information, which is a - * variable-length opaque data structure understood only by the sender. - */ -/* - * 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 = 4 | Chunk Flags | Heartbeat Length | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * \ \ - * / Heartbeat Information TLV (Variable-Length) / - * \ \ - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - sctp_header_t sctp_hdr; - sctp_chunks_common_hdr_t chunk_hdr; - sctp_hb_info_param_t hb_info; - -} sctp_hb_req_chunk_t; - -always_inline void -vnet_sctp_set_hb_request_info (sctp_hb_req_chunk_t * h, - sctp_hb_info_param_t * hb_info) -{ - vnet_sctp_set_chunk_type (&h->chunk_hdr, HEARTBEAT); - memcpy (&h->hb_info, hb_info, sizeof (h->hb_info)); -} - -/* - * Heartbeat Acknowledgement - * - * An endpoint should send this chunk to its peer endpoint as a response - * to a HEARTBEAT chunk. - * A HEARTBEAT ACK is always sent to the source IP address of the IP datagram - * containing the HEARTBEAT chunk to which this ack is responding. - */ -/* - * - * 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 = 5 | Chunk Flags | Heartbeat Ack Length | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * \ \ - * / Heartbeat Information TLV (Variable-Length) / - * \ \ - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef sctp_hb_req_chunk_t sctp_hb_ack_chunk_t; - -always_inline void -vnet_sctp_set_hb_ack_info (sctp_hb_ack_chunk_t * h, - sctp_hb_info_param_t * hb_info) -{ - vnet_sctp_set_chunk_type (&h->chunk_hdr, HEARTBEAT_ACK); - memcpy (&h->hb_info, hb_info, sizeof (h->hb_info)); -} - -/* - * Error cause - */ -/* - * 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 - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Cause Code | Cause Length | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * / Cause-Specific Information / - * \ \ - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ -*/ -typedef struct -{ - - sctp_opt_params_hdr_t param_hdr; - u64 cause_info; - -} sctp_err_cause_param_t; - - -/* - * An end-point sends this chunk to its peer end-point to notify it of - * certain error conditions. It contains one or more error causes. - * An Operation Error is not considered fatal in and of itself, but may be - * used with an ABORT chunk to report a fatal condition. It has the - * following parameters: - * - * 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 = 9 | Chunk Flags | Length | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * \ \ - * / one or more Error Causes / - * \ \ - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - sctp_header_t sctp_hdr; - sctp_chunks_common_hdr_t chunk_hdr; - sctp_err_cause_param_t err_causes[]; - -} sctp_operation_error_t; - -/* - * Abort Association (ABORT) - * - * The ABORT chunk is sent to the peer of an association to close the - * association. The ABORT chunk may contain Cause Parameters to inform - * the receiver about the reason of the abort. DATA chunks MUST NOT be - * bundled with ABORT. Control chunks (except for INIT, INIT ACK, and - * SHUTDOWN COMPLETE) MAY be bundled with an ABORT, but they MUST be - * placed before the ABORT in the SCTP packet or they will be ignored by - * the receiver. - * - * If an endpoint receives an ABORT with a format error or no TCB is - * found, it MUST silently discard it. Moreover, under any - * circumstances, an endpoint that receives an ABORT MUST NOT respond to - * that ABORT by sending an ABORT of its own. - */ -/* - * 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 = 6 |Reserved |T| Length | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * \ \ - * / zero or more Error Causes / - * \ \ - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - sctp_header_t sctp_hdr; - sctp_chunks_common_hdr_t chunk_hdr; - sctp_err_cause_param_t err_causes[]; - -} sctp_abort_chunk_t; - -always_inline void -vnet_sctp_set_tbit (sctp_abort_chunk_t * a) -{ - vnet_sctp_set_chunk_type (&a->chunk_hdr, ABORT); - // a->chunk_hdr.flags = clib_host_to_net_u16 (1); -} - -always_inline void -vnet_sctp_unset_tbit (sctp_abort_chunk_t * a) -{ - vnet_sctp_set_chunk_type (&a->chunk_hdr, ABORT); - // a->chunk_hdr.flags = clib_host_to_net_u16 (0); -} - -/* - * Shutdown Association (SHUTDOWN) - * - * An endpoint in an association MUST use this chunk to initiate a - * graceful close of the association with its peer. - */ -/* - * 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 = 7 | Chunk Flags | Length = 8 | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - * | Cumulative TSN Ack | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - sctp_header_t sctp_hdr; - sctp_chunks_common_hdr_t chunk_hdr; - /* - * This parameter contains the TSN of the last chunk received in - * sequence before any gaps. - * - * Note: Since the SHUTDOWN message does not contain Gap Ack Blocks, - * it cannot be used to acknowledge TSNs received out of order. In a - * SACK, lack of Gap Ack Blocks that were previously included - * indicates that the data receiver reneged on the associated DATA - * chunks. Since SHUTDOWN does not contain Gap Ack Blocks, the - * receiver of the SHUTDOWN shouldn't interpret the lack of a Gap Ack - * Block as a renege. - */ - u32 cumulative_tsn_ack; - -} sctp_shutdown_association_chunk_t; - -always_inline void -vnet_sctp_set_tsn_last_received_chunk (sctp_shutdown_association_chunk_t * s, - u32 tsn_last_chunk) -{ - vnet_sctp_set_chunk_type (&s->chunk_hdr, SHUTDOWN); - s->cumulative_tsn_ack = clib_host_to_net_u32 (tsn_last_chunk); -} - -/* - * Shutdown Acknowledgement (SHUTDOWN ACK) - * - * This chunk MUST be used to acknowledge the receipt of the SHUTDOWN - * chunk at the completion of the shutdown process. - */ -/* - * 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 = 8 |Chunk Flags | Length = 4 | - * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ - */ -typedef struct -{ - sctp_header_t sctp_hdr; - sctp_chunks_common_hdr_t chunk_hdr; -} sctp_shutdown_ack_chunk_t; - -always_inline void -vnet_sctp_fill_shutdown_ack (sctp_shutdown_ack_chunk_t * s) -{ - vnet_sctp_set_chunk_type (&s->chunk_hdr, SHUTDOWN_ACK); - vnet_sctp_set_chunk_length (&s->chunk_hdr, 4); -} - -#endif /* included_vnet_sctp_packet_h */ - -/* - * fd.io coding-style-patch-verification: ON - * - * Local Variables: - * eval: (c-set-style "gnu") - * End: - */ |