| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513 | /** * D header file for POSIX. * * Copyright: Public Domain * License: Public Domain * Authors: Sean Kelly * Standards: The Open Group Base Specifications Issue 6, IEEE Std 1003.1, 2004 Edition */ module tango.stdc.posix.netinet.in_; private import tango.stdc.posix.config; public import tango.stdc.inttypes : uint32_t, uint16_t, uint8_t; public import tango.stdc.posix.arpa.inet; public import tango.stdc.posix.sys.socket; // for sa_family_t extern (C): // // Required // /* NOTE: The following must must be defined in tango.stdc.posix.arpa.inet to break a circular import: in_port_t, in_addr_t, struct in_addr, INET_ADDRSTRLEN. in_port_t in_addr_t sa_family_t // from tango.stdc.posix.sys.socket uint8_t // from tango.stdc.inttypes uint32_t // from tango.stdc.inttypes struct in_addr { in_addr_t s_addr; } struct sockaddr_in { sa_family_t sin_family; in_port_t sin_port; in_addr sin_addr; } IPPROTO_IP IPPROTO_ICMP IPPROTO_TCP IPPROTO_UDP INADDR_ANY INADDR_BROADCAST INET_ADDRSTRLEN htonl() // from tango.stdc.posix.arpa.inet htons() // from tango.stdc.posix.arpa.inet ntohl() // from tango.stdc.posix.arpa.inet ntohs() // from tango.stdc.posix.arpa.inet */ version( linux ) { private const __SOCK_SIZE__ = 16; struct sockaddr_in { sa_family_t sin_family; in_port_t sin_port; in_addr sin_addr; /* Pad to size of `struct sockaddr'. */ ubyte[__SOCK_SIZE__ - sa_family_t.sizeof - in_port_t.sizeof - in_addr.sizeof] __pad; } enum { IPPROTO_IP = 0, IPPROTO_ICMP = 1, IPPROTO_TCP = 6, IPPROTO_UDP = 17 } const uint INADDR_ANY = 0x00000000; const uint INADDR_BROADCAST = 0xffffffff; } else version( darwin ) { private const __SOCK_SIZE__ = 16; struct sockaddr_in { ubyte sin_len; sa_family_t sin_family; in_port_t sin_port; in_addr sin_addr; ubyte[8] sin_zero; } enum { IPPROTO_IP = 0, IPPROTO_ICMP = 1, IPPROTO_TCP = 6, IPPROTO_UDP = 17 } const uint INADDR_ANY = 0x00000000; const uint INADDR_BROADCAST = 0xffffffff; } else version( FreeBSD ) { private const __SOCK_SIZE__ = 16; struct sockaddr_in { ubyte sin_len; sa_family_t sin_family; in_port_t sin_port; in_addr sin_addr; ubyte[8] sin_zero; } enum { IPPROTO_IP = 0, IPPROTO_ICMP = 1, IPPROTO_TCP = 6, IPPROTO_UDP = 17 } const uint INADDR_ANY = 0x00000000; const uint INADDR_BROADCAST = 0xffffffff; } else version( solaris ) { struct sockaddr_in { sa_family_t sin_family; in_port_t sin_port; in_addr sin_addr; ubyte[8] sin_zero; } enum { IPPROTO_IP = 0, IPPROTO_ICMP = 1, IPPROTO_TCP = 6, IPPROTO_UDP = 17 } const uint INADDR_ANY = 0x00000000; const uint INADDR_BROADCAST = 0xffffffff; } // // IPV6 (IP6) // /* NOTE: The following must must be defined in tango.stdc.posix.arpa.inet to break a circular import: INET6_ADDRSTRLEN. struct in6_addr { uint8_t[16] s6_addr; } struct sockaddr_in6 { sa_family_t sin6_family; in_port_t sin6_port; uint32_t sin6_flowinfo; in6_addr sin6_addr; uint32_t sin6_scope_id; } extern in6_addr in6addr_any; extern in6_addr in6addr_loopback; struct ipv6_mreq { in6_addr ipv6mr_multiaddr; uint ipv6mr_interface; } IPPROTO_IPV6 INET6_ADDRSTRLEN IPV6_JOIN_GROUP IPV6_LEAVE_GROUP IPV6_MULTICAST_HOPS IPV6_MULTICAST_IF IPV6_MULTICAST_LOOP IPV6_UNICAST_HOPS IPV6_V6ONLY // macros int IN6_IS_ADDR_UNSPECIFIED(in6_addr*) int IN6_IS_ADDR_LOOPBACK(in6_addr*) int IN6_IS_ADDR_MULTICAST(in6_addr*) int IN6_IS_ADDR_LINKLOCAL(in6_addr*) int IN6_IS_ADDR_SITELOCAL(in6_addr*) int IN6_IS_ADDR_V4MAPPED(in6_addr*) int IN6_IS_ADDR_V4COMPAT(in6_addr*) int IN6_IS_ADDR_MC_NODELOCAL(in6_addr*) int IN6_IS_ADDR_MC_LINKLOCAL(in6_addr*) int IN6_IS_ADDR_MC_SITELOCAL(in6_addr*) int IN6_IS_ADDR_MC_ORGLOCAL(in6_addr*) int IN6_IS_ADDR_MC_GLOBAL(in6_addr*) */ version ( linux ) { struct in6_addr { union { uint8_t[16] s6_addr; uint16_t[8] s6_addr16; uint32_t[4] s6_addr32; } } struct sockaddr_in6 { sa_family_t sin6_family; in_port_t sin6_port; uint32_t sin6_flowinfo; in6_addr sin6_addr; uint32_t sin6_scope_id; } extern in6_addr in6addr_any; extern in6_addr in6addr_loopback; struct ipv6_mreq { in6_addr ipv6mr_multiaddr; uint ipv6mr_interface; } enum : uint { IPPROTO_IPV6 = 41, INET6_ADDRSTRLEN = 46, IPV6_JOIN_GROUP = 20, IPV6_LEAVE_GROUP = 21, IPV6_MULTICAST_HOPS = 18, IPV6_MULTICAST_IF = 17, IPV6_MULTICAST_LOOP = 19, IPV6_UNICAST_HOPS = 16, IPV6_V6ONLY = 26 } // macros extern (D) int IN6_IS_ADDR_UNSPECIFIED( in6_addr* addr ) { return (cast(uint32_t*) addr)[0] == 0 && (cast(uint32_t*) addr)[1] == 0 && (cast(uint32_t*) addr)[2] == 0 && (cast(uint32_t*) addr)[3] == 0; } extern (D) int IN6_IS_ADDR_LOOPBACK( in6_addr* addr ) { return (cast(uint32_t*) addr)[0] == 0 && (cast(uint32_t*) addr)[1] == 0 && (cast(uint32_t*) addr)[2] == 0 && (cast(uint32_t*) addr)[3] == htonl( 1 ); } extern (D) int IN6_IS_ADDR_MULTICAST( in6_addr* addr ) { return (cast(uint8_t*) addr)[0] == 0xff; } extern (D) int IN6_IS_ADDR_LINKLOCAL( in6_addr* addr ) { return ((cast(uint32_t*) addr)[0] & htonl( 0xffc00000 )) == htonl( 0xfe800000 ); } extern (D) int IN6_IS_ADDR_SITELOCAL( in6_addr* addr ) { return ((cast(uint32_t*) addr)[0] & htonl( 0xffc00000 )) == htonl( 0xfec00000 ); } extern (D) int IN6_IS_ADDR_V4MAPPED( in6_addr* addr ) { return (cast(uint32_t*) addr)[0] == 0 && (cast(uint32_t*) addr)[1] == 0 && (cast(uint32_t*) addr)[2] == htonl( 0xffff ); } extern (D) int IN6_IS_ADDR_V4COMPAT( in6_addr* addr ) { return (cast(uint32_t*) addr)[0] == 0 && (cast(uint32_t*) addr)[1] == 0 && (cast(uint32_t*) addr)[2] == 0 && ntohl( (cast(uint32_t*) addr)[3] ) > 1; } extern (D) int IN6_IS_ADDR_MC_NODELOCAL( in6_addr* addr ) { return IN6_IS_ADDR_MULTICAST( addr ) && ((cast(uint8_t*) addr)[1] & 0xf) == 0x1; } extern (D) int IN6_IS_ADDR_MC_LINKLOCAL( in6_addr* addr ) { return IN6_IS_ADDR_MULTICAST( addr ) && ((cast(uint8_t*) addr)[1] & 0xf) == 0x2; } extern (D) int IN6_IS_ADDR_MC_SITELOCAL( in6_addr* addr ) { return IN6_IS_ADDR_MULTICAST(addr) && ((cast(uint8_t*) addr)[1] & 0xf) == 0x5; } extern (D) int IN6_IS_ADDR_MC_ORGLOCAL( in6_addr* addr ) { return IN6_IS_ADDR_MULTICAST( addr) && ((cast(uint8_t*) addr)[1] & 0xf) == 0x8; } extern (D) int IN6_IS_ADDR_MC_GLOBAL( in6_addr* addr ) { return IN6_IS_ADDR_MULTICAST( addr ) && ((cast(uint8_t*) addr)[1] & 0xf) == 0xe; } } version ( solaris ) { struct in6_addr { union { uint8_t[16] s6_addr; uint32_t[4] s6_addr32; uint32_t __S6_align; } } struct sockaddr_in6 { sa_family_t sin6_family; in_port_t sin6_port; uint32_t sin6_flowinfo; in6_addr sin6_addr; uint32_t sin6_scope_id; uint32_t __sin6_src_id; /* Impl. specific - UDP replies */ } extern in6_addr in6addr_any; extern in6_addr in6addr_loopback; struct ipv6_mreq { in6_addr ipv6mr_multiaddr; uint ipv6mr_interface; } enum : uint { IPPROTO_IPV6 = 41, INET6_ADDRSTRLEN = 46, IPV6_JOIN_GROUP = 0x9, IPV6_LEAVE_GROUP = 0xa, IPV6_MULTICAST_HOPS = 0x7, IPV6_MULTICAST_IF = 0x6, IPV6_MULTICAST_LOOP = 0x8, IPV6_UNICAST_HOPS = 0x5, IPV6_V6ONLY = 0x27 } // macros extern (D) int IN6_IS_ADDR_UNSPECIFIED( in6_addr* addr ) { return addr.s6_addr32[3] == 0 && addr.s6_addr32[2] == 0 && addr.s6_addr32[1] == 0 && addr.s6_addr32[0] == 0; } version(BigEndian) { extern (D) int IN6_IS_ADDR_LOOPBACK( in6_addr* addr ) { version(BigEndian) enum : uint { N = 0x00000001 } else enum : uint { N = 0x01000000 } return addr.s6_addr32[3] == N && addr.s6_addr32[2] == 0 && addr.s6_addr32[1] == 0 && addr.s6_addr32[0] == 0; } } // // Note to tango devs: // These macros seem alot more efficient then the Linux ones! // extern (D) int IN6_IS_ADDR_MULTICAST( in6_addr* addr ) { version(BigEndian) return (addr.s6_addr32[0] & 0xff000000) == 0xff000000; else return (addr.s6_addr32[0] & 0x000000ff) == 0x000000ff; } extern (D) int IN6_IS_ADDR_LINKLOCAL( in6_addr* addr ) { version(BigEndian) return (addr.s6_addr32[0] & 0xffc00000) == 0xfe800000; else return (addr.s6_addr32[0] & 0x0000c0ff) == 0x000080fe; } extern (D) int IN6_IS_ADDR_SITELOCAL( in6_addr* addr ) { version(BigEndian) return (addr.s6_addr32[0] & 0xffc00000) == 0xfec00000; else return (addr.s6_addr32[0] & 0x0000c0ff) == 0x0000c0fe; } extern (D) int IN6_IS_ADDR_V4MAPPED( in6_addr* addr ) { version(BigEndian) enum : uint { N = 0x0000ffff } else enum : uint { N = 0xffff0000 } return addr.s6_addr32[2] == N && addr.s6_addr32[1] == 0 && addr.s6_addr32[0] == 0; } extern (D) int IN6_IS_ADDR_V4COMPAT( in6_addr* addr ) { version(BigEndian) enum : uint { N = 0x00000001 } else enum : uint { N = 0x01000000 } return addr.s6_addr32[2] == 0 && addr.s6_addr32[1] == 0 && addr.s6_addr32[0] == 0 && addr.s6_addr32[3] != 0 && addr.s6_addr32[3] != N; } extern (D) int IN6_IS_ADDR_MC_NODELOCAL( in6_addr* addr ) { version(BigEndian) return (addr.s6_addr32[0] & 0xff0f0000) == 0xff010000; else return (addr.s6_addr32[0] & 0x00000fff) == 0x000001ff; } extern (D) int IN6_IS_ADDR_MC_LINKLOCAL( in6_addr* addr ) { version(BigEndian) return (addr.s6_addr32[0] & 0xff0f0000) == 0xff020000; else return (addr.s6_addr32[0] & 0x00000fff) == 0x000002ff; } extern (D) int IN6_IS_ADDR_MC_SITELOCAL( in6_addr* addr ) { version(BigEndian) return (addr.s6_addr32[0] & 0xff0f0000) == 0xff050000; else return (addr.s6_addr32[0] & 0x00000fff) == 0x000005ff; } extern (D) int IN6_IS_ADDR_MC_ORGLOCAL( in6_addr* addr ) { version(BigEndian) return (addr.s6_addr32[0] & 0xff0f0000) == 0xff080000; else return (addr.s6_addr32[0] & 0x00000fff) == 0x000008ff; } extern (D) int IN6_IS_ADDR_MC_GLOBAL( in6_addr* addr ) { version(BigEndian) return (addr.s6_addr32[0] & 0xff0f0000) == 0xff0e0000; else return (addr.s6_addr32[0] & 0x00000fff) == 0x00000eff; } } // // Raw Sockets (RS) // /* IPPROTO_RAW */ version ( linux ) { const uint IPPROTO_RAW = 255; } else version ( solaris ) { const uint IPPROTO_RAW = 255; } |