2019-11-11 20:40:57 +01:00
|
|
|
/****************************************************************************
|
2021-06-16 09:22:16 +02:00
|
|
|
* apps/examples/netlink_route/netlink_route_main.c
|
2019-11-11 20:40:57 +01:00
|
|
|
*
|
2021-06-15 09:09:58 +02:00
|
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
|
|
* contributor license agreements. See the NOTICE file distributed with
|
|
|
|
* this work for additional information regarding copyright ownership. The
|
|
|
|
* ASF licenses this file to you under the Apache License, Version 2.0 (the
|
|
|
|
* "License"); you may not use this file except in compliance with the
|
|
|
|
* License. You may obtain a copy of the License at
|
2019-11-11 20:40:57 +01:00
|
|
|
*
|
2021-06-15 09:09:58 +02:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2019-11-11 20:40:57 +01:00
|
|
|
*
|
2021-06-15 09:09:58 +02:00
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
|
|
* License for the specific language governing permissions and limitations
|
|
|
|
* under the License.
|
2019-11-11 20:40:57 +01:00
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Included Files
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <inttypes.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <sched.h>
|
|
|
|
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
#include <net/if.h>
|
|
|
|
#include <net/route.h>
|
|
|
|
#include <net/ethernet.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
|
|
|
|
#include <nuttx/net/arp.h>
|
|
|
|
#include <nuttx/net/neighbor.h>
|
|
|
|
|
|
|
|
#include "netutils/netlib.h"
|
|
|
|
|
|
|
|
#ifdef CONFIG_NETLINK_ROUTE
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Pre-processor Definitions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#define MAX_DEVICES 16
|
|
|
|
#define MAX_ROUTES 64
|
|
|
|
|
|
|
|
#ifdef CONFIG_NET_IPv6
|
|
|
|
# define ROUTE_BUFSIZE INET6_ADDRSTRLEN
|
|
|
|
#else
|
|
|
|
# define ROUTE_BUFSIZE INET_ADDRSTRLEN
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
2019-11-12 17:20:26 +01:00
|
|
|
#ifndef CONFIG_NETLINK_DISABLE_GETLINK
|
2019-11-11 20:40:57 +01:00
|
|
|
static void dump_devices(sa_family_t family)
|
|
|
|
{
|
|
|
|
struct netlib_device_s devlist[MAX_DEVICES];
|
|
|
|
ssize_t nentries;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
printf("\nDevice List");
|
|
|
|
|
|
|
|
/* Read the device list */
|
|
|
|
|
|
|
|
nentries = netlib_get_devices(devlist, MAX_DEVICES, family);
|
|
|
|
if (nentries < 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "\nERROR: netlib_get_devices failed: %d\n",
|
|
|
|
(int)nentries);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf(" (Entries: %ld)\n", (long)nentries);
|
|
|
|
|
|
|
|
/* Dump the device list */
|
|
|
|
|
|
|
|
for (i = 0; i < nentries; i++)
|
|
|
|
{
|
|
|
|
FAR struct netlib_device_s *dev = &devlist[i];
|
|
|
|
|
|
|
|
#ifdef CONFIG_NETDEV_IFINDEX
|
2020-02-25 13:17:51 +01:00
|
|
|
printf(" Index: %2d ", dev->ifindex);
|
2019-11-11 20:40:57 +01:00
|
|
|
#else
|
|
|
|
printf(" ");
|
|
|
|
#endif
|
|
|
|
printf("Name: \"%s\"\n", dev->ifname);
|
|
|
|
}
|
|
|
|
}
|
2019-11-12 17:20:26 +01:00
|
|
|
#else
|
|
|
|
# define dump_devices(f)
|
|
|
|
#endif
|
2019-11-11 20:40:57 +01:00
|
|
|
|
2019-11-12 17:20:26 +01:00
|
|
|
#if defined(CONFIG_NET_IPv6) && !defined(CONFIG_NETLINK_DISABLE_GETNEIGH)
|
2019-11-11 20:40:57 +01:00
|
|
|
static void dump_neighbor(void)
|
|
|
|
{
|
|
|
|
FAR struct neighbor_entry_s *nbtab;
|
|
|
|
FAR char buffer[INET6_ADDRSTRLEN];
|
|
|
|
size_t allocsize;
|
|
|
|
ssize_t nentries;
|
|
|
|
int i;
|
|
|
|
int j;
|
|
|
|
|
|
|
|
printf("\nIPv6 Neighbor Table");
|
|
|
|
|
|
|
|
/* Allocate a buffer to hold the neighbor table */
|
|
|
|
|
|
|
|
allocsize = CONFIG_NET_IPv6_NCONF_ENTRIES *
|
|
|
|
sizeof(struct neighbor_entry_s);
|
|
|
|
nbtab = (FAR struct neighbor_entry_s *)malloc(allocsize);
|
|
|
|
if (nbtab == NULL)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "\nERROR: Failed to allocate neighbor table\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the neighbor table */
|
|
|
|
|
|
|
|
nentries = netlib_get_nbtable(nbtab, CONFIG_NET_IPv6_NCONF_ENTRIES);
|
|
|
|
if (nentries < 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "\nERROR: netlib_get_nbtable failed: %d\n",
|
|
|
|
(int)nentries);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf(" (Entries: %ld)\n", (long)nentries);
|
|
|
|
|
|
|
|
/* Show each neighbor */
|
|
|
|
|
|
|
|
for (i = 0; i < nentries; i++)
|
|
|
|
{
|
|
|
|
FAR struct neighbor_entry_s *nb = &nbtab[i];
|
|
|
|
|
2020-01-02 13:09:50 +01:00
|
|
|
inet_ntop(AF_INET6, &nb->ne_ipaddr, buffer, INET6_ADDRSTRLEN);
|
2019-11-11 20:40:57 +01:00
|
|
|
printf(" Dest: %s MAC Type: %2u Size: %3u Addr: ",
|
|
|
|
buffer, nb->ne_addr.na_lltype, nb->ne_addr.na_llsize);
|
|
|
|
|
|
|
|
for (j = 0; j < nb->ne_addr.na_llsize; j++)
|
|
|
|
{
|
|
|
|
if ((j + 1) >= nb->ne_addr.na_llsize)
|
|
|
|
{
|
|
|
|
printf("%02x", nb->ne_addr.u.na_addr);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("%02x.", nb->ne_addr.u.na_addr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (; j < 7; j++)
|
|
|
|
{
|
|
|
|
printf(" ");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (j < 8)
|
|
|
|
{
|
|
|
|
printf(" ");
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CONFIG_SYSTEM_TIME64
|
|
|
|
printf("Time 0x%" PRIx64 "\n", nb->ne_time);
|
|
|
|
#else
|
|
|
|
printf("Time 0x%" PRIx32 "\n", nb->ne_time);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
free(nbtab);
|
|
|
|
}
|
2019-11-12 17:20:26 +01:00
|
|
|
#else
|
|
|
|
# define dump_neighbor()
|
2019-11-11 20:40:57 +01:00
|
|
|
#endif
|
|
|
|
|
2019-11-12 17:20:26 +01:00
|
|
|
#if defined(CONFIG_NET_ARP) && !defined(CONFIG_NETLINK_DISABLE_GETNEIGH)
|
2019-11-11 20:40:57 +01:00
|
|
|
static void dump_arp(void)
|
|
|
|
{
|
|
|
|
FAR struct arp_entry_s *arptab;
|
|
|
|
FAR char buffer[INET_ADDRSTRLEN];
|
|
|
|
size_t allocsize;
|
|
|
|
ssize_t nentries;
|
|
|
|
int i;
|
|
|
|
int j;
|
|
|
|
|
|
|
|
printf("\nIPv4 ARP Table");
|
|
|
|
|
|
|
|
/* Allocate a buffer to hold the ARP table */
|
|
|
|
|
|
|
|
allocsize = CONFIG_NET_ARPTAB_SIZE * sizeof(struct arp_entry_s);
|
|
|
|
arptab = (FAR struct arp_entry_s *)malloc(allocsize);
|
|
|
|
if (arptab == NULL)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "\nERROR: Failed to allocate ARP table\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the ARP table */
|
|
|
|
|
|
|
|
nentries = netlib_get_arptable(arptab, CONFIG_NET_ARPTAB_SIZE);
|
|
|
|
if (nentries < 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "\nERROR: netlib_get_arptable failed: %d\n",
|
|
|
|
(int)nentries);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf(" (Entries: %ld)\n", (long)nentries);
|
|
|
|
|
|
|
|
/* Show each ARP table entry */
|
|
|
|
|
|
|
|
for (i = 0; i < nentries; i++)
|
|
|
|
{
|
|
|
|
FAR struct arp_entry_s *arp = &arptab[i];
|
|
|
|
|
2020-01-02 13:09:50 +01:00
|
|
|
inet_ntop(AF_INET, &arp->at_ipaddr, buffer, INET_ADDRSTRLEN);
|
2019-11-11 20:40:57 +01:00
|
|
|
printf(" Dest: %s MAC Addr: ", buffer);
|
|
|
|
|
|
|
|
for (j = 0; j < ETHER_ADDR_LEN; j++)
|
|
|
|
{
|
|
|
|
if (j == (ETHER_ADDR_LEN - 1))
|
|
|
|
{
|
2019-11-28 23:28:41 +01:00
|
|
|
printf("%02x", arp->at_ethaddr.ether_addr_octet[j]);
|
2019-11-11 20:40:57 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-11-28 23:28:41 +01:00
|
|
|
printf("%02x.", arp->at_ethaddr.ether_addr_octet[j]);
|
2019-11-11 20:40:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CONFIG_SYSTEM_TIME64
|
2019-11-28 23:28:41 +01:00
|
|
|
printf(" Time 0x%" PRIx64 "\n", arp->at_time);
|
2019-11-11 20:40:57 +01:00
|
|
|
#else
|
2019-11-28 23:28:41 +01:00
|
|
|
printf(" Time 0x%" PRIx32 "\n", arp->at_time);
|
2019-11-11 20:40:57 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
free(arptab);
|
|
|
|
}
|
2019-11-12 17:20:26 +01:00
|
|
|
#else
|
|
|
|
# define dump_arp()
|
2019-11-11 20:40:57 +01:00
|
|
|
#endif
|
|
|
|
|
2019-11-12 17:20:26 +01:00
|
|
|
#if defined(CONFIG_NET_ROUTE) && !defined(CONFIG_NETLINK_DISABLE_GETROUTE)
|
2019-11-11 20:40:57 +01:00
|
|
|
static void dump_route(sa_family_t family)
|
|
|
|
{
|
|
|
|
FAR struct rtentry *rttab;
|
|
|
|
FAR char buffer[ROUTE_BUFSIZE];
|
|
|
|
size_t allocsize;
|
|
|
|
ssize_t nentries;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
printf("\nIPv%c Routing Table", family == AF_INET ? '4' : '6');
|
|
|
|
|
|
|
|
/* Allocate a buffer to hold the routing table */
|
|
|
|
|
|
|
|
allocsize = MAX_ROUTES * sizeof(struct rtentry);
|
|
|
|
rttab = (FAR struct rtentry *)malloc(allocsize);
|
|
|
|
if (rttab == NULL)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "\nERROR: Failed to allocate routing table\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the routing table */
|
|
|
|
|
|
|
|
nentries = netlib_get_route(rttab, MAX_ROUTES, family);
|
|
|
|
if (nentries < 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "\nERROR: netlib_get_route failed: %d\n",
|
|
|
|
(int)nentries);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf(" (Entries: %ld)\n", (long)nentries);
|
|
|
|
|
|
|
|
/* Show each routing table entry */
|
|
|
|
|
|
|
|
for (i = 0; i < nentries; i++)
|
|
|
|
{
|
|
|
|
FAR struct rtentry *rte = &rttab[i];
|
|
|
|
|
2020-01-02 13:09:50 +01:00
|
|
|
inet_ntop(rte->rt_dst.ss_family, &rte->rt_dst.ss_data, buffer,
|
|
|
|
ROUTE_BUFSIZE);
|
2019-11-11 20:40:57 +01:00
|
|
|
printf(" Dest: %s ", buffer);
|
|
|
|
|
2020-01-02 13:09:50 +01:00
|
|
|
inet_ntop(rte->rt_gateway.ss_family, &rte->rt_gateway.ss_data,
|
|
|
|
buffer, ROUTE_BUFSIZE);
|
2019-11-11 20:40:57 +01:00
|
|
|
printf("Gateway: %s ", buffer);
|
|
|
|
|
2020-01-02 13:09:50 +01:00
|
|
|
inet_ntop(rte->rt_genmask.ss_family, &rte->rt_genmask.ss_data,
|
|
|
|
buffer, ROUTE_BUFSIZE);
|
2019-11-11 20:40:57 +01:00
|
|
|
printf("GenMASK: %s ", buffer);
|
|
|
|
printf("Flags: %04x\n", rte->rt_flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
free(rttab);
|
|
|
|
}
|
2019-11-12 17:20:26 +01:00
|
|
|
#else
|
|
|
|
# define dump_route(f)
|
2019-11-11 20:40:57 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Public Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* netlink_route_main
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
int main(int argc, FAR char *argv[])
|
|
|
|
{
|
|
|
|
dump_devices(AF_PACKET);
|
|
|
|
dump_neighbor();
|
|
|
|
dump_arp();
|
|
|
|
dump_route(AF_INET);
|
|
|
|
dump_route(AF_INET6);
|
|
|
|
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* CONFIG_NETLINK_ROUTE */
|