]> git.proxmox.com Git - mirror_qemu.git/blob - net/filter-mirror.c
Merge remote-tracking branch 'remotes/kevin/tags/for-upstream' into staging
[mirror_qemu.git] / net / filter-mirror.c
1 /*
2 * Copyright (c) 2016 HUAWEI TECHNOLOGIES CO., LTD.
3 * Copyright (c) 2016 FUJITSU LIMITED
4 * Copyright (c) 2016 Intel Corporation
5 *
6 * Author: Zhang Chen <zhangchen.fnst@cn.fujitsu.com>
7 *
8 * This work is licensed under the terms of the GNU GPL, version 2 or
9 * later. See the COPYING file in the top-level directory.
10 */
11
12 #include "qemu/osdep.h"
13 #include "net/filter.h"
14 #include "net/net.h"
15 #include "qemu-common.h"
16 #include "qapi/error.h"
17 #include "qapi/qmp/qerror.h"
18 #include "qapi-visit.h"
19 #include "qom/object.h"
20 #include "qemu/main-loop.h"
21 #include "qemu/error-report.h"
22 #include "trace.h"
23 #include "sysemu/char.h"
24 #include "qemu/iov.h"
25 #include "qemu/sockets.h"
26
27 #define FILTER_MIRROR(obj) \
28 OBJECT_CHECK(MirrorState, (obj), TYPE_FILTER_MIRROR)
29
30 #define FILTER_REDIRECTOR(obj) \
31 OBJECT_CHECK(MirrorState, (obj), TYPE_FILTER_REDIRECTOR)
32
33 #define TYPE_FILTER_MIRROR "filter-mirror"
34 #define TYPE_FILTER_REDIRECTOR "filter-redirector"
35 #define REDIRECTOR_MAX_LEN NET_BUFSIZE
36
37 typedef struct MirrorState {
38 NetFilterState parent_obj;
39 char *indev;
40 char *outdev;
41 CharDriverState *chr_in;
42 CharDriverState *chr_out;
43 int state; /* 0 = getting length, 1 = getting data */
44 unsigned int index;
45 unsigned int packet_len;
46 uint8_t buf[REDIRECTOR_MAX_LEN];
47 } MirrorState;
48
49 static int filter_mirror_send(CharDriverState *chr_out,
50 const struct iovec *iov,
51 int iovcnt)
52 {
53 int ret = 0;
54 ssize_t size = 0;
55 uint32_t len = 0;
56 char *buf;
57
58 size = iov_size(iov, iovcnt);
59 if (!size) {
60 return 0;
61 }
62
63 len = htonl(size);
64 ret = qemu_chr_fe_write_all(chr_out, (uint8_t *)&len, sizeof(len));
65 if (ret != sizeof(len)) {
66 goto err;
67 }
68
69 buf = g_malloc(size);
70 iov_to_buf(iov, iovcnt, 0, buf, size);
71 ret = qemu_chr_fe_write_all(chr_out, (uint8_t *)buf, size);
72 g_free(buf);
73 if (ret != size) {
74 goto err;
75 }
76
77 return 0;
78
79 err:
80 return ret < 0 ? ret : -EIO;
81 }
82
83 static void
84 redirector_to_filter(NetFilterState *nf, const uint8_t *buf, int len)
85 {
86 struct iovec iov = {
87 .iov_base = (void *)buf,
88 .iov_len = len,
89 };
90
91 if (nf->direction == NET_FILTER_DIRECTION_ALL ||
92 nf->direction == NET_FILTER_DIRECTION_TX) {
93 qemu_netfilter_pass_to_next(nf->netdev, 0, &iov, 1, nf);
94 }
95
96 if (nf->direction == NET_FILTER_DIRECTION_ALL ||
97 nf->direction == NET_FILTER_DIRECTION_RX) {
98 qemu_netfilter_pass_to_next(nf->netdev->peer, 0, &iov, 1, nf);
99 }
100 }
101
102 static int redirector_chr_can_read(void *opaque)
103 {
104 return REDIRECTOR_MAX_LEN;
105 }
106
107 static void redirector_chr_read(void *opaque, const uint8_t *buf, int size)
108 {
109 NetFilterState *nf = opaque;
110 MirrorState *s = FILTER_REDIRECTOR(nf);
111 unsigned int l;
112
113 while (size > 0) {
114 /* reassemble a packet from the network */
115 switch (s->state) { /* 0 = getting length, 1 = getting data */
116 case 0:
117 l = 4 - s->index;
118 if (l > size) {
119 l = size;
120 }
121 memcpy(s->buf + s->index, buf, l);
122 buf += l;
123 size -= l;
124 s->index += l;
125 if (s->index == 4) {
126 /* got length */
127 s->packet_len = ntohl(*(uint32_t *)s->buf);
128 s->index = 0;
129 s->state = 1;
130 }
131 break;
132 case 1:
133 l = s->packet_len - s->index;
134 if (l > size) {
135 l = size;
136 }
137 if (s->index + l <= sizeof(s->buf)) {
138 memcpy(s->buf + s->index, buf, l);
139 } else {
140 error_report("serious error: oversized packet received.");
141 s->index = s->state = 0;
142 qemu_chr_add_handlers(s->chr_in, NULL, NULL, NULL, NULL);
143 return;
144 }
145
146 s->index += l;
147 buf += l;
148 size -= l;
149 if (s->index >= s->packet_len) {
150 s->index = 0;
151 s->state = 0;
152 redirector_to_filter(nf, s->buf, s->packet_len);
153 }
154 break;
155 }
156 }
157 }
158
159 static void redirector_chr_event(void *opaque, int event)
160 {
161 NetFilterState *nf = opaque;
162 MirrorState *s = FILTER_REDIRECTOR(nf);
163
164 switch (event) {
165 case CHR_EVENT_CLOSED:
166 qemu_chr_add_handlers(s->chr_in, NULL, NULL, NULL, NULL);
167 break;
168 default:
169 break;
170 }
171 }
172
173 static ssize_t filter_mirror_receive_iov(NetFilterState *nf,
174 NetClientState *sender,
175 unsigned flags,
176 const struct iovec *iov,
177 int iovcnt,
178 NetPacketSent *sent_cb)
179 {
180 MirrorState *s = FILTER_MIRROR(nf);
181 int ret;
182
183 ret = filter_mirror_send(s->chr_out, iov, iovcnt);
184 if (ret) {
185 error_report("filter_mirror_send failed(%s)", strerror(-ret));
186 }
187
188 /*
189 * we don't hope this error interrupt the normal
190 * path of net packet, so we always return zero.
191 */
192 return 0;
193 }
194
195 static ssize_t filter_redirector_receive_iov(NetFilterState *nf,
196 NetClientState *sender,
197 unsigned flags,
198 const struct iovec *iov,
199 int iovcnt,
200 NetPacketSent *sent_cb)
201 {
202 MirrorState *s = FILTER_REDIRECTOR(nf);
203 int ret;
204
205 if (s->chr_out) {
206 ret = filter_mirror_send(s->chr_out, iov, iovcnt);
207 if (ret) {
208 error_report("filter_mirror_send failed(%s)", strerror(-ret));
209 }
210 return iov_size(iov, iovcnt);
211 } else {
212 return 0;
213 }
214 }
215
216 static void filter_mirror_cleanup(NetFilterState *nf)
217 {
218 MirrorState *s = FILTER_MIRROR(nf);
219
220 if (s->chr_out) {
221 qemu_chr_fe_release(s->chr_out);
222 }
223 }
224
225 static void filter_redirector_cleanup(NetFilterState *nf)
226 {
227 MirrorState *s = FILTER_REDIRECTOR(nf);
228
229 if (s->chr_in) {
230 qemu_chr_add_handlers(s->chr_in, NULL, NULL, NULL, NULL);
231 qemu_chr_fe_release(s->chr_in);
232 }
233 if (s->chr_out) {
234 qemu_chr_fe_release(s->chr_out);
235 }
236 }
237
238 static void filter_mirror_setup(NetFilterState *nf, Error **errp)
239 {
240 MirrorState *s = FILTER_MIRROR(nf);
241
242 if (!s->outdev) {
243 error_setg(errp, "filter filter mirror needs 'outdev' "
244 "property set");
245 return;
246 }
247
248 s->chr_out = qemu_chr_find(s->outdev);
249 if (s->chr_out == NULL) {
250 error_set(errp, ERROR_CLASS_DEVICE_NOT_FOUND,
251 "Device '%s' not found", s->outdev);
252 return;
253 }
254
255 if (qemu_chr_fe_claim(s->chr_out) != 0) {
256 error_setg(errp, QERR_DEVICE_IN_USE, s->outdev);
257 return;
258 }
259 }
260
261 static void filter_redirector_setup(NetFilterState *nf, Error **errp)
262 {
263 MirrorState *s = FILTER_REDIRECTOR(nf);
264
265 if (!s->indev && !s->outdev) {
266 error_setg(errp, "filter redirector needs 'indev' or "
267 "'outdev' at least one property set");
268 return;
269 } else if (s->indev && s->outdev) {
270 if (!strcmp(s->indev, s->outdev)) {
271 error_setg(errp, "'indev' and 'outdev' could not be same "
272 "for filter redirector");
273 return;
274 }
275 }
276
277 s->state = s->index = 0;
278
279 if (s->indev) {
280 s->chr_in = qemu_chr_find(s->indev);
281 if (s->chr_in == NULL) {
282 error_set(errp, ERROR_CLASS_DEVICE_NOT_FOUND,
283 "IN Device '%s' not found", s->indev);
284 return;
285 }
286
287 qemu_chr_fe_claim_no_fail(s->chr_in);
288 qemu_chr_add_handlers(s->chr_in, redirector_chr_can_read,
289 redirector_chr_read, redirector_chr_event, nf);
290 }
291
292 if (s->outdev) {
293 s->chr_out = qemu_chr_find(s->outdev);
294 if (s->chr_out == NULL) {
295 error_set(errp, ERROR_CLASS_DEVICE_NOT_FOUND,
296 "OUT Device '%s' not found", s->outdev);
297 return;
298 }
299 qemu_chr_fe_claim_no_fail(s->chr_out);
300 }
301 }
302
303 static void filter_mirror_class_init(ObjectClass *oc, void *data)
304 {
305 NetFilterClass *nfc = NETFILTER_CLASS(oc);
306
307 nfc->setup = filter_mirror_setup;
308 nfc->cleanup = filter_mirror_cleanup;
309 nfc->receive_iov = filter_mirror_receive_iov;
310 }
311
312 static void filter_redirector_class_init(ObjectClass *oc, void *data)
313 {
314 NetFilterClass *nfc = NETFILTER_CLASS(oc);
315
316 nfc->setup = filter_redirector_setup;
317 nfc->cleanup = filter_redirector_cleanup;
318 nfc->receive_iov = filter_redirector_receive_iov;
319 }
320
321 static char *filter_redirector_get_indev(Object *obj, Error **errp)
322 {
323 MirrorState *s = FILTER_REDIRECTOR(obj);
324
325 return g_strdup(s->indev);
326 }
327
328 static void
329 filter_redirector_set_indev(Object *obj, const char *value, Error **errp)
330 {
331 MirrorState *s = FILTER_REDIRECTOR(obj);
332
333 g_free(s->indev);
334 s->indev = g_strdup(value);
335 }
336
337 static char *filter_mirror_get_outdev(Object *obj, Error **errp)
338 {
339 MirrorState *s = FILTER_MIRROR(obj);
340
341 return g_strdup(s->outdev);
342 }
343
344 static void
345 filter_mirror_set_outdev(Object *obj, const char *value, Error **errp)
346 {
347 MirrorState *s = FILTER_MIRROR(obj);
348
349 g_free(s->outdev);
350 s->outdev = g_strdup(value);
351 if (!s->outdev) {
352 error_setg(errp, "filter filter mirror needs 'outdev' "
353 "property set");
354 return;
355 }
356 }
357
358 static char *filter_redirector_get_outdev(Object *obj, Error **errp)
359 {
360 MirrorState *s = FILTER_REDIRECTOR(obj);
361
362 return g_strdup(s->outdev);
363 }
364
365 static void
366 filter_redirector_set_outdev(Object *obj, const char *value, Error **errp)
367 {
368 MirrorState *s = FILTER_REDIRECTOR(obj);
369
370 g_free(s->outdev);
371 s->outdev = g_strdup(value);
372 }
373
374 static void filter_mirror_init(Object *obj)
375 {
376 object_property_add_str(obj, "outdev", filter_mirror_get_outdev,
377 filter_mirror_set_outdev, NULL);
378 }
379
380 static void filter_redirector_init(Object *obj)
381 {
382 object_property_add_str(obj, "indev", filter_redirector_get_indev,
383 filter_redirector_set_indev, NULL);
384 object_property_add_str(obj, "outdev", filter_redirector_get_outdev,
385 filter_redirector_set_outdev, NULL);
386 }
387
388 static void filter_mirror_fini(Object *obj)
389 {
390 MirrorState *s = FILTER_MIRROR(obj);
391
392 g_free(s->outdev);
393 }
394
395 static void filter_redirector_fini(Object *obj)
396 {
397 MirrorState *s = FILTER_REDIRECTOR(obj);
398
399 g_free(s->indev);
400 g_free(s->outdev);
401 }
402
403 static const TypeInfo filter_redirector_info = {
404 .name = TYPE_FILTER_REDIRECTOR,
405 .parent = TYPE_NETFILTER,
406 .class_init = filter_redirector_class_init,
407 .instance_init = filter_redirector_init,
408 .instance_finalize = filter_redirector_fini,
409 .instance_size = sizeof(MirrorState),
410 };
411
412 static const TypeInfo filter_mirror_info = {
413 .name = TYPE_FILTER_MIRROR,
414 .parent = TYPE_NETFILTER,
415 .class_init = filter_mirror_class_init,
416 .instance_init = filter_mirror_init,
417 .instance_finalize = filter_mirror_fini,
418 .instance_size = sizeof(MirrorState),
419 };
420
421 static void register_types(void)
422 {
423 type_register_static(&filter_mirror_info);
424 type_register_static(&filter_redirector_info);
425 }
426
427 type_init(register_types);