| /* vi: set sw=4 ts=4: */ |
| /* |
| * Copyright (C) 2011 Denys Vlasenko. |
| * |
| * Licensed under GPLv2, see file LICENSE in this source tree. |
| */ |
| #include "common.h" |
| #include "d6_common.h" |
| #include <net/if.h> |
| #include <ifaddrs.h> |
| #include <netpacket/packet.h> |
| |
| int FAST_FUNC d6_read_interface( |
| const char *interface, |
| int *ifindex, |
| struct in6_addr *nip6, |
| uint8_t *mac) |
| { |
| int retval = 3; |
| struct ifaddrs *ifap, *ifa; |
| |
| getifaddrs(&ifap); |
| for (ifa = ifap; ifa; ifa = ifa->ifa_next) { |
| struct sockaddr_in6 *sip6; |
| |
| if (!ifa->ifa_addr || (strcmp(ifa->ifa_name, interface) != 0)) |
| continue; |
| |
| if (ifa->ifa_addr->sa_family == AF_PACKET) { |
| struct sockaddr_ll *sll = (void*)(ifa->ifa_addr); |
| memcpy(mac, sll->sll_addr, 6); |
| log2("MAC %02x:%02x:%02x:%02x:%02x:%02x", |
| mac[0], mac[1], mac[2], mac[3], mac[4], mac[5] |
| ); |
| *ifindex = sll->sll_ifindex; |
| log2("ifindex %d", *ifindex); |
| retval &= (3 - (1<<0)); |
| } |
| #if 0 |
| if (ifa->ifa_addr->sa_family == AF_INET) { |
| *nip = ((struct sockaddr_in *)ifa->ifa_addr)->sin_addr.s_addr; |
| log1("IP %s", inet_ntoa(((struct sockaddr_in *)ifa->ifa_addr)->sin_addr)); |
| } |
| #endif |
| /* RFC 3315 |
| * 16. Client Source Address and Interface Selection |
| * |
| * "When a client sends a DHCP message to the |
| * All_DHCP_Relay_Agents_and_Servers address, ... ... The client |
| * MUST use a link-local address assigned to the interface for which it |
| * is requesting configuration information as the source address in the |
| * header of the IP datagram." |
| */ |
| sip6 = (void*)(ifa->ifa_addr); |
| |
| if (ifa->ifa_addr->sa_family == AF_INET6 |
| && IN6_IS_ADDR_LINKLOCAL(&sip6->sin6_addr) |
| ) { |
| *nip6 = sip6->sin6_addr; /* struct copy */ |
| log1( |
| "IPv6 %02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x", |
| nip6->s6_addr[0], nip6->s6_addr[1], |
| nip6->s6_addr[2], nip6->s6_addr[3], |
| nip6->s6_addr[4], nip6->s6_addr[5], |
| nip6->s6_addr[6], nip6->s6_addr[7], |
| nip6->s6_addr[8], nip6->s6_addr[9], |
| nip6->s6_addr[10], nip6->s6_addr[11], |
| nip6->s6_addr[12], nip6->s6_addr[13], |
| nip6->s6_addr[14], nip6->s6_addr[15] |
| ); |
| retval &= (3 - (1<<1)); |
| } |
| } |
| freeifaddrs(ifap); |
| |
| if (retval & (1<<0)) { |
| /* This iface has no MAC (e.g. ppp), generate a random one */ |
| struct ifreq ifr; |
| int fd; |
| |
| /*memset(&ifr, 0, sizeof(ifr)); - SIOCGIFINDEX does not need to clear all */ |
| strncpy_IFNAMSIZ(ifr.ifr_name, interface); |
| fd = xsocket(AF_INET6, SOCK_RAW, IPPROTO_RAW); |
| if (ioctl(fd, SIOCGIFINDEX, &ifr) == 0) { |
| *ifindex = ifr.ifr_ifindex; |
| log2("ifindex %d", *ifindex); |
| if (((uint32_t*)mac)[0] == 0) { |
| /* invent a fictitious MAC (once) */ |
| ((uint32_t*)mac)[0] = rand(); |
| ((uint16_t*)mac)[2] = rand(); |
| mac[0] &= 0xfc; /* make sure it's not bcast */ |
| } |
| retval &= (3 - (1<<0)); |
| } |
| close(fd); |
| } |
| |
| if (retval == 0) |
| return retval; |
| |
| if (retval & (1<<0)) |
| bb_error_msg("can't get %s", "MAC"); |
| if (retval & (1<<1)) |
| bb_error_msg("can't get %s", "link-local IPv6 address"); |
| return retval; |
| } |
| |
| int FAST_FUNC d6_listen_socket(int port, const char *inf) |
| { |
| int fd; |
| struct sockaddr_in6 addr; |
| |
| log2("opening listen socket on *:%d %s", port, inf); |
| fd = xsocket(PF_INET6, SOCK_DGRAM, IPPROTO_UDP); |
| |
| setsockopt_reuseaddr(fd); |
| if (setsockopt_broadcast(fd) == -1) |
| bb_simple_perror_msg_and_die("SO_BROADCAST"); |
| |
| /* NB: bug 1032 says this doesn't work on ethernet aliases (ethN:M) */ |
| if (setsockopt_bindtodevice(fd, inf)) |
| xfunc_die(); /* warning is already printed */ |
| |
| memset(&addr, 0, sizeof(addr)); |
| addr.sin6_family = AF_INET6; |
| addr.sin6_port = htons(port); |
| /* addr.sin6_addr is all-zeros */ |
| xbind(fd, (struct sockaddr *)&addr, sizeof(addr)); |
| |
| return fd; |
| } |