xref: /openbmc/qemu/hw/net/rocker/rocker_fp.c (revision bcad45de6a0b5bf10a274872d2e45da3403232da)
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 
17 #include "qemu/osdep.h"
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 
25 enum duplex {
26     DUPLEX_HALF = 0,
27     DUPLEX_FULL
28 };
29 
30 struct 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 
45 char *fp_port_get_name(FpPort *port)
46 {
47     return port->name;
48 }
49 
50 bool fp_port_get_link_up(FpPort *port)
51 {
52     return !qemu_get_queue(port->nic)->link_down;
53 }
54 
55 void 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 
65 void fp_port_get_macaddr(FpPort *port, MACAddr *macaddr)
66 {
67     memcpy(macaddr->a, port->conf.macaddr.a, sizeof(macaddr->a));
68 }
69 
70 void 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 
77 uint8_t fp_port_get_learning(FpPort *port)
78 {
79     return port->learning;
80 }
81 
82 void fp_port_set_learning(FpPort *port, uint8_t learning)
83 {
84     port->learning = learning;
85 }
86 
87 int 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 
97 int 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 
109 bool 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 
118 int 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 
129 static 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 
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 
144     return world_ingress(port->world, port->pport, iov, iovcnt);
145 }
146 
147 static 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 
158 static void fp_port_cleanup(NetClientState *nc)
159 {
160 }
161 
162 static 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 
169 static NetClientInfo fp_port_info = {
170     .type = NET_CLIENT_DRIVER_NIC,
171     .size = sizeof(NICState),
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 
178 World *fp_port_get_world(FpPort *port)
179 {
180     return port->world;
181 }
182 
183 void 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 
189 bool fp_port_check_world(FpPort *port, World *world)
190 {
191     return port->world == world;
192 }
193 
194 bool fp_port_enabled(FpPort *port)
195 {
196     return port->enabled;
197 }
198 
199 static 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 
209 void fp_port_enable(FpPort *port)
210 {
211     fp_port_set_link(port, true);
212     port->enabled = true;
213     DPRINTF("port %d enabled\n", port->index);
214 }
215 
216 void fp_port_disable(FpPort *port)
217 {
218     port->enabled = false;
219     fp_port_set_link(port, false);
220     DPRINTF("port %d disabled\n", port->index);
221 }
222 
223 FpPort *fp_port_alloc(Rocker *r, char *sw_name,
224                       MACAddr *start_mac, unsigned int index,
225                       NICPeers *peers)
226 {
227     FpPort *port = g_new0(FpPort, 1);
228 
229     if (!port) {
230         return NULL;
231     }
232 
233     port->r = r;
234     port->index = index;
235     port->pport = index + 1;
236 
237     /* front-panel switch port names are 1-based */
238 
239     port->name = g_strdup_printf("%sp%d", sw_name, port->pport);
240 
241     memcpy(port->conf.macaddr.a, start_mac, sizeof(port->conf.macaddr.a));
242     port->conf.macaddr.a[5] += index;
243     port->conf.bootindex = -1;
244     port->conf.peers = *peers;
245 
246     port->nic = qemu_new_nic(&fp_port_info, &port->conf,
247                              sw_name, NULL, port);
248     qemu_format_nic_info_str(qemu_get_queue(port->nic),
249                              port->conf.macaddr.a);
250 
251     fp_port_reset(port);
252 
253     return port;
254 }
255 
256 void fp_port_free(FpPort *port)
257 {
258     qemu_del_nic(port->nic);
259     g_free(port->name);
260     g_free(port);
261 }
262 
263 void fp_port_reset(FpPort *port)
264 {
265     fp_port_disable(port);
266     port->speed = 10000;   /* 10Gbps */
267     port->duplex = DUPLEX_FULL;
268     port->autoneg = 0;
269 }
270