| /* | 
 |  * QEMU System Emulator | 
 |  * | 
 |  * Copyright (c) 2003-2008 Fabrice Bellard | 
 |  * Copyright (c) 2022 Red Hat, Inc. | 
 |  * | 
 |  * Permission is hereby granted, free of charge, to any person obtaining a copy | 
 |  * of this software and associated documentation files (the "Software"), to deal | 
 |  * in the Software without restriction, including without limitation the rights | 
 |  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | 
 |  * copies of the Software, and to permit persons to whom the Software is | 
 |  * furnished to do so, subject to the following conditions: | 
 |  * | 
 |  * The above copyright notice and this permission notice shall be included in | 
 |  * all copies or substantial portions of the Software. | 
 |  * | 
 |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | 
 |  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
 |  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL | 
 |  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | 
 |  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | 
 |  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | 
 |  * THE SOFTWARE. | 
 |  */ | 
 |  | 
 | #include "qemu/osdep.h" | 
 |  | 
 | #include "net/net.h" | 
 | #include "clients.h" | 
 | #include "monitor/monitor.h" | 
 | #include "qapi/error.h" | 
 | #include "qemu/error-report.h" | 
 | #include "qemu/option.h" | 
 | #include "qemu/sockets.h" | 
 | #include "qemu/iov.h" | 
 | #include "qemu/main-loop.h" | 
 | #include "qemu/cutils.h" | 
 |  | 
 | typedef struct NetDgramState { | 
 |     NetClientState nc; | 
 |     int fd; | 
 |     SocketReadState rs; | 
 |     bool read_poll;               /* waiting to receive data? */ | 
 |     bool write_poll;              /* waiting to transmit data? */ | 
 |     /* contains destination iff connectionless */ | 
 |     struct sockaddr *dest_addr; | 
 |     socklen_t dest_len; | 
 | } NetDgramState; | 
 |  | 
 | static void net_dgram_send(void *opaque); | 
 | static void net_dgram_writable(void *opaque); | 
 |  | 
 | static void net_dgram_update_fd_handler(NetDgramState *s) | 
 | { | 
 |     qemu_set_fd_handler(s->fd, | 
 |                         s->read_poll ? net_dgram_send : NULL, | 
 |                         s->write_poll ? net_dgram_writable : NULL, | 
 |                         s); | 
 | } | 
 |  | 
 | static void net_dgram_read_poll(NetDgramState *s, bool enable) | 
 | { | 
 |     s->read_poll = enable; | 
 |     net_dgram_update_fd_handler(s); | 
 | } | 
 |  | 
 | static void net_dgram_write_poll(NetDgramState *s, bool enable) | 
 | { | 
 |     s->write_poll = enable; | 
 |     net_dgram_update_fd_handler(s); | 
 | } | 
 |  | 
 | static void net_dgram_writable(void *opaque) | 
 | { | 
 |     NetDgramState *s = opaque; | 
 |  | 
 |     net_dgram_write_poll(s, false); | 
 |  | 
 |     qemu_flush_queued_packets(&s->nc); | 
 | } | 
 |  | 
 | static ssize_t net_dgram_receive(NetClientState *nc, | 
 |                                  const uint8_t *buf, size_t size) | 
 | { | 
 |     NetDgramState *s = DO_UPCAST(NetDgramState, nc, nc); | 
 |     ssize_t ret; | 
 |  | 
 |     do { | 
 |         if (s->dest_addr) { | 
 |             ret = sendto(s->fd, buf, size, 0, s->dest_addr, s->dest_len); | 
 |         } else { | 
 |             ret = send(s->fd, buf, size, 0); | 
 |         } | 
 |     } while (ret == -1 && errno == EINTR); | 
 |  | 
 |     if (ret == -1 && errno == EAGAIN) { | 
 |         net_dgram_write_poll(s, true); | 
 |         return 0; | 
 |     } | 
 |     return ret; | 
 | } | 
 |  | 
 | static void net_dgram_send_completed(NetClientState *nc, ssize_t len) | 
 | { | 
 |     NetDgramState *s = DO_UPCAST(NetDgramState, nc, nc); | 
 |  | 
 |     if (!s->read_poll) { | 
 |         net_dgram_read_poll(s, true); | 
 |     } | 
 | } | 
 |  | 
 | static void net_dgram_rs_finalize(SocketReadState *rs) | 
 | { | 
 |     NetDgramState *s = container_of(rs, NetDgramState, rs); | 
 |  | 
 |     if (qemu_send_packet_async(&s->nc, rs->buf, | 
 |                                rs->packet_len, | 
 |                                net_dgram_send_completed) == 0) { | 
 |         net_dgram_read_poll(s, false); | 
 |     } | 
 | } | 
 |  | 
 | static void net_dgram_send(void *opaque) | 
 | { | 
 |     NetDgramState *s = opaque; | 
 |     int size; | 
 |  | 
 |     size = recv(s->fd, s->rs.buf, sizeof(s->rs.buf), 0); | 
 |     if (size < 0) { | 
 |         return; | 
 |     } | 
 |     if (size == 0) { | 
 |         /* end of connection */ | 
 |         net_dgram_read_poll(s, false); | 
 |         net_dgram_write_poll(s, false); | 
 |         return; | 
 |     } | 
 |     if (qemu_send_packet_async(&s->nc, s->rs.buf, size, | 
 |                                net_dgram_send_completed) == 0) { | 
 |         net_dgram_read_poll(s, false); | 
 |     } | 
 | } | 
 |  | 
 | static int net_dgram_mcast_create(struct sockaddr_in *mcastaddr, | 
 |                                   struct in_addr *localaddr, | 
 |                                   Error **errp) | 
 | { | 
 |     struct ip_mreq imr; | 
 |     int fd; | 
 |     int val, ret; | 
 | #ifdef __OpenBSD__ | 
 |     unsigned char loop; | 
 | #else | 
 |     int loop; | 
 | #endif | 
 |  | 
 |     if (!IN_MULTICAST(ntohl(mcastaddr->sin_addr.s_addr))) { | 
 |         error_setg(errp, "specified mcastaddr %s (0x%08x) " | 
 |                    "does not contain a multicast address", | 
 |                    inet_ntoa(mcastaddr->sin_addr), | 
 |                    (int)ntohl(mcastaddr->sin_addr.s_addr)); | 
 |         return -1; | 
 |     } | 
 |  | 
 |     fd = qemu_socket(PF_INET, SOCK_DGRAM, 0); | 
 |     if (fd < 0) { | 
 |         error_setg_errno(errp, errno, "can't create datagram socket"); | 
 |         return -1; | 
 |     } | 
 |  | 
 |     /* | 
 |      * Allow multiple sockets to bind the same multicast ip and port by setting | 
 |      * SO_REUSEADDR. This is the only situation where SO_REUSEADDR should be set | 
 |      * on windows. Use socket_set_fast_reuse otherwise as it sets SO_REUSEADDR | 
 |      * only on posix systems. | 
 |      */ | 
 |     val = 1; | 
 |     ret = setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &val, sizeof(val)); | 
 |     if (ret < 0) { | 
 |         error_setg_errno(errp, errno, "can't set socket option SO_REUSEADDR"); | 
 |         goto fail; | 
 |     } | 
 |  | 
 |     ret = bind(fd, (struct sockaddr *)mcastaddr, sizeof(*mcastaddr)); | 
 |     if (ret < 0) { | 
 |         error_setg_errno(errp, errno, "can't bind ip=%s to socket", | 
 |                          inet_ntoa(mcastaddr->sin_addr)); | 
 |         goto fail; | 
 |     } | 
 |  | 
 |     /* Add host to multicast group */ | 
 |     imr.imr_multiaddr = mcastaddr->sin_addr; | 
 |     if (localaddr) { | 
 |         imr.imr_interface = *localaddr; | 
 |     } else { | 
 |         imr.imr_interface.s_addr = htonl(INADDR_ANY); | 
 |     } | 
 |  | 
 |     ret = setsockopt(fd, IPPROTO_IP, IP_ADD_MEMBERSHIP, | 
 |                      &imr, sizeof(struct ip_mreq)); | 
 |     if (ret < 0) { | 
 |         error_setg_errno(errp, errno, | 
 |                          "can't add socket to multicast group %s", | 
 |                          inet_ntoa(imr.imr_multiaddr)); | 
 |         goto fail; | 
 |     } | 
 |  | 
 |     /* Force mcast msgs to loopback (eg. several QEMUs in same host */ | 
 |     loop = 1; | 
 |     ret = setsockopt(fd, IPPROTO_IP, IP_MULTICAST_LOOP, | 
 |                      &loop, sizeof(loop)); | 
 |     if (ret < 0) { | 
 |         error_setg_errno(errp, errno, | 
 |                          "can't force multicast message to loopback"); | 
 |         goto fail; | 
 |     } | 
 |  | 
 |     /* If a bind address is given, only send packets from that address */ | 
 |     if (localaddr != NULL) { | 
 |         ret = setsockopt(fd, IPPROTO_IP, IP_MULTICAST_IF, | 
 |                          localaddr, sizeof(*localaddr)); | 
 |         if (ret < 0) { | 
 |             error_setg_errno(errp, errno, | 
 |                              "can't set the default network send interface"); | 
 |             goto fail; | 
 |         } | 
 |     } | 
 |  | 
 |     qemu_socket_set_nonblock(fd); | 
 |     return fd; | 
 | fail: | 
 |     if (fd >= 0) { | 
 |         close(fd); | 
 |     } | 
 |     return -1; | 
 | } | 
 |  | 
 | static void net_dgram_cleanup(NetClientState *nc) | 
 | { | 
 |     NetDgramState *s = DO_UPCAST(NetDgramState, nc, nc); | 
 |     if (s->fd != -1) { | 
 |         net_dgram_read_poll(s, false); | 
 |         net_dgram_write_poll(s, false); | 
 |         close(s->fd); | 
 |         s->fd = -1; | 
 |     } | 
 |     g_free(s->dest_addr); | 
 |     s->dest_addr = NULL; | 
 |     s->dest_len = 0; | 
 | } | 
 |  | 
 | static NetClientInfo net_dgram_socket_info = { | 
 |     .type = NET_CLIENT_DRIVER_DGRAM, | 
 |     .size = sizeof(NetDgramState), | 
 |     .receive = net_dgram_receive, | 
 |     .cleanup = net_dgram_cleanup, | 
 | }; | 
 |  | 
 | static NetDgramState *net_dgram_fd_init(NetClientState *peer, | 
 |                                         const char *model, | 
 |                                         const char *name, | 
 |                                         int fd, | 
 |                                         Error **errp) | 
 | { | 
 |     NetClientState *nc; | 
 |     NetDgramState *s; | 
 |  | 
 |     nc = qemu_new_net_client(&net_dgram_socket_info, peer, model, name); | 
 |  | 
 |     s = DO_UPCAST(NetDgramState, nc, nc); | 
 |  | 
 |     s->fd = fd; | 
 |     net_socket_rs_init(&s->rs, net_dgram_rs_finalize, false); | 
 |     net_dgram_read_poll(s, true); | 
 |  | 
 |     return s; | 
 | } | 
 |  | 
 | static int net_dgram_mcast_init(NetClientState *peer, | 
 |                                 const char *model, | 
 |                                 const char *name, | 
 |                                 SocketAddress *remote, | 
 |                                 SocketAddress *local, | 
 |                                 Error **errp) | 
 | { | 
 |     NetDgramState *s; | 
 |     int fd, ret; | 
 |     struct sockaddr_in *saddr; | 
 |  | 
 |     if (remote->type != SOCKET_ADDRESS_TYPE_INET) { | 
 |         error_setg(errp, "multicast only support inet type"); | 
 |         return -1; | 
 |     } | 
 |  | 
 |     saddr = g_new(struct sockaddr_in, 1); | 
 |     if (convert_host_port(saddr, remote->u.inet.host, remote->u.inet.port, | 
 |                           errp) < 0) { | 
 |         g_free(saddr); | 
 |         return -1; | 
 |     } | 
 |  | 
 |     if (!local) { | 
 |         fd = net_dgram_mcast_create(saddr, NULL, errp); | 
 |         if (fd < 0) { | 
 |             g_free(saddr); | 
 |             return -1; | 
 |         } | 
 |     } else { | 
 |         switch (local->type) { | 
 |         case SOCKET_ADDRESS_TYPE_INET: { | 
 |             struct in_addr localaddr; | 
 |  | 
 |             if (inet_aton(local->u.inet.host, &localaddr) == 0) { | 
 |                 g_free(saddr); | 
 |                 error_setg(errp, "localaddr '%s' is not a valid IPv4 address", | 
 |                            local->u.inet.host); | 
 |                 return -1; | 
 |             } | 
 |  | 
 |             fd = net_dgram_mcast_create(saddr, &localaddr, errp); | 
 |             if (fd < 0) { | 
 |                 g_free(saddr); | 
 |                 return -1; | 
 |             } | 
 |             break; | 
 |         } | 
 |         case SOCKET_ADDRESS_TYPE_FD: { | 
 |             int newfd; | 
 |  | 
 |             fd = monitor_fd_param(monitor_cur(), local->u.fd.str, errp); | 
 |             if (fd == -1) { | 
 |                 g_free(saddr); | 
 |                 return -1; | 
 |             } | 
 |             ret = qemu_socket_try_set_nonblock(fd); | 
 |             if (ret < 0) { | 
 |                 g_free(saddr); | 
 |                 error_setg_errno(errp, -ret, "%s: Can't use file descriptor %d", | 
 |                                  name, fd); | 
 |                 return -1; | 
 |             } | 
 |  | 
 |             /* | 
 |              * fd passed: multicast: "learn" dest_addr address from bound | 
 |              * address and save it. Because this may be "shared" socket from a | 
 |              * "master" process, datagrams would be recv() by ONLY ONE process: | 
 |              * we must "clone" this dgram socket --jjo | 
 |              */ | 
 |  | 
 |             saddr = g_new(struct sockaddr_in, 1); | 
 |  | 
 |             if (convert_host_port(saddr, local->u.inet.host, local->u.inet.port, | 
 |                                   errp) < 0) { | 
 |                 g_free(saddr); | 
 |                 close(fd); | 
 |                 return -1; | 
 |             } | 
 |  | 
 |             /* must be bound */ | 
 |             if (saddr->sin_addr.s_addr == 0) { | 
 |                 error_setg(errp, "can't setup multicast destination address"); | 
 |                 g_free(saddr); | 
 |                 close(fd); | 
 |                 return -1; | 
 |             } | 
 |             /* clone dgram socket */ | 
 |             newfd = net_dgram_mcast_create(saddr, NULL, errp); | 
 |             if (newfd < 0) { | 
 |                 g_free(saddr); | 
 |                 close(fd); | 
 |                 return -1; | 
 |             } | 
 |             /* clone newfd to fd, close newfd */ | 
 |             dup2(newfd, fd); | 
 |             close(newfd); | 
 |             break; | 
 |         } | 
 |         default: | 
 |             g_free(saddr); | 
 |             error_setg(errp, "only support inet or fd type for local"); | 
 |             return -1; | 
 |         } | 
 |     } | 
 |  | 
 |     s = net_dgram_fd_init(peer, model, name, fd, errp); | 
 |     if (!s) { | 
 |         g_free(saddr); | 
 |         return -1; | 
 |     } | 
 |  | 
 |     g_assert(s->dest_addr == NULL); | 
 |     s->dest_addr = (struct sockaddr *)saddr; | 
 |     s->dest_len = sizeof(*saddr); | 
 |  | 
 |     if (!local) { | 
 |         qemu_set_info_str(&s->nc, "mcast=%s:%d", | 
 |                           inet_ntoa(saddr->sin_addr), | 
 |                           ntohs(saddr->sin_port)); | 
 |     } else { | 
 |         switch (local->type) { | 
 |         case SOCKET_ADDRESS_TYPE_INET: | 
 |             qemu_set_info_str(&s->nc, "mcast=%s:%d", | 
 |                               inet_ntoa(saddr->sin_addr), | 
 |                               ntohs(saddr->sin_port)); | 
 |             break; | 
 |         case SOCKET_ADDRESS_TYPE_FD: | 
 |             qemu_set_info_str(&s->nc, "fd=%d (cloned mcast=%s:%d)", | 
 |                               fd, inet_ntoa(saddr->sin_addr), | 
 |                               ntohs(saddr->sin_port)); | 
 |             break; | 
 |         default: | 
 |             g_assert_not_reached(); | 
 |         } | 
 |     } | 
 |  | 
 |     return 0; | 
 |  | 
 | } | 
 |  | 
 |  | 
 | int net_init_dgram(const Netdev *netdev, const char *name, | 
 |                    NetClientState *peer, Error **errp) | 
 | { | 
 |     NetDgramState *s; | 
 |     int fd, ret; | 
 |     SocketAddress *remote, *local; | 
 |     struct sockaddr *dest_addr; | 
 |     struct sockaddr_in laddr_in, raddr_in; | 
 |     struct sockaddr_un laddr_un, raddr_un; | 
 |     socklen_t dest_len; | 
 |  | 
 |     assert(netdev->type == NET_CLIENT_DRIVER_DGRAM); | 
 |  | 
 |     remote = netdev->u.dgram.remote; | 
 |     local = netdev->u.dgram.local; | 
 |  | 
 |     /* detect multicast address */ | 
 |     if (remote && remote->type == SOCKET_ADDRESS_TYPE_INET) { | 
 |         struct sockaddr_in mcastaddr; | 
 |  | 
 |         if (convert_host_port(&mcastaddr, remote->u.inet.host, | 
 |                               remote->u.inet.port, errp) < 0) { | 
 |             return -1; | 
 |         } | 
 |  | 
 |         if (IN_MULTICAST(ntohl(mcastaddr.sin_addr.s_addr))) { | 
 |             return net_dgram_mcast_init(peer, "dram", name, remote, local, | 
 |                                            errp); | 
 |         } | 
 |     } | 
 |  | 
 |     /* unicast address */ | 
 |     if (!local) { | 
 |         error_setg(errp, "dgram requires local= parameter"); | 
 |         return -1; | 
 |     } | 
 |  | 
 |     if (remote) { | 
 |         if (local->type == SOCKET_ADDRESS_TYPE_FD) { | 
 |             error_setg(errp, "don't set remote with local.fd"); | 
 |             return -1; | 
 |         } | 
 |         if (remote->type != local->type) { | 
 |             error_setg(errp, "remote and local types must be the same"); | 
 |             return -1; | 
 |         } | 
 |     } else { | 
 |         if (local->type != SOCKET_ADDRESS_TYPE_FD) { | 
 |             error_setg(errp, | 
 |                        "type=inet or type=unix requires remote parameter"); | 
 |             return -1; | 
 |         } | 
 |     } | 
 |  | 
 |     switch (local->type) { | 
 |     case SOCKET_ADDRESS_TYPE_INET: | 
 |         if (convert_host_port(&laddr_in, local->u.inet.host, local->u.inet.port, | 
 |                               errp) < 0) { | 
 |             return -1; | 
 |         } | 
 |  | 
 |         if (convert_host_port(&raddr_in, remote->u.inet.host, | 
 |                               remote->u.inet.port, errp) < 0) { | 
 |             return -1; | 
 |         } | 
 |  | 
 |         fd = qemu_socket(PF_INET, SOCK_DGRAM, 0); | 
 |         if (fd < 0) { | 
 |             error_setg_errno(errp, errno, "can't create datagram socket"); | 
 |             return -1; | 
 |         } | 
 |  | 
 |         ret = socket_set_fast_reuse(fd); | 
 |         if (ret < 0) { | 
 |             error_setg_errno(errp, errno, | 
 |                              "can't set socket option SO_REUSEADDR"); | 
 |             close(fd); | 
 |             return -1; | 
 |         } | 
 |         ret = bind(fd, (struct sockaddr *)&laddr_in, sizeof(laddr_in)); | 
 |         if (ret < 0) { | 
 |             error_setg_errno(errp, errno, "can't bind ip=%s to socket", | 
 |                              inet_ntoa(laddr_in.sin_addr)); | 
 |             close(fd); | 
 |             return -1; | 
 |         } | 
 |         qemu_socket_set_nonblock(fd); | 
 |  | 
 |         dest_len = sizeof(raddr_in); | 
 |         dest_addr = g_malloc(dest_len); | 
 |         memcpy(dest_addr, &raddr_in, dest_len); | 
 |         break; | 
 |     case SOCKET_ADDRESS_TYPE_UNIX: | 
 |         ret = unlink(local->u.q_unix.path); | 
 |         if (ret < 0 && errno != ENOENT) { | 
 |             error_setg_errno(errp, errno, "failed to unlink socket %s", | 
 |                              local->u.q_unix.path); | 
 |             return -1; | 
 |         } | 
 |  | 
 |         laddr_un.sun_family = PF_UNIX; | 
 |         ret = snprintf(laddr_un.sun_path, sizeof(laddr_un.sun_path), "%s", | 
 |                        local->u.q_unix.path); | 
 |         if (ret < 0 || ret >= sizeof(laddr_un.sun_path)) { | 
 |             error_setg(errp, "UNIX socket path '%s' is too long", | 
 |                        local->u.q_unix.path); | 
 |             error_append_hint(errp, "Path must be less than %zu bytes\n", | 
 |                               sizeof(laddr_un.sun_path)); | 
 |         } | 
 |  | 
 |         raddr_un.sun_family = PF_UNIX; | 
 |         ret = snprintf(raddr_un.sun_path, sizeof(raddr_un.sun_path), "%s", | 
 |                        remote->u.q_unix.path); | 
 |         if (ret < 0 || ret >= sizeof(raddr_un.sun_path)) { | 
 |             error_setg(errp, "UNIX socket path '%s' is too long", | 
 |                        remote->u.q_unix.path); | 
 |             error_append_hint(errp, "Path must be less than %zu bytes\n", | 
 |                               sizeof(raddr_un.sun_path)); | 
 |         } | 
 |  | 
 |         fd = qemu_socket(PF_UNIX, SOCK_DGRAM, 0); | 
 |         if (fd < 0) { | 
 |             error_setg_errno(errp, errno, "can't create datagram socket"); | 
 |             return -1; | 
 |         } | 
 |  | 
 |         ret = bind(fd, (struct sockaddr *)&laddr_un, sizeof(laddr_un)); | 
 |         if (ret < 0) { | 
 |             error_setg_errno(errp, errno, "can't bind unix=%s to socket", | 
 |                              laddr_un.sun_path); | 
 |             close(fd); | 
 |             return -1; | 
 |         } | 
 |         qemu_socket_set_nonblock(fd); | 
 |  | 
 |         dest_len = sizeof(raddr_un); | 
 |         dest_addr = g_malloc(dest_len); | 
 |         memcpy(dest_addr, &raddr_un, dest_len); | 
 |         break; | 
 |     case SOCKET_ADDRESS_TYPE_FD: | 
 |         fd = monitor_fd_param(monitor_cur(), local->u.fd.str, errp); | 
 |         if (fd == -1) { | 
 |             return -1; | 
 |         } | 
 |         ret = qemu_socket_try_set_nonblock(fd); | 
 |         if (ret < 0) { | 
 |             error_setg_errno(errp, -ret, "%s: Can't use file descriptor %d", | 
 |                              name, fd); | 
 |             return -1; | 
 |         } | 
 |         dest_addr = NULL; | 
 |         dest_len = 0; | 
 |         break; | 
 |     default: | 
 |         error_setg(errp, "only support inet or fd type for local"); | 
 |         return -1; | 
 |     } | 
 |  | 
 |     s = net_dgram_fd_init(peer, "dgram", name, fd, errp); | 
 |     if (!s) { | 
 |         return -1; | 
 |     } | 
 |  | 
 |     if (remote) { | 
 |         g_assert(s->dest_addr == NULL); | 
 |         s->dest_addr = dest_addr; | 
 |         s->dest_len = dest_len; | 
 |     } | 
 |  | 
 |     switch (local->type) { | 
 |     case SOCKET_ADDRESS_TYPE_INET: | 
 |         qemu_set_info_str(&s->nc, "udp=%s:%d/%s:%d", | 
 |                           inet_ntoa(laddr_in.sin_addr), | 
 |                           ntohs(laddr_in.sin_port), | 
 |                           inet_ntoa(raddr_in.sin_addr), | 
 |                           ntohs(raddr_in.sin_port)); | 
 |         break; | 
 |     case SOCKET_ADDRESS_TYPE_UNIX: | 
 |         qemu_set_info_str(&s->nc, "udp=%s:%s", | 
 |                           laddr_un.sun_path, raddr_un.sun_path); | 
 |         break; | 
 |     case SOCKET_ADDRESS_TYPE_FD: { | 
 |         SocketAddress *sa; | 
 |         SocketAddressType sa_type; | 
 |  | 
 |         sa = socket_local_address(fd, errp); | 
 |         if (sa) { | 
 |             sa_type = sa->type; | 
 |             qapi_free_SocketAddress(sa); | 
 |  | 
 |             qemu_set_info_str(&s->nc, "fd=%d %s", fd, | 
 |                               SocketAddressType_str(sa_type)); | 
 |         } else { | 
 |             qemu_set_info_str(&s->nc, "fd=%d", fd); | 
 |         } | 
 |         break; | 
 |     } | 
 |     default: | 
 |         g_assert_not_reached(); | 
 |     } | 
 |  | 
 |     return 0; | 
 | } |