123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322 |
- /*
- * Hub net client
- *
- * Copyright IBM, Corp. 2012
- *
- * Authors:
- * Stefan Hajnoczi <stefanha@linux.vnet.ibm.com>
- * Zhi Yong Wu <wuzhy@linux.vnet.ibm.com>
- *
- * This work is licensed under the terms of the GNU LGPL, version 2 or later.
- * See the COPYING.LIB file in the top-level directory.
- *
- */
- #include "qemu/osdep.h"
- #include "qapi/error.h"
- #include "monitor/monitor.h"
- #include "net/net.h"
- #include "clients.h"
- #include "hub.h"
- #include "qemu/iov.h"
- #include "qemu/error-report.h"
- #include "system/qtest.h"
- /*
- * A hub broadcasts incoming packets to all its ports except the source port.
- * Hubs can be used to provide independent emulated network segments.
- */
- typedef struct NetHub NetHub;
- typedef struct NetHubPort {
- NetClientState nc;
- QLIST_ENTRY(NetHubPort) next;
- NetHub *hub;
- int id;
- } NetHubPort;
- struct NetHub {
- int id;
- QLIST_ENTRY(NetHub) next;
- int num_ports;
- QLIST_HEAD(, NetHubPort) ports;
- };
- static QLIST_HEAD(, NetHub) hubs = QLIST_HEAD_INITIALIZER(&hubs);
- static ssize_t net_hub_receive(NetHub *hub, NetHubPort *source_port,
- const uint8_t *buf, size_t len)
- {
- NetHubPort *port;
- QLIST_FOREACH(port, &hub->ports, next) {
- if (port == source_port) {
- continue;
- }
- qemu_send_packet(&port->nc, buf, len);
- }
- return len;
- }
- static ssize_t net_hub_receive_iov(NetHub *hub, NetHubPort *source_port,
- const struct iovec *iov, int iovcnt)
- {
- NetHubPort *port;
- ssize_t len = iov_size(iov, iovcnt);
- QLIST_FOREACH(port, &hub->ports, next) {
- if (port == source_port) {
- continue;
- }
- qemu_sendv_packet(&port->nc, iov, iovcnt);
- }
- return len;
- }
- static NetHub *net_hub_new(int id)
- {
- NetHub *hub;
- hub = g_malloc(sizeof(*hub));
- hub->id = id;
- hub->num_ports = 0;
- QLIST_INIT(&hub->ports);
- QLIST_INSERT_HEAD(&hubs, hub, next);
- return hub;
- }
- static bool net_hub_port_can_receive(NetClientState *nc)
- {
- NetHubPort *port;
- NetHubPort *src_port = DO_UPCAST(NetHubPort, nc, nc);
- NetHub *hub = src_port->hub;
- QLIST_FOREACH(port, &hub->ports, next) {
- if (port == src_port) {
- continue;
- }
- if (qemu_can_send_packet(&port->nc)) {
- return true;
- }
- }
- return false;
- }
- static ssize_t net_hub_port_receive(NetClientState *nc,
- const uint8_t *buf, size_t len)
- {
- NetHubPort *port = DO_UPCAST(NetHubPort, nc, nc);
- return net_hub_receive(port->hub, port, buf, len);
- }
- static ssize_t net_hub_port_receive_iov(NetClientState *nc,
- const struct iovec *iov, int iovcnt)
- {
- NetHubPort *port = DO_UPCAST(NetHubPort, nc, nc);
- return net_hub_receive_iov(port->hub, port, iov, iovcnt);
- }
- static void net_hub_port_cleanup(NetClientState *nc)
- {
- NetHubPort *port = DO_UPCAST(NetHubPort, nc, nc);
- QLIST_REMOVE(port, next);
- }
- static NetClientInfo net_hub_port_info = {
- .type = NET_CLIENT_DRIVER_HUBPORT,
- .size = sizeof(NetHubPort),
- .can_receive = net_hub_port_can_receive,
- .receive = net_hub_port_receive,
- .receive_iov = net_hub_port_receive_iov,
- .cleanup = net_hub_port_cleanup,
- };
- static NetHubPort *net_hub_port_new(NetHub *hub, const char *name,
- NetClientState *hubpeer)
- {
- NetClientState *nc;
- NetHubPort *port;
- int id = hub->num_ports++;
- char default_name[128];
- if (!name) {
- snprintf(default_name, sizeof(default_name),
- "hub%dport%d", hub->id, id);
- name = default_name;
- }
- nc = qemu_new_net_client(&net_hub_port_info, hubpeer, "hub", name);
- port = DO_UPCAST(NetHubPort, nc, nc);
- port->id = id;
- port->hub = hub;
- QLIST_INSERT_HEAD(&hub->ports, port, next);
- return port;
- }
- /**
- * Create a port on a given hub
- * @hub_id: Number of the hub
- * @name: Net client name or NULL for default name.
- * @hubpeer: Peer to use (if "netdev=id" has been specified)
- *
- * If there is no existing hub with the given id then a new hub is created.
- */
- NetClientState *net_hub_add_port(int hub_id, const char *name,
- NetClientState *hubpeer)
- {
- NetHub *hub;
- NetHubPort *port;
- QLIST_FOREACH(hub, &hubs, next) {
- if (hub->id == hub_id) {
- break;
- }
- }
- if (!hub) {
- hub = net_hub_new(hub_id);
- }
- port = net_hub_port_new(hub, name, hubpeer);
- return &port->nc;
- }
- /**
- * Print hub configuration
- */
- void net_hub_info(Monitor *mon)
- {
- NetHub *hub;
- NetHubPort *port;
- QLIST_FOREACH(hub, &hubs, next) {
- monitor_printf(mon, "hub %d\n", hub->id);
- QLIST_FOREACH(port, &hub->ports, next) {
- monitor_printf(mon, " \\ %s", port->nc.name);
- if (port->nc.peer) {
- monitor_printf(mon, ": ");
- print_net_client(mon, port->nc.peer);
- } else {
- monitor_printf(mon, "\n");
- }
- }
- }
- }
- /**
- * Get the hub id that a client is connected to
- *
- * @id: Pointer for hub id output, may be NULL
- */
- int net_hub_id_for_client(NetClientState *nc, int *id)
- {
- NetHubPort *port;
- if (nc->info->type == NET_CLIENT_DRIVER_HUBPORT) {
- port = DO_UPCAST(NetHubPort, nc, nc);
- } else if (nc->peer != NULL && nc->peer->info->type ==
- NET_CLIENT_DRIVER_HUBPORT) {
- port = DO_UPCAST(NetHubPort, nc, nc->peer);
- } else {
- return -ENOENT;
- }
- if (id) {
- *id = port->hub->id;
- }
- return 0;
- }
- int net_init_hubport(const Netdev *netdev, const char *name,
- NetClientState *peer, Error **errp)
- {
- const NetdevHubPortOptions *hubport;
- NetClientState *hubpeer = NULL;
- assert(netdev->type == NET_CLIENT_DRIVER_HUBPORT);
- assert(!peer);
- hubport = &netdev->u.hubport;
- if (hubport->netdev) {
- hubpeer = qemu_find_netdev(hubport->netdev);
- if (!hubpeer) {
- error_setg(errp, "netdev '%s' not found", hubport->netdev);
- return -1;
- }
- }
- net_hub_add_port(hubport->hubid, name, hubpeer);
- return 0;
- }
- /**
- * Warn if hub configurations are likely wrong
- */
- void net_hub_check_clients(void)
- {
- NetHub *hub;
- NetHubPort *port;
- NetClientState *peer;
- QLIST_FOREACH(hub, &hubs, next) {
- int has_nic = 0, has_host_dev = 0;
- QLIST_FOREACH(port, &hub->ports, next) {
- peer = port->nc.peer;
- if (!peer) {
- warn_report("hub port %s has no peer", port->nc.name);
- continue;
- }
- switch (peer->info->type) {
- case NET_CLIENT_DRIVER_NIC:
- has_nic = 1;
- break;
- case NET_CLIENT_DRIVER_USER:
- case NET_CLIENT_DRIVER_TAP:
- case NET_CLIENT_DRIVER_SOCKET:
- case NET_CLIENT_DRIVER_STREAM:
- case NET_CLIENT_DRIVER_DGRAM:
- case NET_CLIENT_DRIVER_VDE:
- case NET_CLIENT_DRIVER_VHOST_USER:
- has_host_dev = 1;
- break;
- default:
- break;
- }
- }
- if (has_host_dev && !has_nic) {
- warn_report("hub %d with no nics", hub->id);
- }
- if (has_nic && !has_host_dev && !qtest_enabled()) {
- warn_report("hub %d is not connected to host network", hub->id);
- }
- }
- }
- bool net_hub_flush(NetClientState *nc)
- {
- NetHubPort *port;
- NetHubPort *source_port = DO_UPCAST(NetHubPort, nc, nc);
- int ret = 0;
- QLIST_FOREACH(port, &source_port->hub->ports, next) {
- if (port != source_port) {
- ret += qemu_net_queue_flush(port->nc.incoming_queue);
- }
- }
- return ret ? true : false;
- }
|