qemu-devel
[Top][All Lists]
Advanced

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

Re: [PATCH v2 4/4] hw: hyperv: Initial commit for Synthetic Debugging de


From: Emanuele Giuseppe Esposito
Subject: Re: [PATCH v2 4/4] hw: hyperv: Initial commit for Synthetic Debugging device
Date: Thu, 24 Feb 2022 17:36:30 +0100
User-agent: Mozilla/5.0 (X11; Linux x86_64; rv:91.0) Gecko/20100101 Thunderbird/91.2.0


On 16/02/2022 11:25, Jon Doron wrote:
> Signed-off-by: Jon Doron <arilou@gmail.com>
> ---
>  hw/hyperv/Kconfig     |   5 +
>  hw/hyperv/meson.build |   1 +
>  hw/hyperv/syndbg.c    | 402 ++++++++++++++++++++++++++++++++++++++++++
>  3 files changed, 408 insertions(+)
>  create mode 100644 hw/hyperv/syndbg.c
> 
> diff --git a/hw/hyperv/Kconfig b/hw/hyperv/Kconfig
> index 3fbfe41c9e..fcf65903bd 100644
> --- a/hw/hyperv/Kconfig
> +++ b/hw/hyperv/Kconfig
> @@ -11,3 +11,8 @@ config VMBUS
>      bool
>      default y
>      depends on HYPERV
> +
> +config SYNDBG
> +    bool
> +    default y
> +    depends on VMBUS
> diff --git a/hw/hyperv/meson.build b/hw/hyperv/meson.build
> index 1367e2994f..b43f119ea5 100644
> --- a/hw/hyperv/meson.build
> +++ b/hw/hyperv/meson.build
> @@ -1,3 +1,4 @@
>  specific_ss.add(when: 'CONFIG_HYPERV', if_true: files('hyperv.c'))
>  specific_ss.add(when: 'CONFIG_HYPERV_TESTDEV', if_true: 
> files('hyperv_testdev.c'))
>  specific_ss.add(when: 'CONFIG_VMBUS', if_true: files('vmbus.c'))
> +specific_ss.add(when: 'CONFIG_SYNDBG', if_true: files('syndbg.c'))
> diff --git a/hw/hyperv/syndbg.c b/hw/hyperv/syndbg.c
> new file mode 100644
> index 0000000000..8816bc4082
> --- /dev/null
> +++ b/hw/hyperv/syndbg.c
> @@ -0,0 +1,402 @@
> +/*
> + * QEMU Hyper-V Synthetic Debugging device
> + *
> + * 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/ctype.h"
> +#include "qemu/osdep.h"
> +#include "qemu/error-report.h"
> +#include "qemu/main-loop.h"
> +#include "qemu/sockets.h"
> +#include "qemu-common.h"
> +#include "qapi/error.h"
> +#include "migration/vmstate.h"
> +#include "hw/qdev-properties.h"
> +#include "hw/loader.h"
> +#include "cpu.h"
> +#include "hw/hyperv/hyperv.h"
> +#include "hw/hyperv/vmbus-bridge.h"
> +#include "hw/hyperv/hyperv-proto.h"
> +#include "net/net.h"
> +#include "net/eth.h"
> +#include "net/checksum.h"
> +#include "trace.h"
> +
> +#define TYPE_HV_SYNDBG       "hv-syndbg"
> +
> +typedef struct HvSynDbg {
> +    DeviceState parent_obj;
> +
> +    char *host_ip;
> +    uint16_t host_port;
> +    bool use_hcalls;
> +
> +    uint32_t target_ip;
> +    struct sockaddr_in servaddr;
> +    int socket;
> +    bool has_data_pending;
> +    uint64_t pending_page_gpa;
> +} HvSynDbg;
> +
> +#define HVSYNDBG(obj) OBJECT_CHECK(HvSynDbg, (obj), TYPE_HV_SYNDBG)
> +
> +/* returns NULL unless there is exactly one HV Synth debug device */
> +static HvSynDbg *hv_syndbg_find(void)
> +{
> +    /* Returns NULL unless there is exactly one hvsd device */
> +    return HVSYNDBG(object_resolve_path_type("", TYPE_HV_SYNDBG, NULL));
> +}
> +
> +static void set_pending_state(HvSynDbg *syndbg, bool has_pending)
> +{
> +    hwaddr out_len;
> +    void *out_data;
> +
> +    syndbg->has_data_pending = has_pending;
> +
> +    if (!syndbg->pending_page_gpa) {
> +        return;
> +    }
> +
> +    out_len = 1;
> +    out_data = cpu_physical_memory_map(syndbg->pending_page_gpa, &out_len, 
> 1);
> +    if (out_data) {
> +        *(uint8_t *)out_data = !!has_pending;
> +        cpu_physical_memory_unmap(out_data, out_len, 1, out_len);
> +    }
> +}
> +
> +static bool get_udb_pkt_data(void *p, uint32_t len, uint32_t *data_ofs,
> +                             uint32_t *src_ip)
> +{
> +    uint32_t offset, curr_len = len;
> +
> +    if (curr_len < sizeof(struct eth_header) ||
> +        (be16_to_cpu(PKT_GET_ETH_HDR(p)->h_proto) != ETH_P_IP)) {
> +        return false;
> +    }
> +    offset = sizeof(struct eth_header);
> +    curr_len -= sizeof(struct eth_header);
> +
> +    if (curr_len < sizeof(struct ip_header) ||
> +        PKT_GET_IP_HDR(p)->ip_p != IP_PROTO_UDP) {
> +        return false;
> +    }
> +    offset += PKT_GET_IP_HDR_LEN(p);
> +    curr_len -= PKT_GET_IP_HDR_LEN(p);
> +
> +    if (curr_len < sizeof(struct udp_header)) {
> +        return false;
> +    }
> +
> +    offset += sizeof(struct udp_header);
> +    *data_ofs = offset;
> +    *src_ip = PKT_GET_IP_HDR(p)->ip_src;
> +    return true;
> +}
> +
> +static uint16_t handle_send_msg(HvSynDbg *syndbg, uint64_t ingpa,
> +                                uint32_t count, bool is_raw,
> +                                uint32_t *pending_count)
> +{
> +    uint16_t ret;
> +    hwaddr data_len;
> +    void *debug_data = NULL;
> +    uint32_t udp_data_ofs = 0;
> +    const void *pkt_data;
> +    int sent_count;
> +
> +    data_len = count;
> +    debug_data = cpu_physical_memory_map(ingpa, &data_len, 0);
> +    if (!debug_data || data_len < count) {
> +        ret = HV_STATUS_INSUFFICIENT_MEMORY;
> +        goto cleanup;
> +    }
> +
> +    if (is_raw &&
> +        !get_udb_pkt_data(debug_data, count, &udp_data_ofs,
> +                          &syndbg->target_ip)) {
> +        ret = HV_STATUS_SUCCESS;
> +        goto cleanup;
> +    }
> +
> +    pkt_data = (const void *)((uintptr_t)debug_data + udp_data_ofs);
> +    sent_count = qemu_sendto(syndbg->socket, pkt_data, count - udp_data_ofs,
> +                             MSG_NOSIGNAL, NULL, 0);
> +    if (sent_count == -1) {
> +        ret = HV_STATUS_INSUFFICIENT_MEMORY;
> +        goto cleanup;
> +    }
> +
> +    *pending_count = count - (sent_count + udp_data_ofs);
> +    ret = HV_STATUS_SUCCESS;
> +cleanup:
> +    if (debug_data) {
> +        cpu_physical_memory_unmap(debug_data, count, 0, data_len);
> +    }
> +
> +    return ret;
> +}
> +
> +#define UDP_PKT_HEADER_SIZE \
> +    (sizeof(struct eth_header) + sizeof(struct ip_header) +\
> +     sizeof(struct udp_header))
> +
> +static bool create_udp_pkt(HvSynDbg *syndbg, void *pkt, uint32_t pkt_len,
> +                           void *udp_data, uint32_t udp_data_len)
> +{
> +    struct udp_header *udp_part;
> +
> +    if (pkt_len < (UDP_PKT_HEADER_SIZE + udp_data_len)) {
> +        return false;
> +    }
> +
> +    /* Setup the eth */
> +    memset(&PKT_GET_ETH_HDR(pkt)->h_source, 0, ETH_ALEN);
> +    memset(&PKT_GET_ETH_HDR(pkt)->h_dest, 0, ETH_ALEN);
> +    PKT_GET_ETH_HDR(pkt)->h_proto = cpu_to_be16(ETH_P_IP);
> +
> +    /* Setup the ip */
> +    PKT_GET_IP_HDR(pkt)->ip_ver_len =
> +        (4 << 4) | (sizeof(struct ip_header) >> 2);
> +    PKT_GET_IP_HDR(pkt)->ip_tos = 0;
> +    PKT_GET_IP_HDR(pkt)->ip_id = 0;
> +    PKT_GET_IP_HDR(pkt)->ip_off = 0;
> +    PKT_GET_IP_HDR(pkt)->ip_ttl = 64; /* IPDEFTTL */
> +    PKT_GET_IP_HDR(pkt)->ip_p = IP_PROTO_UDP;
> +    PKT_GET_IP_HDR(pkt)->ip_src = syndbg->servaddr.sin_addr.s_addr;
> +    PKT_GET_IP_HDR(pkt)->ip_dst = syndbg->target_ip;
> +    PKT_GET_IP_HDR(pkt)->ip_len =
> +        cpu_to_be16(sizeof(struct ip_header) + sizeof(struct udp_header) +
> +                    udp_data_len);
> +    eth_fix_ip4_checksum(PKT_GET_IP_HDR(pkt), PKT_GET_IP_HDR_LEN(pkt));
> +
> +    udp_part = (struct udp_header *)((uintptr_t)pkt +
> +                                     sizeof(struct eth_header) +
> +                                     PKT_GET_IP_HDR_LEN(pkt));
> +    udp_part->uh_sport = syndbg->servaddr.sin_port;
> +    udp_part->uh_dport = syndbg->servaddr.sin_port;
> +    udp_part->uh_ulen = cpu_to_be16(sizeof(struct udp_header) + 
> udp_data_len);
> +    memcpy(udp_part + 1, udp_data, udp_data_len);
> +    net_checksum_calculate(pkt, UDP_PKT_HEADER_SIZE + udp_data_len, 
> CSUM_ALL);
> +    return true;
> +}
> +
> +static uint16_t handle_recv_msg(HvSynDbg *syndbg, uint64_t outgpa,
> +                                uint32_t count, bool is_raw, uint32_t 
> options,
> +                                uint64_t timeout, uint32_t *retrieved_count)
> +{
> +    uint16_t ret;
> +    uint8_t data_buf[TARGET_PAGE_SIZE - UDP_PKT_HEADER_SIZE];
> +    hwaddr out_len;
> +    void *out_data;
> +    ssize_t recv_byte_count;
> +
> +    /* TODO: Handle options and timeout */
> +    (void)options;
> +    (void)timeout;
> +
> +    if (!syndbg->has_data_pending) {
> +        recv_byte_count = 0;
> +    } else {
> +        recv_byte_count = qemu_recv(syndbg->socket, data_buf,
> +                                    MIN(sizeof(data_buf), count), 
> MSG_WAITALL);
> +        if (recv_byte_count == -1) {
> +            return HV_STATUS_INVALID_PARAMETER;
> +        }
> +    }
> +
> +    if (!recv_byte_count) {
> +        *retrieved_count = 0;
> +        return HV_STATUS_NO_DATA;
> +    }
> +
> +    set_pending_state(syndbg, false);
> +
> +    out_len = recv_byte_count;
> +    if (is_raw) {
> +        out_len += UDP_PKT_HEADER_SIZE;
> +    }
> +    out_data = cpu_physical_memory_map(outgpa, &out_len, 1);
> +    if (!out_data) {
> +        return HV_STATUS_INSUFFICIENT_MEMORY;
> +    }
> +
> +    if (is_raw &&
> +        !create_udp_pkt(syndbg, out_data,
> +                        recv_byte_count + UDP_PKT_HEADER_SIZE,
> +                        data_buf, recv_byte_count)) {
> +        ret = HV_STATUS_INSUFFICIENT_MEMORY;
> +        goto cleanup_out_data;
> +    } else if (!is_raw) {
> +        memcpy(out_data, data_buf, recv_byte_count);
> +    }
> +
> +    *retrieved_count = recv_byte_count;
> +    if (is_raw) {
> +        *retrieved_count += UDP_PKT_HEADER_SIZE;
> +    }
> +    ret = HV_STATUS_SUCCESS;
> +
> +cleanup_out_data:
> +    cpu_physical_memory_unmap(out_data, out_len, 1, out_len);
> +    return ret;
> +}
> +
> +static uint16_t hv_syndbg_handler(void *context, HvSynDbgMsg *msg)
> +{
> +    HvSynDbg *syndbg = context;
> +    uint16_t ret = HV_STATUS_INVALID_HYPERCALL_CODE;
> +
> +    switch (msg->type) {
> +    case HV_SYNDBG_MSG_CONNECTION_INFO:
> +        msg->u.connection_info.host_ip =
> +            ntohl(syndbg->servaddr.sin_addr.s_addr);
> +        msg->u.connection_info.host_port =
> +            ntohs(syndbg->servaddr.sin_port);
> +        ret = HV_STATUS_SUCCESS;
> +        break;
> +    case HV_SYNDBG_MSG_SEND:
> +        ret = handle_send_msg(syndbg, msg->u.send.buf_gpa, msg->u.send.count,
> +                              msg->u.send.is_raw, 
> &msg->u.send.pending_count);
> +        break;
> +    case HV_SYNDBG_MSG_RECV:
> +        ret = handle_recv_msg(syndbg, msg->u.recv.buf_gpa, msg->u.recv.count,
> +                              msg->u.recv.is_raw, msg->u.recv.options,
> +                              msg->u.recv.timeout,
> +                              &msg->u.recv.retrieved_count);
> +        break;
> +    case HV_SYNDBG_MSG_SET_PENDING_PAGE:
> +        syndbg->pending_page_gpa = msg->u.pending_page.buf_gpa;
> +        ret = HV_STATUS_SUCCESS;
> +        break;
> +    case HV_SYNDBG_MSG_QUERY_OPTIONS:
> +        msg->u.query_options.options = 0;
> +        if (syndbg->use_hcalls) {
> +            msg->u.query_options.options = HV_X64_SYNDBG_OPTION_USE_HCALLS;
> +        }
> +        ret = HV_STATUS_SUCCESS;
> +        break;
> +    default:
> +        break;
> +    }
> +
> +    return ret;
> +}
> +
> +static void hv_syndbg_recv_event(void *opaque)
> +{
> +    HvSynDbg *syndbg = opaque;
> +    struct timeval tv;
> +    fd_set rfds;
> +
> +    tv.tv_sec = 0;
> +    tv.tv_usec = 0;
> +    FD_ZERO(&rfds);
> +    FD_SET(syndbg->socket, &rfds);
> +    if (select(syndbg->socket + 1, &rfds, NULL, NULL, &tv) > 0) {
> +        set_pending_state(syndbg, true);
> +    }
> +}
> +
> +static void hv_syndbg_realize(DeviceState *dev, Error **errp)
> +{
> +    HvSynDbg *syndbg = HVSYNDBG(dev);
> +
> +    if (!hv_syndbg_find()) {
> +        error_setg(errp, "at most one %s device is permitted", 
> TYPE_HV_SYNDBG);
> +        return;
> +    }
> +
> +    if (!vmbus_bridge_find()) {
> +        error_setg(errp, "%s device requires vmbus-bridge device",
> +                   TYPE_HV_SYNDBG);
> +        return;
> +    }
> +
> +    /* Parse and host_ip */
> +    if (qemu_isdigit(syndbg->host_ip[0])) {
> +        syndbg->servaddr.sin_addr.s_addr = inet_addr(syndbg->host_ip);
> +    } else {
> +        struct hostent *he = gethostbyname(syndbg->host_ip);
> +        if (!he) {
> +            error_setg(errp, "%s failed to resolve host name %s",
> +                       TYPE_HV_SYNDBG, syndbg->host_ip);
> +            return;
> +        }
> +        syndbg->servaddr.sin_addr = *(struct in_addr *)he->h_addr;
> +    }
> +
> +    syndbg->socket = socket(AF_INET, SOCK_DGRAM, 0);
> +    if (syndbg->socket < 0) {
> +        error_setg(errp, "%s failed to create socket", TYPE_HV_SYNDBG);
> +        return;
> +    }
> +
> +    qemu_set_nonblock(syndbg->socket);
> +
> +    syndbg->servaddr.sin_port = htons(syndbg->host_port);
> +    syndbg->servaddr.sin_family = AF_INET;
> +    if (connect(syndbg->socket, (struct sockaddr *)&syndbg->servaddr,
> +                sizeof(syndbg->servaddr)) < 0) {
> +        closesocket(syndbg->socket);
> +        error_setg(errp, "%s failed to connect to socket", TYPE_HV_SYNDBG);
> +        return;
> +    }
> +
> +    syndbg->pending_page_gpa = 0;
> +    syndbg->has_data_pending = false;
> +    hyperv_set_syndbg_handler(hv_syndbg_handler, syndbg);
> +    qemu_set_fd_handler(syndbg->socket, hv_syndbg_recv_event, NULL, syndbg);
> +}
> +
> +static void hv_syndbg_unrealize(DeviceState *dev)
> +{
> +    HvSynDbg *syndbg = HVSYNDBG(dev);
> +
> +    if (syndbg->socket > 0) {
> +        qemu_set_fd_handler(syndbg->socket, NULL, NULL, NULL);
> +        closesocket(syndbg->socket);
> +    }
> +}
> +
> +static const VMStateDescription vmstate_hv_syndbg = {
> +    .name = TYPE_HV_SYNDBG,
> +    .unmigratable = 1,
> +};
> +
> +static Property hv_syndbg_properties[] = {
> +    DEFINE_PROP_STRING("host_ip", HvSynDbg, host_ip),
> +    DEFINE_PROP_UINT16("host_port", HvSynDbg, host_port, 50000),
> +    DEFINE_PROP_BOOL("use_hcalls", HvSynDbg, use_hcalls, false),
> +    DEFINE_PROP_END_OF_LIST(),
> +};
> +
> +static void hv_syndbg_class_init(ObjectClass *klass, void *data)
> +{
> +    DeviceClass *dc = DEVICE_CLASS(klass);
> +
> +    device_class_set_props(dc, hv_syndbg_properties);
> +    dc->fw_name = TYPE_HV_SYNDBG;
> +    dc->vmsd = &vmstate_hv_syndbg;
> +    dc->realize = hv_syndbg_realize;
> +    dc->unrealize = hv_syndbg_unrealize;
> +    dc->user_creatable = true;
> +    set_bit(DEVICE_CATEGORY_MISC, dc->categories);
> +}
> +
> +static const TypeInfo hv_syndbg_type_info = {
> +    .name = TYPE_HV_SYNDBG,
> +    .parent = TYPE_DEVICE,
> +    .instance_size = sizeof(HvSynDbg),
> +    .class_init = hv_syndbg_class_init,
> +};
> +
> +static void hv_syndbg_register_types(void)
> +{
> +    type_register_static(&hv_syndbg_type_info);
> +}
> +
> +type_init(hv_syndbg_register_types)
> 

Reviewed-by: Emanuele Giuseppe Esposito <eesposit@redhat.com>




reply via email to

[Prev in Thread] Current Thread [Next in Thread]