VirtualBox

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

Last change on this file since 60680 was 60517, checked in by vboxsync, 9 years ago

ValidationKit/usb: Updates for UsbTestService

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