提交 1ceef9f2 编写于 作者: J Jason Wang 提交者: Anthony Liguori

net: multiqueue support

This patch adds basic multiqueue support for qemu. The idea is simple, an array
of NetClientStates were introduced in NICState, parse_netdev() were extended to
find and match all NetClientStates belongs to the backend and place their
pointers in NICConf. Then qemu_new_nic can setup a N:N mapping between NICStates
that belongs to a nic and NICStates belongs to the netdev. And a queue_index
were introduced in NetClientState to track its index. After this, each peers of
a NICState were abstracted as a queue.

After this change, all NetClientState that belongs to the same backend/nic has
the same id. When use want to change the link status, all NetClientStates that
belongs to the same backend/nic will be also changed. When user want to delete
a device or netdev, all NetClientStates that belongs to the same backend/nic
will be deleted also. Changing or deleting an specific queue is not allowed.
Signed-off-by: NJason Wang <jasowang@redhat.com>
Signed-off-by: NAnthony Liguori <aliguori@us.ibm.com>
上级 f7860455
...@@ -900,7 +900,7 @@ void dp83932_init(NICInfo *nd, hwaddr base, int it_shift, ...@@ -900,7 +900,7 @@ void dp83932_init(NICInfo *nd, hwaddr base, int it_shift,
s->regs[SONIC_SR] = 0x0004; /* only revision recognized by Linux */ s->regs[SONIC_SR] = 0x0004; /* only revision recognized by Linux */
s->conf.macaddr = nd->macaddr; s->conf.macaddr = nd->macaddr;
s->conf.peer = nd->netdev; s->conf.peers.ncs[0] = nd->netdev;
s->nic = qemu_new_nic(&net_dp83932_info, &s->conf, nd->model, nd->name, s); s->nic = qemu_new_nic(&net_dp83932_info, &s->conf, nd->model, nd->name, s);
......
...@@ -472,7 +472,7 @@ void mcf_fec_init(MemoryRegion *sysmem, NICInfo *nd, ...@@ -472,7 +472,7 @@ void mcf_fec_init(MemoryRegion *sysmem, NICInfo *nd,
memory_region_add_subregion(sysmem, base, &s->iomem); memory_region_add_subregion(sysmem, base, &s->iomem);
s->conf.macaddr = nd->macaddr; s->conf.macaddr = nd->macaddr;
s->conf.peer = nd->netdev; s->conf.peers.ncs[0] = nd->netdev;
s->nic = qemu_new_nic(&net_mcf_fec_info, &s->conf, nd->model, nd->name, s); s->nic = qemu_new_nic(&net_mcf_fec_info, &s->conf, nd->model, nd->name, s);
......
...@@ -173,16 +173,47 @@ PropertyInfo qdev_prop_chr = { ...@@ -173,16 +173,47 @@ PropertyInfo qdev_prop_chr = {
static int parse_netdev(DeviceState *dev, const char *str, void **ptr) static int parse_netdev(DeviceState *dev, const char *str, void **ptr)
{ {
NetClientState *netdev = qemu_find_netdev(str); NICPeers *peers_ptr = (NICPeers *)ptr;
NICConf *conf = container_of(peers_ptr, NICConf, peers);
NetClientState **ncs = peers_ptr->ncs;
NetClientState *peers[MAX_QUEUE_NUM];
int queues, i = 0;
int ret;
if (netdev == NULL) { queues = qemu_find_net_clients_except(str, peers,
return -ENOENT; NET_CLIENT_OPTIONS_KIND_NIC,
MAX_QUEUE_NUM);
if (queues == 0) {
ret = -ENOENT;
goto err;
} }
if (netdev->peer) {
return -EEXIST; if (queues > MAX_QUEUE_NUM) {
ret = -E2BIG;
goto err;
}
for (i = 0; i < queues; i++) {
if (peers[i] == NULL) {
ret = -ENOENT;
goto err;
}
if (peers[i]->peer) {
ret = -EEXIST;
goto err;
}
ncs[i] = peers[i];
ncs[i]->queue_index = i;
} }
*ptr = netdev;
conf->queues = queues;
return 0; return 0;
err:
return ret;
} }
static const char *print_netdev(void *ptr) static const char *print_netdev(void *ptr)
...@@ -249,7 +280,8 @@ static void set_vlan(Object *obj, Visitor *v, void *opaque, ...@@ -249,7 +280,8 @@ static void set_vlan(Object *obj, Visitor *v, void *opaque,
{ {
DeviceState *dev = DEVICE(obj); DeviceState *dev = DEVICE(obj);
Property *prop = opaque; Property *prop = opaque;
NetClientState **ptr = qdev_get_prop_ptr(dev, prop); NICPeers *peers_ptr = qdev_get_prop_ptr(dev, prop);
NetClientState **ptr = &peers_ptr->ncs[0];
Error *local_err = NULL; Error *local_err = NULL;
int32_t id; int32_t id;
NetClientState *hubport; NetClientState *hubport;
......
...@@ -31,7 +31,7 @@ extern PropertyInfo qdev_prop_pci_host_devaddr; ...@@ -31,7 +31,7 @@ extern PropertyInfo qdev_prop_pci_host_devaddr;
.name = (_name), \ .name = (_name), \
.info = &(_prop), \ .info = &(_prop), \
.offset = offsetof(_state, _field) \ .offset = offsetof(_state, _field) \
+ type_check(_type,typeof_field(_state, _field)), \ + type_check(_type, typeof_field(_state, _field)), \
} }
#define DEFINE_PROP_DEFAULT(_name, _state, _field, _defval, _prop, _type) { \ #define DEFINE_PROP_DEFAULT(_name, _state, _field, _defval, _prop, _type) { \
.name = (_name), \ .name = (_name), \
...@@ -77,9 +77,9 @@ extern PropertyInfo qdev_prop_pci_host_devaddr; ...@@ -77,9 +77,9 @@ extern PropertyInfo qdev_prop_pci_host_devaddr;
#define DEFINE_PROP_STRING(_n, _s, _f) \ #define DEFINE_PROP_STRING(_n, _s, _f) \
DEFINE_PROP(_n, _s, _f, qdev_prop_string, char*) DEFINE_PROP(_n, _s, _f, qdev_prop_string, char*)
#define DEFINE_PROP_NETDEV(_n, _s, _f) \ #define DEFINE_PROP_NETDEV(_n, _s, _f) \
DEFINE_PROP(_n, _s, _f, qdev_prop_netdev, NetClientState*) DEFINE_PROP(_n, _s, _f, qdev_prop_netdev, NICPeers)
#define DEFINE_PROP_VLAN(_n, _s, _f) \ #define DEFINE_PROP_VLAN(_n, _s, _f) \
DEFINE_PROP(_n, _s, _f, qdev_prop_vlan, NetClientState*) DEFINE_PROP(_n, _s, _f, qdev_prop_vlan, NICPeers)
#define DEFINE_PROP_DRIVE(_n, _s, _f) \ #define DEFINE_PROP_DRIVE(_n, _s, _f) \
DEFINE_PROP(_n, _s, _f, qdev_prop_drive, BlockDriverState *) DEFINE_PROP(_n, _s, _f, qdev_prop_drive, BlockDriverState *)
#define DEFINE_PROP_MACADDR(_n, _s, _f) \ #define DEFINE_PROP_MACADDR(_n, _s, _f) \
......
...@@ -9,24 +9,32 @@ ...@@ -9,24 +9,32 @@
#include "migration/vmstate.h" #include "migration/vmstate.h"
#include "qapi-types.h" #include "qapi-types.h"
#define MAX_QUEUE_NUM 1024
struct MACAddr { struct MACAddr {
uint8_t a[6]; uint8_t a[6];
}; };
/* qdev nic properties */ /* qdev nic properties */
typedef struct NICPeers {
NetClientState *ncs[MAX_QUEUE_NUM];
} NICPeers;
typedef struct NICConf { typedef struct NICConf {
MACAddr macaddr; MACAddr macaddr;
NetClientState *peer; NICPeers peers;
int32_t bootindex; int32_t bootindex;
int32_t queues;
} NICConf; } NICConf;
#define DEFINE_NIC_PROPERTIES(_state, _conf) \ #define DEFINE_NIC_PROPERTIES(_state, _conf) \
DEFINE_PROP_MACADDR("mac", _state, _conf.macaddr), \ DEFINE_PROP_MACADDR("mac", _state, _conf.macaddr), \
DEFINE_PROP_VLAN("vlan", _state, _conf.peer), \ DEFINE_PROP_VLAN("vlan", _state, _conf.peers), \
DEFINE_PROP_NETDEV("netdev", _state, _conf.peer), \ DEFINE_PROP_NETDEV("netdev", _state, _conf.peers), \
DEFINE_PROP_INT32("bootindex", _state, _conf.bootindex, -1) DEFINE_PROP_INT32("bootindex", _state, _conf.bootindex, -1)
/* Net clients */ /* Net clients */
typedef void (NetPoll)(NetClientState *, bool enable); typedef void (NetPoll)(NetClientState *, bool enable);
...@@ -60,10 +68,11 @@ struct NetClientState { ...@@ -60,10 +68,11 @@ struct NetClientState {
char info_str[256]; char info_str[256];
unsigned receive_disabled : 1; unsigned receive_disabled : 1;
NetClientDestructor *destructor; NetClientDestructor *destructor;
unsigned int queue_index;
}; };
typedef struct NICState { typedef struct NICState {
NetClientState nc; NetClientState ncs[MAX_QUEUE_NUM];
NICConf *conf; NICConf *conf;
void *opaque; void *opaque;
bool peer_deleted; bool peer_deleted;
...@@ -82,6 +91,7 @@ NICState *qemu_new_nic(NetClientInfo *info, ...@@ -82,6 +91,7 @@ NICState *qemu_new_nic(NetClientInfo *info,
const char *name, const char *name,
void *opaque); void *opaque);
void qemu_del_nic(NICState *nic); void qemu_del_nic(NICState *nic);
NetClientState *qemu_get_subqueue(NICState *nic, int queue_index);
NetClientState *qemu_get_queue(NICState *nic); NetClientState *qemu_get_queue(NICState *nic);
NICState *qemu_get_nic(NetClientState *nc); NICState *qemu_get_nic(NetClientState *nc);
void *qemu_get_nic_opaque(NetClientState *nc); void *qemu_get_nic_opaque(NetClientState *nc);
......
...@@ -236,28 +236,44 @@ NICState *qemu_new_nic(NetClientInfo *info, ...@@ -236,28 +236,44 @@ NICState *qemu_new_nic(NetClientInfo *info,
void *opaque) void *opaque)
{ {
NetClientState *nc; NetClientState *nc;
NetClientState **peers = conf->peers.ncs;
NICState *nic; NICState *nic;
int i;
assert(info->type == NET_CLIENT_OPTIONS_KIND_NIC); assert(info->type == NET_CLIENT_OPTIONS_KIND_NIC);
assert(info->size >= sizeof(NICState)); assert(info->size >= sizeof(NICState));
nc = qemu_new_net_client(info, conf->peer, model, name); nc = qemu_new_net_client(info, peers[0], model, name);
nc->queue_index = 0;
nic = qemu_get_nic(nc); nic = qemu_get_nic(nc);
nic->conf = conf; nic->conf = conf;
nic->opaque = opaque; nic->opaque = opaque;
for (i = 1; i < conf->queues; i++) {
qemu_net_client_setup(&nic->ncs[i], info, peers[i], model, nc->name,
NULL);
nic->ncs[i].queue_index = i;
}
return nic; return nic;
} }
NetClientState *qemu_get_subqueue(NICState *nic, int queue_index)
{
return &nic->ncs[queue_index];
}
NetClientState *qemu_get_queue(NICState *nic) NetClientState *qemu_get_queue(NICState *nic)
{ {
return &nic->nc; return qemu_get_subqueue(nic, 0);
} }
NICState *qemu_get_nic(NetClientState *nc) NICState *qemu_get_nic(NetClientState *nc)
{ {
return DO_UPCAST(NICState, nc, nc); NetClientState *nc0 = nc - nc->queue_index;
return DO_UPCAST(NICState, ncs[0], nc0);
} }
void *qemu_get_nic_opaque(NetClientState *nc) void *qemu_get_nic_opaque(NetClientState *nc)
...@@ -271,9 +287,7 @@ static void qemu_cleanup_net_client(NetClientState *nc) ...@@ -271,9 +287,7 @@ static void qemu_cleanup_net_client(NetClientState *nc)
{ {
QTAILQ_REMOVE(&net_clients, nc, next); QTAILQ_REMOVE(&net_clients, nc, next);
if (nc->info->cleanup) { nc->info->cleanup(nc);
nc->info->cleanup(nc);
}
} }
static void qemu_free_net_client(NetClientState *nc) static void qemu_free_net_client(NetClientState *nc)
...@@ -293,6 +307,17 @@ static void qemu_free_net_client(NetClientState *nc) ...@@ -293,6 +307,17 @@ static void qemu_free_net_client(NetClientState *nc)
void qemu_del_net_client(NetClientState *nc) void qemu_del_net_client(NetClientState *nc)
{ {
NetClientState *ncs[MAX_QUEUE_NUM];
int queues, i;
/* If the NetClientState belongs to a multiqueue backend, we will change all
* other NetClientStates also.
*/
queues = qemu_find_net_clients_except(nc->name, ncs,
NET_CLIENT_OPTIONS_KIND_NIC,
MAX_QUEUE_NUM);
assert(queues != 0);
/* If there is a peer NIC, delete and cleanup client, but do not free. */ /* If there is a peer NIC, delete and cleanup client, but do not free. */
if (nc->peer && nc->peer->info->type == NET_CLIENT_OPTIONS_KIND_NIC) { if (nc->peer && nc->peer->info->type == NET_CLIENT_OPTIONS_KIND_NIC) {
NICState *nic = qemu_get_nic(nc->peer); NICState *nic = qemu_get_nic(nc->peer);
...@@ -300,34 +325,47 @@ void qemu_del_net_client(NetClientState *nc) ...@@ -300,34 +325,47 @@ void qemu_del_net_client(NetClientState *nc)
return; return;
} }
nic->peer_deleted = true; nic->peer_deleted = true;
/* Let NIC know peer is gone. */
nc->peer->link_down = true; for (i = 0; i < queues; i++) {
ncs[i]->peer->link_down = true;
}
if (nc->peer->info->link_status_changed) { if (nc->peer->info->link_status_changed) {
nc->peer->info->link_status_changed(nc->peer); nc->peer->info->link_status_changed(nc->peer);
} }
qemu_cleanup_net_client(nc);
for (i = 0; i < queues; i++) {
qemu_cleanup_net_client(ncs[i]);
}
return; return;
} }
assert(nc->info->type != NET_CLIENT_OPTIONS_KIND_NIC); assert(nc->info->type != NET_CLIENT_OPTIONS_KIND_NIC);
qemu_cleanup_net_client(nc); for (i = 0; i < queues; i++) {
qemu_free_net_client(nc); qemu_cleanup_net_client(ncs[i]);
qemu_free_net_client(ncs[i]);
}
} }
void qemu_del_nic(NICState *nic) void qemu_del_nic(NICState *nic)
{ {
NetClientState *nc = qemu_get_queue(nic); int i, queues = nic->conf->queues;
/* If this is a peer NIC and peer has already been deleted, free it now. */ /* If this is a peer NIC and peer has already been deleted, free it now. */
if (nc->peer && nc->info->type == NET_CLIENT_OPTIONS_KIND_NIC) { if (nic->peer_deleted) {
NICState *nic = qemu_get_nic(nc); for (i = 0; i < queues; i++) {
if (nic->peer_deleted) { qemu_free_net_client(qemu_get_subqueue(nic, i)->peer);
qemu_free_net_client(nc->peer);
} }
} }
qemu_cleanup_net_client(nc); for (i = queues - 1; i >= 0; i--) {
qemu_free_net_client(nc); NetClientState *nc = qemu_get_subqueue(nic, i);
qemu_cleanup_net_client(nc);
qemu_free_net_client(nc);
}
} }
void qemu_foreach_nic(qemu_nic_foreach func, void *opaque) void qemu_foreach_nic(qemu_nic_foreach func, void *opaque)
...@@ -336,7 +374,9 @@ void qemu_foreach_nic(qemu_nic_foreach func, void *opaque) ...@@ -336,7 +374,9 @@ void qemu_foreach_nic(qemu_nic_foreach func, void *opaque)
QTAILQ_FOREACH(nc, &net_clients, next) { QTAILQ_FOREACH(nc, &net_clients, next) {
if (nc->info->type == NET_CLIENT_OPTIONS_KIND_NIC) { if (nc->info->type == NET_CLIENT_OPTIONS_KIND_NIC) {
func(qemu_get_nic(nc), opaque); if (nc->queue_index == 0) {
func(qemu_get_nic(nc), opaque);
}
} }
} }
} }
...@@ -911,8 +951,10 @@ void qmp_netdev_del(const char *id, Error **errp) ...@@ -911,8 +951,10 @@ void qmp_netdev_del(const char *id, Error **errp)
void print_net_client(Monitor *mon, NetClientState *nc) void print_net_client(Monitor *mon, NetClientState *nc)
{ {
monitor_printf(mon, "%s: type=%s,%s\n", nc->name, monitor_printf(mon, "%s: index=%d,type=%s,%s\n", nc->name,
NetClientOptionsKind_lookup[nc->info->type], nc->info_str); nc->queue_index,
NetClientOptionsKind_lookup[nc->info->type],
nc->info_str);
} }
void do_info_network(Monitor *mon, const QDict *qdict) void do_info_network(Monitor *mon, const QDict *qdict)
...@@ -943,20 +985,23 @@ void do_info_network(Monitor *mon, const QDict *qdict) ...@@ -943,20 +985,23 @@ void do_info_network(Monitor *mon, const QDict *qdict)
void qmp_set_link(const char *name, bool up, Error **errp) void qmp_set_link(const char *name, bool up, Error **errp)
{ {
NetClientState *nc = NULL; NetClientState *ncs[MAX_QUEUE_NUM];
NetClientState *nc;
int queues, i;
QTAILQ_FOREACH(nc, &net_clients, next) { queues = qemu_find_net_clients_except(name, ncs,
if (!strcmp(nc->name, name)) { NET_CLIENT_OPTIONS_KIND_MAX,
goto done; MAX_QUEUE_NUM);
}
} if (queues == 0) {
done:
if (!nc) {
error_set(errp, QERR_DEVICE_NOT_FOUND, name); error_set(errp, QERR_DEVICE_NOT_FOUND, name);
return; return;
} }
nc = ncs[0];
nc->link_down = !up; for (i = 0; i < queues; i++) {
ncs[i]->link_down = !up;
}
if (nc->info->link_status_changed) { if (nc->info->link_status_changed) {
nc->info->link_status_changed(nc); nc->info->link_status_changed(nc);
...@@ -976,9 +1021,13 @@ done: ...@@ -976,9 +1021,13 @@ done:
void net_cleanup(void) void net_cleanup(void)
{ {
NetClientState *nc, *next_vc; NetClientState *nc;
QTAILQ_FOREACH_SAFE(nc, &net_clients, next, next_vc) { /* We may del multiple entries during qemu_del_net_client(),
* so QTAILQ_FOREACH_SAFE() is also not safe here.
*/
while (!QTAILQ_EMPTY(&net_clients)) {
nc = QTAILQ_FIRST(&net_clients);
if (nc->info->type == NET_CLIENT_OPTIONS_KIND_NIC) { if (nc->info->type == NET_CLIENT_OPTIONS_KIND_NIC) {
qemu_del_nic(qemu_get_nic(nc)); qemu_del_nic(qemu_get_nic(nc));
} else { } else {
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册