]> git.proxmox.com Git - mirror_qemu.git/blame - hw/net/rocker/rocker_fp.c
Include qapi/error.h exactly where needed
[mirror_qemu.git] / hw / net / rocker / rocker_fp.c
CommitLineData
dc488f88
SF
1/*
2 * QEMU rocker switch emulation - front-panel ports
3 *
4 * Copyright (c) 2014 Scott Feldman <sfeldma@gmail.com>
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 */
16
e8d40465 17#include "qemu/osdep.h"
dc488f88
SF
18#include "net/clients.h"
19
20#include "rocker.h"
21#include "rocker_hw.h"
22#include "rocker_fp.h"
23#include "rocker_world.h"
24
25enum duplex {
26 DUPLEX_HALF = 0,
27 DUPLEX_FULL
28};
29
30struct fp_port {
31 Rocker *r;
32 World *world;
33 unsigned int index;
34 char *name;
35 uint32_t pport;
36 bool enabled;
37 uint32_t speed;
38 uint8_t duplex;
39 uint8_t autoneg;
40 uint8_t learning;
41 NICState *nic;
42 NICConf conf;
43};
44
77349536
DA
45char *fp_port_get_name(FpPort *port)
46{
47 return port->name;
48}
49
dc488f88
SF
50bool fp_port_get_link_up(FpPort *port)
51{
52 return !qemu_get_queue(port->nic)->link_down;
53}
54
fafa4d50
SF
55void fp_port_get_info(FpPort *port, RockerPortList *info)
56{
57 info->value->name = g_strdup(port->name);
58 info->value->enabled = port->enabled;
59 info->value->link_up = fp_port_get_link_up(port);
60 info->value->speed = port->speed;
61 info->value->duplex = port->duplex;
62 info->value->autoneg = port->autoneg;
63}
64
dc488f88
SF
65void fp_port_get_macaddr(FpPort *port, MACAddr *macaddr)
66{
67 memcpy(macaddr->a, port->conf.macaddr.a, sizeof(macaddr->a));
68}
69
70void fp_port_set_macaddr(FpPort *port, MACAddr *macaddr)
71{
72/* XXX TODO implement and test setting mac addr
73 * XXX memcpy(port->conf.macaddr.a, macaddr.a, sizeof(port->conf.macaddr.a));
74 */
75}
76
77uint8_t fp_port_get_learning(FpPort *port)
78{
79 return port->learning;
80}
81
82void fp_port_set_learning(FpPort *port, uint8_t learning)
83{
84 port->learning = learning;
85}
86
87int fp_port_get_settings(FpPort *port, uint32_t *speed,
88 uint8_t *duplex, uint8_t *autoneg)
89{
90 *speed = port->speed;
91 *duplex = port->duplex;
92 *autoneg = port->autoneg;
93
94 return ROCKER_OK;
95}
96
97int fp_port_set_settings(FpPort *port, uint32_t speed,
98 uint8_t duplex, uint8_t autoneg)
99{
100 /* XXX validate inputs */
101
102 port->speed = speed;
103 port->duplex = duplex;
104 port->autoneg = autoneg;
105
106 return ROCKER_OK;
107}
108
109bool fp_port_from_pport(uint32_t pport, uint32_t *port)
110{
111 if (pport < 1 || pport > ROCKER_FP_PORTS_MAX) {
112 return false;
113 }
114 *port = pport - 1;
115 return true;
116}
117
118int fp_port_eg(FpPort *port, const struct iovec *iov, int iovcnt)
119{
120 NetClientState *nc = qemu_get_queue(port->nic);
121
122 if (port->enabled) {
123 qemu_sendv_packet(nc, iov, iovcnt);
124 }
125
126 return ROCKER_OK;
127}
128
dc488f88
SF
129static ssize_t fp_port_receive_iov(NetClientState *nc, const struct iovec *iov,
130 int iovcnt)
131{
132 FpPort *port = qemu_get_nic_opaque(nc);
133
66851f64
SF
134 /* If the port is disabled, we want to drop this pkt
135 * now rather than queing it for later. We don't want
136 * any stale pkts getting into the device when the port
137 * transitions to enabled.
138 */
139
140 if (!port->enabled) {
141 return -1;
142 }
143
dc488f88
SF
144 return world_ingress(port->world, port->pport, iov, iovcnt);
145}
146
147static ssize_t fp_port_receive(NetClientState *nc, const uint8_t *buf,
148 size_t size)
149{
150 const struct iovec iov = {
151 .iov_base = (uint8_t *)buf,
152 .iov_len = size
153 };
154
155 return fp_port_receive_iov(nc, &iov, 1);
156}
157
158static void fp_port_cleanup(NetClientState *nc)
159{
160}
161
162static void fp_port_set_link_status(NetClientState *nc)
163{
164 FpPort *port = qemu_get_nic_opaque(nc);
165
166 rocker_event_link_changed(port->r, port->pport, !nc->link_down);
167}
168
169static NetClientInfo fp_port_info = {
f394b2e2 170 .type = NET_CLIENT_DRIVER_NIC,
dc488f88 171 .size = sizeof(NICState),
dc488f88
SF
172 .receive = fp_port_receive,
173 .receive_iov = fp_port_receive_iov,
174 .cleanup = fp_port_cleanup,
175 .link_status_changed = fp_port_set_link_status,
176};
177
178World *fp_port_get_world(FpPort *port)
179{
180 return port->world;
181}
182
183void fp_port_set_world(FpPort *port, World *world)
184{
185 DPRINTF("port %d setting world \"%s\"\n", port->index, world_name(world));
186 port->world = world;
187}
188
0ab9cd9a
JP
189bool fp_port_check_world(FpPort *port, World *world)
190{
191 return port->world == world;
192}
193
dc488f88
SF
194bool fp_port_enabled(FpPort *port)
195{
196 return port->enabled;
197}
198
5ff1547b
SF
199static void fp_port_set_link(FpPort *port, bool up)
200{
201 NetClientState *nc = qemu_get_queue(port->nic);
202
203 if (up == nc->link_down) {
204 nc->link_down = !up;
205 nc->info->link_status_changed(nc);
206 }
207}
208
dc488f88
SF
209void fp_port_enable(FpPort *port)
210{
5ff1547b 211 fp_port_set_link(port, true);
dc488f88
SF
212 port->enabled = true;
213 DPRINTF("port %d enabled\n", port->index);
214}
215
216void fp_port_disable(FpPort *port)
217{
218 port->enabled = false;
5ff1547b 219 fp_port_set_link(port, false);
dc488f88
SF
220 DPRINTF("port %d disabled\n", port->index);
221}
222
223FpPort *fp_port_alloc(Rocker *r, char *sw_name,
224 MACAddr *start_mac, unsigned int index,
225 NICPeers *peers)
226{
778358d0 227 FpPort *port = g_new0(FpPort, 1);
dc488f88 228
dc488f88
SF
229 port->r = r;
230 port->index = index;
231 port->pport = index + 1;
232
233 /* front-panel switch port names are 1-based */
234
77349536 235 port->name = g_strdup_printf("%sp%d", sw_name, port->pport);
dc488f88
SF
236
237 memcpy(port->conf.macaddr.a, start_mac, sizeof(port->conf.macaddr.a));
238 port->conf.macaddr.a[5] += index;
239 port->conf.bootindex = -1;
240 port->conf.peers = *peers;
241
242 port->nic = qemu_new_nic(&fp_port_info, &port->conf,
243 sw_name, NULL, port);
244 qemu_format_nic_info_str(qemu_get_queue(port->nic),
245 port->conf.macaddr.a);
246
247 fp_port_reset(port);
248
249 return port;
250}
251
252void fp_port_free(FpPort *port)
253{
254 qemu_del_nic(port->nic);
255 g_free(port->name);
256 g_free(port);
257}
258
259void fp_port_reset(FpPort *port)
260{
261 fp_port_disable(port);
262 port->speed = 10000; /* 10Gbps */
263 port->duplex = DUPLEX_FULL;
264 port->autoneg = 0;
265}