1 /* $NetBSD: ntp_net.h,v 1.5 2020/05/25 20:47:19 christos Exp $ */ 2 3 /* 4 * ntp_net.h - definitions for NTP network stuff 5 */ 6 7 #ifndef NTP_NET_H 8 #define NTP_NET_H 9 10 #include <sys/types.h> 11 #ifdef HAVE_SYS_SOCKET_H 12 #include <sys/socket.h> 13 #endif 14 #ifdef HAVE_NET_IF_H 15 #include <net/if.h> 16 #endif 17 #ifdef HAVE_NETINET_IN_H 18 #include <netinet/in.h> 19 #endif 20 #ifdef HAVE_NET_IF_VAR_H 21 #include <net/if_var.h> 22 #endif 23 #ifdef HAVE_NETINET_IN_VAR_H 24 #include <netinet/in_var.h> 25 #endif 26 27 #include "ntp_rfc2553.h" 28 #include "ntp_malloc.h" 29 30 typedef union { 31 struct sockaddr sa; 32 struct sockaddr_in sa4; 33 struct sockaddr_in6 sa6; 34 } sockaddr_u; 35 36 /* 37 * Utilities for manipulating sockaddr_u v4/v6 unions 38 */ 39 #define SOCK_ADDR4(psau) ((psau)->sa4.sin_addr) 40 #define SOCK_ADDR6(psau) ((psau)->sa6.sin6_addr) 41 42 #define PSOCK_ADDR4(psau) (&SOCK_ADDR4(psau)) 43 #define PSOCK_ADDR6(psau) (&SOCK_ADDR6(psau)) 44 45 #define AF(psau) ((psau)->sa.sa_family) 46 47 #define IS_IPV4(psau) (AF_INET == AF(psau)) 48 #define IS_IPV6(psau) (AF_INET6 == AF(psau)) 49 50 /* sockaddr_u v4 address in network byte order */ 51 #define NSRCADR(psau) (SOCK_ADDR4(psau).s_addr) 52 53 /* sockaddr_u v4 address in host byte order */ 54 #define SRCADR(psau) (ntohl(NSRCADR(psau))) 55 56 /* sockaddr_u v6 address in network byte order */ 57 #define NSRCADR6(psau) (SOCK_ADDR6(psau).s6_addr) 58 59 /* assign sockaddr_u v4 address from host byte order */ 60 #define SET_ADDR4(psau, addr4) (NSRCADR(psau) = htonl(addr4)) 61 62 /* assign sockaddr_u v4 address from network byte order */ 63 #define SET_ADDR4N(psau, addr4n) (NSRCADR(psau) = (addr4n)); 64 65 /* assign sockaddr_u v6 address from network byte order */ 66 #define SET_ADDR6N(psau, s6_addr) \ 67 (SOCK_ADDR6(psau) = (s6_addr)) 68 69 /* sockaddr_u v4/v6 port in network byte order */ 70 #define NSRCPORT(psau) ((psau)->sa4.sin_port) 71 72 /* sockaddr_u v4/v6 port in host byte order */ 73 #define SRCPORT(psau) (ntohs(NSRCPORT(psau))) 74 75 /* assign sockaddr_u v4/v6 port from host byte order */ 76 #define SET_PORT(psau, port) (NSRCPORT(psau) = htons(port)) 77 78 /* sockaddr_u v6 scope */ 79 #define SCOPE_VAR(psau) ((psau)->sa6.sin6_scope_id) 80 81 #ifdef ISC_PLATFORM_HAVESCOPEID 82 /* v4/v6 scope (always zero for v4) */ 83 # define SCOPE(psau) (IS_IPV4(psau) \ 84 ? 0 \ 85 : SCOPE_VAR(psau)) 86 87 /* are two v6 sockaddr_u scopes equal? */ 88 # define SCOPE_EQ(psau1, psau2) \ 89 (SCOPE_VAR(psau1) == SCOPE_VAR(psau2)) 90 91 /* assign scope if supported */ 92 # define SET_SCOPE(psau, s) \ 93 do \ 94 if (IS_IPV6(psau)) \ 95 SCOPE_VAR(psau) = (s); \ 96 while (0) 97 #else /* ISC_PLATFORM_HAVESCOPEID not defined */ 98 # define SCOPE(psau) (0) 99 # define SCOPE_EQ(psau1, psau2) (1) 100 # define SET_SCOPE(psau, s) do { } while (0) 101 #endif /* ISC_PLATFORM_HAVESCOPEID */ 102 103 /* v4/v6 is multicast address */ 104 #define IS_MCAST(psau) \ 105 (IS_IPV4(psau) \ 106 ? IN_CLASSD(SRCADR(psau)) \ 107 : IN6_IS_ADDR_MULTICAST(PSOCK_ADDR6(psau))) 108 109 /* v6 is interface ID scope universal, as with MAC-derived addresses */ 110 #define IS_IID_UNIV(psau) \ 111 (!!(0x02 & NSRCADR6(psau)[8])) 112 113 #define SIZEOF_INADDR(fam) \ 114 ((AF_INET == (fam)) \ 115 ? sizeof(struct in_addr) \ 116 : sizeof(struct in6_addr)) 117 118 #define SIZEOF_SOCKADDR(fam) \ 119 ((AF_INET == (fam)) \ 120 ? sizeof(struct sockaddr_in) \ 121 : sizeof(struct sockaddr_in6)) 122 123 #define SOCKLEN(psau) \ 124 (IS_IPV4(psau) \ 125 ? sizeof((psau)->sa4) \ 126 : sizeof((psau)->sa6)) 127 128 #define ZERO_SOCK(psau) \ 129 ZERO(*(psau)) 130 131 /* blast a byte value across sockaddr_u v6 address */ 132 #define MEMSET_ADDR6(psau, v) \ 133 memset((psau)->sa6.sin6_addr.s6_addr, (v), \ 134 sizeof((psau)->sa6.sin6_addr.s6_addr)) 135 136 #define SET_ONESMASK(psau) \ 137 do { \ 138 if (IS_IPV6(psau)) \ 139 MEMSET_ADDR6((psau), 0xff); \ 140 else \ 141 NSRCADR(psau) = 0xffffffff; \ 142 } while(0) 143 144 /* zero sockaddr_u, fill in family and all-ones (host) mask */ 145 #define SET_HOSTMASK(psau, family) \ 146 do { \ 147 ZERO_SOCK(psau); \ 148 AF(psau) = (family); \ 149 SET_ONESMASK(psau); \ 150 } while (0) 151 152 /* 153 * compare two in6_addr returning negative, 0, or positive. 154 * ADDR6_CMP is negative if *pin6A is lower than *pin6B, zero if they 155 * are equal, positive if *pin6A is higher than *pin6B. IN6ADDR_ANY 156 * is the lowest address (128 zero bits). 157 */ 158 #define ADDR6_CMP(pin6A, pin6B) \ 159 memcmp((pin6A)->s6_addr, (pin6B)->s6_addr, \ 160 sizeof(pin6A)->s6_addr) 161 162 /* compare two in6_addr for equality only */ 163 #if !defined(SYS_WINNT) || !defined(in_addr6) 164 #define ADDR6_EQ(pin6A, pin6B) \ 165 (!ADDR6_CMP(pin6A, pin6B)) 166 #else 167 #define ADDR6_EQ(pin6A, pin6B) \ 168 IN6_ADDR_EQUAL(pin6A, pin6B) 169 #endif 170 171 /* compare a in6_addr with socket address */ 172 #define S_ADDR6_EQ(psau, pin6) \ 173 ADDR6_EQ(&(psau)->sa6.sin6_addr, pin6) 174 175 /* are two sockaddr_u's addresses equal? (port excluded) */ 176 #define SOCK_EQ(psau1, psau2) \ 177 ((AF(psau1) != AF(psau2)) \ 178 ? 0 \ 179 : IS_IPV4(psau1) \ 180 ? (NSRCADR(psau1) == NSRCADR(psau2)) \ 181 : (S_ADDR6_EQ((psau1), PSOCK_ADDR6(psau2)) \ 182 && SCOPE_EQ((psau1), (psau2)))) 183 184 /* are two sockaddr_u's addresses and ports equal? */ 185 #define ADDR_PORT_EQ(psau1, psau2) \ 186 ((NSRCPORT(psau1) != NSRCPORT(psau2) \ 187 ? 0 \ 188 : SOCK_EQ((psau1), (psau2)))) 189 190 /* is sockaddr_u address unspecified? */ 191 #define SOCK_UNSPEC(psau) \ 192 (IS_IPV4(psau) \ 193 ? !NSRCADR(psau) \ 194 : IN6_IS_ADDR_UNSPECIFIED(PSOCK_ADDR6(psau))) 195 196 /* just how unspecified do you mean? (scope 0/unspec too) */ 197 #define SOCK_UNSPEC_S(psau) \ 198 (SOCK_UNSPEC(psau) && !SCOPE(psau)) 199 200 /* choose a default net interface (struct interface) for v4 or v6 */ 201 #define ANY_INTERFACE_BYFAM(family) \ 202 ((AF_INET == family) \ 203 ? any_interface \ 204 : any6_interface) 205 206 /* choose a default interface for addresses' protocol (addr family) */ 207 #define ANY_INTERFACE_CHOOSE(psau) \ 208 ANY_INTERFACE_BYFAM(AF(psau)) 209 210 211 /* 212 * We tell reference clocks from real peers by giving the reference 213 * clocks an address of the form 127.127.t.u, where t is the type and 214 * u is the unit number. We define some of this here since we will need 215 * some sanity checks to make sure this address isn't interpretted as 216 * that of a normal peer. 217 */ 218 #define REFCLOCK_ADDR 0x7f7f0000 /* 127.127.0.0 */ 219 #define REFCLOCK_MASK 0xffff0000 /* 255.255.0.0 */ 220 221 #define ISREFCLOCKADR(srcadr) \ 222 (IS_IPV4(srcadr) && \ 223 (SRCADR(srcadr) & REFCLOCK_MASK) == REFCLOCK_ADDR) 224 225 /* 226 * Macro for checking for invalid addresses. This is really, really 227 * gross, but is needed so no one configures a host on net 127 now that 228 * we're encouraging it the the configuration file. 229 */ 230 #define LOOPBACKADR 0x7f000001 231 #define LOOPNETMASK 0xff000000 232 233 #define ISBADADR(srcadr) \ 234 (IS_IPV4(srcadr) \ 235 && ((SRCADR(srcadr) & LOOPNETMASK) \ 236 == (LOOPBACKADR & LOOPNETMASK)) \ 237 && SRCADR(srcadr) != LOOPBACKADR) 238 239 240 #endif /* NTP_NET_H */ 241