sandbox: eth-raw: Add a SIMPLE_BUS to enumerate host interfaces
authorJoe Hershberger <joe.hershberger@ni.com>
Mon, 2 Jul 2018 19:47:54 +0000 (14:47 -0500)
committerJoe Hershberger <joe.hershberger@ni.com>
Thu, 26 Jul 2018 19:08:19 +0000 (14:08 -0500)
Ask the OS for each of its interfaces and for each one, bind a U-Boot
device and then probe it. This will allocate the priv data structure
that is then populated.

Signed-off-by: Joe Hershberger <joe.hershberger@ni.com>
Reviewed-by: Simon Glass <sjg@chromium.org>
arch/sandbox/cpu/eth-raw-os.c
arch/sandbox/dts/sandbox.dts
arch/sandbox/dts/sandbox64.dts
arch/sandbox/include/asm/eth-raw-os.h
drivers/net/Makefile
drivers/net/sandbox-raw-bus.c [new file with mode: 0644]

index df7acaa0bc8480ec01cad2a0b1d3543bbf3841f9..75bfaa4c90a759d218a243dafafc43212f2e4b1e 100644 (file)
 #include <linux/if_ether.h>
 #include <linux/if_packet.h>
 
+struct sandbox_eth_raw_if_nameindex *sandbox_eth_raw_if_nameindex(void)
+{
+       return (struct sandbox_eth_raw_if_nameindex *)if_nameindex();
+}
+
+void sandbox_eth_raw_if_freenameindex(struct sandbox_eth_raw_if_nameindex *ptr)
+{
+       if_freenameindex((struct if_nameindex *)ptr);
+}
+
 int sandbox_eth_raw_os_is_local(const char *ifname)
 {
        int fd = socket(AF_INET, SOCK_DGRAM, 0);
index 0ea2452742d33ee039b4e014dfd41072c34db7b1..9f444c96a9ecead08d1745d2e82089e19eea3498 100644 (file)
@@ -8,7 +8,6 @@
        model = "sandbox";
 
        aliases {
-               eth5 = "/eth@90000000";
                i2c0 = &i2c_0;
                pci0 = &pci;
                rtc0 = &rtc_0;
                };
        };
 
+       ethrawbus {
+               compatible = "sandbox,eth-raw-bus";
+               skip-localhost = <0>;
+       };
+
        eth@10002000 {
                compatible = "sandbox,eth";
                reg = <0x10002000 0x1000>;
                fake-host-hwaddr = [00 00 66 44 22 00];
        };
 
-       eth@80000000 {
-               compatible = "sandbox,eth-raw";
-               reg = <0x80000000 0x1000>;
-               host-raw-interface = "eth0";
-       };
-
-       eth@90000000 {
-               compatible = "sandbox,eth-raw";
-               reg = <0x90000000 0x1000>;
-               host-raw-interface = "lo";
-       };
-
        gpio_a: gpios@0 {
                gpio-controller;
                compatible = "sandbox,gpio";
index 48e420e721ee6e1ec3985eaf4ef2842aa7f0efe8..9e65d2fda3d3eadfff3a303801ee17bb1b0854d9 100644 (file)
@@ -8,7 +8,6 @@
        model = "sandbox";
 
        aliases {
-               eth5 = "/eth@90000000";
                i2c0 = &i2c_0;
                pci0 = &pci;
                rtc0 = &rtc_0;
                };
        };
 
+       ethrawbus {
+               compatible = "sandbox,eth-raw-bus";
+               skip-localhost = <1>;
+       };
+
        eth@10002000 {
                compatible = "sandbox,eth";
                reg = <0x0 0x10002000 0x0 0x1000>;
                fake-host-hwaddr = [00 00 66 44 22 00];
        };
 
-       eth@80000000 {
-               compatible = "sandbox,eth-raw";
-               reg = <0x0 0x80000000 0x0 0x1000>;
-               host-raw-interface = "eth0";
-       };
-
-       eth@90000000 {
-               compatible = "sandbox,eth-raw";
-               reg = <0x0 0x90000000 0x0 0x1000>;
-               host-raw-interface = "lo";
-       };
-
        gpio_a: gpios@0 {
                gpio-controller;
                compatible = "sandbox,gpio";
index 99f674e82e645f8110ab3f5cae623a128ea5c994..0b511db70c39c210076bff68ee246db406a1d948 100644 (file)
@@ -34,6 +34,17 @@ struct eth_sandbox_raw_priv {
        unsigned short local_bind_udp_port;
 };
 
+/* A struct to mimic if_nameindex but that does not depend on Linux headers */
+struct sandbox_eth_raw_if_nameindex {
+       unsigned int if_index; /* Index of interface (1, 2, ...) */
+       char        *if_name;  /* Null-terminated name ("eth0", etc.) */
+};
+
+/* Enumerate host network interfaces */
+struct sandbox_eth_raw_if_nameindex *sandbox_eth_raw_if_nameindex(void);
+/* Free the data structure of enumerated network interfaces */
+void sandbox_eth_raw_if_freenameindex(struct sandbox_eth_raw_if_nameindex *ptr);
+
 /*
  * Check if the interface named "ifname" is a localhost interface or not.
  * ifname - the interface name on the host to check
index 058dd0076886259db7997ffcbbe72851c1d93efb..c1ed44e21f15cf30c60c4ba97c9f38b23ade1088 100644 (file)
@@ -50,6 +50,7 @@ obj-$(CONFIG_RTL8139) += rtl8139.o
 obj-$(CONFIG_RTL8169) += rtl8169.o
 obj-$(CONFIG_ETH_SANDBOX) += sandbox.o
 obj-$(CONFIG_ETH_SANDBOX_RAW) += sandbox-raw.o
+obj-$(CONFIG_ETH_SANDBOX_RAW) += sandbox-raw-bus.o
 obj-$(CONFIG_SH_ETHER) += sh_eth.o
 obj-$(CONFIG_RENESAS_RAVB) += ravb.o
 obj-$(CONFIG_SMC91111) += smc91111.o
diff --git a/drivers/net/sandbox-raw-bus.c b/drivers/net/sandbox-raw-bus.c
new file mode 100644 (file)
index 0000000..76d65af
--- /dev/null
@@ -0,0 +1,66 @@
+// SPDX-License-Identifier: GPL-2.0+
+/*
+ * Copyright (c) 2018 National Instruments
+ * Copyright (c) 2018 Joe Hershberger <joe.hershberger@ni.com>
+ */
+
+#include <common.h>
+#include <asm/eth-raw-os.h>
+#include <dm.h>
+#include <errno.h>
+#include <dm/device-internal.h>
+#include <dm/lists.h>
+
+static int eth_raw_bus_post_bind(struct udevice *dev)
+{
+       struct sandbox_eth_raw_if_nameindex *ni, *i;
+       struct udevice *child;
+       struct eth_sandbox_raw_priv *priv;
+       char *ub_ifname;
+       static const char ub_ifname_pfx[] = "host_";
+       u32 skip_localhost = 0;
+
+       ni = sandbox_eth_raw_if_nameindex();
+       if (!ni)
+               return -EINVAL;
+
+       dev_read_u32(dev, "skip-localhost", &skip_localhost);
+       for (i = ni; !(i->if_index == 0 && !i->if_name); i++) {
+               int local = sandbox_eth_raw_os_is_local(i->if_name);
+
+               if (local < 0)
+                       continue;
+               if (skip_localhost && local)
+                       continue;
+
+               ub_ifname = calloc(IFNAMSIZ + sizeof(ub_ifname_pfx), 1);
+               strcpy(ub_ifname, ub_ifname_pfx);
+               strncat(ub_ifname, i->if_name, IFNAMSIZ);
+               device_bind_driver(dev, "eth_sandbox_raw", ub_ifname, &child);
+
+               device_set_name_alloced(child);
+               device_probe(child);
+               priv = dev_get_priv(child);
+               if (priv) {
+                       memcpy(priv->host_ifname, i->if_name, IFNAMSIZ);
+                       priv->host_ifindex = i->if_index;
+                       priv->local = local;
+               }
+       }
+
+       sandbox_eth_raw_if_freenameindex(ni);
+
+       return 0;
+}
+
+static const struct udevice_id sandbox_eth_raw_bus_ids[] = {
+       { .compatible = "sandbox,eth-raw-bus" },
+       { }
+};
+
+U_BOOT_DRIVER(sandbox_eth_raw_bus) = {
+       .name       = "sb_eth_raw_bus",
+       .id         = UCLASS_SIMPLE_BUS,
+       .of_match   = sandbox_eth_raw_bus_ids,
+       .bind       = eth_raw_bus_post_bind,
+};