⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 output.c

📁 在linux环境下的流控制传输协议(sctp)的源代码
💻 C
📖 第 1 页 / 共 2 页
字号:
/* SCTP kernel implementation * (C) Copyright IBM Corp. 2001, 2004 * Copyright (c) 1999-2000 Cisco, Inc. * Copyright (c) 1999-2001 Motorola, Inc. * * This file is part of the SCTP kernel implementation * * These functions handle output processing. * * This SCTP 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. * * This SCTP 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 <net/sock.h>#include <net/sctp/sctp.h>#include <net/sctp/sm.h>#include <net/sctp/checksum.h>/* Forward declarations for private helpers. */static sctp_xmit_t sctp_packet_append_data(struct sctp_packet *packet,					   struct sctp_chunk *chunk);/* Config a packet. * This appears to be a followup set of initializations. */struct sctp_packet *sctp_packet_config(struct sctp_packet *packet,				       __u32 vtag, int ecn_capable){	struct sctp_chunk *chunk = NULL;	SCTP_DEBUG_PRINTK("%s: packet:%p vtag:0x%x\n", __FUNCTION__,			  packet, vtag);	packet->vtag = vtag;	packet->has_cookie_echo = 0;	packet->has_sack = 0;	packet->has_auth = 0;	packet->has_data = 0;	packet->ipfragok = 0;	packet->auth = NULL;	if (ecn_capable && sctp_packet_empty(packet)) {		chunk = sctp_get_ecne_prepend(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);	}	return packet;}/* Initialize the packet structure. */struct sctp_packet *sctp_packet_init(struct sctp_packet *packet,				     struct sctp_transport *transport,				     __u16 sport, __u16 dport){	struct sctp_association *asoc = transport->asoc;	size_t overhead;	SCTP_DEBUG_PRINTK("%s: packet:%p transport:%p\n", __FUNCTION__,			  packet, transport);	packet->transport = transport;	packet->source_port = sport;	packet->destination_port = dport;	INIT_LIST_HEAD(&packet->chunk_list);	if (asoc) {		struct sctp_sock *sp = sctp_sk(asoc->base.sk);		overhead = sp->pf->af->net_header_len;	} else {		overhead = sizeof(struct ipv6hdr);	}	overhead += sizeof(struct sctphdr);	packet->overhead = overhead;	packet->size = overhead;	packet->vtag = 0;	packet->has_cookie_echo = 0;	packet->has_sack = 0;	packet->has_auth = 0;	packet->has_data = 0;	packet->ipfragok = 0;	packet->malloced = 0;	packet->auth = NULL;	return packet;}/* Free a packet.  */void sctp_packet_free(struct sctp_packet *packet){	struct sctp_chunk *chunk, *tmp;	SCTP_DEBUG_PRINTK("%s: packet:%p\n", __FUNCTION__, packet);	list_for_each_entry_safe(chunk, tmp, &packet->chunk_list, list) {		list_del_init(&chunk->list);		sctp_chunk_free(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(struct sctp_packet *packet,				       struct sctp_chunk *chunk){	sctp_xmit_t retval;	int error = 0;	SCTP_DEBUG_PRINTK("%s: packet:%p chunk:%p\n", __FUNCTION__,			  packet, chunk);	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->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_RWND_FULL:	case SCTP_XMIT_OK:	case SCTP_XMIT_NAGLE_DELAY:		break;	}	return retval;}/* Try to bundle an auth chunk into the packet. */static sctp_xmit_t sctp_packet_bundle_auth(struct sctp_packet *pkt,					   struct sctp_chunk *chunk){	struct sctp_association *asoc = pkt->transport->asoc;	struct sctp_chunk *auth;	sctp_xmit_t retval = SCTP_XMIT_OK;	/* if we don't have an association, we can't do authentication */	if (!asoc)		return retval;	/* See if this is an auth chunk we are bundling or if	 * auth is already bundled.	 */	if (chunk->chunk_hdr->type == SCTP_CID_AUTH || pkt->auth)		return retval;	/* if the peer did not request this chunk to be authenticated,	 * don't do it	 */	if (!chunk->auth)		return retval;	auth = sctp_make_auth(asoc);	if (!auth)		return retval;	retval = sctp_packet_append_chunk(pkt, auth);	return retval;}/* Try to bundle a SACK with the packet. */static sctp_xmit_t sctp_packet_bundle_sack(struct sctp_packet *pkt,					   struct sctp_chunk *chunk){	sctp_xmit_t retval = SCTP_XMIT_OK;	/* If sending DATA and haven't aleady bundled a SACK, try to	 * bundle one in to the packet.	 */	if (sctp_chunk_is_data(chunk) && !pkt->has_sack &&	    !pkt->has_cookie_echo) {		struct sctp_association *asoc;		asoc = pkt->transport->asoc;		if (asoc->a_rwnd > asoc->rwnd) {			struct sctp_chunk *sack;			asoc->a_rwnd = asoc->rwnd;			sack = sctp_make_sack(asoc);			if (sack) {				struct timer_list *timer;				retval = sctp_packet_append_chunk(pkt, sack);				asoc->peer.sack_needed = 0;				timer = &asoc->timers[SCTP_EVENT_TIMEOUT_SACK];				if (timer_pending(timer) && del_timer(timer))					sctp_association_put(asoc);			}		}	}	return retval;}/* Append a chunk to the offered packet reporting back any inability to do * so. */sctp_xmit_t sctp_packet_append_chunk(struct sctp_packet *packet,				     struct sctp_chunk *chunk){	sctp_xmit_t retval = SCTP_XMIT_OK;	__u16 chunk_len = WORD_ROUND(ntohs(chunk->chunk_hdr->length));	size_t psize;	size_t pmtu;	int too_big;	SCTP_DEBUG_PRINTK("%s: packet:%p chunk:%p\n", __FUNCTION__, packet,			  chunk);	/* Try to bundle AUTH chunk */	retval = sctp_packet_bundle_auth(packet, chunk);	if (retval != SCTP_XMIT_OK)		goto finish;	/* Try to bundle SACK chunk */	retval = sctp_packet_bundle_sack(packet, chunk);	if (retval != SCTP_XMIT_OK)		goto finish;	psize = packet->size;	pmtu  = ((packet->transport->asoc) ?		 (packet->transport->asoc->pathmtu) :		 (packet->transport->pathmtu));	too_big = (psize + chunk_len > pmtu);	/* Decide if we need to fragment or resubmit later. */	if (too_big) {		/* It's OK to fragmet at IP level if any one of the following		 * is true:		 * 	1. The packet is empty (meaning this chunk is greater		 * 	   the MTU)		 * 	2. The chunk we are adding is a control chunk		 * 	3. The packet doesn't have any data in it yet and data		 * 	requires authentication.		 */		if (sctp_packet_empty(packet) || !sctp_chunk_is_data(chunk) ||		    (!packet->has_data && chunk->auth)) {			/* We no longer do re-fragmentation.			 * Just fragment at the IP layer, if we			 * actually hit this condition			 */			packet->ipfragok = 1;			goto append;		} else {			retval = SCTP_XMIT_PMTU_FULL;			goto finish;		}	}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.	 */	switch (chunk->chunk_hdr->type) {	    case SCTP_CID_DATA:		retval = sctp_packet_append_data(packet, chunk);		/* Disallow SACK bundling after DATA. */		packet->has_sack = 1;		/* Disallow AUTH bundling after DATA */		packet->has_auth = 1;		/* Let it be knows that packet has DATA in it */		packet->has_data = 1;		if (SCTP_XMIT_OK != retval)			goto finish;		break;	    case SCTP_CID_COOKIE_ECHO:		packet->has_cookie_echo = 1;		break;	    case SCTP_CID_SACK:		packet->has_sack = 1;		break;	    case SCTP_CID_AUTH:		packet->has_auth = 1;		packet->auth = chunk;		break;	}	/* It is OK to send this chunk.  */	list_add_tail(&chunk->list, &packet->chunk_list);	packet->size += chunk_len;	chunk->transport = packet->transport;finish:	return retval;}/* All packets are sent to the network through this function from * sctp_outq_tail(). * * The return value is a normal kernel error return value. */int sctp_packet_transmit(struct sctp_packet *packet){	struct sctp_transport *tp = packet->transport;	struct sctp_association *asoc = tp->asoc;	struct sctphdr *sh;	__u32 crc32 = 0;	struct sk_buff *nskb;	struct sctp_chunk *chunk, *tmp;	struct sock *sk;	int err = 0;	int padding;		/* How much padding do we need?  */	__u8 has_data = 0;	struct dst_entry *dst = tp->dst;	unsigned char *auth = NULL;	/* pointer to auth in skb data */	__u32 cksum_buf_len = sizeof(struct sctphdr);	SCTP_DEBUG_PRINTK("%s: packet:%p\n", __FUNCTION__, packet);

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -