we can get the network interface statistics inside a virtual machine by guest-network-get-interfaces command. it is very userful for us to monitor and analyze network traff.
Signed-off-by: ZhiPeng Lu <lu.zhip...@zte.com.cn> Signed-off-by: DanielP.Berrange <berra...@redhat.com> Reviewed-by: EricBlake <ebl...@redhat.com> --- qga/commands-posix.c | 73 +++++++++++++++++++++++++++++++++++++++++++++++++++- qga/qapi-schema.json | 28 +++++++++++++++++++- 2 files changed, 99 insertions(+), 2 deletions(-) diff --git a/qga/commands-posix.c b/qga/commands-posix.c index 915df9e..0c48707 100644 --- a/qga/commands-posix.c +++ b/qga/commands-posix.c @@ -1638,6 +1638,65 @@ guest_find_interface(GuestNetworkInterfaceList *head, return head; } + static int guest_get_network_stats(const char *path, + GuestNetworkInterfaceStat *stats) +{ + int path_len; + char const *devinfo = "/proc/net/dev"; + FILE *fp; + char *line = NULL, *colon; + size_t n; + fp = fopen(devinfo, "r"); + if (!fp) { + return -1; + } + path_len = strlen(path); + while (getline(&line, &n, fp) != -1) { + long long dummy; + long long rx_bytes; + long long rx_packets; + long long rx_errs; + long long rx_drop; + long long tx_bytes; + long long tx_packets; + long long tx_errs; + long long tx_drop; + + /* The line looks like: + *" eth0:..." + *Split it at the colon. + */ + colon = strchr(line, ':'); + if (!colon) { + continue; + } + *colon = '\0'; + if (colon - path_len >= line && strcmp(colon - path_len, path) == 0) { + if (sscanf(colon + 1, + "%lld %lld %lld %lld %lld %lld %lld %lld %lld %lld %lld %lld %lld %lld %lld %lld", + &rx_bytes, &rx_packets, &rx_errs, &rx_drop, + &dummy, &dummy, &dummy, &dummy, + &tx_bytes, &tx_packets, &tx_errs, &tx_drop, + &dummy, &dummy, &dummy, &dummy) != 16) { + continue; + } + stats->rx_bytes = rx_bytes; + stats->rx_packets = rx_packets; + stats->rx_errs = rx_errs; + stats->rx_drop = rx_drop; + stats->tx_bytes = tx_bytes; + stats->tx_packets = tx_packets; + stats->tx_errs = tx_errs; + stats->tx_drop = tx_drop; + fclose(fp); + return 0; + } + } + fclose(fp); + g_debug("/proc/net/dev: Interface not found"); + return -1; +} + /* * Build information about guest interfaces */ @@ -1654,6 +1713,7 @@ GuestNetworkInterfaceList *qmp_guest_network_get_interfaces(Error **errp) for (ifa = ifap; ifa; ifa = ifa->ifa_next) { GuestNetworkInterfaceList *info; GuestIpAddressList **address_list = NULL, *address_item = NULL; + GuestNetworkInterfaceStatList *interface_stat_item = NULL; char addr4[INET_ADDRSTRLEN]; char addr6[INET6_ADDRSTRLEN]; int sock; @@ -1770,9 +1830,20 @@ GuestNetworkInterfaceList *qmp_guest_network_get_interfaces(Error **errp) } else { (*address_list)->next = address_item; } - info->value->has_ip_addresses = true; + if (!info->value->has_interface_statics) { + interface_stat_item = g_malloc0(sizeof(*interface_stat_item)); + interface_stat_item->value = g_malloc0( + sizeof(*interface_stat_item->value)); + if (guest_get_network_stats(info->value->name, + interface_stat_item->value) == -1) { + error_setg_errno(errp, errno, "guest_get_network_stats failed"); + goto error; + } + info->value->interface_statics = interface_stat_item; + } + info->value->has_interface_statics = true; } diff --git a/qga/qapi-schema.json b/qga/qapi-schema.json index a02dbf2..44bcced 100644 --- a/qga/qapi-schema.json +++ b/qga/qapi-schema.json @@ -635,6 +635,31 @@ 'prefix': 'int'} } ## +# @GuestNetworkInterfaceStat: +# +# @rx-bytes: received bytes of interface +# @rx-packets: received packets of interface +# @rx-errs: received error packets of interface +# @rx-drop: received drop packets of interface +# +# @tx-bytes: send bytes of interface +# @tx-packets: send packets of interface +# @tx-errs: send error packets of interface +# @tx-drop: send drop packets of interface +# Since: 2.10 +## +{ 'struct': 'GuestNetworkInterfaceStat', + 'data': {'rx-bytes': 'uint64', + 'rx-packets': 'uint64', + 'rx-errs': 'uint64', + 'rx-drop': 'uint64', + 'tx-bytes': 'uint64', + 'tx-packets': 'uint64', + 'tx-errs': 'uint64', + 'tx-drop': 'uint64' + } } + +## # @GuestNetworkInterface: # # @name: The name of interface for which info are being delivered @@ -648,7 +673,8 @@ { 'struct': 'GuestNetworkInterface', 'data': {'name': 'str', '*hardware-address': 'str', - '*ip-addresses': ['GuestIpAddress'] } } + '*ip-addresses': ['GuestIpAddress'], + '*interface-statics': ['GuestNetworkInterfaceStat'] } } ## # @guest-network-get-interfaces: -- 1.8.3.1