VirtualBox

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

Last change on this file since 61239 was 61024, checked in by vboxsync, 9 years ago

copyright header fix

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 8.3 KB
Line 
1/* $Id: UsbTestServiceGadgetHostUsbIp.cpp 61024 2016-05-18 07:45:51Z 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 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* Constants And Macros, Structures and Typedefs *
49*********************************************************************************************************************************/
50
51/**
52 * Internal UTS gadget host instance data.
53 */
54typedef struct UTSGADGETHOSTTYPEINT
55{
56 /** Handle to the USB/IP daemon process. */
57 RTPROCESS hProcUsbIp;
58} UTSGADGETHOSTTYPEINT;
59
60/** Default port of the USB/IP server. */
61#define UTS_GADGET_HOST_USBIP_PORT_DEF 3240
62
63/*********************************************************************************************************************************
64* Internal Functions *
65*********************************************************************************************************************************/
66
67
68/**
69 * Worker for binding/unbinding the given gadget from the USB/IP server.
70 *
71 * @returns IPRT status code.
72 * @param pThis The gadget host instance.
73 * @param hGadget The gadget handle.
74 * @param fBind Flag whether to do a bind or unbind.
75 */
76static int usbGadgetHostUsbIpBindUnbind(PUTSGADGETHOSTTYPEINT pThis, UTSGADGET hGadget, bool fBind)
77{
78 uint32_t uBusId, uDevId;
79 char aszBus[32];
80
81 uBusId = utsGadgetGetBusId(hGadget);
82 uDevId = utsGadgetGetDevId(hGadget);
83
84 /* Create the busid argument string. */
85 size_t cbRet = RTStrPrintf(&aszBus[0], RT_ELEMENTS(aszBus), "%u-%u", uBusId, uDevId);
86 if (cbRet == RT_ELEMENTS(aszBus))
87 return VERR_BUFFER_OVERFLOW;
88
89 /* Bind to the USB/IP server. */
90 RTPROCESS hProcUsbIp = NIL_RTPROCESS;
91 const char *apszArgv[5];
92
93 apszArgv[0] = "usbip";
94 apszArgv[1] = fBind ? "bind" : "unbind";
95 apszArgv[2] = "-b";
96 apszArgv[3] = &aszBus[0];
97 apszArgv[4] = NULL;
98
99 int rc = RTProcCreate("usbip", apszArgv, RTENV_DEFAULT, RTPROC_FLAGS_SEARCH_PATH, &hProcUsbIp);
100 if (RT_SUCCESS(rc))
101 {
102 RTPROCSTATUS ProcSts;
103 rc = RTProcWait(hProcUsbIp, RTPROCWAIT_FLAGS_BLOCK, &ProcSts);
104 if (RT_SUCCESS(rc))
105 {
106 /* Evaluate the process status. */
107 if ( ProcSts.enmReason != RTPROCEXITREASON_NORMAL
108 || ProcSts.iStatus != 0)
109 rc = VERR_UNRESOLVED_ERROR; /** @todo: Log and give finer grained status code. */
110 }
111 }
112
113 return rc;
114}
115
116/**
117 * @interface_method_impl{UTSGADGETHOSTIF,pfnInit}
118 */
119static DECLCALLBACK(int) utsGadgetHostUsbIpInit(PUTSGADGETHOSTTYPEINT pIf, PCUTSGADGETCFGITEM paCfg)
120{
121 int rc = VINF_SUCCESS;
122 uint16_t uPort = 0;
123
124 pIf->hProcUsbIp = NIL_RTPROCESS;
125
126 rc = utsGadgetCfgQueryU16Def(paCfg, "UsbIp/Port", &uPort, UTS_GADGET_HOST_USBIP_PORT_DEF);
127 if (RT_SUCCESS(rc))
128 {
129 /* Make sure the kernel drivers are loaded. */
130 rc = utsPlatformModuleLoad("usbip-core", NULL, 0);
131 if (RT_SUCCESS(rc))
132 {
133 rc = utsPlatformModuleLoad("usbip-host", NULL, 0);
134 if (RT_SUCCESS(rc))
135 {
136 char aszPort[10];
137 char aszPidFile[64];
138 const char *apszArgv[6];
139
140 RTStrPrintf(aszPort, RT_ELEMENTS(aszPort), "%u", uPort);
141 RTStrPrintf(aszPidFile, RT_ELEMENTS(aszPidFile), "/var/run/usbipd-%u.pid", uPort);
142 /* Start the USB/IP server process. */
143 apszArgv[0] = "usbipd";
144 apszArgv[1] = "--tcp-port";
145 apszArgv[2] = aszPort;
146 apszArgv[3] = "--pid";
147 apszArgv[4] = aszPidFile;
148 apszArgv[5] = NULL;
149 rc = RTProcCreate("usbipd", apszArgv, RTENV_DEFAULT, RTPROC_FLAGS_SEARCH_PATH, &pIf->hProcUsbIp);
150 if (RT_SUCCESS(rc))
151 {
152 /* Wait for a bit to make sure the server started up successfully. */
153 uint64_t tsStart = RTTimeMilliTS();
154 do
155 {
156 RTPROCSTATUS ProcSts;
157 rc = RTProcWait(pIf->hProcUsbIp, RTPROCWAIT_FLAGS_NOBLOCK, &ProcSts);
158 if (rc != VERR_PROCESS_RUNNING)
159 {
160 rc = VERR_INVALID_HANDLE;
161 break;
162 }
163 RTThreadSleep(1);
164 rc = VINF_SUCCESS;
165 } while (RTTimeMilliTS() - tsStart < 2 * 1000); /* 2 seconds. */
166 }
167 }
168 }
169 }
170
171 return rc;
172}
173
174
175/**
176 * @interface_method_impl{UTSGADGETHOSTIF,pfnTerm}
177 */
178static DECLCALLBACK(void) utsGadgetHostUsbIpTerm(PUTSGADGETHOSTTYPEINT pIf)
179{
180 /* Kill the process and wait for it to terminate. */
181 RTProcTerminate(pIf->hProcUsbIp);
182
183 RTPROCSTATUS ProcSts;
184 RTProcWait(pIf->hProcUsbIp, RTPROCWAIT_FLAGS_BLOCK, &ProcSts);
185}
186
187
188/**
189 * @interface_method_impl{UTSGADGETHOSTIF,pfnGadgetAdd}
190 */
191static DECLCALLBACK(int) utsGadgetHostUsbIpGadgetAdd(PUTSGADGETHOSTTYPEINT pIf, UTSGADGET hGadget)
192{
193 /* Nothing to do so far. */
194 return VINF_SUCCESS;
195}
196
197
198/**
199 * @interface_method_impl{UTSGADGETHOSTIF,pfnGadgetRemove}
200 */
201static DECLCALLBACK(int) utsGadgetHostUsbIpGadgetRemove(PUTSGADGETHOSTTYPEINT pIf, UTSGADGET hGadget)
202{
203 /* Nothing to do so far. */
204 return VINF_SUCCESS;
205}
206
207
208/**
209 * @interface_method_impl{UTSGADGETHOSTIF,pfnGadgetConnect}
210 */
211static DECLCALLBACK(int) utsGadgetHostUsbIpGadgetConnect(PUTSGADGETHOSTTYPEINT pIf, UTSGADGET hGadget)
212{
213 return usbGadgetHostUsbIpBindUnbind(pIf, hGadget, true /* fBind */);
214}
215
216
217/**
218 * @interface_method_impl{UTSGADGETHOSTIF,pfnGadgetDisconnect}
219 */
220static DECLCALLBACK(int) utsGadgetHostUsbIpGadgetDisconnect(PUTSGADGETHOSTTYPEINT pIf, UTSGADGET hGadget)
221{
222 return usbGadgetHostUsbIpBindUnbind(pIf, hGadget, false /* fBind */);
223}
224
225
226
227/**
228 * The gadget host interface callback table.
229 */
230const UTSGADGETHOSTIF g_UtsGadgetHostIfUsbIp =
231{
232 /** enmType */
233 UTSGADGETHOSTTYPE_USBIP,
234 /** pszDesc */
235 "UTS USB/IP gadget host",
236 /** cbIf */
237 sizeof(UTSGADGETHOSTTYPEINT),
238 /** pfnInit */
239 utsGadgetHostUsbIpInit,
240 /** pfnTerm */
241 utsGadgetHostUsbIpTerm,
242 /** pfnGadgetAdd */
243 utsGadgetHostUsbIpGadgetAdd,
244 /** pfnGadgetRemove */
245 utsGadgetHostUsbIpGadgetRemove,
246 /** pfnGadgetConnect */
247 utsGadgetHostUsbIpGadgetConnect,
248 /** pfnGadgetDisconnect */
249 utsGadgetHostUsbIpGadgetDisconnect
250};
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