fafa4d508b
Add QMP/HMP support for rocker devices. This is mostly for debugging purposes to see inside the device's tables and port configurations. Some examples: (qemu) info rocker sw1 name: sw1 id: 0x0000013512005452 ports: 4 (qemu) info rocker-ports sw1 ena/ speed/ auto port link duplex neg? sw1.1 up 10G FD No sw1.2 up 10G FD No sw1.3 !ena 10G FD No sw1.4 !ena 10G FD No (qemu) info rocker-of-dpa-flows sw1 prio tbl hits key(mask) --> actions 2 60 pport 1 vlan 1 LLDP src 00:02:00:00:02:00 dst 01:80:c2:00:00:0e 2 60 pport 1 vlan 1 ARP src 00:02:00:00:02:00 dst 00:02:00:00:03:00 2 60 pport 2 vlan 2 IPv6 src 00:02:00:00:03:00 dst 33:33:ff:00:00:02 proto 58 3 50 vlan 2 dst 33:33:ff:00:00:02 --> write group 0x32000001 goto tbl 60 2 60 pport 2 vlan 2 IPv6 src 00:02:00:00:03:00 dst 33:33:ff:00:03:00 proto 58 3 50 1 vlan 2 dst 33:33:ff:00:03:00 --> write group 0x32000001 goto tbl 60 2 60 pport 2 vlan 2 ARP src 00:02:00:00:03:00 dst 00:02:00:00:02:00 3 50 2 vlan 2 dst 00:02:00:00:02:00 --> write group 0x02000001 goto tbl 60 2 60 1 pport 2 vlan 2 IP src 00:02:00:00:03:00 dst 00:02:00:00:02:00 proto 1 3 50 2 vlan 1 dst 00:02:00:00:03:00 --> write group 0x01000002 goto tbl 60 2 60 1 pport 1 vlan 1 IP src 00:02:00:00:02:00 dst 00:02:00:00:03:00 proto 1 2 60 pport 1 vlan 1 IPv6 src 00:02:00:00:02:00 dst 33:33:ff:00:00:01 proto 58 3 50 vlan 1 dst 33:33:ff:00:00:01 --> write group 0x31000000 goto tbl 60 2 60 pport 1 vlan 1 IPv6 src 00:02:00:00:02:00 dst 33:33:ff:00:02:00 proto 58 3 50 1 vlan 1 dst 33:33:ff:00:02:00 --> write group 0x31000000 goto tbl 60 1 60 173 pport 2 vlan 2 LLDP src <any> dst 01:80:c2:00:00:0e --> write group 0x02000000 1 60 6 pport 2 vlan 2 IPv6 src <any> dst <any> --> write group 0x02000000 1 60 174 pport 1 vlan 1 LLDP src <any> dst 01:80:c2:00:00:0e --> write group 0x01000000 1 60 174 pport 2 vlan 2 IP src <any> dst <any> --> write group 0x02000000 1 60 6 pport 1 vlan 1 IPv6 src <any> dst <any> --> write group 0x01000000 1 60 181 pport 2 vlan 2 ARP src <any> dst <any> --> write group 0x02000000 1 10 715 pport 2 --> apply new vlan 2 goto tbl 20 1 60 177 pport 1 vlan 1 ARP src <any> dst <any> --> write group 0x01000000 1 60 174 pport 1 vlan 1 IP src <any> dst <any> --> write group 0x01000000 1 10 717 pport 1 --> apply new vlan 1 goto tbl 20 1 0 1432 pport 0(0xffff) --> goto tbl 10 (qemu) info rocker-of-dpa-groups sw1 id (decode) --> buckets 0x32000001 (type L2 multicast vlan 2 index 1) --> groups [0x02000001,0x02000000] 0x02000001 (type L2 interface vlan 2 pport 1) --> pop vlan out pport 1 0x01000002 (type L2 interface vlan 1 pport 2) --> pop vlan out pport 2 0x02000000 (type L2 interface vlan 2 pport 0) --> pop vlan out pport 0 0x01000000 (type L2 interface vlan 1 pport 0) --> pop vlan out pport 0 0x31000000 (type L2 multicast vlan 1 index 0) --> groups [0x01000002,0x01000000] [Added "query-" prefixes to rocker.json commands as suggested by Eric Blake <eblake@redhat.com>. --Stefan] Signed-off-by: Scott Feldman <sfeldma@gmail.com> Signed-off-by: Jiri Pirko <jiri@resnulli.us> Message-id: 1433985681-56138-5-git-send-email-sfeldma@gmail.com Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
262 lines
5.9 KiB
C
262 lines
5.9 KiB
C
/*
|
|
* QEMU rocker switch emulation - front-panel ports
|
|
*
|
|
* Copyright (c) 2014 Scott Feldman <sfeldma@gmail.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*/
|
|
|
|
#include "net/clients.h"
|
|
|
|
#include "rocker.h"
|
|
#include "rocker_hw.h"
|
|
#include "rocker_fp.h"
|
|
#include "rocker_world.h"
|
|
|
|
enum duplex {
|
|
DUPLEX_HALF = 0,
|
|
DUPLEX_FULL
|
|
};
|
|
|
|
struct fp_port {
|
|
Rocker *r;
|
|
World *world;
|
|
unsigned int index;
|
|
char *name;
|
|
uint32_t pport;
|
|
bool enabled;
|
|
uint32_t speed;
|
|
uint8_t duplex;
|
|
uint8_t autoneg;
|
|
uint8_t learning;
|
|
NICState *nic;
|
|
NICConf conf;
|
|
};
|
|
|
|
char *fp_port_get_name(FpPort *port)
|
|
{
|
|
return port->name;
|
|
}
|
|
|
|
bool fp_port_get_link_up(FpPort *port)
|
|
{
|
|
return !qemu_get_queue(port->nic)->link_down;
|
|
}
|
|
|
|
void fp_port_get_info(FpPort *port, RockerPortList *info)
|
|
{
|
|
info->value->name = g_strdup(port->name);
|
|
info->value->enabled = port->enabled;
|
|
info->value->link_up = fp_port_get_link_up(port);
|
|
info->value->speed = port->speed;
|
|
info->value->duplex = port->duplex;
|
|
info->value->autoneg = port->autoneg;
|
|
}
|
|
|
|
void fp_port_get_macaddr(FpPort *port, MACAddr *macaddr)
|
|
{
|
|
memcpy(macaddr->a, port->conf.macaddr.a, sizeof(macaddr->a));
|
|
}
|
|
|
|
void fp_port_set_macaddr(FpPort *port, MACAddr *macaddr)
|
|
{
|
|
/* XXX TODO implement and test setting mac addr
|
|
* XXX memcpy(port->conf.macaddr.a, macaddr.a, sizeof(port->conf.macaddr.a));
|
|
*/
|
|
}
|
|
|
|
uint8_t fp_port_get_learning(FpPort *port)
|
|
{
|
|
return port->learning;
|
|
}
|
|
|
|
void fp_port_set_learning(FpPort *port, uint8_t learning)
|
|
{
|
|
port->learning = learning;
|
|
}
|
|
|
|
int fp_port_get_settings(FpPort *port, uint32_t *speed,
|
|
uint8_t *duplex, uint8_t *autoneg)
|
|
{
|
|
*speed = port->speed;
|
|
*duplex = port->duplex;
|
|
*autoneg = port->autoneg;
|
|
|
|
return ROCKER_OK;
|
|
}
|
|
|
|
int fp_port_set_settings(FpPort *port, uint32_t speed,
|
|
uint8_t duplex, uint8_t autoneg)
|
|
{
|
|
/* XXX validate inputs */
|
|
|
|
port->speed = speed;
|
|
port->duplex = duplex;
|
|
port->autoneg = autoneg;
|
|
|
|
return ROCKER_OK;
|
|
}
|
|
|
|
bool fp_port_from_pport(uint32_t pport, uint32_t *port)
|
|
{
|
|
if (pport < 1 || pport > ROCKER_FP_PORTS_MAX) {
|
|
return false;
|
|
}
|
|
*port = pport - 1;
|
|
return true;
|
|
}
|
|
|
|
int fp_port_eg(FpPort *port, const struct iovec *iov, int iovcnt)
|
|
{
|
|
NetClientState *nc = qemu_get_queue(port->nic);
|
|
|
|
if (port->enabled) {
|
|
qemu_sendv_packet(nc, iov, iovcnt);
|
|
}
|
|
|
|
return ROCKER_OK;
|
|
}
|
|
|
|
static int fp_port_can_receive(NetClientState *nc)
|
|
{
|
|
FpPort *port = qemu_get_nic_opaque(nc);
|
|
|
|
return port->enabled;
|
|
}
|
|
|
|
static ssize_t fp_port_receive_iov(NetClientState *nc, const struct iovec *iov,
|
|
int iovcnt)
|
|
{
|
|
FpPort *port = qemu_get_nic_opaque(nc);
|
|
|
|
return world_ingress(port->world, port->pport, iov, iovcnt);
|
|
}
|
|
|
|
static ssize_t fp_port_receive(NetClientState *nc, const uint8_t *buf,
|
|
size_t size)
|
|
{
|
|
const struct iovec iov = {
|
|
.iov_base = (uint8_t *)buf,
|
|
.iov_len = size
|
|
};
|
|
|
|
return fp_port_receive_iov(nc, &iov, 1);
|
|
}
|
|
|
|
static void fp_port_cleanup(NetClientState *nc)
|
|
{
|
|
}
|
|
|
|
static void fp_port_set_link_status(NetClientState *nc)
|
|
{
|
|
FpPort *port = qemu_get_nic_opaque(nc);
|
|
|
|
rocker_event_link_changed(port->r, port->pport, !nc->link_down);
|
|
}
|
|
|
|
static NetClientInfo fp_port_info = {
|
|
.type = NET_CLIENT_OPTIONS_KIND_NIC,
|
|
.size = sizeof(NICState),
|
|
.can_receive = fp_port_can_receive,
|
|
.receive = fp_port_receive,
|
|
.receive_iov = fp_port_receive_iov,
|
|
.cleanup = fp_port_cleanup,
|
|
.link_status_changed = fp_port_set_link_status,
|
|
};
|
|
|
|
World *fp_port_get_world(FpPort *port)
|
|
{
|
|
return port->world;
|
|
}
|
|
|
|
void fp_port_set_world(FpPort *port, World *world)
|
|
{
|
|
DPRINTF("port %d setting world \"%s\"\n", port->index, world_name(world));
|
|
port->world = world;
|
|
}
|
|
|
|
bool fp_port_enabled(FpPort *port)
|
|
{
|
|
return port->enabled;
|
|
}
|
|
|
|
static void fp_port_set_link(FpPort *port, bool up)
|
|
{
|
|
NetClientState *nc = qemu_get_queue(port->nic);
|
|
|
|
if (up == nc->link_down) {
|
|
nc->link_down = !up;
|
|
nc->info->link_status_changed(nc);
|
|
}
|
|
}
|
|
|
|
void fp_port_enable(FpPort *port)
|
|
{
|
|
fp_port_set_link(port, true);
|
|
port->enabled = true;
|
|
DPRINTF("port %d enabled\n", port->index);
|
|
}
|
|
|
|
void fp_port_disable(FpPort *port)
|
|
{
|
|
port->enabled = false;
|
|
fp_port_set_link(port, false);
|
|
DPRINTF("port %d disabled\n", port->index);
|
|
}
|
|
|
|
FpPort *fp_port_alloc(Rocker *r, char *sw_name,
|
|
MACAddr *start_mac, unsigned int index,
|
|
NICPeers *peers)
|
|
{
|
|
FpPort *port = g_malloc0(sizeof(FpPort));
|
|
|
|
if (!port) {
|
|
return NULL;
|
|
}
|
|
|
|
port->r = r;
|
|
port->index = index;
|
|
port->pport = index + 1;
|
|
|
|
/* front-panel switch port names are 1-based */
|
|
|
|
port->name = g_strdup_printf("%sp%d", sw_name, port->pport);
|
|
|
|
memcpy(port->conf.macaddr.a, start_mac, sizeof(port->conf.macaddr.a));
|
|
port->conf.macaddr.a[5] += index;
|
|
port->conf.bootindex = -1;
|
|
port->conf.peers = *peers;
|
|
|
|
port->nic = qemu_new_nic(&fp_port_info, &port->conf,
|
|
sw_name, NULL, port);
|
|
qemu_format_nic_info_str(qemu_get_queue(port->nic),
|
|
port->conf.macaddr.a);
|
|
|
|
fp_port_reset(port);
|
|
|
|
return port;
|
|
}
|
|
|
|
void fp_port_free(FpPort *port)
|
|
{
|
|
qemu_del_nic(port->nic);
|
|
g_free(port->name);
|
|
g_free(port);
|
|
}
|
|
|
|
void fp_port_reset(FpPort *port)
|
|
{
|
|
fp_port_disable(port);
|
|
port->speed = 10000; /* 10Gbps */
|
|
port->duplex = DUPLEX_FULL;
|
|
port->autoneg = 0;
|
|
}
|