2
0

syndbg.c 12 KB

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