| /* | 
 |  * Copyright (c) 2015 FUJITSU LIMITED | 
 |  * Author: Yang Hongyang <yanghy@cn.fujitsu.com> | 
 |  * | 
 |  * This work is licensed under the terms of the GNU GPL, version 2 or | 
 |  * later.  See the COPYING file in the top-level directory. | 
 |  */ | 
 |  | 
 | #include "qemu/osdep.h" | 
 | #include "net/filter.h" | 
 | #include "net/queue.h" | 
 | #include "qapi/error.h" | 
 | #include "qemu/timer.h" | 
 | #include "qemu/iov.h" | 
 | #include "qapi/qapi-builtin-visit.h" | 
 | #include "qapi/qmp/qerror.h" | 
 | #include "qom/object.h" | 
 |  | 
 | #define TYPE_FILTER_BUFFER "filter-buffer" | 
 |  | 
 | OBJECT_DECLARE_SIMPLE_TYPE(FilterBufferState, FILTER_BUFFER) | 
 |  | 
 | struct FilterBufferState { | 
 |     NetFilterState parent_obj; | 
 |  | 
 |     NetQueue *incoming_queue; | 
 |     uint32_t interval; | 
 |     QEMUTimer release_timer; | 
 | }; | 
 |  | 
 | static void filter_buffer_flush(NetFilterState *nf) | 
 | { | 
 |     FilterBufferState *s = FILTER_BUFFER(nf); | 
 |  | 
 |     if (!qemu_net_queue_flush(s->incoming_queue)) { | 
 |         /* Unable to empty the queue, purge remaining packets */ | 
 |         qemu_net_queue_purge(s->incoming_queue, nf->netdev); | 
 |     } | 
 | } | 
 |  | 
 | static void filter_buffer_release_timer(void *opaque) | 
 | { | 
 |     NetFilterState *nf = opaque; | 
 |     FilterBufferState *s = FILTER_BUFFER(nf); | 
 |  | 
 |     /* | 
 |      * Note: filter_buffer_flush() drops packets that can't be sent | 
 |      * TODO: We should leave them queued.  But currently there's no way | 
 |      * for the next filter or receiver to notify us that it can receive | 
 |      * more packets. | 
 |      */ | 
 |     filter_buffer_flush(nf); | 
 |     /* Timer rearmed to fire again in s->interval microseconds. */ | 
 |     timer_mod(&s->release_timer, | 
 |               qemu_clock_get_us(QEMU_CLOCK_VIRTUAL) + s->interval); | 
 | } | 
 |  | 
 | /* filter APIs */ | 
 | static ssize_t filter_buffer_receive_iov(NetFilterState *nf, | 
 |                                          NetClientState *sender, | 
 |                                          unsigned flags, | 
 |                                          const struct iovec *iov, | 
 |                                          int iovcnt, | 
 |                                          NetPacketSent *sent_cb) | 
 | { | 
 |     FilterBufferState *s = FILTER_BUFFER(nf); | 
 |  | 
 |     /* | 
 |      * We return size when buffer a packet, the sender will take it as | 
 |      * a already sent packet, so sent_cb should not be called later. | 
 |      * | 
 |      * FIXME: Even if the guest can't receive packets for some reasons, | 
 |      * the filter can still accept packets until its internal queue is full. | 
 |      * For example: | 
 |      *   For some reason, receiver could not receive more packets | 
 |      * (.can_receive() returns false). Without a filter, at most one packet | 
 |      * will be queued in incoming queue and sender's poll will be disabled | 
 |      * unit its sent_cb() was called. With a filter, it will keep receiving | 
 |      * the packets without caring about the receiver. This is suboptimal. | 
 |      * May need more thoughts (e.g keeping sent_cb). | 
 |      */ | 
 |     qemu_net_queue_append_iov(s->incoming_queue, sender, flags, | 
 |                               iov, iovcnt, NULL); | 
 |     return iov_size(iov, iovcnt); | 
 | } | 
 |  | 
 | static void filter_buffer_cleanup(NetFilterState *nf) | 
 | { | 
 |     FilterBufferState *s = FILTER_BUFFER(nf); | 
 |  | 
 |     if (s->interval) { | 
 |         timer_del(&s->release_timer); | 
 |     } | 
 |  | 
 |     /* flush packets */ | 
 |     if (s->incoming_queue) { | 
 |         filter_buffer_flush(nf); | 
 |         g_free(s->incoming_queue); | 
 |     } | 
 | } | 
 |  | 
 | static void filter_buffer_setup_timer(NetFilterState *nf) | 
 | { | 
 |     FilterBufferState *s = FILTER_BUFFER(nf); | 
 |  | 
 |     if (s->interval) { | 
 |         timer_init_us(&s->release_timer, QEMU_CLOCK_VIRTUAL, | 
 |                       filter_buffer_release_timer, nf); | 
 |         /* Timer armed to fire in s->interval microseconds. */ | 
 |         timer_mod(&s->release_timer, | 
 |                   qemu_clock_get_us(QEMU_CLOCK_VIRTUAL) + s->interval); | 
 |     } | 
 | } | 
 |  | 
 | static void filter_buffer_setup(NetFilterState *nf, Error **errp) | 
 | { | 
 |     FilterBufferState *s = FILTER_BUFFER(nf); | 
 |  | 
 |     /* | 
 |      * We may want to accept zero interval when VM FT solutions like MC | 
 |      * or COLO use this filter to release packets on demand. | 
 |      */ | 
 |     if (!s->interval) { | 
 |         error_setg(errp, QERR_INVALID_PARAMETER_VALUE, "interval", | 
 |                    "a non-zero interval"); | 
 |         return; | 
 |     } | 
 |  | 
 |     s->incoming_queue = qemu_new_net_queue(qemu_netfilter_pass_to_next, nf); | 
 |     filter_buffer_setup_timer(nf); | 
 | } | 
 |  | 
 | static void filter_buffer_status_changed(NetFilterState *nf, Error **errp) | 
 | { | 
 |     FilterBufferState *s = FILTER_BUFFER(nf); | 
 |  | 
 |     if (!nf->on) { | 
 |         if (s->interval) { | 
 |             timer_del(&s->release_timer); | 
 |         } | 
 |         filter_buffer_flush(nf); | 
 |     } else { | 
 |         filter_buffer_setup_timer(nf); | 
 |     } | 
 | } | 
 |  | 
 | static void filter_buffer_get_interval(Object *obj, Visitor *v, | 
 |                                        const char *name, void *opaque, | 
 |                                        Error **errp) | 
 | { | 
 |     FilterBufferState *s = FILTER_BUFFER(obj); | 
 |     uint32_t value = s->interval; | 
 |  | 
 |     visit_type_uint32(v, name, &value, errp); | 
 | } | 
 |  | 
 | static void filter_buffer_set_interval(Object *obj, Visitor *v, | 
 |                                        const char *name, void *opaque, | 
 |                                        Error **errp) | 
 | { | 
 |     FilterBufferState *s = FILTER_BUFFER(obj); | 
 |     uint32_t value; | 
 |  | 
 |     if (!visit_type_uint32(v, name, &value, errp)) { | 
 |         return; | 
 |     } | 
 |     if (!value) { | 
 |         error_setg(errp, "Property '%s.%s' requires a positive value", | 
 |                    object_get_typename(obj), name); | 
 |         return; | 
 |     } | 
 |     s->interval = value; | 
 | } | 
 |  | 
 | static void filter_buffer_class_init(ObjectClass *oc, const void *data) | 
 | { | 
 |     NetFilterClass *nfc = NETFILTER_CLASS(oc); | 
 |  | 
 |     object_class_property_add(oc, "interval", "uint32", | 
 |                               filter_buffer_get_interval, | 
 |                               filter_buffer_set_interval, NULL, NULL); | 
 |  | 
 |     nfc->setup = filter_buffer_setup; | 
 |     nfc->cleanup = filter_buffer_cleanup; | 
 |     nfc->receive_iov = filter_buffer_receive_iov; | 
 |     nfc->status_changed = filter_buffer_status_changed; | 
 | } | 
 |  | 
 | static const TypeInfo filter_buffer_info = { | 
 |     .name = TYPE_FILTER_BUFFER, | 
 |     .parent = TYPE_NETFILTER, | 
 |     .class_init = filter_buffer_class_init, | 
 |     .instance_size = sizeof(FilterBufferState), | 
 | }; | 
 |  | 
 | static void register_types(void) | 
 | { | 
 |     type_register_static(&filter_buffer_info); | 
 | } | 
 |  | 
 | type_init(register_types); |