VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/utils/usb/UsbTestServiceGadgetHostUsbIp.cpp@ 74978

Last change on this file since 74978 was 69111, checked in by vboxsync, 7 years ago

(C) year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 8.4 KB
Line 
1/* $Id: UsbTestServiceGadgetHostUsbIp.cpp 69111 2017-10-17 14:26:02Z vboxsync $ */
2/** @file
3 * UsbTestServ - Remote USB test configuration and execution server, USB gadget host interface
4 * for USB/IP.
5 */
6
7/*
8 * Copyright (C) 2016-2017 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 *
18 * The contents of this file may alternatively be used under the terms
19 * of the Common Development and Distribution License Version 1.0
20 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
21 * VirtualBox OSE distribution, in which case the provisions of the
22 * CDDL are applicable instead of those of the GPL.
23 *
24 * You may elect to license modified versions of this file under the
25 * terms and conditions of either the GPL or the CDDL or both.
26 */
27
28
29/*********************************************************************************************************************************
30* Header Files *
31*********************************************************************************************************************************/
32
33#include <iprt/asm.h>
34#include <iprt/cdefs.h>
35#include <iprt/ctype.h>
36#include <iprt/env.h>
37#include <iprt/mem.h>
38#include <iprt/process.h>
39#include <iprt/string.h>
40#include <iprt/thread.h>
41#include <iprt/time.h>
42#include <iprt/types.h>
43
44#include "UsbTestServiceGadgetHostInternal.h"
45#include "UsbTestServicePlatform.h"
46
47
48/*********************************************************************************************************************************
49* Constants And Macros, Structures and Typedefs *
50*********************************************************************************************************************************/
51
52/**
53 * Internal UTS gadget host instance data.
54 */
55typedef struct UTSGADGETHOSTTYPEINT
56{
57 /** Handle to the USB/IP daemon process. */
58 RTPROCESS hProcUsbIp;
59} UTSGADGETHOSTTYPEINT;
60
61/** Default port of the USB/IP server. */
62#define UTS_GADGET_HOST_USBIP_PORT_DEF 3240
63
64
65/*********************************************************************************************************************************
66* Internal Functions *
67*********************************************************************************************************************************/
68
69
70/**
71 * Worker for binding/unbinding the given gadget from the USB/IP server.
72 *
73 * @returns IPRT status code.
74 * @param pThis The gadget host instance.
75 * @param hGadget The gadget handle.
76 * @param fBind Flag whether to do a bind or unbind.
77 */
78static int usbGadgetHostUsbIpBindUnbind(PUTSGADGETHOSTTYPEINT pThis, UTSGADGET hGadget, bool fBind)
79{
80 RT_NOREF1(pThis);
81 uint32_t uBusId, uDevId;
82 char aszBus[32];
83
84 uBusId = utsGadgetGetBusId(hGadget);
85 uDevId = utsGadgetGetDevId(hGadget);
86
87 /* Create the busid argument string. */
88 size_t cbRet = RTStrPrintf(&aszBus[0], RT_ELEMENTS(aszBus), "%u-%u", uBusId, uDevId);
89 if (cbRet == RT_ELEMENTS(aszBus))
90 return VERR_BUFFER_OVERFLOW;
91
92 /* Bind to the USB/IP server. */
93 RTPROCESS hProcUsbIp = NIL_RTPROCESS;
94 const char *apszArgv[5];
95
96 apszArgv[0] = "usbip";
97 apszArgv[1] = fBind ? "bind" : "unbind";
98 apszArgv[2] = "-b";
99 apszArgv[3] = &aszBus[0];
100 apszArgv[4] = NULL;
101
102 int rc = RTProcCreate("usbip", apszArgv, RTENV_DEFAULT, RTPROC_FLAGS_SEARCH_PATH, &hProcUsbIp);
103 if (RT_SUCCESS(rc))
104 {
105 RTPROCSTATUS ProcSts;
106 rc = RTProcWait(hProcUsbIp, RTPROCWAIT_FLAGS_BLOCK, &ProcSts);
107 if (RT_SUCCESS(rc))
108 {
109 /* Evaluate the process status. */
110 if ( ProcSts.enmReason != RTPROCEXITREASON_NORMAL
111 || ProcSts.iStatus != 0)
112 rc = VERR_UNRESOLVED_ERROR; /** @todo Log and give finer grained status code. */
113 }
114 }
115
116 return rc;
117}
118
119/**
120 * @interface_method_impl{UTSGADGETHOSTIF,pfnInit}
121 */
122static DECLCALLBACK(int) utsGadgetHostUsbIpInit(PUTSGADGETHOSTTYPEINT pIf, PCUTSGADGETCFGITEM paCfg)
123{
124 int rc = VINF_SUCCESS;
125 uint16_t uPort = 0;
126
127 pIf->hProcUsbIp = NIL_RTPROCESS;
128
129 rc = utsGadgetCfgQueryU16Def(paCfg, "UsbIp/Port", &uPort, UTS_GADGET_HOST_USBIP_PORT_DEF);
130 if (RT_SUCCESS(rc))
131 {
132 /* Make sure the kernel drivers are loaded. */
133 rc = utsPlatformModuleLoad("usbip-core", NULL, 0);
134 if (RT_SUCCESS(rc))
135 {
136 rc = utsPlatformModuleLoad("usbip-host", NULL, 0);
137 if (RT_SUCCESS(rc))
138 {
139 char aszPort[10];
140 char aszPidFile[64];
141 const char *apszArgv[6];
142
143 RTStrPrintf(aszPort, RT_ELEMENTS(aszPort), "%u", uPort);
144 RTStrPrintf(aszPidFile, RT_ELEMENTS(aszPidFile), "/var/run/usbipd-%u.pid", uPort);
145 /* Start the USB/IP server process. */
146 apszArgv[0] = "usbipd";
147 apszArgv[1] = "--tcp-port";
148 apszArgv[2] = aszPort;
149 apszArgv[3] = "--pid";
150 apszArgv[4] = aszPidFile;
151 apszArgv[5] = NULL;
152 rc = RTProcCreate("usbipd", apszArgv, RTENV_DEFAULT, RTPROC_FLAGS_SEARCH_PATH, &pIf->hProcUsbIp);
153 if (RT_SUCCESS(rc))
154 {
155 /* Wait for a bit to make sure the server started up successfully. */
156 uint64_t tsStart = RTTimeMilliTS();
157 do
158 {
159 RTPROCSTATUS ProcSts;
160 rc = RTProcWait(pIf->hProcUsbIp, RTPROCWAIT_FLAGS_NOBLOCK, &ProcSts);
161 if (rc != VERR_PROCESS_RUNNING)
162 {
163 rc = VERR_INVALID_HANDLE;
164 break;
165 }
166 RTThreadSleep(1);
167 rc = VINF_SUCCESS;
168 } while (RTTimeMilliTS() - tsStart < 2 * 1000); /* 2 seconds. */
169 }
170 }
171 }
172 }
173
174 return rc;
175}
176
177
178/**
179 * @interface_method_impl{UTSGADGETHOSTIF,pfnTerm}
180 */
181static DECLCALLBACK(void) utsGadgetHostUsbIpTerm(PUTSGADGETHOSTTYPEINT pIf)
182{
183 /* Kill the process and wait for it to terminate. */
184 RTProcTerminate(pIf->hProcUsbIp);
185
186 RTPROCSTATUS ProcSts;
187 RTProcWait(pIf->hProcUsbIp, RTPROCWAIT_FLAGS_BLOCK, &ProcSts);
188}
189
190
191/**
192 * @interface_method_impl{UTSGADGETHOSTIF,pfnGadgetAdd}
193 */
194static DECLCALLBACK(int) utsGadgetHostUsbIpGadgetAdd(PUTSGADGETHOSTTYPEINT pIf, UTSGADGET hGadget)
195{
196 /* Nothing to do so far. */
197 RT_NOREF2(pIf, hGadget);
198 return VINF_SUCCESS;
199}
200
201
202/**
203 * @interface_method_impl{UTSGADGETHOSTIF,pfnGadgetRemove}
204 */
205static DECLCALLBACK(int) utsGadgetHostUsbIpGadgetRemove(PUTSGADGETHOSTTYPEINT pIf, UTSGADGET hGadget)
206{
207 /* Nothing to do so far. */
208 RT_NOREF2(pIf, hGadget);
209 return VINF_SUCCESS;
210}
211
212
213/**
214 * @interface_method_impl{UTSGADGETHOSTIF,pfnGadgetConnect}
215 */
216static DECLCALLBACK(int) utsGadgetHostUsbIpGadgetConnect(PUTSGADGETHOSTTYPEINT pIf, UTSGADGET hGadget)
217{
218 return usbGadgetHostUsbIpBindUnbind(pIf, hGadget, true /* fBind */);
219}
220
221
222/**
223 * @interface_method_impl{UTSGADGETHOSTIF,pfnGadgetDisconnect}
224 */
225static DECLCALLBACK(int) utsGadgetHostUsbIpGadgetDisconnect(PUTSGADGETHOSTTYPEINT pIf, UTSGADGET hGadget)
226{
227 return usbGadgetHostUsbIpBindUnbind(pIf, hGadget, false /* fBind */);
228}
229
230
231
232/**
233 * The gadget host interface callback table.
234 */
235const UTSGADGETHOSTIF g_UtsGadgetHostIfUsbIp =
236{
237 /** enmType */
238 UTSGADGETHOSTTYPE_USBIP,
239 /** pszDesc */
240 "UTS USB/IP gadget host",
241 /** cbIf */
242 sizeof(UTSGADGETHOSTTYPEINT),
243 /** pfnInit */
244 utsGadgetHostUsbIpInit,
245 /** pfnTerm */
246 utsGadgetHostUsbIpTerm,
247 /** pfnGadgetAdd */
248 utsGadgetHostUsbIpGadgetAdd,
249 /** pfnGadgetRemove */
250 utsGadgetHostUsbIpGadgetRemove,
251 /** pfnGadgetConnect */
252 utsGadgetHostUsbIpGadgetConnect,
253 /** pfnGadgetDisconnect */
254 utsGadgetHostUsbIpGadgetDisconnect
255};
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette