2022-02-06 20:05:29 -05:00
|
|
|
// Copyright 2011-2018 Nicholas J. Kain <njkain at gmail dot com>
|
|
|
|
// SPDX-License-Identifier: MIT
|
2014-03-17 05:56:30 -04:00
|
|
|
#include <unistd.h>
|
2011-07-02 01:34:50 -04:00
|
|
|
#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>
|
2014-03-15 00:30:39 -04:00
|
|
|
#include <time.h>
|
2011-07-02 01:34:50 -04:00
|
|
|
#include <errno.h>
|
2011-07-03 18:10:00 -04:00
|
|
|
#include <poll.h>
|
2014-03-30 17:02:48 -04:00
|
|
|
#include "nk/log.h"
|
2011-07-02 01:34:50 -04:00
|
|
|
|
|
|
|
#include "netlink.h"
|
2011-07-03 17:30:55 -04:00
|
|
|
#include "nl.h"
|
2011-07-04 22:10:14 -04:00
|
|
|
#include "state.h"
|
2011-07-02 01:34:50 -04:00
|
|
|
|
2017-01-12 06:05:00 -05:00
|
|
|
// Returns true if the current interface state is UP.
|
|
|
|
bool nl_event_carrier_wentup(int state)
|
2011-07-02 01:34:50 -04:00
|
|
|
{
|
2015-02-15 06:38:03 -05:00
|
|
|
switch (state) {
|
|
|
|
case IFS_UP:
|
2017-01-12 06:05:00 -05:00
|
|
|
log_line("%s: Carrier up.", client_config.interface);
|
|
|
|
return true;
|
2015-02-15 06:38:03 -05:00
|
|
|
case IFS_DOWN:
|
|
|
|
// Interface configured, but no hardware carrier.
|
2015-02-18 05:31:13 -05:00
|
|
|
log_line("%s: Carrier down.", client_config.interface);
|
2017-01-12 06:05:00 -05:00
|
|
|
return false;
|
2015-02-15 06:38:03 -05:00
|
|
|
case IFS_SHUT:
|
|
|
|
// User shut down the interface.
|
2017-01-12 06:05:00 -05:00
|
|
|
log_line("%s: Interface shut down.", client_config.interface);
|
|
|
|
return false;
|
2015-02-15 06:38:03 -05:00
|
|
|
case IFS_REMOVED:
|
|
|
|
log_line("Interface removed. Exiting.");
|
|
|
|
exit(EXIT_SUCCESS);
|
2017-01-12 06:05:00 -05:00
|
|
|
default: return false;
|
2011-07-02 01:34:50 -04:00
|
|
|
}
|
2015-02-15 06:38:03 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static int nl_process_msgs_return;
|
|
|
|
static void nl_process_msgs(const struct nlmsghdr *nlh, void *data)
|
|
|
|
{
|
|
|
|
(void)data;
|
|
|
|
struct ifinfomsg *ifm = NLMSG_DATA(nlh);
|
|
|
|
|
|
|
|
if (ifm->ifi_index != client_config.ifindex)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (nlh->nlmsg_type == RTM_NEWLINK) {
|
|
|
|
// IFF_UP corresponds to ifconfig down or ifconfig up.
|
|
|
|
// IFF_RUNNING is the hardware carrier.
|
|
|
|
if (ifm->ifi_flags & IFF_UP) {
|
|
|
|
if (ifm->ifi_flags & IFF_RUNNING)
|
|
|
|
nl_process_msgs_return = IFS_UP;
|
|
|
|
else
|
|
|
|
nl_process_msgs_return = IFS_DOWN;
|
|
|
|
} else {
|
|
|
|
nl_process_msgs_return = IFS_SHUT;
|
|
|
|
}
|
|
|
|
} else if (nlh->nlmsg_type == RTM_DELLINK)
|
|
|
|
nl_process_msgs_return = IFS_REMOVED;
|
2011-07-02 01:34:50 -04:00
|
|
|
}
|
|
|
|
|
2022-01-11 22:35:19 -05:00
|
|
|
int nl_event_get(struct client_state_t *cs)
|
2011-07-02 01:34:50 -04:00
|
|
|
{
|
2012-04-12 20:06:05 -04:00
|
|
|
char nlbuf[8192];
|
2011-07-03 17:30:55 -04:00
|
|
|
ssize_t ret;
|
2011-07-02 01:34:50 -04:00
|
|
|
assert(cs->nlFd != -1);
|
2015-02-15 06:38:03 -05:00
|
|
|
nl_process_msgs_return = IFS_NONE;
|
2011-07-02 01:34:50 -04:00
|
|
|
do {
|
2011-07-03 17:30:55 -04:00
|
|
|
ret = nl_recv_buf(cs->nlFd, nlbuf, sizeof nlbuf);
|
2014-04-06 06:33:14 -04:00
|
|
|
if (ret < 0)
|
2011-07-03 17:30:55 -04:00
|
|
|
break;
|
2018-02-09 02:39:46 -05:00
|
|
|
if (nl_foreach_nlmsg(nlbuf, (size_t)ret, 0, cs->nlPortId, nl_process_msgs, 0) < 0)
|
2011-07-03 17:30:55 -04:00
|
|
|
break;
|
2011-07-02 01:34:50 -04:00
|
|
|
} while (ret > 0);
|
2015-02-15 06:38:03 -05:00
|
|
|
return nl_process_msgs_return;
|
2011-07-02 01:34:50 -04:00
|
|
|
}
|
|
|
|
|
2014-03-17 05:56:30 -04:00
|
|
|
static int get_if_index_and_mac(const struct nlmsghdr *nlh,
|
|
|
|
struct ifinfomsg *ifm)
|
|
|
|
{
|
2014-03-17 20:22:20 -04:00
|
|
|
struct rtattr *tb[IFLA_MAX] = {0};
|
|
|
|
nl_rtattr_parse(nlh, sizeof *ifm, rtattr_assign, tb);
|
2014-03-17 05:56:30 -04:00
|
|
|
if (tb[IFLA_IFNAME] && !strncmp(client_config.interface,
|
2014-03-17 20:28:26 -04:00
|
|
|
RTA_DATA(tb[IFLA_IFNAME]),
|
2014-03-17 05:56:30 -04:00
|
|
|
sizeof client_config.interface)) {
|
|
|
|
client_config.ifindex = ifm->ifi_index;
|
|
|
|
if (!tb[IFLA_ADDRESS])
|
2020-11-24 21:02:51 -05:00
|
|
|
suicide("FATAL: Adapter %s lacks a hardware address.", client_config.interface);
|
2014-03-17 20:22:20 -04:00
|
|
|
int maclen = tb[IFLA_ADDRESS]->rta_len - 4;
|
2014-03-17 05:56:30 -04:00
|
|
|
if (maclen != 6)
|
|
|
|
suicide("FATAL: Adapter hardware address length should be 6, but is %u.",
|
|
|
|
maclen);
|
|
|
|
|
2014-03-17 20:28:26 -04:00
|
|
|
const unsigned char *mac = RTA_DATA(tb[IFLA_ADDRESS]);
|
2014-03-17 05:56:30 -04:00
|
|
|
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;
|
2014-03-17 20:26:37 -04:00
|
|
|
struct ifinfomsg *ifm = NLMSG_DATA(nlh);
|
2014-03-17 05:56:30 -04:00
|
|
|
|
|
|
|
switch(nlh->nlmsg_type) {
|
|
|
|
case RTM_NEWLINK:
|
|
|
|
*got_ifdata |= get_if_index_and_mac(nlh, ifm);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-17 06:15:59 -04:00
|
|
|
static int handle_getifdata(int fd, uint32_t seq)
|
2014-03-17 05:56:30 -04:00
|
|
|
{
|
|
|
|
char nlbuf[8192];
|
|
|
|
ssize_t ret;
|
|
|
|
int got_ifdata = 0;
|
|
|
|
do {
|
|
|
|
ret = nl_recv_buf(fd, nlbuf, sizeof nlbuf);
|
2014-04-06 06:33:14 -04:00
|
|
|
if (ret < 0)
|
2014-03-17 05:56:30 -04:00
|
|
|
return -1;
|
2018-02-09 02:39:46 -05:00
|
|
|
if (nl_foreach_nlmsg(nlbuf, (size_t)ret, seq, 0,
|
2014-04-06 06:33:14 -04:00
|
|
|
do_handle_getifdata, &got_ifdata) < 0)
|
2014-03-17 05:56:30 -04:00
|
|
|
return -1;
|
|
|
|
} while (ret > 0);
|
|
|
|
return got_ifdata ? 0 : -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int nl_getifdata(void)
|
2012-04-03 22:00:47 -04:00
|
|
|
{
|
2014-03-17 05:56:30 -04:00
|
|
|
int ret = -1;
|
2022-08-13 00:42:27 -04:00
|
|
|
int fd = socket(AF_NETLINK, SOCK_DGRAM|SOCK_CLOEXEC, NETLINK_ROUTE);
|
2014-03-17 05:56:30 -04:00
|
|
|
if (fd < 0) {
|
|
|
|
log_line("%s: (%s) netlink socket open failed: %s",
|
|
|
|
client_config.interface, __func__, strerror(errno));
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
2014-03-17 06:15:59 -04:00
|
|
|
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)) {
|
2014-03-17 05:56:30 -04:00
|
|
|
log_line("%s: (%s) nl_sendgetlinks failed",
|
|
|
|
client_config.interface, __func__);
|
|
|
|
goto fail_fd;
|
|
|
|
}
|
2011-07-02 01:34:50 -04:00
|
|
|
|
2011-07-03 18:10:00 -04:00
|
|
|
for (int pr = 0; !pr;) {
|
2014-03-17 05:56:30 -04:00
|
|
|
pr = poll(&((struct pollfd){.fd=fd,.events=POLLIN}), 1, -1);
|
2011-07-03 18:10:00 -04:00
|
|
|
if (pr == 1)
|
2014-03-17 06:15:59 -04:00
|
|
|
ret = handle_getifdata(fd, seq);
|
2014-04-06 06:33:14 -04:00
|
|
|
else if (pr < 0 && errno != EINTR)
|
2014-03-17 05:56:30 -04:00
|
|
|
goto fail_fd;
|
2011-07-03 18:10:00 -04:00
|
|
|
}
|
2014-03-17 05:56:30 -04:00
|
|
|
fail_fd:
|
|
|
|
close(fd);
|
|
|
|
fail:
|
|
|
|
return ret;
|
2011-07-02 01:34:50 -04:00
|
|
|
}
|
|
|
|
|