toaruos/apps/ifconfig.c

253 lines
6.6 KiB
C

/**
* @file apps/ifconfig.c
* @brief Network interface configuration tool.
*
* Manipulates and enumerates network interfaces.
*
* All of the APIs used in this tool are temporary and subject to change.
*
* @copyright
* This file is part of ToaruOS and is released under the terms
* of the NCSA / University of Illinois License - see LICENSE.md
* Copyright (C) 2021 K. Lange
*/
#include <stdio.h>
#include <stdint.h>
#include <string.h>
#include <dirent.h>
#include <fcntl.h>
#include <ctype.h>
#include <unistd.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <net/if.h>
#include <arpa/inet.h>
#include <netinet/in.h>
extern char * _argv_0;
static void ip_ntoa(const uint32_t src_addr, char * out) {
snprintf(out, 16, "%d.%d.%d.%d",
(src_addr & 0xFF000000) >> 24,
(src_addr & 0xFF0000) >> 16,
(src_addr & 0xFF00) >> 8,
(src_addr & 0xFF));
}
static char * flagsToStr(uint32_t flags) {
static char out[1024] = {0};
char * o = out;
#define FLAG(f) if (flags & IFF_ ## f) { \
if (o != out) o += sprintf(o,","); \
o += sprintf(o,"%s",#f); } \
FLAG(UP)
FLAG(BROADCAST)
FLAG(DEBUG)
FLAG(LOOPBACK)
FLAG(RUNNING)
FLAG(MULTICAST)
return out;
}
static int print_human_readable_size(char * _out, size_t s) {
size_t count = 5;
char * prefix = "PTGMK";
for (; count > 0 && *prefix; count--, prefix++) {
size_t base = 1UL << (count * 10);
if (s >= base) {
size_t t = s / base;
return sprintf(_out, "%zu.%1zu %cB", t, (s - t * base) / (base / 10), *prefix);
}
}
return sprintf(_out, "%d B", (int)s);
}
static int open_netdev(const char * if_name) {
char if_path[100];
snprintf(if_path, 100, "/dev/net/%s", if_name);
return open(if_path, O_RDONLY);
}
static int print_interface(const char * if_name) {
int netdev = open_netdev(if_name);
if (netdev < 0) {
perror(_argv_0);
return 1;
}
uint32_t flags = 0;
ioctl(netdev, SIOCGIFFLAGS, &flags);
uint32_t mtu = 0;
ioctl(netdev, SIOCGIFMTU, &mtu);
fprintf(stdout,"%s: flags=%d<%s> mtu %d\n", if_name, flags, flagsToStr(flags), mtu);
/* Get IPv4 address */
uint32_t ip_addr = 0;
if (!ioctl(netdev, SIOCGIFADDR, &ip_addr)) {
char ip_str[16];
ip_ntoa(ntohl(ip_addr), ip_str);
fprintf(stdout," inet %s", ip_str);
/* Netmask ? */
uint32_t netmask = 0;
if (!ioctl(netdev, SIOCGIFNETMASK, &netmask)) {
ip_ntoa(ntohl(netmask), ip_str);
fprintf(stdout, " netmask %s", ip_str);
uint32_t bcast = (ip_addr & netmask) | (~netmask);
ip_ntoa(ntohl(bcast), ip_str);
fprintf(stdout, " broadcast %s", ip_str);
}
fprintf(stdout,"\n");
}
uint8_t ip6_addr[16];
if (!ioctl(netdev, SIOCGIFADDR6, &ip6_addr)) {
/* TODO inet6 address to nice string */
fprintf(stdout," inet6 %02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x\n",
ip6_addr[0], ip6_addr[1], ip6_addr[2], ip6_addr[3],
ip6_addr[4], ip6_addr[5], ip6_addr[6], ip6_addr[7],
ip6_addr[8], ip6_addr[9], ip6_addr[10], ip6_addr[11],
ip6_addr[12], ip6_addr[13], ip6_addr[14], ip6_addr[15]);
}
/* Get ethernet address */
uint8_t mac_addr[6];
if (!ioctl(netdev, SIOCGIFHWADDR, &mac_addr)) {
fprintf(stdout," ether %02x:%02x:%02x:%02x:%02x:%02x\n",
mac_addr[0], mac_addr[1], mac_addr[2],
mac_addr[3], mac_addr[4], mac_addr[5]);
}
netif_counters_t counts;
if (!ioctl(netdev, SIOCGIFCOUNTS, &counts)) {
char _buf[100];
print_human_readable_size(_buf, counts.rx_bytes);
fprintf(stdout," RX packets %zu bytes %zu (%s)\n", counts.rx_count, counts.rx_bytes, _buf);
print_human_readable_size(_buf, counts.tx_bytes);
fprintf(stdout," TX packets %zu bytes %zu (%s)\n", counts.tx_count, counts.tx_bytes, _buf);
}
/* TODO stats */
fprintf(stdout,"\n");
return 0;
}
static int print_all_interfaces(void) {
int retval = 0;
/* Read /dev/net for interfaces */
DIR * d = opendir("/dev/net");
if (!d) {
fprintf(stderr, "%s: no network?\n", _argv_0);
return 1;
}
struct dirent * ent;
while ((ent = readdir(d))) {
if (ent->d_name[0] == '.') continue;
/* Retrieve data for the interface and print the results. */
if (print_interface(ent->d_name)) {
retval = 1;
}
}
closedir(d);
return retval;
}
static int maybe_address(const char * s) {
/* Our inet_addr is not great, let's help it a little... */
int dots = 0;
for (;*s;s++) {
if (*s == '.') {
dots++;
if (dots > 3) return 0;
continue;
}
if (!isdigit(*s)) return 0;
}
return dots == 3;
}
static int parse_address(const char * cmd, const char * addr, in_addr_t * out) {
if (!addr) {
fprintf(stderr, "%s: %s: expected argument\n", _argv_0, cmd);
return 1;
}
if (!maybe_address(addr)) {
fprintf(stderr, "%s: %s: '%s' doesn't look like a valid address\n", _argv_0, cmd, addr);
return 1;
}
*out = inet_addr(addr);
return 0;
}
static int _set_address(int netdev, const char * cmd, const char * arg, const char * ioctlstr, unsigned long ioctltype) {
int status;
in_addr_t ip;
if ((status = parse_address(cmd, arg, &ip))) return status;
if ((status = ioctl(netdev, ioctltype, &ip))) { perror(ioctlstr); return status; }
return 0;
}
#define set_address(cmd, arg, itype) _set_address(netdev, cmd, arg, #itype, itype)
#define command_with_address(cmd, itype) if (!strcmp(argv[i], cmd)) { if (_set_address(netdev, argv[i], argv[i+1], #itype, itype)) { return 1; } continue; }
int main(int argc, char * argv[]) {
/* Figure out what we're trying to do. */
if (argc < 2) return print_all_interfaces();
/* Handle (ignore) some common commands */
if (!strcmp(argv[1], "up") || !strcmp(argv[1],"down")) {
fprintf(stderr, "%s: 'up' and 'down' commands are unsupported\n", argv[0]);
return 1;
}
/* If there is an interface name and nothing else, print and be done with it. */
if (argc == 2) return print_interface(argv[1]);
/* All other options here require a leading interface. */
int netdev = open_netdev(argv[1]);
if (netdev < 0) {
perror(argv[0]);
return 1;
}
/* Now let's figure out what we want to do with remaining options */
int collected_address = 0;
for (int i = 2; i < argc; ++i) {
/* Is this argument an address? */
if (maybe_address(argv[i])) {
/* Try to set IPv4 address */
if (collected_address) {
fprintf(stderr, "%s: expected at most one bare address, but found a second\n", argv[0]);
return 1;
}
if (set_address("inet", argv[i], SIOCSIFADDR)) return 1;
collected_address = 1;
} else {
command_with_address("netmask", SIOCSIFNETMASK);
command_with_address("gw", SIOCSIFGATEWAY);
command_with_address("gateway", SIOCSIFGATEWAY);
command_with_address("inet", SIOCSIFADDR);
fprintf(stderr, "%s: '%s' is not an understood command\n", argv[0], argv[i]);
return 1;
}
}
return 0;
}