| /* |
| * vmnet-bridged.m |
| * |
| * Copyright(c) 2022 Vladislav Yaroshchuk <vladislav.yaroshchuk@jetbrains.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/qapi-types-net.h" |
| #include "qapi/error.h" |
| #include "clients.h" |
| #include "vmnet_int.h" |
| |
| #include <vmnet/vmnet.h> |
| |
| |
| static bool validate_ifname(const char *ifname) |
| { |
| xpc_object_t shared_if_list = vmnet_copy_shared_interface_list(); |
| bool match = false; |
| if (!xpc_array_get_count(shared_if_list)) { |
| goto done; |
| } |
| |
| match = !xpc_array_apply( |
| shared_if_list, |
| ^bool(size_t index, xpc_object_t value) { |
| return strcmp(xpc_string_get_string_ptr(value), ifname) != 0; |
| }); |
| |
| done: |
| xpc_release(shared_if_list); |
| return match; |
| } |
| |
| |
| static char* get_valid_ifnames(void) |
| { |
| xpc_object_t shared_if_list = vmnet_copy_shared_interface_list(); |
| __block char *if_list = NULL; |
| __block char *if_list_prev = NULL; |
| |
| if (!xpc_array_get_count(shared_if_list)) { |
| goto done; |
| } |
| |
| xpc_array_apply( |
| shared_if_list, |
| ^bool(size_t index, xpc_object_t value) { |
| /* build list of strings like "en0 en1 en2 " */ |
| if_list = g_strconcat(xpc_string_get_string_ptr(value), |
| " ", |
| if_list_prev, |
| NULL); |
| g_free(if_list_prev); |
| if_list_prev = if_list; |
| return true; |
| }); |
| |
| done: |
| xpc_release(shared_if_list); |
| return if_list; |
| } |
| |
| |
| static bool validate_options(const Netdev *netdev, Error **errp) |
| { |
| const NetdevVmnetBridgedOptions *options = &(netdev->u.vmnet_bridged); |
| char* if_list; |
| |
| if (!validate_ifname(options->ifname)) { |
| if_list = get_valid_ifnames(); |
| if (if_list) { |
| error_setg(errp, |
| "unsupported ifname '%s', expected one of [ %s]", |
| options->ifname, |
| if_list); |
| g_free(if_list); |
| } else { |
| error_setg(errp, |
| "unsupported ifname '%s', no supported " |
| "interfaces available", |
| options->ifname); |
| } |
| return false; |
| } |
| |
| #if !defined(MAC_OS_VERSION_11_0) || \ |
| MAC_OS_X_VERSION_MIN_REQUIRED < MAC_OS_VERSION_11_0 |
| if (options->has_isolated) { |
| error_setg(errp, |
| "vmnet-bridged.isolated feature is " |
| "unavailable: outdated vmnet.framework API"); |
| return false; |
| } |
| #endif |
| return true; |
| } |
| |
| |
| static xpc_object_t build_if_desc(const Netdev *netdev) |
| { |
| const NetdevVmnetBridgedOptions *options = &(netdev->u.vmnet_bridged); |
| xpc_object_t if_desc = xpc_dictionary_create(NULL, NULL, 0); |
| |
| xpc_dictionary_set_uint64(if_desc, |
| vmnet_operation_mode_key, |
| VMNET_BRIDGED_MODE |
| ); |
| |
| xpc_dictionary_set_string(if_desc, |
| vmnet_shared_interface_name_key, |
| options->ifname); |
| |
| #if defined(MAC_OS_VERSION_11_0) && \ |
| MAC_OS_X_VERSION_MIN_REQUIRED >= MAC_OS_VERSION_11_0 |
| xpc_dictionary_set_bool(if_desc, |
| vmnet_enable_isolation_key, |
| options->isolated); |
| #endif |
| return if_desc; |
| } |
| |
| |
| static NetClientInfo net_vmnet_bridged_info = { |
| .type = NET_CLIENT_DRIVER_VMNET_BRIDGED, |
| .size = sizeof(VmnetState), |
| .receive = vmnet_receive_common, |
| .cleanup = vmnet_cleanup_common, |
| }; |
| |
| |
| int net_init_vmnet_bridged(const Netdev *netdev, const char *name, |
| NetClientState *peer, Error **errp) |
| { |
| NetClientState *nc = qemu_new_net_client(&net_vmnet_bridged_info, |
| peer, "vmnet-bridged", name); |
| xpc_object_t if_desc; |
| int result = -1; |
| |
| if (!validate_options(netdev, errp)) { |
| return result; |
| } |
| |
| if_desc = build_if_desc(netdev); |
| result = vmnet_if_create(nc, if_desc, errp); |
| xpc_release(if_desc); |
| return result; |
| } |