123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131 |
- /* 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;
- log1("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;
- }
|