cregit-Linux how code gets into the kernel

Release 4.11 tools/usb/usbip/src/usbipd.c

/*
 * 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>
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program. If not, see <http://www.gnu.org/licenses/>.
 */

#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); }

Contributors

PersonTokensPropCommitsCommitProp
Matt Mooney15100.00%1100.00%
Total15100.00%1100.00%


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 error = 0; 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 */ rc = usbip_export_device(edev, sockfd); if (rc < 0) error = 1; } else { info("requested device not found: %s", req.busid); error = 1; } rc = usbip_net_send_op_common(sockfd, OP_REP_IMPORT, (!error ? ST_OK : ST_NA)); if (rc < 0) { dbg("usbip_net_send_op_common failed: %#0x", OP_REP_IMPORT); return -1; } if (error) { 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; }

Contributors

PersonTokensPropCommitsCommitProp
Matt Mooney17354.23%240.00%
Takahiro Hirofuchi12539.18%120.00%
Valentina Manea195.96%120.00%
Krzysztof Opasiak20.63%120.00%
Total319100.00%5100.00%


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; reply.ndev = 0; /* number of exported devices */ list_for_each(j, &driver->edev_list) { 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); 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; }

Contributors

PersonTokensPropCommitsCommitProp
Valentina Manea11636.94%228.57%
Matt Mooney10834.39%342.86%
Takahiro Hirofuchi8828.03%114.29%
Krzysztof Opasiak20.64%114.29%
Total314100.00%7100.00%


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; }

Contributors

PersonTokensPropCommitsCommitProp
Matt Mooney5259.09%266.67%
Takahiro Hirofuchi3640.91%133.33%
Total88100.00%3100.00%


static int recv_pdu(int connfd) { uint16_t code = OP_UNSPEC; int ret; ret = usbip_net_recv_op_common(connfd, &code); 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; }

Contributors

PersonTokensPropCommitsCommitProp
Takahiro Hirofuchi9560.90%125.00%
Matt Mooney5736.54%250.00%
Krzysztof Opasiak42.56%125.00%
Total156100.00%4100.00%

#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; }

Contributors

PersonTokensPropCommitsCommitProp
Matt Mooney3864.41%150.00%
Takahiro Hirofuchi2135.59%150.00%
Total59100.00%2100.00%

#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; }

Contributors

PersonTokensPropCommitsCommitProp
Matt Mooney13675.14%266.67%
Takahiro Hirofuchi4524.86%133.33%
Total181100.00%3100.00%


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; }

Contributors

PersonTokensPropCommitsCommitProp
Matt Mooney3653.73%150.00%
Ilija Hadzic3146.27%150.00%
Total67100.00%2100.00%


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); }

Contributors

PersonTokensPropCommitsCommitProp
Matt Mooney7777.78%150.00%
Anthony Foiani2222.22%150.00%
Total99100.00%2100.00%


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; }

Contributors

PersonTokensPropCommitsCommitProp
Takahiro Hirofuchi12251.05%116.67%
Anthony Foiani9841.00%233.33%
Dominik Paulus104.18%116.67%
Matt Mooney93.77%233.33%
Total239100.00%6100.00%


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; }

Contributors

PersonTokensPropCommitsCommitProp
Takahiro Hirofuchi4749.47%125.00%
Matt Mooney4648.42%250.00%
Anthony Foiani22.11%125.00%
Total95100.00%4100.00%


static void signal_handler(int i) { dbg("received '%s' signal", strsignal(i)); }

Contributors

PersonTokensPropCommitsCommitProp
Takahiro Hirofuchi1684.21%150.00%
Ilija Hadzic315.79%150.00%
Total19100.00%2100.00%


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(SIGCLD, &act, NULL); }

Contributors

PersonTokensPropCommitsCommitProp
Matt Mooney3344.00%250.00%
Takahiro Hirofuchi2634.67%125.00%
Ilija Hadzic1621.33%125.00%
Total75100.00%4100.00%

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); } }

Contributors

PersonTokensPropCommitsCommitProp
Anthony Foiani6490.14%133.33%
Pawel Lebioda79.86%266.67%
Total71100.00%3100.00%


static void remove_pid_file(void) { if (pid_file) { dbg("removing pid file %s", pid_file); unlink(pid_file); } }

Contributors

PersonTokensPropCommitsCommitProp
Anthony Foiani2388.46%150.00%
Pawel Lebioda311.54%150.00%
Total26100.00%2100.00%


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; }

Contributors

PersonTokensPropCommitsCommitProp
Ilija Hadzic18440.80%112.50%
Takahiro Hirofuchi11425.28%112.50%
Dominik Paulus7616.85%112.50%
Matt Mooney4810.64%112.50%
Krzysztof Opasiak194.21%112.50%
Stefan Reif51.11%112.50%
Anthony Foiani51.11%225.00%
Total451100.00%8100.00%


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); }

Contributors

PersonTokensPropCommitsCommitProp
Takahiro Hirofuchi18344.74%112.50%
Matt Mooney7718.83%112.50%
Dominik Paulus7317.85%112.50%
Anthony Foiani4811.74%225.00%
Igor Kotrasinski204.89%112.50%
Krzysztof Opasiak51.22%112.50%
Ilija Hadzic30.73%112.50%
Total409100.00%8100.00%


Overall Contributors

PersonTokensPropCommitsCommitProp
Takahiro Hirofuchi97834.39%15.00%
Matt Mooney93732.95%420.00%
Anthony Foiani29310.30%525.00%
Ilija Hadzic2468.65%15.00%
Dominik Paulus1665.84%15.00%
Valentina Manea1384.85%210.00%
Krzysztof Opasiak421.48%15.00%
Igor Kotrasinski280.98%15.00%
Pawel Lebioda100.35%210.00%
Stefan Reif50.18%15.00%
David Chang10.04%15.00%
Total2844100.00%20100.00%
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.