/** @file * IPRT - Network Protocols. */ /* * Copyright (C) 2008 Sun Microsystems, Inc. * * This file is part of VirtualBox Open Source Edition (OSE), as * available from http://www.virtualbox.org. This file is free software; * you can redistribute it and/or modify it under the terms of the GNU * General Public License (GPL) as published by the Free Software * Foundation, in version 2 as it comes in the "COPYING" file of the * VirtualBox OSE distribution. VirtualBox OSE is distributed in the * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. * * The contents of this file may alternatively be used under the terms * of the Common Development and Distribution License Version 1.0 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the * VirtualBox OSE distribution, in which case the provisions of the * CDDL are applicable instead of those of the GPL. * * You may elect to license modified versions of this file under the * terms and conditions of either the GPL or the CDDL or both. * * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa * Clara, CA 95054 USA or visit http://www.sun.com if you need * additional information or have any questions. */ #ifndef ___iprt_net_h #define ___iprt_net_h #include #include #include __BEGIN_DECLS /** @defgroup grp_rt_net RTNet - Network Protocols * @ingroup grp_rt * @{ */ /** * IPv4 address. */ typedef RTUINT32U RTNETADDRIPV4; AssertCompileSize(RTNETADDRIPV4, 4); /** Pointer to a IPv4 address. */ typedef RTNETADDRIPV4 *PRTNETADDRIPV4; /** Pointer to a const IPv4 address. */ typedef RTNETADDRIPV4 const *PCRTNETADDRIPV4; /** * IPv6 address. */ typedef RTUINT128U RTNETADDRIPV6; AssertCompileSize(RTNETADDRIPV6, 16); /** Pointer to a IPv4 address. */ typedef RTNETADDRIPV6 *PRTNETADDRIPV6; /** Pointer to a const IPv4 address. */ typedef RTNETADDRIPV6 const *PCRTNETADDRIPV6; /** * IPX address. */ #pragma pack(1) typedef struct RTNETADDRIPX { /** The network ID. */ uint32_t Network; /** The node ID. (Defaults to the MAC address apparently.) */ RTMAC Node; } RTNETADDRIPX; #pragma pack() AssertCompileSize(RTNETADDRIPX, 4+6); /** Pointer to an IPX address. */ typedef RTNETADDRIPX *PRTNETADDRIPX; /** Pointer to a const IPX address. */ typedef RTNETADDRIPX const *PCRTNETADDRIPX; /** * Address union. */ typedef union RTNETADDRU { /** 64-bit view. */ uint64_t au64[2]; /** 32-bit view. */ uint32_t au32[4]; /** 16-bit view. */ uint16_t au16[8]; /** 8-bit view. */ uint8_t au8[16]; /** IPv4 view. */ RTNETADDRIPV4 IPv4; /** IPv6 view. */ RTNETADDRIPV6 IPv6; /** IPX view. */ RTNETADDRIPX Ipx; /** MAC address view. */ RTMAC Mac; } RTNETADDRU; AssertCompileSize(RTNETADDRU, 16); /** Pointer to an address union. */ typedef RTNETADDRU *PRTNETADDRU; /** Pointer to a const address union. */ typedef RTNETADDRU const *PCRTNETADDRU; /** * Ethernet header. */ #pragma pack(1) typedef struct RTNETETHERHDR { RTMAC DstMac; RTMAC SrcMac; /** Ethernet frame type or frame size, depending on the kind of ethernet. * This is big endian on the wire. */ uint16_t EtherType; } RTNETETHERHDR; #pragma pack() AssertCompileSize(RTNETETHERHDR, 14); /** Pointer to an ethernet header. */ typedef RTNETETHERHDR *PRTNETETHERHDR; /** Pointer to a const ethernet header. */ typedef RTNETETHERHDR const *PCRTNETETHERHDR; /** @name EtherType (RTNETETHERHDR::EtherType) * @{ */ #define RTNET_ETHERTYPE_IPV4 UINT16_C(0x0800) #define RTNET_ETHERTYPE_ARP UINT16_C(0x0806) #define RTNET_ETHERTYPE_IPV6 UINT16_C(0x86dd) #define RTNET_ETHERTYPE_VLAN UINT16_C(0x8100) #define RTNET_ETHERTYPE_IPX_1 UINT16_C(0x8037) #define RTNET_ETHERTYPE_IPX_2 UINT16_C(0x8137) #define RTNET_ETHERTYPE_IPX_3 UINT16_C(0x8138) /** @} */ /** * IPv4 header. * All is bigendian on the wire. */ #pragma pack(1) typedef struct RTNETIPV4 { #ifdef RT_BIG_ENDIAN unsigned int ip_v : 4; unsigned int ip_hl : 4; unsigned int ip_tos : 8; unsigned int ip_len : 16; #else /** 00:0 - Header length given as a 32-bit word count. */ unsigned int ip_hl : 4; /** 00:4 - Header version. */ unsigned int ip_v : 4; /** 01 - Type of service. */ unsigned int ip_tos : 8; /** 02 - Total length (header + data). */ unsigned int ip_len : 16; #endif /** 04 - Packet idenficiation. */ uint16_t ip_id; /** 06 - Offset if fragmented. */ uint16_t ip_off; /** 08 - Time to live. */ uint8_t ip_ttl; /** 09 - Protocol. */ uint8_t ip_p; /** 0a - Header check sum. */ uint16_t ip_sum; /** 0c - Source address. */ RTNETADDRIPV4 ip_src; /** 10 - Destination address. */ RTNETADDRIPV4 ip_dst; /** 14 - Options (optional). */ uint32_t ip_options[1]; } RTNETIPV4; #pragma pack() AssertCompileSize(RTNETIPV4, 6 * 4); /** Pointer to a IPv4 header. */ typedef RTNETIPV4 *PRTNETIPV4; /** Pointer to a const IPv4 header. */ typedef RTNETIPV4 const *PCRTNETIPV4; /** The minimum IPv4 header length (in bytes). * Up to and including RTNETIPV4::ip_dst. */ #define RTNETIPV4_MIN_LEN (20) /** @name IPv4 Protocol Numbers * @{ */ /** IPv4: ICMP */ #define RTNETIPV4_PROT_ICMP (1) /** IPv4: TCP */ #define RTNETIPV4_PROT_TCP (6) /** IPv4: UDP */ #define RTNETIPV4_PROT_UDP (17) /** @} */ /** @name Common IPv4 Port Assignments * @{ */ /** Boostrap Protocol / DHCP) Server. */ #define RTNETIPV4_PORT_BOOTPS (67) /** Boostrap Protocol / DHCP) Client. */ #define RTNETIPV4_PORT_BOOTPC (68) /** @} */ RTDECL(uint16_t) RTNetIPv4HdrChecksum(PCRTNETIPV4 pIpHdr); RTDECL(bool) RTNetIPv4IsHdrValid(PCRTNETIPV4 pIpHdr, size_t cbHdrMax, size_t cbPktMax); RTDECL(uint32_t) RTNetIPv4PseudoChecksum(PCRTNETIPV4 pIpHdr); RTDECL(uint32_t) RTNetIPv4PseudoChecksumBits(RTNETADDRIPV4 SrcAddr, RTNETADDRIPV4 DstAddr, uint8_t bProtocol, uint16_t cbPkt); RTDECL(uint32_t) RTNetIPv4AddDataChecksum(void const *pvData, size_t cbData, uint32_t u32Sum, bool *pfOdd); RTDECL(uint16_t) RTNetIPv4FinalizeChecksum(uint32_t u32Sum); /** * UDP header. */ #pragma pack(1) typedef struct RTNETUDP { /** The source port. */ uint16_t uh_sport; /** The destination port. */ uint16_t uh_dport; /** The length of the UDP header and associated data. */ uint16_t uh_ulen; /** The checksum of the pseudo header, the UDP header and the data. */ uint16_t uh_sum; } RTNETUDP; #pragma pack() AssertCompileSize(RTNETUDP, 8); /** Pointer to an UDP header. */ typedef RTNETUDP *PRTNETUDP; /** Pointer to a const UDP header. */ typedef RTNETUDP const *PCRTNETUDP; /** The minimum UDP packet length (in bytes). (RTNETUDP::uh_ulen) */ #define RTNETUDP_MIN_LEN (8) RTDECL(uint32_t) RTNetIPv4AddUDPChecksum(PCRTNETUDP pUdpHdr, uint32_t u32Sum); RTDECL(uint16_t) RTNetIPv4UDPChecksum(PCRTNETIPV4 pIpHdr, PCRTNETUDP pUdpHdr, void const *pvData); RTDECL(bool) RTNetIPv4IsUDPSizeValid(PCRTNETIPV4 pIpHdr, PCRTNETUDP pUdpHdr, size_t cbPktMax); RTDECL(bool) RTNetIPv4IsUDPValid(PCRTNETIPV4 pIpHdr, PCRTNETUDP pUdpHdr, void const *pvData, size_t cbPktMax); /** * IPv4 BOOTP / DHCP packet. */ #pragma pack(1) typedef struct RTNETBOOTP { /** 00 - The packet opcode. */ uint8_t bp_op; /** 01 - Hardware address type. Same as RTNETARPHDR::ar_htype. */ uint8_t bp_htype; /** 02 - Hardware address length. */ uint8_t bp_hlen; /** 03 - Gateway hops. */ uint8_t bp_hops; /** 04 - Transaction ID. */ uint32_t bp_xid; /** 08 - Seconds since boot started. */ uint16_t bp_secs; /** 0a - Unused (BOOTP) / Flags (DHCP). */ uint16_t bp_flags; /** 0c - Client IPv4 address. */ RTNETADDRIPV4 bp_ciaddr; /** 10 - Your IPv4 address. */ RTNETADDRIPV4 bp_yiaddr; /** 14 - Server IPv4 address. */ RTNETADDRIPV4 bp_siaddr; /** 18 - Gateway IPv4 address. */ RTNETADDRIPV4 bp_giaddr; /** 1c - Client hardware address. */ union { uint8_t au8[16]; RTMAC Mac; } bp_chaddr; /** 2c - Server name. */ uint8_t bp_sname[64]; /** 6c - File name / more DHCP options. */ uint8_t bp_file[128]; /** ec - Vendor specific area (BOOTP) / Options (DHCP). * @remark This is really 312 bytes in the DHCP version. */ union { uint8_t au8[128]; struct DHCP { /** ec - The DHCP cookie (RTNET_DHCP_COOKIE). */ uint32_t dhcp_cookie; /** f0 - The DHCP options. */ uint8_t dhcp_opts[124]; } Dhcp; } bp_vend; } RTNETBOOTP; #pragma pack() AssertCompileSize(RTNETBOOTP, 0xec + 128); /** Pointer to a BOOTP / DHCP packet. */ typedef RTNETBOOTP *PRTNETBOOTP; /** Pointer to a const BOOTP / DHCP packet. */ typedef RTNETBOOTP const *PCRTNETBOOTP; /** Minimum BOOTP packet length. For quick validation, no standard thing really. */ #define RTNETBOOTP_MIN_LEN 0xec /** Minimum DHCP packet length. For quick validation, no standard thing really. */ #define RTNETBOOTP_DHCP_MIN_LEN 0xf1 /** @name BOOTP packet opcode values * @{ */ #define RTNETBOOTP_OP_REQUEST 1 #define RTNETBOOTP_OP_REPLY 2 /** @} */ /** @name DHCP flags (RTNETBOOTP::bp_flags) * @{ */ #define RTNET_DHCP_FLAGS_NO_BROADCAST UINT16_C(0x8000) /** @todo check test!!! */ /** @} */ /** The DHCP cookie (network endian). */ #define RTNET_DHCP_COOKIE UINT32_C(0x63825363) /** * An IPv4 DHCP option header. */ typedef struct RTNETDHCPOPT { /** 00 - The DHCP option. */ uint8_t dhcp_opt; /** 01 - The data length (excluding this header). */ uint8_t dhcp_len; /* 02 - The option data follows here, optional and of variable length. */ } RTNETDHCPOPT; AssertCompileSize(RTNETDHCPOPT, 2); /** Pointer to a DHCP option header. */ typedef RTNETDHCPOPT *PRTNETDHCPOPT; /** Pointer to a const DHCP option header. */ typedef RTNETDHCPOPT const *PCRTNETDHCPOPT; /** @name DHCP options * @{ */ /** 1 byte padding, this has no dhcp_len field. */ #define RTNET_DHCP_OPT_PAD 0 /** Have a 8-bit message type value as data, see RTNET_DHCP_MT_*. */ #define RTNET_DHCP_OPT_MSG_TYPE 53 /** Marks the end of the DHCP options, this has no dhcp_len field. */ #define RTNET_DHCP_OPT_END 255 /** @} */ /** @name DHCP Message Types (option 53) * @{ */ #define RTNET_DHCP_MT_DISCOVER 1 #define RTNET_DHCP_MT_OFFER 2 #define RTNET_DHCP_MT_REQUEST 3 #define RTNET_DHCP_MT_DECLINE 4 #define RTNET_DHCP_MT_ACK 5 #define RTNET_DHCP_MT_NAC 6 #define RTNET_DHCP_MT_RELEASE 7 #define RTNET_DHCP_MT_INFORM 8 /** @} */ /** @name DHCP Flags * @{ */ #define RTNET_DHCP_FLAG_BROADCAST 0x8000 /** @} */ RTDECL(bool) RTNetIPv4IsDHCPValid(PCRTNETUDP pUdpHdr, PCRTNETBOOTP pDhcp, size_t cbDhcp, uint8_t *pMsgType); /** * IPv4 DHCP packet. * @obsolete Use RTNETBOOTP. */ #pragma pack(1) typedef struct RTNETDHCP { /** 00 - The packet opcode. */ uint8_t Op; /** Hardware address type. */ uint8_t HType; /** Hardware address length. */ uint8_t HLen; uint8_t Hops; uint32_t XID; uint16_t Secs; uint16_t Flags; /** Client IPv4 address. */ RTNETADDRIPV4 CIAddr; /** Your IPv4 address. */ RTNETADDRIPV4 YIAddr; /** Server IPv4 address. */ RTNETADDRIPV4 SIAddr; /** Gateway IPv4 address. */ RTNETADDRIPV4 GIAddr; /** Client hardware address. */ uint8_t CHAddr[16]; /** Server name. */ uint8_t SName[64]; uint8_t File[128]; uint8_t abMagic[4]; uint8_t DhcpOpt; uint8_t DhcpLen; /* 1 */ uint8_t DhcpReq; uint8_t abOptions[57]; } RTNETDHCP; #pragma pack() /** @todo AssertCompileSize(RTNETDHCP, ); */ /** Pointer to a DHCP packet. */ typedef RTNETDHCP *PRTNETDHCP; /** Pointer to a const DHCP packet. */ typedef RTNETDHCP const *PCRTNETDHCP; /** * TCP packet. */ #pragma pack(1) typedef struct RTNETTCP { /** 00 - The source port. */ uint16_t th_sport; /** 02 - The destination port. */ uint16_t th_dport; /** 04 - The sequence number. */ uint32_t th_seq; /** 08 - The acknowledgement number. */ uint32_t th_ack; #ifdef RT_BIG_ENDIAN unsigned int th_win : 16; unsigned int th_flags : 8; unsigned int th_off : 4; unsigned int th_x2 : 4; #else /** 0c:0 - Reserved. */ unsigned int th_x2 : 4; /** 0c:4 - The data offset given as a dword count from the start of this header. */ unsigned int th_off : 4; /** 0d - flags. */ unsigned int th_flags : 8; /** 0e - The window. */ unsigned int th_win : 16; #endif /** 10 - The checksum of the pseudo header, the TCP header and the data. */ uint16_t th_sum; /** 12 - The urgent pointer. */ uint16_t th_urp; /* (options follows here and then the data (aka text).) */ } RTNETTCP; #pragma pack() AssertCompileSize(RTNETTCP, 20); /** Pointer to a TCP packet. */ typedef RTNETTCP *PRTNETTCP; /** Pointer to a const TCP packet. */ typedef RTNETTCP const *PCRTNETTCP; /** The minimum TCP header length (in bytes). (RTNETTCP::th_off * 4) */ #define RTNETTCP_MIN_LEN (20) RTDECL(uint32_t) RTNetIPv4AddTCPChecksum(PCRTNETTCP pTcpHdr, uint32_t u32Sum); RTDECL(uint16_t) RTNetIPv4TCPChecksum(PCRTNETIPV4 pIpHdr, PCRTNETTCP pTcpHdr, void const *pvData); RTDECL(bool) RTNetIPv4IsTCPSizeValid(PCRTNETIPV4 pIpHdr, PCRTNETTCP pTcpHdr, size_t cbHdrMax, size_t cbPktMax); RTDECL(bool) RTNetIPv4IsTCPValid(PCRTNETIPV4 pIpHdr, PCRTNETTCP pTcpHdr, size_t cbHdrMax, void const *pvData, size_t cbPktMax); /** * IPv4 ICMP packet header. */ #pragma pack(1) typedef struct RTNETICMPV4HDR { /** 00 - The ICMP message type. */ uint8_t icmp_type; /** 01 - Type specific code that further qualifies the message. */ uint8_t icmp_code; /** 02 - Checksum of the ICMP message. */ uint16_t icmp_cksum; } RTNETICMPV4HDR; #pragma pack() AssertCompileSize(RTNETICMPV4HDR, 4); /** Pointer to an ICMP packet header. */ typedef RTNETICMPV4HDR *PRTNETICMPV4HDR; /** Pointer to a const ICMP packet header. */ typedef RTNETICMPV4HDR const *PCRTNETICMPV4HDR; /** @name ICMP (v4) message types. * @{ */ #define RTNETICMPV4_TYPE_ECHO_REPLY 0 #define RTNETICMPV4_TYPE_ECHO_REQUEST 8 #define RTNETICMPV4_TYPE_TRACEROUTE 30 /** @} */ /** * IPv4 ICMP ECHO Reply & Request packet. */ #pragma pack(1) typedef struct RTNETICMPV4ECHO { /** 00 - The ICMP header. */ RTNETICMPV4HDR Hdr; /** 04 - The identifier to help the requestor match up the reply. * Can be 0. Typically fixed value. */ uint16_t icmp_id; /** 06 - The sequence number to help the requestor match up the reply. * Can be 0. Typically incrementing between requests. */ uint16_t icmp_seq; /** 08 - Variable length data that is to be returned unmodified in the reply. */ uint8_t icmp_data[1]; } RTNETICMPV4ECHO; #pragma pack() AssertCompileSize(RTNETICMPV4ECHO, 9); /** Pointer to an ICMP ECHO packet. */ typedef RTNETICMPV4ECHO *PRTNETICMPV4ECHO; /** Pointer to a const ICMP ECHO packet. */ typedef RTNETICMPV4ECHO const *PCRTNETICMPV4ECHO; /** * IPv4 ICMP TRACEROUTE packet. * This is an reply to an IP packet with the traceroute option set. */ #pragma pack(1) typedef struct RTNETICMPV4TRACEROUTE { /** 00 - The ICMP header. */ RTNETICMPV4HDR Hdr; /** 04 - Identifier copied from the traceroute option's ID number. */ uint16_t icmp_id; /** 06 - Unused. (Possibly an icmp_seq?) */ uint16_t icmp_void; /** 08 - Outbound hop count. From the IP packet causing this message. */ uint16_t icmp_ohc; /** 0a - Return hop count. From the IP packet causing this message. */ uint16_t icmp_rhc; /** 0c - Output link speed, 0 if not known. */ uint32_t icmp_speed; /** 10 - Output link MTU, 0 if not known. */ uint32_t icmp_mtu; } RTNETICMPV4TRACEROUTE; #pragma pack() AssertCompileSize(RTNETICMPV4TRACEROUTE, 20); /** Pointer to an ICMP TRACEROUTE packet. */ typedef RTNETICMPV4TRACEROUTE *PRTNETICMPV4TRACEROUTE; /** Pointer to a const ICMP TRACEROUTE packet. */ typedef RTNETICMPV4TRACEROUTE const *PCRTNETICMPV4TRACEROUTE; /** @todo add more ICMPv4 as needed. */ /** * IPv4 ICMP union packet. */ typedef union RTNETICMPV4 { RTNETICMPV4HDR Hdr; RTNETICMPV4ECHO Echo; RTNETICMPV4TRACEROUTE Traceroute; } RTNETICMPV4; /** Pointer to an ICMP union packet. */ typedef RTNETICMPV4 *PRTNETICMPV4; /** Pointer to a const ICMP union packet. */ typedef RTNETICMPV4 const *PCRTNETICMPV4; /** @todo add ICMPv6 when needed. */ /** * Ethernet ARP header. */ #pragma pack(1) typedef struct RTNETARPHDR { /** The hardware type. */ uint16_t ar_htype; /** The protocol type (ethertype). */ uint16_t ar_ptype; /** The hardware address length. */ uint8_t ar_hlen; /** The protocol address length. */ uint8_t ar_plen; /** The operation. */ uint16_t ar_oper; } RTNETARPHDR; #pragma pack() AssertCompileSize(RTNETARPHDR, 8); /** Pointer to an ethernet ARP header. */ typedef RTNETARPHDR *PRTNETARPHDR; /** Pointer to a const ethernet ARP header. */ typedef RTNETARPHDR const *PCRTNETARPHDR; /** ARP hardware type - ethernet. */ #define RTNET_ARP_ETHER UINT16_C(1) /** @name ARP operations * @{ */ #define RTNET_ARPOP_REQUEST UINT16_C(1) /**< Request hardward address given a protocol address (ARP). */ #define RTNET_ARPOP_REPLY UINT16_C(2) #define RTNET_ARPOP_REVREQUEST UINT16_C(3) /**< Request protocol address given a hardware address (RARP). */ #define RTNET_ARPOP_REVREPLY UINT16_C(4) #define RTNET_ARPOP_INVREQUEST UINT16_C(8) /**< Inverse ARP. */ #define RTNET_ARPOP_INVREPLY UINT16_C(9) /** Check if an ARP operation is a request or not. */ #define RTNET_ARPOP_IS_REQUEST(Op) ((Op) & 1) /** Check if an ARP operation is a reply or not. */ #define RTNET_ARPOP_IS_REPLY(Op) (!RTNET_ARPOP_IS_REQUEST(Op)) /** @} */ /** * Ethernet IPv4 + 6-byte MAC ARP request packet. */ #pragma pack(1) typedef struct RTNETARPIPV4 { /** ARP header. */ RTNETARPHDR Hdr; /** The sender hardware address. */ RTMAC ar_sha; /** The sender protocol address. */ RTNETADDRIPV4 ar_spa; /** The target hardware address. */ RTMAC ar_tha; /** The arget protocol address. */ RTNETADDRIPV4 ar_tpa; } RTNETARPIPV4; #pragma pack() AssertCompileSize(RTNETARPIPV4, 8+6+4+6+4); /** Pointer to an ethernet IPv4+MAC ARP request packet. */ typedef RTNETARPIPV4 *PRTNETARPIPV4; /** Pointer to a const ethernet IPv4+MAC ARP request packet. */ typedef RTNETARPIPV4 const *PCRTNETARPIPV4; /** @todo RTNETNDP (IPv6)*/ /** @} */ __END_DECLS #endif