1d8701185SJon Doron /* 2d8701185SJon Doron * QEMU Hyper-V Synthetic Debugging device 3d8701185SJon Doron * 4d8701185SJon Doron * This work is licensed under the terms of the GNU GPL, version 2 or later. 5d8701185SJon Doron * See the COPYING file in the top-level directory. 6d8701185SJon Doron */ 7d8701185SJon Doron 8d8701185SJon Doron #include "qemu/osdep.h" 92ca10faeSMarkus Armbruster #include "qemu/ctype.h" 10d8701185SJon Doron #include "qemu/error-report.h" 11d8701185SJon Doron #include "qemu/main-loop.h" 12d8701185SJon Doron #include "qemu/sockets.h" 13d8701185SJon Doron #include "qapi/error.h" 14d8701185SJon Doron #include "migration/vmstate.h" 15d8701185SJon Doron #include "hw/qdev-properties.h" 16d8701185SJon Doron #include "hw/loader.h" 17d8701185SJon Doron #include "cpu.h" 189c2ff9cdSPierrick Bouvier #include "exec/target_page.h" 19d8701185SJon Doron #include "hw/hyperv/hyperv.h" 20d8701185SJon Doron #include "hw/hyperv/vmbus-bridge.h" 21d8701185SJon Doron #include "hw/hyperv/hyperv-proto.h" 22d8701185SJon Doron #include "net/net.h" 23d8701185SJon Doron #include "net/eth.h" 24d8701185SJon Doron #include "net/checksum.h" 25d8701185SJon Doron #include "trace.h" 26d8701185SJon Doron 27d8701185SJon Doron #define TYPE_HV_SYNDBG "hv-syndbg" 28d8701185SJon Doron 29d8701185SJon Doron typedef struct HvSynDbg { 30d8701185SJon Doron DeviceState parent_obj; 31d8701185SJon Doron 32d8701185SJon Doron char *host_ip; 33d8701185SJon Doron uint16_t host_port; 34d8701185SJon Doron bool use_hcalls; 35d8701185SJon Doron 36d8701185SJon Doron uint32_t target_ip; 37d8701185SJon Doron struct sockaddr_in servaddr; 38d8701185SJon Doron int socket; 39d8701185SJon Doron bool has_data_pending; 40d8701185SJon Doron uint64_t pending_page_gpa; 41d8701185SJon Doron } HvSynDbg; 42d8701185SJon Doron 43d8701185SJon Doron #define HVSYNDBG(obj) OBJECT_CHECK(HvSynDbg, (obj), TYPE_HV_SYNDBG) 44d8701185SJon Doron 45d8701185SJon Doron /* returns NULL unless there is exactly one HV Synth debug device */ 46d8701185SJon Doron static HvSynDbg *hv_syndbg_find(void) 47d8701185SJon Doron { 48d8701185SJon Doron /* Returns NULL unless there is exactly one hvsd device */ 49d8701185SJon Doron return HVSYNDBG(object_resolve_path_type("", TYPE_HV_SYNDBG, NULL)); 50d8701185SJon Doron } 51d8701185SJon Doron 52d8701185SJon Doron static void set_pending_state(HvSynDbg *syndbg, bool has_pending) 53d8701185SJon Doron { 54d8701185SJon Doron hwaddr out_len; 55d8701185SJon Doron void *out_data; 56d8701185SJon Doron 57d8701185SJon Doron syndbg->has_data_pending = has_pending; 58d8701185SJon Doron 59d8701185SJon Doron if (!syndbg->pending_page_gpa) { 60d8701185SJon Doron return; 61d8701185SJon Doron } 62d8701185SJon Doron 63d8701185SJon Doron out_len = 1; 64d8701185SJon Doron out_data = cpu_physical_memory_map(syndbg->pending_page_gpa, &out_len, 1); 65d8701185SJon Doron if (out_data) { 66d8701185SJon Doron *(uint8_t *)out_data = !!has_pending; 67d8701185SJon Doron cpu_physical_memory_unmap(out_data, out_len, 1, out_len); 68d8701185SJon Doron } 69d8701185SJon Doron } 70d8701185SJon Doron 71d8701185SJon Doron static bool get_udb_pkt_data(void *p, uint32_t len, uint32_t *data_ofs, 72d8701185SJon Doron uint32_t *src_ip) 73d8701185SJon Doron { 74d8701185SJon Doron uint32_t offset, curr_len = len; 75d8701185SJon Doron 76d8701185SJon Doron if (curr_len < sizeof(struct eth_header) || 77d8701185SJon Doron (be16_to_cpu(PKT_GET_ETH_HDR(p)->h_proto) != ETH_P_IP)) { 78d8701185SJon Doron return false; 79d8701185SJon Doron } 80d8701185SJon Doron offset = sizeof(struct eth_header); 81d8701185SJon Doron curr_len -= sizeof(struct eth_header); 82d8701185SJon Doron 83d8701185SJon Doron if (curr_len < sizeof(struct ip_header) || 84d8701185SJon Doron PKT_GET_IP_HDR(p)->ip_p != IP_PROTO_UDP) { 85d8701185SJon Doron return false; 86d8701185SJon Doron } 87d8701185SJon Doron offset += PKT_GET_IP_HDR_LEN(p); 88d8701185SJon Doron curr_len -= PKT_GET_IP_HDR_LEN(p); 89d8701185SJon Doron 90d8701185SJon Doron if (curr_len < sizeof(struct udp_header)) { 91d8701185SJon Doron return false; 92d8701185SJon Doron } 93d8701185SJon Doron 94d8701185SJon Doron offset += sizeof(struct udp_header); 95d8701185SJon Doron *data_ofs = offset; 96d8701185SJon Doron *src_ip = PKT_GET_IP_HDR(p)->ip_src; 97d8701185SJon Doron return true; 98d8701185SJon Doron } 99d8701185SJon Doron 100d8701185SJon Doron static uint16_t handle_send_msg(HvSynDbg *syndbg, uint64_t ingpa, 101d8701185SJon Doron uint32_t count, bool is_raw, 102d8701185SJon Doron uint32_t *pending_count) 103d8701185SJon Doron { 104d8701185SJon Doron uint16_t ret; 105d8701185SJon Doron hwaddr data_len; 106d8701185SJon Doron void *debug_data = NULL; 107d8701185SJon Doron uint32_t udp_data_ofs = 0; 108d8701185SJon Doron const void *pkt_data; 109d8701185SJon Doron int sent_count; 110d8701185SJon Doron 111d8701185SJon Doron data_len = count; 112d8701185SJon Doron debug_data = cpu_physical_memory_map(ingpa, &data_len, 0); 113d8701185SJon Doron if (!debug_data || data_len < count) { 114d8701185SJon Doron ret = HV_STATUS_INSUFFICIENT_MEMORY; 115d8701185SJon Doron goto cleanup; 116d8701185SJon Doron } 117d8701185SJon Doron 118d8701185SJon Doron if (is_raw && 119d8701185SJon Doron !get_udb_pkt_data(debug_data, count, &udp_data_ofs, 120d8701185SJon Doron &syndbg->target_ip)) { 121d8701185SJon Doron ret = HV_STATUS_SUCCESS; 122d8701185SJon Doron goto cleanup; 123d8701185SJon Doron } 124d8701185SJon Doron 125d8701185SJon Doron pkt_data = (const void *)((uintptr_t)debug_data + udp_data_ofs); 126d8701185SJon Doron sent_count = sendto(syndbg->socket, pkt_data, count - udp_data_ofs, 127d8701185SJon Doron MSG_NOSIGNAL, NULL, 0); 128d8701185SJon Doron if (sent_count == -1) { 129d8701185SJon Doron ret = HV_STATUS_INSUFFICIENT_MEMORY; 130d8701185SJon Doron goto cleanup; 131d8701185SJon Doron } 132d8701185SJon Doron 133d8701185SJon Doron *pending_count = count - (sent_count + udp_data_ofs); 134d8701185SJon Doron ret = HV_STATUS_SUCCESS; 135d8701185SJon Doron cleanup: 136d8701185SJon Doron if (debug_data) { 137d8701185SJon Doron cpu_physical_memory_unmap(debug_data, count, 0, data_len); 138d8701185SJon Doron } 139d8701185SJon Doron 140d8701185SJon Doron return ret; 141d8701185SJon Doron } 142d8701185SJon Doron 143d8701185SJon Doron #define UDP_PKT_HEADER_SIZE \ 144d8701185SJon Doron (sizeof(struct eth_header) + sizeof(struct ip_header) +\ 145d8701185SJon Doron sizeof(struct udp_header)) 146d8701185SJon Doron 147d8701185SJon Doron static bool create_udp_pkt(HvSynDbg *syndbg, void *pkt, uint32_t pkt_len, 148d8701185SJon Doron void *udp_data, uint32_t udp_data_len) 149d8701185SJon Doron { 150d8701185SJon Doron struct udp_header *udp_part; 151d8701185SJon Doron 152d8701185SJon Doron if (pkt_len < (UDP_PKT_HEADER_SIZE + udp_data_len)) { 153d8701185SJon Doron return false; 154d8701185SJon Doron } 155d8701185SJon Doron 156d8701185SJon Doron /* Setup the eth */ 157d8701185SJon Doron memset(&PKT_GET_ETH_HDR(pkt)->h_source, 0, ETH_ALEN); 158d8701185SJon Doron memset(&PKT_GET_ETH_HDR(pkt)->h_dest, 0, ETH_ALEN); 159d8701185SJon Doron PKT_GET_ETH_HDR(pkt)->h_proto = cpu_to_be16(ETH_P_IP); 160d8701185SJon Doron 161d8701185SJon Doron /* Setup the ip */ 162d8701185SJon Doron PKT_GET_IP_HDR(pkt)->ip_ver_len = 163d8701185SJon Doron (4 << 4) | (sizeof(struct ip_header) >> 2); 164d8701185SJon Doron PKT_GET_IP_HDR(pkt)->ip_tos = 0; 165d8701185SJon Doron PKT_GET_IP_HDR(pkt)->ip_id = 0; 166d8701185SJon Doron PKT_GET_IP_HDR(pkt)->ip_off = 0; 167d8701185SJon Doron PKT_GET_IP_HDR(pkt)->ip_ttl = 64; /* IPDEFTTL */ 168d8701185SJon Doron PKT_GET_IP_HDR(pkt)->ip_p = IP_PROTO_UDP; 169d8701185SJon Doron PKT_GET_IP_HDR(pkt)->ip_src = syndbg->servaddr.sin_addr.s_addr; 170d8701185SJon Doron PKT_GET_IP_HDR(pkt)->ip_dst = syndbg->target_ip; 171d8701185SJon Doron PKT_GET_IP_HDR(pkt)->ip_len = 172d8701185SJon Doron cpu_to_be16(sizeof(struct ip_header) + sizeof(struct udp_header) + 173d8701185SJon Doron udp_data_len); 174d8701185SJon Doron eth_fix_ip4_checksum(PKT_GET_IP_HDR(pkt), PKT_GET_IP_HDR_LEN(pkt)); 175d8701185SJon Doron 176d8701185SJon Doron udp_part = (struct udp_header *)((uintptr_t)pkt + 177d8701185SJon Doron sizeof(struct eth_header) + 178d8701185SJon Doron PKT_GET_IP_HDR_LEN(pkt)); 179d8701185SJon Doron udp_part->uh_sport = syndbg->servaddr.sin_port; 180d8701185SJon Doron udp_part->uh_dport = syndbg->servaddr.sin_port; 181d8701185SJon Doron udp_part->uh_ulen = cpu_to_be16(sizeof(struct udp_header) + udp_data_len); 182d8701185SJon Doron memcpy(udp_part + 1, udp_data, udp_data_len); 183d8701185SJon Doron net_checksum_calculate(pkt, UDP_PKT_HEADER_SIZE + udp_data_len, CSUM_ALL); 184d8701185SJon Doron return true; 185d8701185SJon Doron } 186d8701185SJon Doron 187d8701185SJon Doron static uint16_t handle_recv_msg(HvSynDbg *syndbg, uint64_t outgpa, 188d8701185SJon Doron uint32_t count, bool is_raw, uint32_t options, 189d8701185SJon Doron uint64_t timeout, uint32_t *retrieved_count) 190d8701185SJon Doron { 191d8701185SJon Doron uint16_t ret; 192d8701185SJon Doron uint8_t data_buf[TARGET_PAGE_SIZE - UDP_PKT_HEADER_SIZE]; 193d8701185SJon Doron hwaddr out_len; 194d8701185SJon Doron void *out_data; 195d8701185SJon Doron ssize_t recv_byte_count; 196d8701185SJon Doron 197d8701185SJon Doron /* TODO: Handle options and timeout */ 198d8701185SJon Doron (void)options; 199d8701185SJon Doron (void)timeout; 200d8701185SJon Doron 201d8701185SJon Doron if (!syndbg->has_data_pending) { 202d8701185SJon Doron recv_byte_count = 0; 203d8701185SJon Doron } else { 204d8701185SJon Doron recv_byte_count = recv(syndbg->socket, data_buf, 205d8701185SJon Doron MIN(sizeof(data_buf), count), MSG_WAITALL); 206d8701185SJon Doron if (recv_byte_count == -1) { 207d8701185SJon Doron return HV_STATUS_INVALID_PARAMETER; 208d8701185SJon Doron } 209d8701185SJon Doron } 210d8701185SJon Doron 211d8701185SJon Doron if (!recv_byte_count) { 212d8701185SJon Doron *retrieved_count = 0; 213d8701185SJon Doron return HV_STATUS_NO_DATA; 214d8701185SJon Doron } 215d8701185SJon Doron 216d8701185SJon Doron set_pending_state(syndbg, false); 217d8701185SJon Doron 218d8701185SJon Doron out_len = recv_byte_count; 219d8701185SJon Doron if (is_raw) { 220d8701185SJon Doron out_len += UDP_PKT_HEADER_SIZE; 221d8701185SJon Doron } 222d8701185SJon Doron out_data = cpu_physical_memory_map(outgpa, &out_len, 1); 223d8701185SJon Doron if (!out_data) { 224d8701185SJon Doron return HV_STATUS_INSUFFICIENT_MEMORY; 225d8701185SJon Doron } 226d8701185SJon Doron 227d8701185SJon Doron if (is_raw && 228d8701185SJon Doron !create_udp_pkt(syndbg, out_data, 229d8701185SJon Doron recv_byte_count + UDP_PKT_HEADER_SIZE, 230d8701185SJon Doron data_buf, recv_byte_count)) { 231d8701185SJon Doron ret = HV_STATUS_INSUFFICIENT_MEMORY; 232d8701185SJon Doron goto cleanup_out_data; 233d8701185SJon Doron } else if (!is_raw) { 234d8701185SJon Doron memcpy(out_data, data_buf, recv_byte_count); 235d8701185SJon Doron } 236d8701185SJon Doron 237d8701185SJon Doron *retrieved_count = recv_byte_count; 238d8701185SJon Doron if (is_raw) { 239d8701185SJon Doron *retrieved_count += UDP_PKT_HEADER_SIZE; 240d8701185SJon Doron } 241d8701185SJon Doron ret = HV_STATUS_SUCCESS; 242d8701185SJon Doron 243d8701185SJon Doron cleanup_out_data: 244d8701185SJon Doron cpu_physical_memory_unmap(out_data, out_len, 1, out_len); 245d8701185SJon Doron return ret; 246d8701185SJon Doron } 247d8701185SJon Doron 248d8701185SJon Doron static uint16_t hv_syndbg_handler(void *context, HvSynDbgMsg *msg) 249d8701185SJon Doron { 250d8701185SJon Doron HvSynDbg *syndbg = context; 251d8701185SJon Doron uint16_t ret = HV_STATUS_INVALID_HYPERCALL_CODE; 252d8701185SJon Doron 253d8701185SJon Doron switch (msg->type) { 254d8701185SJon Doron case HV_SYNDBG_MSG_CONNECTION_INFO: 255d8701185SJon Doron msg->u.connection_info.host_ip = 256d8701185SJon Doron ntohl(syndbg->servaddr.sin_addr.s_addr); 257d8701185SJon Doron msg->u.connection_info.host_port = 258d8701185SJon Doron ntohs(syndbg->servaddr.sin_port); 259d8701185SJon Doron ret = HV_STATUS_SUCCESS; 260d8701185SJon Doron break; 261d8701185SJon Doron case HV_SYNDBG_MSG_SEND: 262d8701185SJon Doron ret = handle_send_msg(syndbg, msg->u.send.buf_gpa, msg->u.send.count, 263d8701185SJon Doron msg->u.send.is_raw, &msg->u.send.pending_count); 264d8701185SJon Doron break; 265d8701185SJon Doron case HV_SYNDBG_MSG_RECV: 266d8701185SJon Doron ret = handle_recv_msg(syndbg, msg->u.recv.buf_gpa, msg->u.recv.count, 267d8701185SJon Doron msg->u.recv.is_raw, msg->u.recv.options, 268d8701185SJon Doron msg->u.recv.timeout, 269d8701185SJon Doron &msg->u.recv.retrieved_count); 270d8701185SJon Doron break; 271d8701185SJon Doron case HV_SYNDBG_MSG_SET_PENDING_PAGE: 272d8701185SJon Doron syndbg->pending_page_gpa = msg->u.pending_page.buf_gpa; 273d8701185SJon Doron ret = HV_STATUS_SUCCESS; 274d8701185SJon Doron break; 275d8701185SJon Doron case HV_SYNDBG_MSG_QUERY_OPTIONS: 276d8701185SJon Doron msg->u.query_options.options = 0; 277d8701185SJon Doron if (syndbg->use_hcalls) { 278d8701185SJon Doron msg->u.query_options.options = HV_X64_SYNDBG_OPTION_USE_HCALLS; 279d8701185SJon Doron } 280d8701185SJon Doron ret = HV_STATUS_SUCCESS; 281d8701185SJon Doron break; 282d8701185SJon Doron default: 283d8701185SJon Doron break; 284d8701185SJon Doron } 285d8701185SJon Doron 286d8701185SJon Doron return ret; 287d8701185SJon Doron } 288d8701185SJon Doron 289d8701185SJon Doron static void hv_syndbg_recv_event(void *opaque) 290d8701185SJon Doron { 291d8701185SJon Doron HvSynDbg *syndbg = opaque; 292d8701185SJon Doron struct timeval tv; 293d8701185SJon Doron fd_set rfds; 294d8701185SJon Doron 295d8701185SJon Doron tv.tv_sec = 0; 296d8701185SJon Doron tv.tv_usec = 0; 297d8701185SJon Doron FD_ZERO(&rfds); 298d8701185SJon Doron FD_SET(syndbg->socket, &rfds); 299d8701185SJon Doron if (select(syndbg->socket + 1, &rfds, NULL, NULL, &tv) > 0) { 300d8701185SJon Doron set_pending_state(syndbg, true); 301d8701185SJon Doron } 302d8701185SJon Doron } 303d8701185SJon Doron 304d8701185SJon Doron static void hv_syndbg_realize(DeviceState *dev, Error **errp) 305d8701185SJon Doron { 306d8701185SJon Doron HvSynDbg *syndbg = HVSYNDBG(dev); 307d8701185SJon Doron 308d8701185SJon Doron if (!hv_syndbg_find()) { 309d8701185SJon Doron error_setg(errp, "at most one %s device is permitted", TYPE_HV_SYNDBG); 310d8701185SJon Doron return; 311d8701185SJon Doron } 312d8701185SJon Doron 313d8701185SJon Doron if (!vmbus_bridge_find()) { 314d8701185SJon Doron error_setg(errp, "%s device requires vmbus-bridge device", 315d8701185SJon Doron TYPE_HV_SYNDBG); 316d8701185SJon Doron return; 317d8701185SJon Doron } 318d8701185SJon Doron 319d8701185SJon Doron /* Parse and host_ip */ 320d8701185SJon Doron if (qemu_isdigit(syndbg->host_ip[0])) { 321d8701185SJon Doron syndbg->servaddr.sin_addr.s_addr = inet_addr(syndbg->host_ip); 322d8701185SJon Doron } else { 323d8701185SJon Doron struct hostent *he = gethostbyname(syndbg->host_ip); 324d8701185SJon Doron if (!he) { 325d8701185SJon Doron error_setg(errp, "%s failed to resolve host name %s", 326d8701185SJon Doron TYPE_HV_SYNDBG, syndbg->host_ip); 327d8701185SJon Doron return; 328d8701185SJon Doron } 329d8701185SJon Doron syndbg->servaddr.sin_addr = *(struct in_addr *)he->h_addr; 330d8701185SJon Doron } 331d8701185SJon Doron 332d8701185SJon Doron syndbg->socket = socket(AF_INET, SOCK_DGRAM, 0); 333d8701185SJon Doron if (syndbg->socket < 0) { 334d8701185SJon Doron error_setg(errp, "%s failed to create socket", TYPE_HV_SYNDBG); 335d8701185SJon Doron return; 336d8701185SJon Doron } 337d8701185SJon Doron 338ff5927baSMarc-André Lureau qemu_socket_set_nonblock(syndbg->socket); 339d8701185SJon Doron 340d8701185SJon Doron syndbg->servaddr.sin_port = htons(syndbg->host_port); 341d8701185SJon Doron syndbg->servaddr.sin_family = AF_INET; 342d8701185SJon Doron if (connect(syndbg->socket, (struct sockaddr *)&syndbg->servaddr, 343d8701185SJon Doron sizeof(syndbg->servaddr)) < 0) { 34425657fc6SMarc-André Lureau close(syndbg->socket); 345d8701185SJon Doron error_setg(errp, "%s failed to connect to socket", TYPE_HV_SYNDBG); 346d8701185SJon Doron return; 347d8701185SJon Doron } 348d8701185SJon Doron 349d8701185SJon Doron syndbg->pending_page_gpa = 0; 350d8701185SJon Doron syndbg->has_data_pending = false; 351d8701185SJon Doron hyperv_set_syndbg_handler(hv_syndbg_handler, syndbg); 352d8701185SJon Doron qemu_set_fd_handler(syndbg->socket, hv_syndbg_recv_event, NULL, syndbg); 353d8701185SJon Doron } 354d8701185SJon Doron 355d8701185SJon Doron static void hv_syndbg_unrealize(DeviceState *dev) 356d8701185SJon Doron { 357d8701185SJon Doron HvSynDbg *syndbg = HVSYNDBG(dev); 358d8701185SJon Doron 359d8701185SJon Doron if (syndbg->socket > 0) { 360d8701185SJon Doron qemu_set_fd_handler(syndbg->socket, NULL, NULL, NULL); 36125657fc6SMarc-André Lureau close(syndbg->socket); 362d8701185SJon Doron } 363d8701185SJon Doron } 364d8701185SJon Doron 365d8701185SJon Doron static const VMStateDescription vmstate_hv_syndbg = { 366d8701185SJon Doron .name = TYPE_HV_SYNDBG, 367d8701185SJon Doron .unmigratable = 1, 368d8701185SJon Doron }; 369d8701185SJon Doron 370e923f5e1SRichard Henderson static const Property hv_syndbg_properties[] = { 371d8701185SJon Doron DEFINE_PROP_STRING("host_ip", HvSynDbg, host_ip), 372d8701185SJon Doron DEFINE_PROP_UINT16("host_port", HvSynDbg, host_port, 50000), 373d8701185SJon Doron DEFINE_PROP_BOOL("use_hcalls", HvSynDbg, use_hcalls, false), 374d8701185SJon Doron }; 375d8701185SJon Doron 376*12d1a768SPhilippe Mathieu-Daudé static void hv_syndbg_class_init(ObjectClass *klass, const void *data) 377d8701185SJon Doron { 378d8701185SJon Doron DeviceClass *dc = DEVICE_CLASS(klass); 379d8701185SJon Doron 380d8701185SJon Doron device_class_set_props(dc, hv_syndbg_properties); 381d8701185SJon Doron dc->fw_name = TYPE_HV_SYNDBG; 382d8701185SJon Doron dc->vmsd = &vmstate_hv_syndbg; 383d8701185SJon Doron dc->realize = hv_syndbg_realize; 384d8701185SJon Doron dc->unrealize = hv_syndbg_unrealize; 385d8701185SJon Doron dc->user_creatable = true; 386d8701185SJon Doron set_bit(DEVICE_CATEGORY_MISC, dc->categories); 387d8701185SJon Doron } 388d8701185SJon Doron 389d8701185SJon Doron static const TypeInfo hv_syndbg_type_info = { 390d8701185SJon Doron .name = TYPE_HV_SYNDBG, 391d8701185SJon Doron .parent = TYPE_DEVICE, 392d8701185SJon Doron .instance_size = sizeof(HvSynDbg), 393d8701185SJon Doron .class_init = hv_syndbg_class_init, 394d8701185SJon Doron }; 395d8701185SJon Doron 396d8701185SJon Doron static void hv_syndbg_register_types(void) 397d8701185SJon Doron { 398d8701185SJon Doron type_register_static(&hv_syndbg_type_info); 399d8701185SJon Doron } 400d8701185SJon Doron 401d8701185SJon Doron type_init(hv_syndbg_register_types) 402