// SPDX-License-Identifier: GPL-2.0-or-later /* * Copyright (C) 2011 matt mooney <mfm@muteddisk.com> * 2005-2007 Takahiro Hirofuchi * Copyright (C) 2015-2016 Samsung Electronics * Igor Kotrasinski <i.kotrasinsk@samsung.com> * Krzysztof Opasiak <k.opasiak@samsung.com> */ #ifdef HAVE_CONFIG_H #include "../config.h" #endif #define _GNU_SOURCE #include <errno.h> #include <unistd.h> #include <netdb.h> #include <string.h> #include <stdlib.h> #include <sys/types.h> #include <sys/stat.h> #include <arpa/inet.h> #include <sys/socket.h> #include <netinet/in.h> #ifdef HAVE_LIBWRAP #include <tcpd.h> #endif #include <getopt.h> #include <signal.h> #include <poll.h> #include "usbip_host_driver.h" #include "usbip_host_common.h" #include "usbip_device_driver.h" #include "usbip_common.h" #include "usbip_network.h" #include "list.h" #undef PROGNAME #define PROGNAME "usbipd" #define MAXSOCKFD 20 #define MAIN_LOOP_TIMEOUT 10 #define DEFAULT_PID_FILE "/var/run/" PROGNAME ".pid" static const char usbip_version_string[] = PACKAGE_STRING; static const char usbipd_help_string[] = "usage: usbipd [options]\n" "\n" " -4, --ipv4\n" " Bind to IPv4. Default is both.\n" "\n" " -6, --ipv6\n" " Bind to IPv6. Default is both.\n" "\n" " -e, --device\n" " Run in device mode.\n" " Rather than drive an attached device, create\n" " a virtual UDC to bind gadgets to.\n" "\n" " -D, --daemon\n" " Run as a daemon process.\n" "\n" " -d, --debug\n" " Print debugging information.\n" "\n" " -PFILE, --pid FILE\n" " Write process id to FILE.\n" " If no FILE specified, use " DEFAULT_PID_FILE "\n" "\n" " -tPORT, --tcp-port PORT\n" " Listen on TCP/IP port PORT.\n" "\n" " -h, --help\n" " Print this help.\n" "\n" " -v, --version\n" " Show version.\n"; static struct usbip_host_driver *driver; static void usbipd_help(void) { printf("%s\n", usbipd_help_string); } static int recv_request_import(int sockfd) { struct op_import_request req; struct usbip_exported_device *edev; struct usbip_usb_device pdu_udev; struct list_head *i; int found = 0; int status = ST_OK; int rc; memset(&req, 0, sizeof(req)); rc = usbip_net_recv(sockfd, &req, sizeof(req)); if (rc < 0) { dbg("usbip_net_recv failed: import request"); return -1; } PACK_OP_IMPORT_REQUEST(0, &req); list_for_each(i, &driver->edev_list) { edev = list_entry(i, struct usbip_exported_device, node); if (!strncmp(req.busid, edev->udev.busid, SYSFS_BUS_ID_SIZE)) { info("found requested device: %s", req.busid); found = 1; break; } } if (found) { /* should set TCP_NODELAY for usbip */ usbip_net_set_nodelay(sockfd); /* export device needs a TCP/IP socket descriptor */ status = usbip_export_device(edev, sockfd); if (status < 0) status = ST_NA; } else { info("requested device not found: %s", req.busid); status = ST_NODEV; } rc = usbip_net_send_op_common(sockfd, OP_REP_IMPORT, status); if (rc < 0) { dbg("usbip_net_send_op_common failed: %#0x", OP_REP_IMPORT); return -1; } if (status) { dbg("import request busid %s: failed", req.busid); return -1; } memcpy(&pdu_udev, &edev->udev, sizeof(pdu_udev)); usbip_net_pack_usb_device(1, &pdu_udev); rc = usbip_net_send(sockfd, &pdu_udev, sizeof(pdu_udev)); if (rc < 0) { dbg("usbip_net_send failed: devinfo"); return -1; } dbg("import request busid %s: complete", req.busid); return 0; } static int send_reply_devlist(int connfd) { struct usbip_exported_device *edev; struct usbip_usb_device pdu_udev; struct usbip_usb_interface pdu_uinf; struct op_devlist_reply reply; struct list_head *j; int rc, i; /* * Exclude devices that are already exported to a client from * the exportable device list to avoid: * - import requests for devices that are exported only to * fail the request. * - revealing devices that are imported by a client to * another client. */ reply.ndev = 0; /* number of exported devices */ list_for_each(j, &driver->edev_list) { edev = list_entry(j, struct usbip_exported_device, node); if (edev->status != SDEV_ST_USED) reply.ndev += 1; } info("exportable devices: %d", reply.ndev); rc = usbip_net_send_op_common(connfd, OP_REP_DEVLIST, ST_OK); if (rc < 0) { dbg("usbip_net_send_op_common failed: %#0x", OP_REP_DEVLIST); return -1; } PACK_OP_DEVLIST_REPLY(1, &reply); rc = usbip_net_send(connfd, &reply, sizeof(reply)); if (rc < 0) { dbg("usbip_net_send failed: %#0x", OP_REP_DEVLIST); return -1; } list_for_each(j, &driver->edev_list) { edev = list_entry(j, struct usbip_exported_device, node); if (edev->status == SDEV_ST_USED) continue; dump_usb_device(&edev->udev); memcpy(&pdu_udev, &edev->udev, sizeof(pdu_udev)); usbip_net_pack_usb_device(1, &pdu_udev); rc = usbip_net_send(connfd, &pdu_udev, sizeof(pdu_udev)); if (rc < 0) { dbg("usbip_net_send failed: pdu_udev"); return -1; } for (i = 0; i < edev->udev.bNumInterfaces; i++) { dump_usb_interface(&edev->uinf[i]); memcpy(&pdu_uinf, &edev->uinf[i], sizeof(pdu_uinf)); usbip_net_pack_usb_interface(1, &pdu_uinf); rc = usbip_net_send(connfd, &pdu_uinf, sizeof(pdu_uinf)); if (rc < 0) { err("usbip_net_send failed: pdu_uinf"); return -1; } } } return 0; } static int recv_request_devlist(int connfd) { struct op_devlist_request req; int rc; memset(&req, 0, sizeof(req)); rc = usbip_net_recv(connfd, &req, sizeof(req)); if (rc < 0) { dbg("usbip_net_recv failed: devlist request"); return -1; } rc = send_reply_devlist(connfd); if (rc < 0) { dbg("send_reply_devlist failed"); return -1; } return 0; } static int recv_pdu(int connfd) { uint16_t code = OP_UNSPEC; int ret; int status; ret = usbip_net_recv_op_common(connfd, &code, &status); if (ret < 0) { dbg("could not receive opcode: %#0x", code); return -1; } ret = usbip_refresh_device_list(driver); if (ret < 0) { dbg("could not refresh device list: %d", ret); return -1; } info("received request: %#0x(%d)", code, connfd); switch (code) { case OP_REQ_DEVLIST: ret = recv_request_devlist(connfd); break; case OP_REQ_IMPORT: ret = recv_request_import(connfd); break; case OP_REQ_DEVINFO: case OP_REQ_CRYPKEY: default: err("received an unknown opcode: %#0x", code); ret = -1; } if (ret == 0) info("request %#0x(%d): complete", code, connfd); else info("request %#0x(%d): failed", code, connfd); return ret; } #ifdef HAVE_LIBWRAP static int tcpd_auth(int connfd) { struct request_info request; int rc; request_init(&request, RQ_DAEMON, PROGNAME, RQ_FILE, connfd, 0); fromhost(&request); rc = hosts_access(&request); if (rc == 0) return -1; return 0; } #endif static int do_accept(int listenfd) { int connfd; struct sockaddr_storage ss; socklen_t len = sizeof(ss); char host[NI_MAXHOST], port[NI_MAXSERV]; int rc; memset(&ss, 0, sizeof(ss)); connfd = accept(listenfd, (struct sockaddr *)&ss, &len); if (connfd < 0) { err("failed to accept connection"); return -1; } rc = getnameinfo((struct sockaddr *)&ss, len, host, sizeof(host), port, sizeof(port), NI_NUMERICHOST | NI_NUMERICSERV); if (rc) err("getnameinfo: %s", gai_strerror(rc)); #ifdef HAVE_LIBWRAP rc = tcpd_auth(connfd); if (rc < 0) { info("denied access from %s", host); close(connfd); return -1; } #endif info("connection from %s:%s", host, port); return connfd; } int process_request(int listenfd) { pid_t childpid; int connfd; connfd = do_accept(listenfd); if (connfd < 0) return -1; childpid = fork(); if (childpid == 0) { close(listenfd); recv_pdu(connfd); exit(0); } close(connfd); return 0; } static void addrinfo_to_text(struct addrinfo *ai, char buf[], const size_t buf_size) { char hbuf[NI_MAXHOST]; char sbuf[NI_MAXSERV]; int rc; buf[0] = '\0'; rc = getnameinfo(ai->ai_addr, ai->ai_addrlen, hbuf, sizeof(hbuf), sbuf, sizeof(sbuf), NI_NUMERICHOST | NI_NUMERICSERV); if (rc) err("getnameinfo: %s", gai_strerror(rc)); snprintf(buf, buf_size, "%s:%s", hbuf, sbuf); } static int listen_all_addrinfo(struct addrinfo *ai_head, int sockfdlist[], int maxsockfd) { struct addrinfo *ai; int ret, nsockfd = 0; const size_t ai_buf_size = NI_MAXHOST + NI_MAXSERV + 2; char ai_buf[ai_buf_size]; for (ai = ai_head; ai && nsockfd < maxsockfd; ai = ai->ai_next) { int sock; addrinfo_to_text(ai, ai_buf, ai_buf_size); dbg("opening %s", ai_buf); sock = socket(ai->ai_family, ai->ai_socktype, ai->ai_protocol); if (sock < 0) { err("socket: %s: %d (%s)", ai_buf, errno, strerror(errno)); continue; } usbip_net_set_reuseaddr(sock); usbip_net_set_nodelay(sock); /* We use seperate sockets for IPv4 and IPv6 * (see do_standalone_mode()) */ usbip_net_set_v6only(sock); ret = bind(sock, ai->ai_addr, ai->ai_addrlen); if (ret < 0) { err("bind: %s: %d (%s)", ai_buf, errno, strerror(errno)); close(sock); continue; } ret = listen(sock, SOMAXCONN); if (ret < 0) { err("listen: %s: %d (%s)", ai_buf, errno, strerror(errno)); close(sock); continue; } info("listening on %s", ai_buf); sockfdlist[nsockfd++] = sock; } return nsockfd; } static struct addrinfo *do_getaddrinfo(char *host, int ai_family) { struct addrinfo hints, *ai_head; int rc; memset(&hints, 0, sizeof(hints)); hints.ai_family = ai_family; hints.ai_socktype = SOCK_STREAM; hints.ai_flags = AI_PASSIVE; rc = getaddrinfo(host, usbip_port_string, &hints, &ai_head); if (rc) { err("failed to get a network address %s: %s", usbip_port_string, gai_strerror(rc)); return NULL; } return ai_head; } static void signal_handler(int i) { dbg("received '%s' signal", strsignal(i)); } static void set_signal(void) { struct sigaction act; memset(&act, 0, sizeof(act)); act.sa_handler = signal_handler; sigemptyset(&act.sa_mask); sigaction(SIGTERM, &act, NULL); sigaction(SIGINT, &act, NULL); act.sa_handler = SIG_IGN; sigaction(SIGCHLD, &act, NULL); } static const char *pid_file; static void write_pid_file(void) { if (pid_file) { dbg("creating pid file %s", pid_file); FILE *fp; fp = fopen(pid_file, "w"); if (!fp) { err("pid_file: %s: %d (%s)", pid_file, errno, strerror(errno)); return; } fprintf(fp, "%d\n", getpid()); fclose(fp); } } static void remove_pid_file(void) { if (pid_file) { dbg("removing pid file %s", pid_file); unlink(pid_file); } } static int do_standalone_mode(int daemonize, int ipv4, int ipv6) { struct addrinfo *ai_head; int sockfdlist[MAXSOCKFD]; int nsockfd, family; int i, terminate; struct pollfd *fds; struct timespec timeout; sigset_t sigmask; if (usbip_driver_open(driver)) return -1; if (daemonize) { if (daemon(0, 0) < 0) { err("daemonizing failed: %s", strerror(errno)); usbip_driver_close(driver); return -1; } umask(0); usbip_use_syslog = 1; } set_signal(); write_pid_file(); info("starting " PROGNAME " (%s)", usbip_version_string); /* * To suppress warnings on systems with bindv6only disabled * (default), we use seperate sockets for IPv6 and IPv4 and set * IPV6_V6ONLY on the IPv6 sockets. */ if (ipv4 && ipv6) family = AF_UNSPEC; else if (ipv4) family = AF_INET; else family = AF_INET6; ai_head = do_getaddrinfo(NULL, family); if (!ai_head) { usbip_driver_close(driver); return -1; } nsockfd = listen_all_addrinfo(ai_head, sockfdlist, sizeof(sockfdlist) / sizeof(*sockfdlist)); freeaddrinfo(ai_head); if (nsockfd <= 0) { err("failed to open a listening socket"); usbip_driver_close(driver); return -1; } dbg("listening on %d address%s", nsockfd, (nsockfd == 1) ? "" : "es"); fds = calloc(nsockfd, sizeof(struct pollfd)); for (i = 0; i < nsockfd; i++) { fds[i].fd = sockfdlist[i]; fds[i].events = POLLIN; } timeout.tv_sec = MAIN_LOOP_TIMEOUT; timeout.tv_nsec = 0; sigfillset(&sigmask); sigdelset(&sigmask, SIGTERM); sigdelset(&sigmask, SIGINT); terminate = 0; while (!terminate) { int r; r = ppoll(fds, nsockfd, &timeout, &sigmask); if (r < 0) { dbg("%s", strerror(errno)); terminate = 1; } else if (r) { for (i = 0; i < nsockfd; i++) { if (fds[i].revents & POLLIN) { dbg("read event on fd[%d]=%d", i, sockfdlist[i]); process_request(sockfdlist[i]); } } } else { dbg("heartbeat timeout on ppoll()"); } } info("shutting down " PROGNAME); free(fds); usbip_driver_close(driver); return 0; } int main(int argc, char *argv[]) { static const struct option longopts[] = { { "ipv4", no_argument, NULL, '4' }, { "ipv6", no_argument, NULL, '6' }, { "daemon", no_argument, NULL, 'D' }, { "daemon", no_argument, NULL, 'D' }, { "debug", no_argument, NULL, 'd' }, { "device", no_argument, NULL, 'e' }, { "pid", optional_argument, NULL, 'P' }, { "tcp-port", required_argument, NULL, 't' }, { "help", no_argument, NULL, 'h' }, { "version", no_argument, NULL, 'v' }, { NULL, 0, NULL, 0 } }; enum { cmd_standalone_mode = 1, cmd_help, cmd_version } cmd; int daemonize = 0; int ipv4 = 0, ipv6 = 0; int opt, rc = -1; pid_file = NULL; usbip_use_stderr = 1; usbip_use_syslog = 0; if (geteuid() != 0) err("not running as root?"); cmd = cmd_standalone_mode; driver = &host_driver; for (;;) { opt = getopt_long(argc, argv, "46DdeP::t:hv", longopts, NULL); if (opt == -1) break; switch (opt) { case '4': ipv4 = 1; break; case '6': ipv6 = 1; break; case 'D': daemonize = 1; break; case 'd': usbip_use_debug = 1; break; case 'h': cmd = cmd_help; break; case 'P': pid_file = optarg ? optarg : DEFAULT_PID_FILE; break; case 't': usbip_setup_port_number(optarg); break; case 'v': cmd = cmd_version; break; case 'e': driver = &device_driver; break; case '?': usbipd_help(); default: goto err_out; } } if (!ipv4 && !ipv6) ipv4 = ipv6 = 1; switch (cmd) { case cmd_standalone_mode: rc = do_standalone_mode(daemonize, ipv4, ipv6); remove_pid_file(); break; case cmd_version: printf(PROGNAME " (%s)\n", usbip_version_string); rc = 0; break; case cmd_help: usbipd_help(); rc = 0; break; default: usbipd_help(); goto err_out; } err_out: return (rc > -1 ? EXIT_SUCCESS : EXIT_FAILURE); }