Move source from ndhc/ to src/ since ifchd is no longer a separate program.
This commit is contained in:
194
src/netlink.c
Normal file
194
src/netlink.c
Normal file
@ -0,0 +1,194 @@
|
||||
/* netlink.c - netlink physical link notification handling and info retrieval
|
||||
*
|
||||
* Copyright (c) 2011-2014 Nicholas J. Kain <njkain at gmail dot com>
|
||||
* All rights reserved.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
* modification, are permitted provided that the following conditions are met:
|
||||
*
|
||||
* - Redistributions of source code must retain the above copyright notice,
|
||||
* this list of conditions and the following disclaimer.
|
||||
*
|
||||
* - Redistributions in binary form must reproduce the above copyright notice,
|
||||
* this list of conditions and the following disclaimer in the documentation
|
||||
* and/or other materials provided with the distribution.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||||
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
|
||||
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
||||
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
||||
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
||||
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
||||
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
||||
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||||
* POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
#include <unistd.h>
|
||||
#include <arpa/inet.h>
|
||||
#include <assert.h>
|
||||
#include <asm/types.h>
|
||||
#include <sys/socket.h>
|
||||
#include <net/if.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <time.h>
|
||||
#include <errno.h>
|
||||
#include <poll.h>
|
||||
#include "nk/log.h"
|
||||
|
||||
#include "netlink.h"
|
||||
#include "nl.h"
|
||||
#include "state.h"
|
||||
|
||||
static void nl_process_msgs(const struct nlmsghdr *nlh, void *data)
|
||||
{
|
||||
struct ifinfomsg *ifm = NLMSG_DATA(nlh);
|
||||
struct client_state_t *cs = data;
|
||||
|
||||
switch(nlh->nlmsg_type) {
|
||||
case RTM_NEWLINK:
|
||||
if (ifm->ifi_index != client_config.ifindex)
|
||||
break;
|
||||
// IFF_UP corresponds to ifconfig down or ifconfig up.
|
||||
if (ifm->ifi_flags & IFF_UP) {
|
||||
// IFF_RUNNING is the hardware carrier.
|
||||
if (ifm->ifi_flags & IFF_RUNNING) {
|
||||
if (cs->ifsPrevState != IFS_UP) {
|
||||
cs->ifsPrevState = IFS_UP;
|
||||
ifup_action(cs);
|
||||
}
|
||||
} else if (cs->ifsPrevState != IFS_DOWN) {
|
||||
// Interface configured, but no hardware carrier.
|
||||
cs->ifsPrevState = IFS_DOWN;
|
||||
ifnocarrier_action(cs);
|
||||
}
|
||||
} else if (cs->ifsPrevState != IFS_SHUT) {
|
||||
// User shut down the interface.
|
||||
cs->ifsPrevState = IFS_SHUT;
|
||||
ifdown_action(cs);
|
||||
}
|
||||
break;
|
||||
case RTM_DELLINK:
|
||||
if (ifm->ifi_index != client_config.ifindex)
|
||||
break;
|
||||
if (cs->ifsPrevState != IFS_REMOVED) {
|
||||
cs->ifsPrevState = IFS_REMOVED;
|
||||
log_line("Interface removed. Exiting.");
|
||||
exit(EXIT_SUCCESS);
|
||||
}
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void handle_nl_message(struct client_state_t *cs)
|
||||
{
|
||||
char nlbuf[8192];
|
||||
ssize_t ret;
|
||||
assert(cs->nlFd != -1);
|
||||
do {
|
||||
ret = nl_recv_buf(cs->nlFd, nlbuf, sizeof nlbuf);
|
||||
if (ret < 0)
|
||||
break;
|
||||
if (nl_foreach_nlmsg(nlbuf, ret, 0, cs->nlPortId, nl_process_msgs, cs)
|
||||
< 0)
|
||||
break;
|
||||
} while (ret > 0);
|
||||
}
|
||||
|
||||
static int get_if_index_and_mac(const struct nlmsghdr *nlh,
|
||||
struct ifinfomsg *ifm)
|
||||
{
|
||||
struct rtattr *tb[IFLA_MAX] = {0};
|
||||
nl_rtattr_parse(nlh, sizeof *ifm, rtattr_assign, tb);
|
||||
if (tb[IFLA_IFNAME] && !strncmp(client_config.interface,
|
||||
RTA_DATA(tb[IFLA_IFNAME]),
|
||||
sizeof client_config.interface)) {
|
||||
client_config.ifindex = ifm->ifi_index;
|
||||
if (!tb[IFLA_ADDRESS])
|
||||
suicide("FATAL: Adapter %s lacks a hardware address.");
|
||||
int maclen = tb[IFLA_ADDRESS]->rta_len - 4;
|
||||
if (maclen != 6)
|
||||
suicide("FATAL: Adapter hardware address length should be 6, but is %u.",
|
||||
maclen);
|
||||
|
||||
const unsigned char *mac = RTA_DATA(tb[IFLA_ADDRESS]);
|
||||
log_line("%s hardware address %x:%x:%x:%x:%x:%x",
|
||||
client_config.interface,
|
||||
mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]);
|
||||
memcpy(client_config.arp, mac, 6);
|
||||
return 1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void do_handle_getifdata(const struct nlmsghdr *nlh, void *data)
|
||||
{
|
||||
int *got_ifdata = (int *)data;
|
||||
struct ifinfomsg *ifm = NLMSG_DATA(nlh);
|
||||
|
||||
switch(nlh->nlmsg_type) {
|
||||
case RTM_NEWLINK:
|
||||
*got_ifdata |= get_if_index_and_mac(nlh, ifm);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
static int handle_getifdata(int fd, uint32_t seq)
|
||||
{
|
||||
char nlbuf[8192];
|
||||
ssize_t ret;
|
||||
int got_ifdata = 0;
|
||||
do {
|
||||
ret = nl_recv_buf(fd, nlbuf, sizeof nlbuf);
|
||||
if (ret < 0)
|
||||
return -1;
|
||||
if (nl_foreach_nlmsg(nlbuf, ret, seq, 0,
|
||||
do_handle_getifdata, &got_ifdata) < 0)
|
||||
return -1;
|
||||
} while (ret > 0);
|
||||
return got_ifdata ? 0 : -1;
|
||||
}
|
||||
|
||||
int nl_getifdata(void)
|
||||
{
|
||||
int ret = -1;
|
||||
int fd = socket(AF_NETLINK, SOCK_DGRAM, NETLINK_ROUTE);
|
||||
if (fd < 0) {
|
||||
log_line("%s: (%s) netlink socket open failed: %s",
|
||||
client_config.interface, __func__, strerror(errno));
|
||||
goto fail;
|
||||
}
|
||||
|
||||
struct timespec ts;
|
||||
if (clock_gettime(CLOCK_REALTIME, &ts) < 0) {
|
||||
log_line("%s: (%s) clock_gettime failed",
|
||||
client_config.interface, __func__);
|
||||
goto fail_fd;
|
||||
}
|
||||
uint32_t seq = ts.tv_nsec;
|
||||
if (nl_sendgetlinks(fd, seq)) {
|
||||
log_line("%s: (%s) nl_sendgetlinks failed",
|
||||
client_config.interface, __func__);
|
||||
goto fail_fd;
|
||||
}
|
||||
|
||||
for (int pr = 0; !pr;) {
|
||||
pr = poll(&((struct pollfd){.fd=fd,.events=POLLIN}), 1, -1);
|
||||
if (pr == 1)
|
||||
ret = handle_getifdata(fd, seq);
|
||||
else if (pr < 0 && errno != EINTR)
|
||||
goto fail_fd;
|
||||
}
|
||||
fail_fd:
|
||||
close(fd);
|
||||
fail:
|
||||
return ret;
|
||||
}
|
||||
|
Reference in New Issue
Block a user