| /* | 
 |  * 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 "qapi/error.h" | 
 | #include "qapi/qmp/qerror.h" | 
 | #include "qemu/error-report.h" | 
 |  | 
 | #include "net/filter.h" | 
 | #include "net/net.h" | 
 | #include "net/vhost_net.h" | 
 | #include "qom/object_interfaces.h" | 
 | #include "qemu/iov.h" | 
 | #include "qemu/module.h" | 
 | #include "net/colo.h" | 
 | #include "migration/colo.h" | 
 |  | 
 | static inline bool qemu_can_skip_netfilter(NetFilterState *nf) | 
 | { | 
 |     return !nf->on; | 
 | } | 
 |  | 
 | ssize_t qemu_netfilter_receive(NetFilterState *nf, | 
 |                                NetFilterDirection direction, | 
 |                                NetClientState *sender, | 
 |                                unsigned flags, | 
 |                                const struct iovec *iov, | 
 |                                int iovcnt, | 
 |                                NetPacketSent *sent_cb) | 
 | { | 
 |     if (qemu_can_skip_netfilter(nf)) { | 
 |         return 0; | 
 |     } | 
 |     if (nf->direction == direction || | 
 |         nf->direction == NET_FILTER_DIRECTION_ALL) { | 
 |         return NETFILTER_GET_CLASS(OBJECT(nf))->receive_iov( | 
 |                                    nf, sender, flags, iov, iovcnt, sent_cb); | 
 |     } | 
 |  | 
 |     return 0; | 
 | } | 
 |  | 
 | static NetFilterState *netfilter_next(NetFilterState *nf, | 
 |                                       NetFilterDirection dir) | 
 | { | 
 |     NetFilterState *next; | 
 |  | 
 |     if (dir == NET_FILTER_DIRECTION_TX) { | 
 |         /* forward walk through filters */ | 
 |         next = QTAILQ_NEXT(nf, next); | 
 |     } else { | 
 |         /* reverse order */ | 
 |         next = QTAILQ_PREV(nf, next); | 
 |     } | 
 |  | 
 |     return next; | 
 | } | 
 |  | 
 | ssize_t qemu_netfilter_pass_to_next(NetClientState *sender, | 
 |                                     unsigned flags, | 
 |                                     const struct iovec *iov, | 
 |                                     int iovcnt, | 
 |                                     void *opaque) | 
 | { | 
 |     int ret = 0; | 
 |     int direction; | 
 |     NetFilterState *nf = opaque; | 
 |     NetFilterState *next = NULL; | 
 |  | 
 |     if (!sender || !sender->peer) { | 
 |         /* no receiver, or sender been deleted, no need to pass it further */ | 
 |         goto out; | 
 |     } | 
 |  | 
 |     if (nf->direction == NET_FILTER_DIRECTION_ALL) { | 
 |         if (sender == nf->netdev) { | 
 |             /* This packet is sent by netdev itself */ | 
 |             direction = NET_FILTER_DIRECTION_TX; | 
 |         } else { | 
 |             direction = NET_FILTER_DIRECTION_RX; | 
 |         } | 
 |     } else { | 
 |         direction = nf->direction; | 
 |     } | 
 |  | 
 |     next = netfilter_next(nf, direction); | 
 |     while (next) { | 
 |         /* | 
 |          * if qemu_netfilter_pass_to_next been called, means that | 
 |          * the packet has been hold by filter and has already retured size | 
 |          * to the sender, so sent_cb shouldn't be called later, just | 
 |          * pass NULL to next. | 
 |          */ | 
 |         ret = qemu_netfilter_receive(next, direction, sender, flags, iov, | 
 |                                      iovcnt, NULL); | 
 |         if (ret) { | 
 |             return ret; | 
 |         } | 
 |         next = netfilter_next(next, direction); | 
 |     } | 
 |  | 
 |     /* | 
 |      * We have gone through all filters, pass it to receiver. | 
 |      * Do the valid check again incase sender or receiver been | 
 |      * deleted while we go through filters. | 
 |      */ | 
 |     if (sender && sender->peer) { | 
 |         qemu_net_queue_send_iov(sender->peer->incoming_queue, | 
 |                                 sender, flags, iov, iovcnt, NULL); | 
 |     } | 
 |  | 
 | out: | 
 |     /* no receiver, or sender been deleted */ | 
 |     return iov_size(iov, iovcnt); | 
 | } | 
 |  | 
 | static char *netfilter_get_netdev_id(Object *obj, Error **errp) | 
 | { | 
 |     NetFilterState *nf = NETFILTER(obj); | 
 |  | 
 |     return g_strdup(nf->netdev_id); | 
 | } | 
 |  | 
 | static void netfilter_set_netdev_id(Object *obj, const char *str, Error **errp) | 
 | { | 
 |     NetFilterState *nf = NETFILTER(obj); | 
 |  | 
 |     nf->netdev_id = g_strdup(str); | 
 | } | 
 |  | 
 | static int netfilter_get_direction(Object *obj, Error **errp G_GNUC_UNUSED) | 
 | { | 
 |     NetFilterState *nf = NETFILTER(obj); | 
 |     return nf->direction; | 
 | } | 
 |  | 
 | static void netfilter_set_direction(Object *obj, int direction, Error **errp) | 
 | { | 
 |     NetFilterState *nf = NETFILTER(obj); | 
 |     nf->direction = direction; | 
 | } | 
 |  | 
 | static char *netfilter_get_status(Object *obj, Error **errp) | 
 | { | 
 |     NetFilterState *nf = NETFILTER(obj); | 
 |  | 
 |     return nf->on ? g_strdup("on") : g_strdup("off"); | 
 | } | 
 |  | 
 | static void netfilter_set_status(Object *obj, const char *str, Error **errp) | 
 | { | 
 |     NetFilterState *nf = NETFILTER(obj); | 
 |     NetFilterClass *nfc = NETFILTER_GET_CLASS(obj); | 
 |  | 
 |     if (strcmp(str, "on") && strcmp(str, "off")) { | 
 |         error_setg(errp, "Invalid value for netfilter status, " | 
 |                          "should be 'on' or 'off'"); | 
 |         return; | 
 |     } | 
 |     if (nf->on == !strcmp(str, "on")) { | 
 |         return; | 
 |     } | 
 |     nf->on = !nf->on; | 
 |     if (nf->netdev && nfc->status_changed) { | 
 |         nfc->status_changed(nf, errp); | 
 |     } | 
 | } | 
 |  | 
 | static char *netfilter_get_position(Object *obj, Error **errp) | 
 | { | 
 |     NetFilterState *nf = NETFILTER(obj); | 
 |  | 
 |     return g_strdup(nf->position); | 
 | } | 
 |  | 
 | static void netfilter_set_position(Object *obj, const char *str, Error **errp) | 
 | { | 
 |     NetFilterState *nf = NETFILTER(obj); | 
 |  | 
 |     nf->position = g_strdup(str); | 
 | } | 
 |  | 
 | static char *netfilter_get_insert(Object *obj, Error **errp) | 
 | { | 
 |     NetFilterState *nf = NETFILTER(obj); | 
 |  | 
 |     return nf->insert_before_flag ? g_strdup("before") : g_strdup("behind"); | 
 | } | 
 |  | 
 | static void netfilter_set_insert(Object *obj, const char *str, Error **errp) | 
 | { | 
 |     NetFilterState *nf = NETFILTER(obj); | 
 |  | 
 |     if (strcmp(str, "before") && strcmp(str, "behind")) { | 
 |         error_setg(errp, "Invalid value for netfilter insert, " | 
 |                          "should be 'before' or 'behind'"); | 
 |         return; | 
 |     } | 
 |  | 
 |     nf->insert_before_flag = !strcmp(str, "before"); | 
 | } | 
 |  | 
 | static void netfilter_init(Object *obj) | 
 | { | 
 |     NetFilterState *nf = NETFILTER(obj); | 
 |  | 
 |     nf->on = true; | 
 |     nf->insert_before_flag = false; | 
 |     nf->position = g_strdup("tail"); | 
 | } | 
 |  | 
 | static void netfilter_complete(UserCreatable *uc, Error **errp) | 
 | { | 
 |     NetFilterState *nf = NETFILTER(uc); | 
 |     NetFilterState *position = NULL; | 
 |     NetClientState *ncs[MAX_QUEUE_NUM]; | 
 |     NetFilterClass *nfc = NETFILTER_GET_CLASS(uc); | 
 |     int queues; | 
 |     Error *local_err = NULL; | 
 |  | 
 |     if (!nf->netdev_id) { | 
 |         error_setg(errp, "Parameter 'netdev' is required"); | 
 |         return; | 
 |     } | 
 |  | 
 |     queues = qemu_find_net_clients_except(nf->netdev_id, ncs, | 
 |                                           NET_CLIENT_DRIVER_NIC, | 
 |                                           MAX_QUEUE_NUM); | 
 |     if (queues < 1) { | 
 |         error_setg(errp, QERR_INVALID_PARAMETER_VALUE, "netdev", | 
 |                    "a network backend id"); | 
 |         return; | 
 |     } else if (queues > 1) { | 
 |         error_setg(errp, "multiqueue is not supported"); | 
 |         return; | 
 |     } | 
 |  | 
 |     if (get_vhost_net(ncs[0])) { | 
 |         error_setg(errp, "Vhost is not supported"); | 
 |         return; | 
 |     } | 
 |  | 
 |     if (strcmp(nf->position, "head") && strcmp(nf->position, "tail")) { | 
 |         Object *container; | 
 |         Object *obj; | 
 |         char *position_id; | 
 |  | 
 |         if (!g_str_has_prefix(nf->position, "id=")) { | 
 |             error_setg(errp, QERR_INVALID_PARAMETER_VALUE, "position", | 
 |                        "'head', 'tail' or 'id=<id>'"); | 
 |             return; | 
 |         } | 
 |  | 
 |         /* get the id from the string */ | 
 |         position_id = g_strndup(nf->position + 3, strlen(nf->position) - 3); | 
 |  | 
 |         /* Search for the position to insert before/behind */ | 
 |         container = object_get_objects_root(); | 
 |         obj = object_resolve_path_component(container, position_id); | 
 |         if (!obj) { | 
 |             error_setg(errp, "filter '%s' not found", position_id); | 
 |             g_free(position_id); | 
 |             return; | 
 |         } | 
 |  | 
 |         position = NETFILTER(obj); | 
 |  | 
 |         if (position->netdev != ncs[0]) { | 
 |             error_setg(errp, "filter '%s' belongs to a different netdev", | 
 |                         position_id); | 
 |             g_free(position_id); | 
 |             return; | 
 |         } | 
 |  | 
 |         g_free(position_id); | 
 |     } | 
 |  | 
 |     nf->netdev = ncs[0]; | 
 |  | 
 |     if (nfc->setup) { | 
 |         nfc->setup(nf, &local_err); | 
 |         if (local_err) { | 
 |             error_propagate(errp, local_err); | 
 |             return; | 
 |         } | 
 |     } | 
 |  | 
 |     if (position) { | 
 |         if (nf->insert_before_flag) { | 
 |             QTAILQ_INSERT_BEFORE(position, nf, next); | 
 |         } else { | 
 |             QTAILQ_INSERT_AFTER(&nf->netdev->filters, position, nf, next); | 
 |         } | 
 |     } else if (!strcmp(nf->position, "head")) { | 
 |         QTAILQ_INSERT_HEAD(&nf->netdev->filters, nf, next); | 
 |     } else if (!strcmp(nf->position, "tail")) { | 
 |         QTAILQ_INSERT_TAIL(&nf->netdev->filters, nf, next); | 
 |     } | 
 | } | 
 |  | 
 | static void netfilter_finalize(Object *obj) | 
 | { | 
 |     NetFilterState *nf = NETFILTER(obj); | 
 |     NetFilterClass *nfc = NETFILTER_GET_CLASS(obj); | 
 |  | 
 |     if (nfc->cleanup) { | 
 |         nfc->cleanup(nf); | 
 |     } | 
 |  | 
 |     if (nf->netdev && !QTAILQ_EMPTY(&nf->netdev->filters) && | 
 |         QTAILQ_IN_USE(nf, next)) { | 
 |         QTAILQ_REMOVE(&nf->netdev->filters, nf, next); | 
 |     } | 
 |     g_free(nf->netdev_id); | 
 |     g_free(nf->position); | 
 | } | 
 |  | 
 | static void default_handle_event(NetFilterState *nf, int event, Error **errp) | 
 | { | 
 |     switch (event) { | 
 |     case COLO_EVENT_CHECKPOINT: | 
 |         break; | 
 |     case COLO_EVENT_FAILOVER: | 
 |         object_property_set_str(OBJECT(nf), "status", "off", errp); | 
 |         break; | 
 |     default: | 
 |         break; | 
 |     } | 
 | } | 
 |  | 
 | static void netfilter_class_init(ObjectClass *oc, void *data) | 
 | { | 
 |     UserCreatableClass *ucc = USER_CREATABLE_CLASS(oc); | 
 |     NetFilterClass *nfc = NETFILTER_CLASS(oc); | 
 |  | 
 |     object_class_property_add_str(oc, "netdev", | 
 |                                   netfilter_get_netdev_id, netfilter_set_netdev_id); | 
 |     object_class_property_add_enum(oc, "queue", "NetFilterDirection", | 
 |                                    &NetFilterDirection_lookup, | 
 |                                    netfilter_get_direction, netfilter_set_direction); | 
 |     object_class_property_add_str(oc, "status", | 
 |                                   netfilter_get_status, netfilter_set_status); | 
 |     object_class_property_add_str(oc, "position", | 
 |                                   netfilter_get_position, netfilter_set_position); | 
 |     object_class_property_add_str(oc, "insert", | 
 |                                   netfilter_get_insert, netfilter_set_insert); | 
 |  | 
 |     ucc->complete = netfilter_complete; | 
 |     nfc->handle_event = default_handle_event; | 
 | } | 
 |  | 
 | static const TypeInfo netfilter_info = { | 
 |     .name = TYPE_NETFILTER, | 
 |     .parent = TYPE_OBJECT, | 
 |     .abstract = true, | 
 |     .class_size = sizeof(NetFilterClass), | 
 |     .class_init = netfilter_class_init, | 
 |     .instance_size = sizeof(NetFilterState), | 
 |     .instance_init = netfilter_init, | 
 |     .instance_finalize = netfilter_finalize, | 
 |     .interfaces = (InterfaceInfo[]) { | 
 |         { TYPE_USER_CREATABLE }, | 
 |         { } | 
 |     } | 
 | }; | 
 |  | 
 | static void register_types(void) | 
 | { | 
 |     type_register_static(&netfilter_info); | 
 | } | 
 |  | 
 | type_init(register_types); |