2  * Copyright (c) 2004, Swedish Institute of Computer Science. 
   5  * Redistribution and use in source and binary forms, with or without 
   6  * modification, are permitted provided that the following conditions 
   8  * 1. Redistributions of source code must retain the above copyright 
   9  *    notice, this list of conditions and the following disclaimer. 
  10  * 2. Redistributions in binary form must reproduce the above copyright 
  11  *    notice, this list of conditions and the following disclaimer in the 
  12  *    documentation and/or other materials provided with the distribution. 
  13  * 3. Neither the name of the Institute nor the names of its contributors 
  14  *    may be used to endorse or promote products derived from this software 
  15  *    without specific prior written permission. 
  17  * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND 
  18  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 
  19  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 
  20  * ARE DISCLAIMED.  IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE 
  21  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 
  22  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 
  23  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 
  24  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 
  25  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 
  26  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 
  29  * This file is part of the Contiki operating system. 
  31  * Author: Adam Dunkels <adam@sics.se> 
  33  * $Id: uip-split.c,v 1.2 2008/10/14 13:39:12 julienabeille Exp $ 
  36 #include "uip-split.h" 
  39 #define BUF ((struct uip_tcpip_hdr *)&uip_buf[UIP_LLH_LEN]) 
  41 /*-----------------------------------------------------------------------------*/ 
  43 uip_split_output(void) 
  46   u16_t tcplen
, len1
, len2
; 
  48   /* We only try to split maximum sized TCP segments. */ 
  49   if(BUF
->proto 
== UIP_PROTO_TCP  
&& uip_len 
== UIP_BUFSIZE
) { 
  51     tcplen 
= uip_len 
- UIP_TCPIP_HLEN 
- UIP_LLH_LEN
; 
  52     /* Split the segment in two. If the original packet length was 
  53        odd, we make the second packet one byte larger. */ 
  54     len1 
= len2 
= tcplen 
/ 2; 
  55     if(len1 
+ len2 
< tcplen
) { 
  59     /* Create the first packet. This is done by altering the length 
  60        field of the IP header and updating the checksums. */ 
  61     uip_len 
= len1 
+ UIP_TCPIP_HLEN 
+ UIP_LLH_LEN
; 
  63     /* For IPv6, the IP length field does not include the IPv6 IP header 
  65     BUF
->len
[0] = ((uip_len 
- UIP_IPH_LEN
) >> 8); 
  66     BUF
->len
[1] = ((uip_len 
- UIP_IPH_LEN
) & 0xff); 
  67 #else /* UIP_CONF_IPV6 */ 
  68     BUF
->len
[0] = (uip_len 
- UIP_LLH_LEN
) >> 8; 
  69     BUF
->len
[1] = (uip_len 
- UIP_LLH_LEN
) & 0xff; 
  70 #endif /* UIP_CONF_IPV6 */ 
  72     /* Recalculate the TCP checksum. */ 
  74     BUF
->tcpchksum 
= ~(uip_tcpchksum()); 
  77     /* Recalculate the IP checksum. */ 
  79     BUF
->ipchksum 
= ~(uip_ipchksum()); 
  80 #endif /* UIP_CONF_IPV6 */ 
  82     /* Transmit the first packet. */ 
  86         if (USB_CurrentMode 
== USB_MODE_Device
) 
  87           RNDIS_Device_SendPacket(&Ethernet_RNDIS_Interface_Device
, uip_buf
, uip_len
); 
  89           RNDIS_Host_SendPacket(&Ethernet_RNDIS_Interface_Host
, uip_buf
, uip_len
); 
  90 #endif /* UIP_CONF_IPV6 */ 
  92     /* Now, create the second packet. To do this, it is not enough to 
  93        just alter the length field, but we must also update the TCP 
  94        sequence number and point the uip_appdata to a new place in 
  95        memory. This place is determined by the length of the first 
  97     uip_len 
= len2 
+ UIP_TCPIP_HLEN 
+ UIP_LLH_LEN
; 
  99     /* For IPv6, the IP length field does not include the IPv6 IP header 
 101     BUF
->len
[0] = ((uip_len 
- UIP_IPH_LEN
) >> 8); 
 102     BUF
->len
[1] = ((uip_len 
- UIP_IPH_LEN
) & 0xff); 
 103 #else /* UIP_CONF_IPV6 */ 
 104     BUF
->len
[0] = (uip_len  
- UIP_LLH_LEN
) >> 8; 
 105     BUF
->len
[1] = (uip_len 
- UIP_LLH_LEN
) & 0xff; 
 106 #endif /* UIP_CONF_IPV6 */ 
 108     memcpy(uip_appdata
, (u8_t 
*)uip_appdata 
+ len1
, len2
); 
 110     uip_add32(BUF
->seqno
, len1
); 
 111     BUF
->seqno
[0] = uip_acc32
[0]; 
 112     BUF
->seqno
[1] = uip_acc32
[1]; 
 113     BUF
->seqno
[2] = uip_acc32
[2]; 
 114     BUF
->seqno
[3] = uip_acc32
[3]; 
 116     /* Recalculate the TCP checksum. */ 
 118     BUF
->tcpchksum 
= ~(uip_tcpchksum()); 
 121     /* Recalculate the IP checksum. */ 
 123     BUF
->ipchksum 
= ~(uip_ipchksum()); 
 124 #endif /* UIP_CONF_IPV6 */ 
 126     /* Transmit the second packet. */ 
 130         if (USB_CurrentMode 
== USB_MODE_Device
) 
 131           RNDIS_Device_SendPacket(&Ethernet_RNDIS_Interface_Device
, uip_buf
, uip_len
); 
 133           RNDIS_Host_SendPacket(&Ethernet_RNDIS_Interface_Host
, uip_buf
, uip_len
); 
 134 #endif /* UIP_CONF_IPV6 */ 
 139   /*    uip_fw_output();*/ 
 143         if (USB_CurrentMode 
== USB_MODE_Device
) 
 144           RNDIS_Device_SendPacket(&Ethernet_RNDIS_Interface_Device
, uip_buf
, uip_len
); 
 146           RNDIS_Host_SendPacket(&Ethernet_RNDIS_Interface_Host
, uip_buf
, uip_len
); 
 147 #endif /* UIP_CONF_IPV6 */ 
 150 /*-----------------------------------------------------------------------------*/