123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160 |
- #include <stdint.h>
- #include <string.h>
- #include <errno.h>
- #include <malloc.h>
- #include <byteswap.h>
- #include <gpxe/in.h>
- #include <gpxe/ip.h>
- #include <gpxe/pkbuff.h>
- #include <gpxe/tables.h>
- #include <gpxe/netdevice.h>
- #include <gpxe/tcpip.h>
-
- /** @file
- *
- * Transport-network layer interface
- *
- * This file contains functions and utilities for the transport-network layer interface
- */
-
- /** Registered network-layer protocols that support TCPIP */
- static struct tcpip_net_protocol tcpip_net_protocols[0] __table_start ( tcpip_net_protocols );
- static struct tcpip_net_protocol tcpip_net_protocols_end[0] __table_end ( tcpip_net_protocols );
-
- struct tcpip_protocol;
-
- /** Registered transport-layer protocols that support TCPIP */
- static struct tcpip_protocol tcpip_protocols[0] __table_start ( tcpip_protocols );
- static struct tcpip_protocol tcpip_protocols_end[0] __table_end ( tcpip_protocols );
-
- /** Identify TCPIP network-layer protocol
- *
- * @v sa_family Network address family
- * @ret tcpip Protocol supporting TCPIP, or NULL
- */
- static struct tcpip_net_protocol * tcpip_find_protocol ( sa_family_t sa_family ) {
- struct tcpip_net_protocol *tcpip_net;
-
- for ( tcpip_net = tcpip_net_protocols;
- tcpip_net < tcpip_net_protocols_end; ++tcpip_net ) {
- if ( tcpip_net->sa_family == sa_family ) {
- return tcpip_net;
- }
- }
- return NULL;
- }
-
- /** Identify TCPIP transport-layer protocol
- *
- * @v trans_proto Transport-layer protocol number, IP_XXX
- * @ret tcpip_protocol Transport-layer protocol, or NULL
- */
- struct tcpip_protocol* find_tcpip_protocol ( uint8_t trans_proto ) {
- struct tcpip_protocol *tcpip;
-
- for ( tcpip = tcpip_protocols; tcpip <= tcpip_protocols_end;
- ++tcpip ) {
- if ( tcpip->trans_proto == trans_proto ) {
- return tcpip;
- }
- }
- return NULL;
- }
-
- /** Process a received packet
- *
- * @v pkb Packet buffer
- * @v trans_proto Transport-layer protocol number
- * @v src Source network-layer address
- * @v dest Destination network-layer address
- *
- * This function expects a transport-layer segment from the network-layer
- */
- void trans_rx ( struct pk_buff *pkb, uint8_t trans_proto, struct in_addr *src,
- struct in_addr *dest ) {
- struct tcpip_protocol *tcpip;
-
- /* Identify the transport layer protocol */
- for ( tcpip = tcpip_protocols; tcpip <= tcpip_protocols_end; ++tcpip ) {
- if ( tcpip->trans_proto == trans_proto ) {
- DBG ( "Packet sent to %s module", tcpip->name );
- tcpip->rx ( pkb, src, dest );
- }
- }
- }
-
- /** Transmit a transport-layer segment
- *
- * @v pkb Packet buffer
- * @v trans_proto Transport-layer protocol
- * @v sock Destination socket address
- * @ret Status
- */
- int trans_tx ( struct pk_buff *pkb, struct tcpip_protocol *tcpip,
- struct sockaddr *sock ) {
-
- /* Identify the network layer protocol and send it using xxx_tx() */
- switch ( sock->sa_family ) {
- case AF_INET: /* IPv4 network family */
- return ipv4_tx ( pkb, tcpip, &sock->sin.sin_addr );
- case AF_INET6: /* IPv6 network family */
- return ipv6_tx ( pkb, tcpip, &sock->sin6.sin6_addr );
- }
- DBG ( "Network family %d not supported", sock->sa_family );
- return -EAFNOSUPPORT;
- }
-
- /**
- * Calculate continued TCP/IP checkum
- *
- * @v partial Checksum of already-summed data, in network byte order
- * @v data Data buffer
- * @v len Length of data buffer
- * @ret cksum Updated checksum, in network byte order
- *
- * Calculates a TCP/IP-style 16-bit checksum over the data block. The
- * checksum is returned in network byte order.
- *
- * This function may be used to add new data to an existing checksum.
- * The function assumes that both the old data and the new data start
- * on even byte offsets; if this is not the case then you will need to
- * byte-swap either the input partial checksum, the output checksum,
- * or both. Deciding which to swap is left as an exercise for the
- * interested reader.
- */
- unsigned int tcpip_continue_chksum ( unsigned int partial, const void *data,
- size_t len ) {
- unsigned int cksum = ( ( ~partial ) & 0xffff );
- unsigned int value;
- unsigned int i;
-
- for ( i = 0 ; i < len ; i++ ) {
- value = * ( ( uint8_t * ) data + i );
- if ( i & 1 ) {
- /* Odd bytes: swap on little-endian systems */
- value = be16_to_cpu ( value );
- } else {
- /* Even bytes: swap on big-endian systems */
- value = le16_to_cpu ( value );
- }
- cksum += value;
- if ( cksum > 0xffff )
- cksum -= 0xffff;
- }
-
- return ( ( ~cksum ) & 0xffff );
- }
-
- /**
- * Calculate TCP/IP checkum
- *
- * @v data Data buffer
- * @v len Length of data buffer
- * @ret cksum Checksum, in network byte order
- *
- * Calculates a TCP/IP-style 16-bit checksum over the data block. The
- * checksum is returned in network byte order.
- */
- unsigned int tcpip_chksum ( const void *data, size_t len ) {
- return tcpip_continue_chksum ( 0xffff, data, len );
- }
|