xref: /qemu/hw/hyperv/syndbg.c (revision f9bb7e53a341d08fd4ec8d7e810ebfd4f6f936bd)
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"
13*3efb9d22SPierrick Bouvier #include "qemu/units.h"
14d8701185SJon Doron #include "qapi/error.h"
15d8701185SJon Doron #include "migration/vmstate.h"
16d8701185SJon Doron #include "hw/qdev-properties.h"
17d8701185SJon Doron #include "hw/loader.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 */
hv_syndbg_find(void)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 
set_pending_state(HvSynDbg * syndbg,bool has_pending)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 
get_udb_pkt_data(void * p,uint32_t len,uint32_t * data_ofs,uint32_t * src_ip)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 
handle_send_msg(HvSynDbg * syndbg,uint64_t ingpa,uint32_t count,bool is_raw,uint32_t * pending_count)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 
create_udp_pkt(HvSynDbg * syndbg,void * pkt,uint32_t pkt_len,void * udp_data,uint32_t udp_data_len)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 
187*3efb9d22SPierrick Bouvier #define MSG_BUFSZ (4 * KiB)
188*3efb9d22SPierrick Bouvier 
handle_recv_msg(HvSynDbg * syndbg,uint64_t outgpa,uint32_t count,bool is_raw,uint32_t options,uint64_t timeout,uint32_t * retrieved_count)189d8701185SJon Doron static uint16_t handle_recv_msg(HvSynDbg *syndbg, uint64_t outgpa,
190d8701185SJon Doron                                 uint32_t count, bool is_raw, uint32_t options,
191d8701185SJon Doron                                 uint64_t timeout, uint32_t *retrieved_count)
192d8701185SJon Doron {
193d8701185SJon Doron     uint16_t ret;
194*3efb9d22SPierrick Bouvier     g_assert(MSG_BUFSZ >= qemu_target_page_size());
195*3efb9d22SPierrick Bouvier     uint8_t data_buf[MSG_BUFSZ];
196d8701185SJon Doron     hwaddr out_len;
197d8701185SJon Doron     void *out_data;
198d8701185SJon Doron     ssize_t recv_byte_count;
199d8701185SJon Doron 
200d8701185SJon Doron     /* TODO: Handle options and timeout */
201d8701185SJon Doron     (void)options;
202d8701185SJon Doron     (void)timeout;
203d8701185SJon Doron 
204d8701185SJon Doron     if (!syndbg->has_data_pending) {
205d8701185SJon Doron         recv_byte_count = 0;
206d8701185SJon Doron     } else {
207d8701185SJon Doron         recv_byte_count = recv(syndbg->socket, data_buf,
208*3efb9d22SPierrick Bouvier                                MIN(MSG_BUFSZ, count), MSG_WAITALL);
209d8701185SJon Doron         if (recv_byte_count == -1) {
210d8701185SJon Doron             return HV_STATUS_INVALID_PARAMETER;
211d8701185SJon Doron         }
212d8701185SJon Doron     }
213d8701185SJon Doron 
214d8701185SJon Doron     if (!recv_byte_count) {
215d8701185SJon Doron         *retrieved_count = 0;
216d8701185SJon Doron         return HV_STATUS_NO_DATA;
217d8701185SJon Doron     }
218d8701185SJon Doron 
219d8701185SJon Doron     set_pending_state(syndbg, false);
220d8701185SJon Doron 
221d8701185SJon Doron     out_len = recv_byte_count;
222d8701185SJon Doron     if (is_raw) {
223d8701185SJon Doron         out_len += UDP_PKT_HEADER_SIZE;
224d8701185SJon Doron     }
225d8701185SJon Doron     out_data = cpu_physical_memory_map(outgpa, &out_len, 1);
226d8701185SJon Doron     if (!out_data) {
227d8701185SJon Doron         return HV_STATUS_INSUFFICIENT_MEMORY;
228d8701185SJon Doron     }
229d8701185SJon Doron 
230d8701185SJon Doron     if (is_raw &&
231d8701185SJon Doron         !create_udp_pkt(syndbg, out_data,
232d8701185SJon Doron                         recv_byte_count + UDP_PKT_HEADER_SIZE,
233d8701185SJon Doron                         data_buf, recv_byte_count)) {
234d8701185SJon Doron         ret = HV_STATUS_INSUFFICIENT_MEMORY;
235d8701185SJon Doron         goto cleanup_out_data;
236d8701185SJon Doron     } else if (!is_raw) {
237d8701185SJon Doron         memcpy(out_data, data_buf, recv_byte_count);
238d8701185SJon Doron     }
239d8701185SJon Doron 
240d8701185SJon Doron     *retrieved_count = recv_byte_count;
241d8701185SJon Doron     if (is_raw) {
242d8701185SJon Doron         *retrieved_count += UDP_PKT_HEADER_SIZE;
243d8701185SJon Doron     }
244d8701185SJon Doron     ret = HV_STATUS_SUCCESS;
245d8701185SJon Doron 
246d8701185SJon Doron cleanup_out_data:
247d8701185SJon Doron     cpu_physical_memory_unmap(out_data, out_len, 1, out_len);
248d8701185SJon Doron     return ret;
249d8701185SJon Doron }
250d8701185SJon Doron 
hv_syndbg_handler(void * context,HvSynDbgMsg * msg)251d8701185SJon Doron static uint16_t hv_syndbg_handler(void *context, HvSynDbgMsg *msg)
252d8701185SJon Doron {
253d8701185SJon Doron     HvSynDbg *syndbg = context;
254d8701185SJon Doron     uint16_t ret = HV_STATUS_INVALID_HYPERCALL_CODE;
255d8701185SJon Doron 
256d8701185SJon Doron     switch (msg->type) {
257d8701185SJon Doron     case HV_SYNDBG_MSG_CONNECTION_INFO:
258d8701185SJon Doron         msg->u.connection_info.host_ip =
259d8701185SJon Doron             ntohl(syndbg->servaddr.sin_addr.s_addr);
260d8701185SJon Doron         msg->u.connection_info.host_port =
261d8701185SJon Doron             ntohs(syndbg->servaddr.sin_port);
262d8701185SJon Doron         ret = HV_STATUS_SUCCESS;
263d8701185SJon Doron         break;
264d8701185SJon Doron     case HV_SYNDBG_MSG_SEND:
265d8701185SJon Doron         ret = handle_send_msg(syndbg, msg->u.send.buf_gpa, msg->u.send.count,
266d8701185SJon Doron                               msg->u.send.is_raw, &msg->u.send.pending_count);
267d8701185SJon Doron         break;
268d8701185SJon Doron     case HV_SYNDBG_MSG_RECV:
269d8701185SJon Doron         ret = handle_recv_msg(syndbg, msg->u.recv.buf_gpa, msg->u.recv.count,
270d8701185SJon Doron                               msg->u.recv.is_raw, msg->u.recv.options,
271d8701185SJon Doron                               msg->u.recv.timeout,
272d8701185SJon Doron                               &msg->u.recv.retrieved_count);
273d8701185SJon Doron         break;
274d8701185SJon Doron     case HV_SYNDBG_MSG_SET_PENDING_PAGE:
275d8701185SJon Doron         syndbg->pending_page_gpa = msg->u.pending_page.buf_gpa;
276d8701185SJon Doron         ret = HV_STATUS_SUCCESS;
277d8701185SJon Doron         break;
278d8701185SJon Doron     case HV_SYNDBG_MSG_QUERY_OPTIONS:
279d8701185SJon Doron         msg->u.query_options.options = 0;
280d8701185SJon Doron         if (syndbg->use_hcalls) {
281d8701185SJon Doron             msg->u.query_options.options = HV_X64_SYNDBG_OPTION_USE_HCALLS;
282d8701185SJon Doron         }
283d8701185SJon Doron         ret = HV_STATUS_SUCCESS;
284d8701185SJon Doron         break;
285d8701185SJon Doron     default:
286d8701185SJon Doron         break;
287d8701185SJon Doron     }
288d8701185SJon Doron 
289d8701185SJon Doron     return ret;
290d8701185SJon Doron }
291d8701185SJon Doron 
hv_syndbg_recv_event(void * opaque)292d8701185SJon Doron static void hv_syndbg_recv_event(void *opaque)
293d8701185SJon Doron {
294d8701185SJon Doron     HvSynDbg *syndbg = opaque;
295d8701185SJon Doron     struct timeval tv;
296d8701185SJon Doron     fd_set rfds;
297d8701185SJon Doron 
298d8701185SJon Doron     tv.tv_sec = 0;
299d8701185SJon Doron     tv.tv_usec = 0;
300d8701185SJon Doron     FD_ZERO(&rfds);
301d8701185SJon Doron     FD_SET(syndbg->socket, &rfds);
302d8701185SJon Doron     if (select(syndbg->socket + 1, &rfds, NULL, NULL, &tv) > 0) {
303d8701185SJon Doron         set_pending_state(syndbg, true);
304d8701185SJon Doron     }
305d8701185SJon Doron }
306d8701185SJon Doron 
hv_syndbg_realize(DeviceState * dev,Error ** errp)307d8701185SJon Doron static void hv_syndbg_realize(DeviceState *dev, Error **errp)
308d8701185SJon Doron {
309d8701185SJon Doron     HvSynDbg *syndbg = HVSYNDBG(dev);
310d8701185SJon Doron 
311d8701185SJon Doron     if (!hv_syndbg_find()) {
312d8701185SJon Doron         error_setg(errp, "at most one %s device is permitted", TYPE_HV_SYNDBG);
313d8701185SJon Doron         return;
314d8701185SJon Doron     }
315d8701185SJon Doron 
316d8701185SJon Doron     if (!vmbus_bridge_find()) {
317d8701185SJon Doron         error_setg(errp, "%s device requires vmbus-bridge device",
318d8701185SJon Doron                    TYPE_HV_SYNDBG);
319d8701185SJon Doron         return;
320d8701185SJon Doron     }
321d8701185SJon Doron 
322d8701185SJon Doron     /* Parse and host_ip */
323d8701185SJon Doron     if (qemu_isdigit(syndbg->host_ip[0])) {
324d8701185SJon Doron         syndbg->servaddr.sin_addr.s_addr = inet_addr(syndbg->host_ip);
325d8701185SJon Doron     } else {
326d8701185SJon Doron         struct hostent *he = gethostbyname(syndbg->host_ip);
327d8701185SJon Doron         if (!he) {
328d8701185SJon Doron             error_setg(errp, "%s failed to resolve host name %s",
329d8701185SJon Doron                        TYPE_HV_SYNDBG, syndbg->host_ip);
330d8701185SJon Doron             return;
331d8701185SJon Doron         }
332d8701185SJon Doron         syndbg->servaddr.sin_addr = *(struct in_addr *)he->h_addr;
333d8701185SJon Doron     }
334d8701185SJon Doron 
335d8701185SJon Doron     syndbg->socket = socket(AF_INET, SOCK_DGRAM, 0);
336d8701185SJon Doron     if (syndbg->socket < 0) {
337d8701185SJon Doron         error_setg(errp, "%s failed to create socket", TYPE_HV_SYNDBG);
338d8701185SJon Doron         return;
339d8701185SJon Doron     }
340d8701185SJon Doron 
341ff5927baSMarc-André Lureau     qemu_socket_set_nonblock(syndbg->socket);
342d8701185SJon Doron 
343d8701185SJon Doron     syndbg->servaddr.sin_port = htons(syndbg->host_port);
344d8701185SJon Doron     syndbg->servaddr.sin_family = AF_INET;
345d8701185SJon Doron     if (connect(syndbg->socket, (struct sockaddr *)&syndbg->servaddr,
346d8701185SJon Doron                 sizeof(syndbg->servaddr)) < 0) {
34725657fc6SMarc-André Lureau         close(syndbg->socket);
348d8701185SJon Doron         error_setg(errp, "%s failed to connect to socket", TYPE_HV_SYNDBG);
349d8701185SJon Doron         return;
350d8701185SJon Doron     }
351d8701185SJon Doron 
352d8701185SJon Doron     syndbg->pending_page_gpa = 0;
353d8701185SJon Doron     syndbg->has_data_pending = false;
354d8701185SJon Doron     hyperv_set_syndbg_handler(hv_syndbg_handler, syndbg);
355d8701185SJon Doron     qemu_set_fd_handler(syndbg->socket, hv_syndbg_recv_event, NULL, syndbg);
356d8701185SJon Doron }
357d8701185SJon Doron 
hv_syndbg_unrealize(DeviceState * dev)358d8701185SJon Doron static void hv_syndbg_unrealize(DeviceState *dev)
359d8701185SJon Doron {
360d8701185SJon Doron     HvSynDbg *syndbg = HVSYNDBG(dev);
361d8701185SJon Doron 
362d8701185SJon Doron     if (syndbg->socket > 0) {
363d8701185SJon Doron         qemu_set_fd_handler(syndbg->socket, NULL, NULL, NULL);
36425657fc6SMarc-André Lureau         close(syndbg->socket);
365d8701185SJon Doron     }
366d8701185SJon Doron }
367d8701185SJon Doron 
368d8701185SJon Doron static const VMStateDescription vmstate_hv_syndbg = {
369d8701185SJon Doron     .name = TYPE_HV_SYNDBG,
370d8701185SJon Doron     .unmigratable = 1,
371d8701185SJon Doron };
372d8701185SJon Doron 
373e923f5e1SRichard Henderson static const Property hv_syndbg_properties[] = {
374d8701185SJon Doron     DEFINE_PROP_STRING("host_ip", HvSynDbg, host_ip),
375d8701185SJon Doron     DEFINE_PROP_UINT16("host_port", HvSynDbg, host_port, 50000),
376d8701185SJon Doron     DEFINE_PROP_BOOL("use_hcalls", HvSynDbg, use_hcalls, false),
377d8701185SJon Doron };
378d8701185SJon Doron 
hv_syndbg_class_init(ObjectClass * klass,const void * data)37912d1a768SPhilippe Mathieu-Daudé static void hv_syndbg_class_init(ObjectClass *klass, const void *data)
380d8701185SJon Doron {
381d8701185SJon Doron     DeviceClass *dc = DEVICE_CLASS(klass);
382d8701185SJon Doron 
383d8701185SJon Doron     device_class_set_props(dc, hv_syndbg_properties);
384d8701185SJon Doron     dc->fw_name = TYPE_HV_SYNDBG;
385d8701185SJon Doron     dc->vmsd = &vmstate_hv_syndbg;
386d8701185SJon Doron     dc->realize = hv_syndbg_realize;
387d8701185SJon Doron     dc->unrealize = hv_syndbg_unrealize;
388d8701185SJon Doron     dc->user_creatable = true;
389d8701185SJon Doron     set_bit(DEVICE_CATEGORY_MISC, dc->categories);
390d8701185SJon Doron }
391d8701185SJon Doron 
392d8701185SJon Doron static const TypeInfo hv_syndbg_type_info = {
393d8701185SJon Doron     .name = TYPE_HV_SYNDBG,
394d8701185SJon Doron     .parent = TYPE_DEVICE,
395d8701185SJon Doron     .instance_size = sizeof(HvSynDbg),
396d8701185SJon Doron     .class_init = hv_syndbg_class_init,
397d8701185SJon Doron };
398d8701185SJon Doron 
hv_syndbg_register_types(void)399d8701185SJon Doron static void hv_syndbg_register_types(void)
400d8701185SJon Doron {
401d8701185SJon Doron     type_register_static(&hv_syndbg_type_info);
402d8701185SJon Doron }
403d8701185SJon Doron 
404d8701185SJon Doron type_init(hv_syndbg_register_types)
405